6
6
package org .opensearch .sql .expression .ip ;
7
7
8
8
import static org .opensearch .sql .data .type .ExprCoreType .BOOLEAN ;
9
+ import static org .opensearch .sql .data .type .ExprCoreType .IP ;
9
10
import static org .opensearch .sql .data .type .ExprCoreType .STRING ;
10
11
import static org .opensearch .sql .expression .function .FunctionDSL .define ;
11
12
import static org .opensearch .sql .expression .function .FunctionDSL .impl ;
12
13
import static org .opensearch .sql .expression .function .FunctionDSL .nullMissingHandling ;
13
14
14
- import inet .ipaddr .AddressStringException ;
15
- import inet .ipaddr .IPAddressString ;
16
- import inet .ipaddr .IPAddressStringParameters ;
15
+ import inet .ipaddr .IPAddress ;
17
16
import lombok .experimental .UtilityClass ;
18
17
import org .opensearch .sql .data .model .ExprValue ;
19
18
import org .opensearch .sql .data .model .ExprValueUtils ;
20
19
import org .opensearch .sql .exception .SemanticCheckException ;
21
20
import org .opensearch .sql .expression .function .BuiltinFunctionName ;
22
21
import org .opensearch .sql .expression .function .BuiltinFunctionRepository ;
23
22
import org .opensearch .sql .expression .function .DefaultFunctionResolver ;
23
+ import org .opensearch .sql .utils .IPUtils ;
24
24
25
25
/** Utility class that defines and registers IP functions. */
26
26
@ UtilityClass
@@ -31,75 +31,30 @@ public void register(BuiltinFunctionRepository repository) {
31
31
}
32
32
33
33
private DefaultFunctionResolver cidrmatch () {
34
-
35
- // TODO #3145: Add support for IP address data type.
36
34
return define (
37
35
BuiltinFunctionName .CIDRMATCH .getName (),
38
- impl (nullMissingHandling (IPFunctions ::exprCidrMatch ), BOOLEAN , STRING , STRING ));
36
+ impl (nullMissingHandling (IPFunctions ::exprCidrMatch ), BOOLEAN , IP , STRING ));
39
37
}
40
38
41
39
/**
42
40
* Returns whether the given IP address is within the specified inclusive CIDR IP address range.
43
41
* Supports both IPv4 and IPv6 addresses.
44
42
*
45
- * @param addressExprValue IP address as a string (e.g. "198.51.100.14" or
46
- * "2001:0db8::ff00:42:8329").
47
- * @param rangeExprValue IP address range in CIDR notation as a string (e.g. "198.51.100.0/24" or
43
+ * @param addressExprValue IP address (e.g. "198.51.100.14" or "2001:0db8::ff00:42:8329").
44
+ * @param rangeExprValue IP address range string in CIDR notation (e.g. "198.51.100.0/24" or
48
45
* "2001:0db8::/32")
49
46
* @return true if the address is in the range; otherwise false.
50
47
* @throws SemanticCheckException if the address or range is not valid, or if they do not use the
51
48
* same version (IPv4 or IPv6).
52
49
*/
53
50
private ExprValue exprCidrMatch (ExprValue addressExprValue , ExprValue rangeExprValue ) {
54
51
55
- // TODO #3145: Update to support IP address data type.
56
- String addressString = addressExprValue .stringValue ();
57
- String rangeString = rangeExprValue .stringValue ();
58
-
59
- final IPAddressStringParameters validationOptions =
60
- new IPAddressStringParameters .Builder ()
61
- .allowEmpty (false )
62
- .setEmptyAsLoopback (false )
63
- .allow_inet_aton (false )
64
- .allowSingleSegment (false )
65
- .toParams ();
66
-
67
- // Get and validate IP address.
68
- IPAddressString address =
69
- new IPAddressString (addressExprValue .stringValue (), validationOptions );
70
-
71
- try {
72
- address .validate ();
73
- } catch (AddressStringException e ) {
74
- String msg =
75
- String .format (
76
- "IP address '%s' is not valid. Error details: %s" , addressString , e .getMessage ());
77
- throw new SemanticCheckException (msg , e );
78
- }
79
-
80
- // Get and validate CIDR IP address range.
81
- IPAddressString range = new IPAddressString (rangeExprValue .stringValue (), validationOptions );
82
-
83
- try {
84
- range .validate ();
85
- } catch (AddressStringException e ) {
86
- String msg =
87
- String .format (
88
- "CIDR IP address range '%s' is not valid. Error details: %s" ,
89
- rangeString , e .getMessage ());
90
- throw new SemanticCheckException (msg , e );
91
- }
92
-
93
- // Address and range must use the same IP version (IPv4 or IPv6).
94
- if (address .isIPv4 () ^ range .isIPv4 ()) {
95
- String msg =
96
- String .format (
97
- "IP address '%s' and CIDR IP address range '%s' are not compatible. Both must be"
98
- + " either IPv4 or IPv6." ,
99
- addressString , rangeString );
100
- throw new SemanticCheckException (msg );
101
- }
52
+ IPAddress address = addressExprValue .ipValue ();
53
+ IPAddress range = IPUtils .toRange (rangeExprValue .stringValue ());
102
54
103
- return ExprValueUtils .booleanValue (range .contains (address ));
55
+ return (IPUtils .compare (address , range .getLower ()) < 0 )
56
+ || (IPUtils .compare (address , range .getUpper ()) > 0 )
57
+ ? ExprValueUtils .LITERAL_FALSE
58
+ : ExprValueUtils .LITERAL_TRUE ;
104
59
}
105
60
}
0 commit comments