ArangoDB v3.13 is under development and not released yet. This documentation is not final and potentially incomplete.
Operators
AQL supports a number of operators that can be used in expressions, such as for arithmetic, comparing values, and logically combining conditions
Comparison operators
Comparison (or relational) operators compare two operands. They can be used with any input data types, and return a boolean result value.
The following comparison operators are supported:
Operator | Description |
---|---|
== | equality |
!= | inequality |
< | less than |
<= | less or equal |
> | greater than |
>= | greater or equal |
IN | test if a value is contained in an array |
NOT IN | test if a value is not contained in an array |
LIKE | tests if a string value matches a pattern |
NOT LIKE | tests if a string value does not match a pattern |
=~ | tests if a string value matches a regular expression |
!~ | tests if a string value does not match a regular expression |
Each of the comparison operators returns a boolean value if the comparison can be evaluated and returns true if the comparison evaluates to true, and false otherwise.
The comparison operators accept any data types for the first and second
operands. However, IN
and NOT IN
only return a meaningful result if
their right-hand operand is an array. LIKE
and NOT LIKE
only execute
if both operands are string values. All four operators do not perform
implicit type casts if the compared operands have different types, i.e.
they test for strict equality or inequality (0
is different to "0"
,
[0]
, false
and null
for example).
0 == null // false
1 > 0 // true
true != null // true
45 <= "yikes!" // true
65 != "65" // true
65 == 65 // true
1.23 > 1.32 // false
1.5 IN [ 2, 3, 1.5 ] // true
"foo" IN null // false
42 NOT IN [ 17, 40, 50 ] // true
"abc" == "abc" // true
"abc" == "ABC" // false
"foo" LIKE "f%" // true
"foo" NOT LIKE "f%" // false
"foo" =~ "^f[o].$" // true
"foo" !~ "[a-z]+bar$" // true
The LIKE
operator checks whether its left operand matches the pattern specified
in its right operand. The pattern can consist of regular characters and wildcards.
The supported wildcards are _
to match a single arbitrary character, and %
to
match any number of arbitrary characters. Literal %
and _
need to be escaped
with a backslash. Backslashes need to be escaped themselves, which effectively
means that two reverse solidus characters need to precede a literal percent sign
or underscore. In arangosh, additional escaping is required, making it four
backslashes in total preceding the to-be-escaped character.
"abc" LIKE "a%" // true
"abc" LIKE "_bc" // true
"a_b_foo" LIKE "a\\_b\\_foo" // true
The pattern matching performed by the LIKE
operator is case-sensitive.
The NOT LIKE
operator has the same characteristics as the LIKE
operator
but with the result negated. It is thus identical to NOT (… LIKE …)
. Note
the parentheses, which are necessary for certain expressions:
FOR doc IN coll
RETURN NOT doc.attr LIKE "…"
The return expression gets transformed into LIKE(!doc.attr, "…")
, leading
to unexpected results. NOT(doc.attr LIKE "…")
gets transformed into the
more reasonable ! LIKE(doc.attr, "…")
.
The regular expression operators =~
and !~
expect their left-hand operands to
be strings, and their right-hand operands to be strings containing valid regular
expressions as specified in the documentation for the AQL function
REGEX_TEST()
.
Array comparison operators
Most comparison operators also exist as an array variant. In the array variant,
a ==
, !=
, >
, >=
, <
, <=
, IN
, or NOT IN
operator is prefixed with
an ALL
, ANY
, or NONE
keyword. This changes the operator’s behavior to
compare the individual array elements of the left-hand argument to the right-hand
argument. Depending on the quantifying keyword, all, any, or none of these
comparisons need to be satisfied to evaluate to true
overall.
You can also combine one of the supported comparison operators with the special
AT LEAST (<expression>)
operator to require an arbitrary number of elements
to satisfy the condition to evaluate to true
. You can use a static number or
calculate it dynamically using an expression.
[ 1, 2, 3 ] ALL IN [ 2, 3, 4 ] // false
[ 1, 2, 3 ] ALL IN [ 1, 2, 3 ] // true
[ 1, 2, 3 ] NONE IN [ 3 ] // false
[ 1, 2, 3 ] NONE IN [ 23, 42 ] // true
[ 1, 2, 3 ] ANY IN [ 4, 5, 6 ] // false
[ 1, 2, 3 ] ANY IN [ 1, 42 ] // true
[ 1, 2, 3 ] ANY == 2 // true
[ 1, 2, 3 ] ANY == 4 // false
[ 1, 2, 3 ] ANY > 0 // true
[ 1, 2, 3 ] ANY <= 1 // true
[ 1, 2, 3 ] NONE < 99 // false
[ 1, 2, 3 ] NONE > 10 // true
[ 1, 2, 3 ] ALL > 2 // false
[ 1, 2, 3 ] ALL > 0 // true
[ 1, 2, 3 ] ALL >= 3 // false
["foo", "bar"] ALL != "moo" // true
["foo", "bar"] NONE == "bar" // false
["foo", "bar"] ANY == "foo" // true
[ 1, 2, 3 ] AT LEAST (2) IN [ 2, 3, 4 ] // true
["foo", "bar"] AT LEAST (1+1) == "foo" // false
Note that these operators do not utilize indexes in regular queries.
The operators are also supported in SEARCH expressions,
where ArangoSearch’s indexes can be utilized. The semantics differ however, see
AQL SEARCH
operation.
Logical operators
The following logical operators are supported in AQL:
&&
logical and operator||
logical or operator!
logical not/negation operator
AQL also supports the following alternative forms for the logical operators:
AND
logical and operatorOR
logical or operatorNOT
logical not/negation operator
The alternative forms are aliases and functionally equivalent to the regular operators.
The two-operand logical operators in AQL are executed with short-circuit evaluation (except if one of the operands is or includes a subquery. In this case the subquery is pulled out an evaluated before the logical operator).
The result of the logical operators in AQL is defined as follows:
lhs && rhs
returnslhs
if it isfalse
or would befalse
when converted to a boolean. Iflhs
istrue
or would betrue
when converted to a boolean,rhs
is returned.lhs || rhs
returnslhs
if it istrue
or would betrue
when converted to a boolean. Iflhs
isfalse
or would befalse
when converted to a boolean,rhs
is returned.! value
returns the negated value ofvalue
converted to a boolean
u.age > 15 && u.address.city != ""
true || false
NOT u.isInvalid
1 || ! 0
Passing non-boolean values to a logical operator is allowed. Any non-boolean operands are casted to boolean implicitly by the operator, without making the query abort.
The conversion to a boolean value works as follows:
null
is converted tofalse
- boolean values remain unchanged
- all numbers unequal to zero are
true
, zero isfalse
- an empty string is
false
, all other strings aretrue
- arrays (
[ ]
) and objects / documents ({ }
) aretrue
, regardless of their contents
The result of logical and and logical or operations can now have any data type and is not necessarily a boolean value.
For example, the following logical operations return boolean values:
25 > 1 && 42 != 7 // true
22 IN [ 23, 42 ] || 23 NOT IN [ 22, 7 ] // true
25 != 25 // false
… whereas the following logical operations do not return boolean values:
1 || 7 // 1
null || "foo" // "foo"
null && true // null
true && 23 // 23
Arithmetic operators
Arithmetic operators perform an arithmetic operation on two numeric operands. The result of an arithmetic operation is again a numeric value.
AQL supports the following arithmetic operators:
+
addition-
subtraction*
multiplication/
division%
modulus
Unary plus and unary minus are supported as well:
LET x = -5
LET y = 1
RETURN [-x, +y]
// [5, 1]
For exponentiation, there is a numeric function POW()
.
The syntax base ** exp
is not supported.
For string concatenation, you must use the CONCAT()
string function.
Combining two strings with a plus operator ("foo" + "bar"
) does not work!
Also see Common Errors.
1 + 1
33 - 99
12.4 * 4.5
13.0 / 0.1
23 % 7
-15
+9.99
The arithmetic operators accept operands of any type. Passing non-numeric values to an
arithmetic operator casts the operands to numbers using the type casting rules
applied by the TO_NUMBER()
function:
null
is converted to0
false
is converted to0
,true
is converted to1
- a valid numeric value remains unchanged, but NaN and Infinity are converted to
0
- string values are converted to a number if they contain a valid string representation
of a number. Any whitespace at the start or the end of the string is ignored. Strings
with any other contents are converted to the number
0
- an empty array is converted to
0
, an array with one member is converted to the numeric representation of its sole member. Arrays with more members are converted to the number0
. - objects / documents are converted to the number
0
.
An arithmetic operation that produces an invalid value, such as 1 / 0
(division by zero), produces a result value of null
. The query is not
aborted, but you may see a warning.
1 + "a" // 1
1 + "99" // 100
1 + null // 1
null + 1 // 1
3 + [ ] // 3
24 + [ 2 ] // 26
24 + [ 2, 4 ] // 24
25 - null // 25
17 - true // 16
23 * { } // 0
5 * [ 7 ] // 35
24 / "12" // 2
1 / 0 // null (with a 'division by zero' warning)
Ternary operator
AQL also supports a ternary operator that can be used for conditional evaluation. The ternary operator expects a boolean condition as its first operand, and it returns the result of the second operand if the condition evaluates to true, and the third operand otherwise. You may use subqueries as operands.
In the following example, the expression returns u.userId
if u.age
is
greater than 15 or if u.active
is true
. Otherwise it returns null
:
u.age > 15 || u.active == true ? u.userId : null
There is also a shortcut variant of the ternary operator with just two operands. This variant can be used if the expression for the boolean condition and the return value should be the same.
In the following example, the expression evaluates to u.value
if u.value
is
truthy. Otherwise, a fixed string is given back:
u.value ? : 'value is null, 0 or not present'
The condition (here just u.value
) is only evaluated once if the second
operand between ?
and :
is omitted, whereas it would be evaluated twice
in case of u.value ? u.value : 'value is null'
.
Up to v3.12.0, subqueries used inside expressions are pulled out of these expressions and executed beforehand. That means that subqueries do not participate in lazy evaluation of operands.
From v3.12.1 onward, short-circuiting is applied.
Also see evaluation of subqueries.
Range operator
AQL supports expressing simple numeric ranges with the ..
operator.
This operator can be used to easily iterate over a sequence of numeric
values.
The ..
operator produces an array of the integer values in the
defined range, with both bounding values included.
2010..2013
The above example produces the following result:
[ 2010, 2011, 2012, 2013 ]
Using the range operator is equivalent to writing an array with the integer values in the range specified by the bounds of the range. If the bounds of the range operator are non-integers, they are converted to integer values first.
There is also a RANGE()
function.
Array operators
AQL provides different array operators:
[n]
to access the array element at indexn
[*]
for expanding array variables[**]
,[***]
etc. for flattening arrays[* ...]
,[** ...]
etc. for filtering, limiting, and projecting arrays using inline expressions[? ...]
for nested search, known as the question mark operatorLET [ ] = [ ]
andFOR [ ] IN [[ ], [ ]]
for array destructuring
Indexed value access
You can access individual array elements by their position using the []
accessor.
The position is called the index and starts at 0
.
When specifying an index, use a numeric integer value. You can use negative index values to access array elements starting from the end of the array. This is convenient if the length of the array is unknown and you want to access elements at the end of the array.
You can also use an expression and calculate the index of an element.
null
value without
raising an error or warning.LET friends = [ "tina", "helga", "alfred" ]
friends[0] // access 1st array element (elements start at index 0)
friends[2] // access 3rd array element
friends[-1] // access last array element
friends[-2] // access second to last array element
friends[LENGTH(friends) / 2] // access array element in the middle (floored)
Array expansion
In order to access a named attribute from all elements in an array easily, AQL
offers the shortcut operator [*]
for array variable expansion.
Using the [*]
operator with an array variable will iterate over all elements
in the array, thus allowing to access a particular attribute of each element. It is
required that the expanded variable is an array. The result of the [*]
operator is again an array.
To demonstrate the array expansion operator, let’s go on with the following three example users documents:
[
{
"name": "john",
"age": 35,
"friends": [
{ "name": "tina", "age": 43 },
{ "name": "helga", "age": 52 },
{ "name": "alfred", "age": 34 }
]
},
{
"name": "yves",
"age": 24,
"friends": [
{ "name": "sergei", "age": 27 },
{ "name": "tiffany", "age": 25 }
]
},
{
"name": "sandra",
"age": 40,
"friends": [
{ "name": "bob", "age": 32 },
{ "name": "elena", "age": 48 }
]
}
]
With the [*]
operator it becomes easy to query just the names of the
friends for each user:
FOR u IN users
RETURN { name: u.name, friends: u.friends[*].name }
This will produce:
[
{ "name" : "john", "friends" : [ "tina", "helga", "alfred" ] },
{ "name" : "yves", "friends" : [ "sergei", "tiffany" ] },
{ "name" : "sandra", "friends" : [ "bob", "elena" ] }
]
This is a shortcut for the longer, semantically equivalent query:
FOR u IN users
RETURN { name: u.name, friends: (FOR f IN u.friends RETURN f.name) }
Array contraction
In order to collapse (or flatten) results in nested arrays, AQL provides the [**]
operator. It works similar to the [*]
operator, but additionally collapses nested
arrays.
How many levels are collapsed is determined by the amount of asterisk characters used.
[**]
collapses one level of nesting - just like FLATTEN(array)
or FLATTEN(array, 1)
would do -, [***]
collapses two levels - the equivalent to FLATTEN(array, 2)
- and
so on.
Let’s compare the array expansion operator with an array contraction operator. For example, the following query produces an array of friend names per user:
FOR u IN users
RETURN u.friends[*].name
As we have multiple users, the overall result is a nested array:
[
[
"tina",
"helga",
"alfred"
],
[
"sergei",
"tiffany"
],
[
"bob",
"elena"
]
]
If the goal is to get rid of the nested array, we can apply the [**]
operator on the
result. But simply appending [**]
to the query won’t help, because u.friends
is not a nested (multi-dimensional) array, but a simple (one-dimensional) array. Still,
the [**]
can be used if it has access to a multi-dimensional nested result.
We can extend above query as follows and still create the same nested result:
RETURN (
FOR u IN users RETURN u.friends[*].name
)
By now appending the [**]
operator at the end of the query…
RETURN (
FOR u IN users RETURN u.friends[*].name
)[**]
… the query result becomes:
[
[
"tina",
"helga",
"alfred",
"sergei",
"tiffany",
"bob",
"elena"
]
]
Note that the elements are not de-duplicated. For a flat array with only unique
elements, a combination of UNIQUE()
and
FLATTEN()
is advisable.
Inline expressions
It is possible to filter elements while iterating over an array, to limit the amount of returned elements and to create a projection using the current array element. Sorting is not supported by this shorthand form.
These inline expressions can follow array expansion and contraction operators
[* ...]
, [** ...]
etc. The keywords FILTER
, LIMIT
and RETURN
must occur in this order if they are used in combination, and can only occur once:
anyArray[* FILTER conditions LIMIT skip,limit RETURN projection]
Example with nested numbers and array contraction:
LET arr = [ [ 1, 2 ], 3, [ 4, 5 ], 6 ]
RETURN arr[** FILTER CURRENT % 2 == 0]
All even numbers are returned in a flat array:
[
[ 2, 4, 6 ]
]
Complex example with multiple conditions, limit and projection:
FOR u IN users
RETURN {
name: u.name,
friends: u.friends[* FILTER CONTAINS(CURRENT.name, "a") AND CURRENT.age > 40
LIMIT 2
RETURN CONCAT(CURRENT.name, " is ", CURRENT.age)
]
}
No more than two computed strings based on friends with an a
in their name and
older than 40 years are returned per user:
[
{
"name": "john",
"friends": [
"tina is 43",
"helga is 52"
]
},
{
"name": "sandra",
"friends": [
"elena is 48"
]
},
{
"name": "yves",
"friends": []
}
]
Inline filter
To return only the names of friends that have an age value
higher than the user herself, an inline FILTER
can be used:
FOR u IN users
RETURN { name: u.name, friends: u.friends[* FILTER CURRENT.age > u.age].name }
The pseudo-variable CURRENT can be used to access the current array element.
The FILTER
condition can refer to CURRENT or any variables valid in the
outer scope.
Inline limit
The number of elements returned can be restricted with LIMIT
. It works the same
as the limit operation. LIMIT
must come after FILTER
and before RETURN
, if they are present.
FOR u IN users
RETURN { name: u.name, friends: u.friends[* LIMIT 1].name }
Above example returns one friend each:
[
{ "name": "john", "friends": [ "tina" ] },
{ "name": "sandra", "friends": [ "bob" ] },
{ "name": "yves", "friends": [ "sergei" ] }
]
A number of elements can also be skipped and up to n returned:
FOR u IN users
RETURN { name: u.name, friends: u.friends[* LIMIT 1,2].name }
The example query skips the first friend and returns two friends at most per user:
[
{ "name": "john", "friends": [ "helga", "alfred" ] },
{ "name": "sandra", "friends": [ "elena" ] },
{ "name": "yves", "friends": [ "tiffany" ] }
]
Inline projection
To return a projection of the current element, use RETURN
. If a FILTER
is
also present, RETURN
must come later.
FOR u IN users
RETURN u.friends[* RETURN CONCAT(CURRENT.name, " is a friend of ", u.name)]
The above will return:
[
[
"tina is a friend of john",
"helga is a friend of john",
"alfred is a friend of john"
],
[
"sergei is a friend of yves",
"tiffany is a friend of yves"
],
[
"bob is a friend of sandra",
"elena is a friend of sandra"
]
]
Question mark operator
You can use the [? ... ]
operator on arrays to check whether the elements
fulfill certain criteria, and you can specify how often they should be satisfied.
The operator is similar to an inline filter but with an additional length check
and it evaluates to true
or false
.
The following example shows how to check whether two of numbers in the array are even:
LET arr = [ 1, 2, 3, 4 ]
RETURN arr[? 2 FILTER CURRENT % 2 == 0] // true
The number 2
after the ?
is the quantifier. It is optional and defaults to
ANY
. The following quantifiers are supported:
- Integer numbers for exact quantities (e.g.
2
) - Number ranges for a quantity between the two values (e.g.
2..3
) NONE
(equivalent to0
)ANY
ALL
AT LEAST
The quantifier needs to be followed by a FILTER
operation if you want to specify
conditions. You can refer to the current array element via the CURRENT
pseudo-variable in the filter expression. If you leave out the quantifier and
FILTER
operation (only arr[?]
), then arr
is checked whether it is an array
and if it has at least one element.
The question mark operator is a shorthand for an inline filter with a surrounding length check. The following table compares both variants:
Question mark operator | Inline filter with length check |
---|---|
arr[? <number> FILTER <conditions>] | LENGTH(arr[* FILTER <conditions>]) == <number> |
arr[? <min>..<max> FILTER <conditions>] | IN_RANGE(LENGTH(arr[* FILTER <conditions>]), <min>, <max>, true, true) |
arr[? NONE FILTER <conditions>] | LENGTH(arr[* FILTER <conditions>]) == 0 |
arr[? ANY FILTER <conditions>] | LENGTH(arr[* FILTER <conditions>]) > 0 |
arr[? ALL FILTER <conditions>] | LENGTH(arr[* FILTER <conditions>]) == LENGTH(arr) |
arr[? AT LEAST (<number>) FILTER <conditions>] | LENGTH(arr[* FILTER <conditions>]) >= <number> |
arr[?] | LENGTH(arr[*]) > 0 |
The question mark operator can be used for nested search (Enterprise Edition only):
- Nested search with ArangoSearch using Views
- Nested search using Inverted indexes
Array destructuring
Introduced in: v3.12.2
Array destructuring lets you assign array values to one or multiple variables
with a single LET
operation. You can also destructure nested arrays you loop
over with a FOR
operation.
Wrap the target assignment variables on the left-hand side of a LET
operation
in square brackets and separate them with commas, like an array. This assigns
the first array element to the first variable, the second element to the second
variable, and so on:
LET [x, y] = [1, 2, 3]
The above example assigns the value 1
to variable x
and the value 2
to
variable y
. The value 3
is not assigned to a variable and thus ignored.
You can skip the assignment of unneeded array values by leaving out variable names but keeping the commas:
LET [, y, z] = [1, 2, 3]
The above example assigns the value 2
to variable y
and the value 3
to
variable z
. The first array element with value 1
is not assigned to any
variable and thus ignored.
If there are more variables assigned than there are array elements, the target
variables that are mapped to non-existing array elements are populated with a
value of null
. The assigned target variables also receive a value of null
if the array destructuring is used on anything other than an array:
LET [x, y] = ["one"]
LET [z] = { obj: true }
The above example assigns the value "one"
to variable x
and the value null
to both variables y
and z
.
You can also destructure nested arrays as follows:
LET [[first1], [second1, second2]] = [["foo", "bar"], [1, 2, 3]]
The above example assigns the value "foo"
to variable first1
, the value
"bar"
is ignored, the value 1
is assigned to variable second1
, the value
2
to variable second2
, and the value 3
is ignored.
You can mix array and object destructuring:
LET [ { obj: [x, y] } ] = [ { obj: [1, 2] } ]
The above example assigns the value 1
to variable x
and the value 2
to
variable y
. It does not create a variable obj
, however.
To use array destructing in a FOR
operation, the value you iterate over must
be a nested array to assign values other than null
. This is because the FOR
loop iterates over an outer array and the destructuring is carried out on the
elements.
FOR [x, y] IN [["foo", 1], ["bar", 2]]
RETURN {x, y}
The above example assigns the value "foo"
to variable key
and the value 1
to variable val
in the first iteration of the loop. In the second iteration,
it assigns "bar"
to key
and 2
to val
. The RETURN
operation constructs
the objects {"x": "foo", "y": 1}
and {"x": "bar", "y": 2}
.
Object operators
.
and[expr]
for accessing an object attributeLET { } = { }
andFOR { } IN [{ }, { }]
for object destructuring
Attribute access
You can access individual object attributes by their names using the
dot accessor .
and the square bracket accessor []
.
The dot accessor lets you specify the attribute name as an unquoted string. This is only possible if the attribute name would be valid as a variable name. Otherwise, you need to quote the name with backticks or forward ticks, or use the square bracket accessor.
You can also use the dot accessor together with a bind parameter to select an attribute or sub-attribute.
LET ob = { name: "sandra", "with space": true }
LET unquoted = ob.name
LET quoted_1 = ob.`with space`
LET quoted_2 = ob.´with space´
LET bindvar = ob.@attr
The square bracket accessor lets you specify an expression to select an attribute. This is usually a quoted string literal but you can also calculate the name dynamically using an arbitrary expression.
You can also use the square bracket accessor together with a bind parameter to select an attribute.
LET ob = { name: "sandra", "with 2 spaces": true }
LET literal_1 = ob["name"]
LET literal_2 = ob["with 2 spaces"]
LET attribute = "name"
LET variable = ob[attribute]
LET expression = ob[CONCAT_SEPARATOR(" ", "with", 1+1, "spaces")]
LET bindvar = ob[@attr]
null
value without raising an error or warning.Object destructuring
Introduced in: v3.12.2
Object destructuring lets you assign object attributes to one or multiple
variables with a single LET
operation. You can also destructure objects as
part of regular FOR
loops.
Wrap the target assignment variables on the left-hand side of a LET
operation
in curly braces and separate them with commas, similar to an object. The attributes
of the source object on the right-hand side are mapped to the target variables
by name:
LET { name, age } = { vip: true, age: 39, name: "Luna Miller" }
The above example assigns the value "Luna Miller"
of the name
attribute to
the name
variable. The value 39
of the age
attribute is assigned to the
age
variable. The vip
attribute of the source object is ignored.
If you specify target variables with no matching attributes in the source object,
or if you try the object destructuring on anything other than an object, then
the variables receive a value of null
.
LET { vip, status } = { vip: true }
LET { email } = [1, 2]
The above example assigns the value true
to the vip
variable and the other
variables status
and email
get a value of null
.
You can also destructure objects with sub-attributes as follows:
LET { name: {first, last} } = { name: { first: "Luna", middle: "R", last: "Miller" } }
The above example assigns the value "Luna"
to the variable first
and the
value "Miller"
to the variable last
. The middle
attribute of the source
object is ignored. Note that no variable called name
is created here. You
could additional assign the sub-object to a variable called name
as follows:
LET { name: {first, last}, name } = { name: { first: "Luna", middle: "R", last: "Miller" } }
You may give the target variables a different name than in the source object. Specify the attribute name you want to map, followed by a colon and the desired variable name:
LET { name: {first: firstName, last: lastName} } = { name: { first: "Luna", middle: "R", last: "Miller" } }
The above example assigns the value "Luna"
to the variable firstName
and the
value "Miller"
to the variable lastName
. Neither of these attributes exist
in the source object.
You can mix object and array destructuring:
LET { obj: [x, y] } = { obj: [1, 2] }
The above example assigns the value 1
to variable x
and the value 2
to
variable y
. It does not create a variable obj
, however.
You can use object destructing in a FOR
operation that iterates over an array
of objects, a collection, or a View. For anything other than documents/objects
emitted in each iteration, the assigned values are null
.
LET names = [
{ name: "Luna Miller"},
{ name: "Sam Miller" },
]
FOR { name } IN names
RETURN name
The above example assigns the value "Luna Miller"
to variable name
in the
first iteration of the loop. In the second iteration, it assigns "Sam Miller"
to name
.
FOR
operations that are used
for graph traversals and path searches.Operator precedence
The operator precedence in AQL is similar as in other familiar languages (highest precedence first):
Operator(s) | Description |
---|---|
:: | scope (user-defined AQL functions) |
[*] | array expansion |
[] | indexed value access (arrays), attribute access (objects) |
. | attribute access (objects) |
() | function call |
! , NOT , + , - | unary not (logical negation), unary plus, unary minus |
* , / , % | multiplication, division, modulus |
+ , - | addition, subtraction |
.. | range operator |
< , <= , >= , > | less than, less equal, greater equal, greater than |
IN , NOT IN | in operator, not in operator |
== , != , LIKE , NOT LIKE , =~ , !~ | equality, inequality, wildcard match, wildcard non-match, regex match, regex non-match |
AT LEAST | at least modifier (array comparison operator, question mark operator) |
OUTBOUND , INBOUND , ANY , ALL , NONE | graph traversal directions, array comparison operators, question mark operator |
&& , AND | logical and |
` | |
INTO | into operator (INSERT / UPDATE / REPLACE / REMOVE / COLLECT operations) |
WITH | with operator (WITH / UPDATE / REPLACE / COLLECT operations) |
= | variable assignment (LET / COLLECT operations, AGGREGATE / PRUNE clauses) |
? , : | ternary operator, object literals |
DISTINCT | distinct modifier (RETURN operations) |
, | comma separator |
The parentheses (
and )
can be used to enforce a different operator
evaluation order.