Executing queries from Arangosh
Within the ArangoDB shell, the _query and _createStatement methods of the db object can be used to execute AQL queries. This chapter also describes how to use bind parameters, counting, statistics and cursors.
with db._query
One can execute queries with the _query method of the db object. This will run the specified query in the context of the currently selected database and return the query results in a cursor. The results of the cursor can be printed using its toArray method:
arangosh> db._create("mycollection")
[ArangoCollection 6600, "mycollection" (type document, status loaded)]
arangosh> db.mycollection.save({ _key: "testKey", Hello : "World" })
{
"_id" : "mycollection/testKey",
"_key" : "testKey",
"_rev" : "_VTxSboK---"
}
arangosh> db._query('FOR my IN mycollection RETURN my._key').toArray()
[
"testKey"
]
arangosh> db._create("mycollection")
arangosh> db.mycollection.save({ _key: "testKey", Hello : "World" })
arangosh> db._query('FOR my IN mycollection RETURN my._key').toArray()
db._query Bind parameters
To pass bind parameters into a query, they can be specified as second argument to the _query method:
arangosh> db._query(
........> 'FOR c IN @@collection FILTER c._key == @key RETURN c._key', {
........> '@collection': 'mycollection',
........> 'key': 'testKey'
........> }).toArray();
[
"testKey"
]
ES6 template strings
It is also possible to use ES6 template strings for generating AQL queries. There is a template string generator function named aql; we call it once to demonstrate its result, and once putting it directly into the query:
var key = 'testKey';
aql`FOR c IN mycollection FILTER c._key == ${key} RETURN c._key`;
{
"query" : "FOR c IN mycollection FILTER c._key == @value0 RETURN c._key",
"bindVars" : {
"value0" : "testKey"
}
}
arangosh> var key = 'testKey';
arangosh> db._query(
........> aql`FOR c IN mycollection FILTER c._key == ${key} RETURN c._key`
........> ).toArray();
[
"testKey"
]
Arbitrary JavaScript expressions can be used in queries that are generated with the aql template string generator. Collection objects are handled automatically:
arangosh> var key = 'testKey';
arangosh> db._query(aql`FOR doc IN ${ db.mycollection } RETURN doc`
........> ).toArray();
[
{
"_key" : "testKey",
"_id" : "mycollection/testKey",
"_rev" : "_VTxSboK---",
"Hello" : "World"
}
]
arangosh> var key = 'testKey';
arangosh> db._query(aql`FOR doc IN ${ db.mycollection } RETURN doc`
........> ).toArray();
Note: data-modification AQL queries normally do not return a result (unless the AQL query contains an extra RETURN statement). When not using a RETURN statement in the query, the toArray method will return an empty array.
Statistics and extra Information
It is always possible to retrieve statistics for a query with the getExtra method:
arangosh> db._query(`FOR i IN 1..100
........> INSERT { _key: CONCAT('test', TO_STRING(i)) }
........> INTO mycollection`
........> ).getExtra();
{
"stats" : {
"writesExecuted" : 100,
"writesIgnored" : 0,
"scannedFull" : 0,
"scannedIndex" : 0,
"filtered" : 0,
"httpRequests" : 0,
"executionTime" : 0.0013070106506347656
},
"warnings" : [ ]
}
arangosh> db._query(`FOR i IN 1..100
........> INSERT { _key: CONCAT('test', TO_STRING(i)) }
........> INTO mycollection`
........> ).getExtra();
The meaning of the statistics values is described in Execution statistics. You also will find warnings in here; If you're designing queries on the shell be sure to also look at it.
Setting a memory limit
To set a memory limit for the query, pass options to the _query method. The memory limit specifies the maximum number of bytes that the query is allowed to use. When a single AQL query reaches the specified limit value, the query will be aborted with a resource limit exceeded exception. In a cluster, the memory accounting is done per shard, so the limit value is effectively a memory limit per query per shard.
arangosh> db._query(
........> 'FOR i IN 1..100000 SORT i RETURN i', {}, {
........> memoryLimit: 100000
........> }).toArray();
[ArangoError 32: query would use more memory than allowed (while executing)]
If no memory limit is specified, then the server default value (controlled by startup option --query.memory-limit will be used for restricting the maximum amount of memory the query can use. A memory limit value of 0 means that the maximum amount of memory for the query is not restricted.
Setting options
There are further options that can be passed in the options attribute of the _query method:
failOnWarning: when set to true, this will make the query throw an exception and abort in case a warning occurs. This option should be used in development to catch errors early. If set to false, warnings will not be propagated to exceptions and will be returned with the query results. There is also a server configuration option
--query.fail-on-warning
for setting the default value for failOnWarning so it does not need to be set on a per-query level.cache: if set to true, this will put the query result into the query result cache if the query result is eligible for caching and the query cache is running in demand mode. If set to false, the query result will not be inserted into the query result cache. Note that query results will never be inserted into the query result cache if the query result cache is disabled, and that they will be automatically inserted into the query result cache when it is active in non-demand mode.
profile: if set to true, returns extra timing information for the query. The timing information is accessible via the getExtra method of the query result.
maxWarningCount: limits the number of warnings that are returned by the query if failOnWarning is not set to true. The default value is 10.
maxNumberOfPlans: limits the number of query execution plans the optimizer will create at most. Reducing the number of query execution plans may speed up query plan creation and optimization for complex queries, but normally there is no need to adjust this value.
The following additional attributes can be passed to queries in the RocksDB storage engine:
maxTransactionSize: transaction size limit in bytes
intermediateCommitSize: maximum total size of operations after which an intermediate commit is performed automatically
intermediateCommitCount: maximum number of operations after which an intermediate commit is performed automatically
with _createStatement (ArangoStatement)
The _query method is a shorthand for creating an ArangoStatement object, executing it and iterating over the resulting cursor. If more control over the result set iteration is needed, it is recommended to first create an ArangoStatement object as follows:
arangosh> stmt = db._createStatement( {
........> "query": "FOR i IN [ 1, 2 ] RETURN i * 2" } );
[object ArangoStatement]
To execute the query, use the execute method of the statement:
arangosh> c = stmt.execute();
[
2,
4
]
[object ArangoQueryCursor, count: 2, cached: false, hasMore: false]
arangosh> c = stmt.execute();
Cursors
Once the query executed the query results are available in a cursor. The cursor can return all its results at once using the toArray method. This is a short-cut that you can use if you want to access the full result set without iterating over it yourself.
arangosh> c.toArray();
[
2,
4
]
Cursors can also be used to iterate over the result set document-by-document. To do so, use the hasNext and next methods of the cursor:
arangosh> while (c.hasNext()) { require("@arangodb").print(c.next()); }
2
4
Please note that you can iterate over the results of a cursor only once, and that the cursor will be empty when you have fully iterated over it. To iterate over the results again, the query needs to be re-executed.
Additionally, the iteration can be done in a forward-only fashion. There is no backwards iteration or random access to elements in a cursor.
ArangoStatement parameters binding
To execute an AQL query using bind parameters, you need to create a statement first and then bind the parameters to it before execution:
arangosh> var stmt = db._createStatement( {
........> "query": "FOR i IN [ @one, @two ] RETURN i * 2" } );
arangosh> stmt.bind("one", 1);
arangosh> stmt.bind("two", 2);
arangosh> c = stmt.execute();
[
2,
4
]
[object ArangoQueryCursor, count: 2, cached: false, hasMore: false]
arangosh> var stmt = db._createStatement( {
........> "query": "FOR i IN [ @one, @two ] RETURN i * 2" } );
arangosh> stmt.bind("one", 1);
arangosh> stmt.bind("two", 2);
arangosh> c = stmt.execute();
The cursor results can then be dumped or iterated over as usual, e.g.:
arangosh> c.toArray();
[
2,
4
]
or
arangosh> while (c.hasNext()) { require("@arangodb").print(c.next()); }
2
4
Please note that bind parameters can also be passed into the _createStatement method directly, making it a bit more convenient:
arangosh> stmt = db._createStatement( {
........> "query": "FOR i IN [ @one, @two ] RETURN i * 2",
........> "bindVars": {
........> "one": 1,
........> "two": 2
........> }
........> } );
[object ArangoStatement]
Counting with a cursor
Cursors also optionally provide the total number of results. By default, they do not. To make the server return the total number of results, you may set the count attribute to true when creating a statement:
arangosh> stmt = db._createStatement( {
........> "query": "FOR i IN [ 1, 2, 3, 4 ] RETURN i",
........> "count": true } );
[object ArangoStatement]
After executing this query, you can use the count method of the cursor to get the number of total results from the result set:
arangosh> var c = stmt.execute();
arangosh> c.count();
4
Please note that the count method returns nothing if you did not specify the count attribute when creating the query.
This is intentional so that the server may apply optimizations when executing the query and construct the result set incrementally. Incremental creation of the result sets is no possible if all of the results need to be shipped to the client anyway. Therefore, the client has the choice to specify count and retrieve the total number of results for a query (and disable potential incremental result set creation on the server), or to not retrieve the total number of results and allow the server to apply optimizations.
Please note that at the moment the server will always create the full result set for each query so specifying or omitting the count attribute currently does not have any impact on query execution. This may change in the future. Future versions of ArangoDB may create result sets incrementally on the server-side and may be able to apply optimizations if a result set is not fully fetched by a client.
Using cursors to obtain additional information on internal timings
Cursors can also optionally provide statistics of the internal execution phases. By default, they do not. To get to know how long parsing, otpimisation, instanciation and execution took, make the server return that by setting the profile attribute to true when creating a statement:
arangosh> stmt = db._createStatement( {
........> "query": "FOR i IN [ 1, 2, 3, 4 ] RETURN i",
........> options: {"profile": true}} );
[object ArangoStatement]
After executing this query, you can use the getExtra() method of the cursor to get the produced statistics:
arangosh> var c = stmt.execute();
arangosh> c.getExtra();
{
"stats" : {
"writesExecuted" : 0,
"writesIgnored" : 0,
"scannedFull" : 0,
"scannedIndex" : 0,
"filtered" : 0,
"httpRequests" : 0,
"executionTime" : 0.00015807151794433594
},
"profile" : {
"initializing" : 0.0000019073486328125,
"parsing" : 0.000030994415283203125,
"optimizing ast" : 0.0000030994415283203125,
"loading collections" : 0.0000030994415283203125,
"instantiating plan" : 0.000016927719116210938,
"optimizing plan" : 0.00004291534423828125,
"executing" : 0.000030040740966796875,
"finalizing" : 0.0000209808349609375
},
"warnings" : [ ]
}
arangosh> var c = stmt.execute();
arangosh> c.getExtra();