Expressions are used in various places in the native query language, including virtual columns and . They are also generated by most Druid SQL functions during the process.
This expression language supports the following operators (listed in decreasing order of precedence).
Long, double, and string data types are supported. If a number contains a dot, it is interpreted as a double, otherwise it is interpreted as a long. That means, always add a ‘.’ to your number if you want it interpreted as a double value. String literals should be quoted by single quotation marks.
Additionally, the expression language supports long, double, and string arrays. Array literals are created by wrapping square brackets around a list of scalar literals values delimited by a comma or space character. All values in an array literal must be the same type, however null values are accepted. Typed empty arrays may be defined by prefixing with their type in angle brackets: , <DOUBLE>[]
, or <LONG>[]
.
Expressions can contain variables. Variable names may contain letters, digits, ‘_‘ and ‘$’. Variable names must not begin with a digit. To escape other special characters, you can quote it with double quotation marks.
Multi-value string dimensions are supported and may be treated as either scalar or array typed values, as follows:
- When treated as a scalar type, the expression is automatically transformed so that the scalar operation is applied across all values of the multi-valued type, mimicking Druid’s native behavior.
- Druid coerces values that result in arrays back into the native Druid string type for grouping and aggregation. Grouping on multi-value string dimensions in Druid groups by the individual values, not the ‘array’. This behavior produces results similar to the
UNNEST
operator available in many SQL dialects. Alternatively, you can use thearray_to_string
function to perform the aggregation on a stringified version of the complete array and therefore preserve the complete row. To transform the stringified dimension back into the true native array type, usestring_to_array
in an expression post-aggregator.
The following built-in functions are available.
name | description |
---|---|
cast | cast(expr,’LONG’ or ‘DOUBLE’ or ‘STRING’ or ‘LONG_ARRAY’, or ‘DOUBLE_ARRAY’ or ‘STRING_ARRAY’) returns expr with specified type. exception can be thrown. Scalar types may be cast to array types and will take the form of a single element list (null will still be null). |
if | if(predicate,then,else) returns ‘then’ if ‘predicate’ evaluates to a positive number, otherwise it returns ‘else’ |
nvl | nvl(expr,expr-for-null) returns ‘expr-for-null’ if ‘expr’ is null (or empty string for string type) |
like | like(expr, pattern[, escape]) is equivalent to SQL expr LIKE pattern |
case_searched | case_searched(expr1, result1, [[expr2, result2, …], else-result]) |
case_simple | case_simple(expr, value1, result1, [[value2, result2, …], else-result]) |
bloom_filter_test | bloom_filter_test(expr, filter) tests the value of ‘expr’ against ‘filter’, a bloom filter serialized as a base64 string. See documentation for additional details. |
String functions
name | description |
---|---|
concat | concat(expr, expr…) concatenate a list of strings |
format | format(pattern[, args…]) returns a string formatted in the manner of Java’s . |
like | like(expr, pattern[, escape]) is equivalent to SQL expr LIKE pattern |
lookup | lookup(expr, lookup-name) looks up expr in a registered query-time lookup |
parse_long | parse_long(string[, radix]) parses a string as a long with the given radix, or 10 (decimal) if a radix is not provided. |
regexp_extract | regexp_extract(expr, pattern[, index]) applies a regular expression pattern and extracts a capture group index, or null if there is no match. If index is unspecified or zero, returns the substring that matched the pattern. The pattern may match anywhere inside expr ; if you want to match the entire string instead, use the ^ and $ markers at the start and end of your pattern. |
regexp_like | regexp_like(expr, pattern) returns whether expr matches regular expression pattern . The pattern may match anywhere inside expr ; if you want to match the entire string instead, use the ^ and $ markers at the start and end of your pattern. |
contains_string | contains_string(expr, string) returns whether expr contains string as a substring. This method is case-sensitive. |
icontains_string | contains_string(expr, string) returns whether expr contains string as a substring. This method is case-insensitive. |
replace | replace(expr, pattern, replacement) replaces pattern with replacement |
substring | substring(expr, index, length) behaves like java.lang.String’s substring |
right | right(expr, length) returns the rightmost length characters from a string |
left | left(expr, length) returns the leftmost length characters from a string |
strlen | strlen(expr) returns length of a string in UTF-16 code units |
strpos | strpos(haystack, needle[, fromIndex]) returns the position of the needle within the haystack, with indexes starting from 0. The search will begin at fromIndex, or 0 if fromIndex is not specified. If the needle is not found then the function returns -1. |
trim | trim(expr[, chars]) remove leading and trailing characters from expr if they are present in chars . chars defaults to ‘ ‘ (space) if not provided. |
ltrim | ltrim(expr[, chars]) remove leading characters from expr if they are present in chars . chars defaults to ‘ ‘ (space) if not provided. |
rtrim | rtrim(expr[, chars]) remove trailing characters from expr if they are present in chars . chars defaults to ‘ ‘ (space) if not provided. |
lower | lower(expr) converts a string to lowercase |
upper | upper(expr) converts a string to uppercase |
reverse | reverse(expr) reverses a string |
repeat | repeat(expr, N) repeats a string N times |
lpad | lpad(expr, length, chars) returns a string of length from expr left-padded with chars . If length is shorter than the length of expr , the result is expr which is truncated to length . The result will be null if either expr or chars is null. If chars is an empty string, no padding is added, however expr may be trimmed if necessary. |
rpad | rpad(expr, length, chars) returns a string of length from expr right-padded with chars . If length is shorter than the length of expr , the result is expr which is truncated to length . The result will be null if either expr or chars is null. If is an empty string, no padding is added, however expr may be trimmed if necessary. |
Time functions
See javadoc of java.lang.Math for detailed explanation for each function.
name | description |
---|---|
abs | abs(x) returns the absolute value of x |
acos | acos(x) returns the arc cosine of x |
asin | asin(x) returns the arc sine of x |
atan | atan(x) returns the arc tangent of x |
bitwiseAnd | bitwiseAnd(x,y) returns the result of x & y. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
bitwiseComplement | bitwiseComplement(x) returns the result of ~x. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
bitwiseConvertDoubleToLongBits | bitwiseConvertDoubleToLongBits(x) converts the bits of an IEEE 754 floating-point double value to a long. If the input is not a double, it is implicitly cast to a double prior to conversion |
bitwiseConvertLongBitsToDouble | bitwiseConvertLongBitsToDouble(x) converts a long to the IEEE 754 floating-point double specified by the bits stored in the long. If the input is not a long, it is implicitly cast to a long prior to conversion |
bitwiseOr | bitwiseOr(x,y) returns the result of x [PIPE] y. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
bitwiseShiftLeft | bitwiseShiftLeft(x,y) returns the result of x << y. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
bitwiseShiftRight | bitwiseShiftRight(x,y) returns the result of x >> y. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
bitwiseXor | bitwiseXor(x,y) returns the result of x ^ y. Double values will be implicitly cast to longs, use bitwiseConvertDoubleToLongBits to perform bitwise operations directly with doubles |
atan2 | atan2(y, x) returns the angle theta from the conversion of rectangular coordinates (x, y) to polar coordinates (r, theta) |
cbrt | cbrt(x) returns the cube root of x |
ceil | ceil(x) returns the smallest (closest to negative infinity) double value that is greater than or equal to x and is equal to a mathematical integer |
copysign | copysign(x) returns the first floating-point argument with the sign of the second floating-point argument |
cos | cos(x) returns the trigonometric cosine of x |
cosh | cosh(x) returns the hyperbolic cosine of x |
cot | cot(x) returns the trigonometric cotangent of an angle x |
div | div(x,y) is integer division of x by y |
exp | exp(x) returns Euler’s number raised to the power of x |
expm1 | expm1(x) returns e^x-1 |
floor | floor(x) returns the largest (closest to positive infinity) double value that is less than or equal to x and is equal to a mathematical integer |
getExponent | getExponent(x) returns the unbiased exponent used in the representation of x |
hypot | hypot(x, y) returns sqrt(x^2+y^2) without intermediate overflow or underflow |
log | log(x) returns the natural logarithm of x |
log10 | log10(x) returns the base 10 logarithm of x |
log1p | log1p(x) will the natural logarithm of x + 1 |
max | max(x, y) returns the greater of two values |
min | min(x, y) returns the smaller of two values |
nextafter | nextafter(x, y) returns the floating-point number adjacent to the x in the direction of the y |
nextUp | nextUp(x) returns the floating-point value adjacent to x in the direction of positive infinity |
pi | pi returns the constant value of the π |
pow | pow(x, y) returns the value of the x raised to the power of y |
remainder | remainder(x, y) returns the remainder operation on two arguments as prescribed by the IEEE 754 standard |
rint | rint(x) returns value that is closest in value to x and is equal to a mathematical integer |
round | round(x, y) returns the value of the x rounded to the y decimal places. While x can be an integer or floating-point number, y must be an integer. The type of the return value is specified by that of x. y defaults to 0 if omitted. When y is negative, x is rounded on the left side of the y decimal points. If x is NaN , x returns 0. If x is infinity, x will be converted to the nearest finite double. |
scalb | scalb(d, sf) returns d 2^sf rounded as if performed by a single correctly rounded floating-point multiply to a member of the double value set |
signum | signum(x) returns the signum function of the argument x |
sin | sin(x) returns the trigonometric sine of an angle x |
sinh | sinh(x) returns the hyperbolic sine of x |
sqrt | sqrt(x) returns the correctly rounded positive square root of x |
tan | tan(x) returns the trigonometric tangent of an angle x |
tanh | tanh(x) returns the hyperbolic tangent of x |
todegrees | todegrees(x) converts an angle measured in radians to an approximately equivalent angle measured in degrees |
toradians | toradians(x) converts an angle measured in degrees to an approximately equivalent angle measured in radians |
ulp | ulp(x) returns the size of an ulp of the argument x |
Array functions
function | description |
---|---|
array(expr1,expr …) | constructs an array from the expression arguments, using the type of the first argument as the output array type |
array_length(arr) | returns length of array expression |
array_offset(arr,long) | returns the array element at the 0 based index supplied, or null for an out of range index |
array_ordinal(arr,long) | returns the array element at the 1 based index supplied, or null for an out of range index |
array_contains(arr,expr) | returns 1 if the array contains the element specified by expr, or contains all elements specified by expr if expr is an array, else 0 |
array_overlap(arr1,arr2) | returns 1 if arr1 and arr2 have any elements in common, else 0 |
array_offset_of(arr,expr) | returns the 0 based index of the first occurrence of expr in the array, or -1 or null if druid.generic.useDefaultValueForNull=false if no matching elements exist in the array. |
array_ordinal_of(arr,expr) | returns the 1 based index of the first occurrence of expr in the array, or -1 or null if druid.generic.useDefaultValueForNull=false if no matching elements exist in the array. |
array_prepend(expr,arr) | adds expr to arr at the beginning, the resulting array type determined by the type of the array |
array_append(arr,expr) | appends expr to arr, the resulting array type determined by the type of the first array |
array_concat(arr1,arr2) | concatenates 2 arrays, the resulting array type determined by the type of the first array |
array_set_add(arr,expr) | adds expr to arr and converts the array to a new array composed of the unique set of elements. The resulting array type determined by the type of the array |
array_set_add_all(arr1,arr2) | combines the unique set of elements of 2 arrays, the resulting array type determined by the type of the first array |
array_slice(arr,start,end) | return the subarray of arr from the 0 based index start(inclusive) to end(exclusive), or null , if start is less than 0, greater than length of arr or less than end |
array_to_string(arr,str) | joins all elements of arr by the delimiter specified by str |
string_to_array(str1,str2) | splits str1 into an array on the delimiter specified by str2 |
Apply functions
Apply functions allow for special ‘lambda’ expressions to be defined and applied to array inputs to enable free-form transformations.
Lambda expressions are a sort of function definition, where new identifiers can be defined and passed as input to the expression body
The identifier arguments of a lambda expression correspond to the elements of the array it is being applied to. For example:
will map each element of some_multi_value_column
to the identifier x
so that the lambda expression body can be evaluated for each x
. The scoping rules are that lambda arguments will override identifiers which are defined externally from the lambda expression body. Using the same example:
in this case, the x
when evaluating x + 1
is the lambda argument, thus an element of the multi-valued column x
, rather than the column x
itself.
Reduction functions operate on zero or more expressions and return a single expression. If no expressions are passed as arguments, then the result is NULL
. The expressions must all be convertible to a common data type, which will be the type of the result:
- If all arguments are
NULL
, the result isNULL
. Otherwise,NULL
arguments are ignored. - If the arguments comprise a mix of numbers and strings, the arguments are interpreted as strings.
- If all arguments are integer numbers, the arguments are interpreted as longs.
- If all arguments are numbers and at least one argument is a double, the arguments are interpreted as doubles.
function | description |
---|---|
greatest([expr1, …]) | Evaluates zero or more expressions and returns the maximum value based on comparisons as described above. |
least([expr1, …]) | Evaluates zero or more expressions and returns the minimum value based on comparisons as described above. |
IP address functions
For the IPv4 address functions, the address
argument can either be an IPv4 dotted-decimal string (e.g., “192.168.0.1”) or an IP address represented as a long (e.g., 3232235521). The subnet
argument should be a string formatted as an IPv4 address subnet in CIDR notation (e.g., “192.168.0.0/16”).
function | description |
---|---|
ipv4_match(address, subnet) | Returns 1 if the address belongs to the subnet literal, else 0. If address is not a valid IPv4 address, then 0 is returned. This function is more efficient if address is a long instead of a string. |
ipv4_parse(address) | Parses address into an IPv4 address stored as a long. If address is a long that is a valid IPv4 address, then it is passed through. Returns null if address cannot be represented as an IPv4 address. |
ipv4_stringify(address) | Converts address into an IPv4 address dotted-decimal string. If address is a string that is a valid IPv4 address, then it is passed through. Returns null if address cannot be represented as an IPv4 address. |
Vectorization Support
supported features:
- constants and identifiers are supported for any column type
cast
is supported for numeric and string types- math operators:
+
,-
,*
,/
,%
,^
are supported for numeric types - math functions:
abs
,acos
,asin
,atan
,cbrt
,ceil
,cos
,cosh
,cot
,exp
,expm1
,floor
,getExponent
,log
,log10
,log1p
,nextUp
,rint
,signum
,sin
,sinh
,sqrt
,tan
,tanh
,toDegrees
,toRadians
,ulp
,atan2
,copySign
,div
,hypot
,max
,min
,nextAfter
,pow
,remainder
,scalb
are supported for numeric types - time functions:
timestamp_floor
(with constant granularity argument) is supported for numeric types - other:
parse_long
is supported for numeric and string types