blob: 9857d8a46622a5ed71c7177176e46d1100cbebb4 [file] [log] [blame]
Ian Maxon49d15b22020-12-06 16:23:00 -08001<!DOCTYPE html>
2<!--
3 | Generated by Apache Maven Doxia Site Renderer 1.8.1 from src/site/markdown/aql/manual.md at 2020-12-06
4 | Rendered using Apache Maven Fluido Skin 1.7
5-->
6<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
7 <head>
8 <meta charset="UTF-8" />
9 <meta name="viewport" content="width=device-width, initial-scale=1.0" />
10 <meta name="Date-Revision-yyyymmdd" content="20201206" />
11 <meta http-equiv="Content-Language" content="en" />
12 <title>AsterixDB &#x2013; The Asterix Query Language, Version 1.0</title>
13 <link rel="stylesheet" href="../css/apache-maven-fluido-1.7.min.css" />
14 <link rel="stylesheet" href="../css/site.css" />
15 <link rel="stylesheet" href="../css/print.css" media="print" />
16 <script type="text/javascript" src="../js/apache-maven-fluido-1.7.min.js"></script>
17
18 </head>
19 <body class="topBarDisabled">
20 <div class="container-fluid">
21 <div id="banner">
22 <div class="pull-left"><a href=".././" id="bannerLeft"><img src="../images/asterixlogo.png" alt="AsterixDB"/></a></div>
23 <div class="pull-right"></div>
24 <div class="clear"><hr/></div>
25 </div>
26
27 <div id="breadcrumbs">
28 <ul class="breadcrumb">
29 <li id="publishDate">Last Published: 2020-12-06</li>
30 <li id="projectVersion" class="pull-right">Version: 0.9.6-SNAPSHOT</li>
31 <li class="pull-right"><a href="../index.html" title="Documentation Home">Documentation Home</a></li>
32 </ul>
33 </div>
34 <div class="row-fluid">
35 <div id="leftColumn" class="span2">
36 <div class="well sidebar-nav">
37 <ul class="nav nav-list">
38 <li class="nav-header">Get Started - Installation</li>
39 <li><a href="../ncservice.html" title="Option 1: using NCService"><span class="none"></span>Option 1: using NCService</a></li>
40 <li><a href="../ansible.html" title="Option 2: using Ansible"><span class="none"></span>Option 2: using Ansible</a></li>
41 <li><a href="../aws.html" title="Option 3: using Amazon Web Services"><span class="none"></span>Option 3: using Amazon Web Services</a></li>
42 <li class="nav-header">AsterixDB Primer</li>
43 <li><a href="../sqlpp/primer-sqlpp.html" title="Using SQL++"><span class="none"></span>Using SQL++</a></li>
44 <li class="nav-header">Data Model</li>
45 <li><a href="../datamodel.html" title="The Asterix Data Model"><span class="none"></span>The Asterix Data Model</a></li>
46 <li class="nav-header">Queries</li>
47 <li><a href="../sqlpp/manual.html" title="The SQL++ Query Language"><span class="none"></span>The SQL++ Query Language</a></li>
48 <li><a href="../SQLPP.html" title="Raw SQL++ Grammar"><span class="none"></span>Raw SQL++ Grammar</a></li>
49 <li><a href="../sqlpp/builtins.html" title="Builtin Functions"><span class="none"></span>Builtin Functions</a></li>
50 <li class="nav-header">API/SDK</li>
51 <li><a href="../api.html" title="HTTP API"><span class="none"></span>HTTP API</a></li>
52 <li><a href="../csv.html" title="CSV Output"><span class="none"></span>CSV Output</a></li>
53 <li class="nav-header">Advanced Features</li>
54 <li><a href="../aql/externaldata.html" title="Accessing External Data"><span class="none"></span>Accessing External Data</a></li>
55 <li><a href="../feeds.html" title="Data Ingestion with Feeds"><span class="none"></span>Data Ingestion with Feeds</a></li>
56 <li><a href="../udf.html" title="User Defined Functions"><span class="none"></span>User Defined Functions</a></li>
57 <li><a href="../sqlpp/filters.html" title="Filter-Based LSM Index Acceleration"><span class="none"></span>Filter-Based LSM Index Acceleration</a></li>
58 <li><a href="../sqlpp/fulltext.html" title="Support of Full-text Queries"><span class="none"></span>Support of Full-text Queries</a></li>
59 <li><a href="../sqlpp/similarity.html" title="Support of Similarity Queries"><span class="none"></span>Support of Similarity Queries</a></li>
60 <li><a href="../interval_join.html" title="Support of Interval Joins"><span class="none"></span>Support of Interval Joins</a></li>
61 <li class="nav-header">Deprecated</li>
62 <li><a href="../aql/primer.html" title="AsterixDB Primer: Using AQL"><span class="none"></span>AsterixDB Primer: Using AQL</a></li>
63 <li class="active"><a href="#"><span class="none"></span>Queries: The Asterix Query Language (AQL)</a></li>
64 <li><a href="../aql/builtins.html" title="Queries: Builtin Functions (AQL)"><span class="none"></span>Queries: Builtin Functions (AQL)</a></li>
65</ul>
66 <hr />
67 <div id="poweredBy">
68 <div class="clear"></div>
69 <div class="clear"></div>
70 <div class="clear"></div>
71 <div class="clear"></div>
72<a href=".././" title="AsterixDB" class="builtBy"><img class="builtBy" alt="AsterixDB" src="../images/asterixlogo.png" /></a>
73 </div>
74 </div>
75 </div>
76 <div id="bodyColumn" class="span10" >
77<!--
78 ! Licensed to the Apache Software Foundation (ASF) under one
79 ! or more contributor license agreements. See the NOTICE file
80 ! distributed with this work for additional information
81 ! regarding copyright ownership. The ASF licenses this file
82 ! to you under the Apache License, Version 2.0 (the
83 ! "License"); you may not use this file except in compliance
84 ! with the License. You may obtain a copy of the License at
85 !
86 ! http://www.apache.org/licenses/LICENSE-2.0
87 !
88 ! Unless required by applicable law or agreed to in writing,
89 ! software distributed under the License is distributed on an
90 ! "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
91 ! KIND, either express or implied. See the License for the
92 ! specific language governing permissions and limitations
93 ! under the License.
94 !-->
95<h1>The Asterix Query Language, Version 1.0</h1>
96<div class="section">
97<h2><a name="Table_of_Contents"></a><a name="toc" id="toc">Table of Contents</a></h2>
98<ul>
99
100<li><a href="#Introduction">1. Introduction</a></li>
101<li><a href="#Expressions">2. Expressions</a></li>
102<li><a href="#Statements">3. Statements</a></li>
103</ul></div>
104<div class="section">
105<h2><a name="a1._Introduction_.5BBack_to_TOC.5D"></a><a name="Introduction" id="Introduction">1. Introduction</a><font size="4"> <a href="#toc">[Back to TOC]</a></font></h2>
106<p>This document is intended as a reference guide to the full syntax and semantics of the Asterix Query Language (AQL), the language for talking to AsterixDB. This guide covers both the data manipulation language (DML) aspects of AQL, including its support for queries and data modification, as well as its data definition language (DDL) aspects. New AsterixDB users are encouraged to read and work through the (friendlier) guide &#x201c;AsterixDB 101: An ADM and AQL Primer&#x201d; before attempting to make use of this document. In addition, readers are advised to read and understand the Asterix Data Model (ADM) reference guide since a basic understanding of ADM concepts is a prerequisite to understanding AQL. In what follows, we detail the features of the AQL language in a grammar-guided manner: We list and briefly explain each of the productions in the AQL grammar, offering examples for clarity in cases where doing so seems needed or helpful.</p></div>
107<div class="section">
108<h2><a name="a2._Expressions_.5BBack_to_TOC.5D"></a><a name="Expressions" id="Expressions">2. Expressions</a> <font size="4"><a href="#toc">[Back to TOC]</a></font></h2>
109
110<div>
111<div>
112<pre class="source">Query ::= Expression
113</pre></div></div>
114
115<p>An AQL query can be any legal AQL expression.</p>
116
117<div>
118<div>
119<pre class="source">Expression ::= ( OperatorExpr | IfThenElse | FLWOR | QuantifiedExpression )
120</pre></div></div>
121
122<p>AQL is a fully composable expression language. Each AQL expression returns zero or more Asterix Data Model (ADM) instances. There are four major kinds of expressions in AQL. At the topmost level, an AQL expression can be an OperatorExpr (similar to a mathematical expression), an IfThenElse (to choose between two alternative values), a FLWOR expression (the heart of AQL, pronounced &#x201c;flower expression&#x201d;), or a QuantifiedExpression (which yields a boolean value). Each will be detailed as we explore the full AQL grammar.</p>
123<div class="section">
124<h3><a name="Primary_Expressions"></a>Primary Expressions</h3>
125
126<div>
127<div>
128<pre class="source">PrimaryExpr ::= Literal
129 | VariableRef
130 | ParenthesizedExpression
131 | FunctionCallExpr
132 | DatasetAccessExpression
133 | ListConstructor
134 | ObjectConstructor
135</pre></div></div>
136
137<p>The most basic building block for any AQL expression is the PrimaryExpr. This can be a simple literal (constant) value, a reference to a query variable that is in scope, a parenthesized expression, a function call, an expression accessing the ADM contents of a dataset, a newly constructed list of ADM instances, or a newly constructed ADM object.</p>
138<div class="section">
139<h4><a name="Literals"></a>Literals</h4>
140
141<div>
142<div>
143<pre class="source">Literal ::= StringLiteral
144 | IntegerLiteral
145 | FloatLiteral
146 | DoubleLiteral
147 | &quot;null&quot;
148 | &quot;true&quot;
149 | &quot;false&quot;
150StringLiteral ::= (&quot;\&quot;&quot; (&lt;ESCAPE_QUOT&gt; | ~[&quot;\&quot;&quot;])* &quot;\&quot;&quot;)
151 | (&quot;\'&quot; (&lt;ESCAPE_APOS&gt; | ~[&quot;\'&quot;])* &quot;\'&quot;)
152&lt;ESCAPE_QUOT&gt; ::= &quot;\\\&quot;&quot;
153&lt;ESCAPE_APOS&gt; ::= &quot;\\\'&quot;
154IntegerLiteral ::= &lt;DIGITS&gt;
155&lt;DIGITS&gt; ::= [&quot;0&quot; - &quot;9&quot;]+
156FloatLiteral ::= &lt;DIGITS&gt; ( &quot;f&quot; | &quot;F&quot; )
157 | &lt;DIGITS&gt; ( &quot;.&quot; &lt;DIGITS&gt; ( &quot;f&quot; | &quot;F&quot; ) )?
158 | &quot;.&quot; &lt;DIGITS&gt; ( &quot;f&quot; | &quot;F&quot; )
159DoubleLiteral ::= &lt;DIGITS&gt;
160 | &lt;DIGITS&gt; ( &quot;.&quot; &lt;DIGITS&gt; )?
161 | &quot;.&quot; &lt;DIGITS&gt;
162</pre></div></div>
163
164<p>Literals (constants) in AQL can be strings, integers, floating point values, double values, boolean constants, or the constant value null. The null value in AQL has &#x201c;unknown&#x201d; or &#x201c;missing&#x201d; value semantics, similar to (though not identical to) nulls in the relational query language SQL.</p>
165<p>The following are some simple examples of AQL literals. Since AQL is an expression language, each example is also a complete, legal AQL query (!).</p>
166<div class="section">
167<h5><a name="Examples"></a>Examples</h5>
168
169<div>
170<div>
171<pre class="source">&quot;a string&quot;
17242
173</pre></div></div>
174</div></div>
175<div class="section">
176<h4><a name="Variable_References"></a>Variable References</h4>
177
178<div>
179<div>
180<pre class="source">VariableRef ::= &lt;VARIABLE&gt;
181&lt;VARIABLE&gt; ::= &quot;$&quot; &lt;LETTER&gt; (&lt;LETTER&gt; | &lt;DIGIT&gt; | &quot;_&quot;)*
182&lt;LETTER&gt; ::= [&quot;A&quot; - &quot;Z&quot;, &quot;a&quot; - &quot;z&quot;]
183</pre></div></div>
184
185<p>A variable in AQL can be bound to any legal ADM value. A variable reference refers to the value to which an in-scope variable is bound. (E.g., a variable binding may originate from one of the for or let clauses of a FLWOR expression or from an input parameter in the context of an AQL function body.)</p>
186<div class="section">
187<h5><a name="Examples"></a>Examples</h5>
188
189<div>
190<div>
191<pre class="source">$tweet
192$id
193</pre></div></div>
194</div></div>
195<div class="section">
196<h4><a name="Parenthesized_Expressions"></a>Parenthesized Expressions</h4>
197
198<div>
199<div>
200<pre class="source">ParenthesizedExpression ::= &quot;(&quot; Expression &quot;)&quot;
201</pre></div></div>
202
203<p>As in most languages, an expression may be parenthesized.</p>
204<p>Since AQL is an expression language, the following example expression is actually also a complete, legal AQL query whose result is the value 2. (As such, you can have Big Fun explaining to your boss how AsterixDB and AQL can turn your 1000-node shared-nothing Big Data cluster into a $5M calculator in its spare time.)</p>
205<div class="section">
206<h5><a name="Example"></a>Example</h5>
207
208<div>
209<div>
210<pre class="source">( 1 + 1 )
211</pre></div></div>
212</div></div>
213<div class="section">
214<h4><a name="Function_Calls"></a>Function Calls</h4>
215
216<div>
217<div>
218<pre class="source">FunctionCallExpr ::= FunctionOrTypeName &quot;(&quot; ( Expression ( &quot;,&quot; Expression )* )? &quot;)&quot;
219</pre></div></div>
220
221<p>Functions are included in AQL, like most languages, as a way to package useful functionality or to componentize complicated or reusable AQL computations. A function call is a legal AQL query expression that represents the ADM value resulting from the evaluation of its body expression with the given parameter bindings; the parameter value bindings can themselves be any AQL expressions.</p>
222<p>The following example is a (built-in) function call expression whose value is 8.</p>
223<div class="section">
224<h5><a name="Example"></a>Example</h5>
225
226<div>
227<div>
228<pre class="source">string-length(&quot;a string&quot;)
229</pre></div></div>
230</div></div>
231<div class="section">
232<h4><a name="Dataset_Access"></a>Dataset Access</h4>
233
234<div>
235<div>
236<pre class="source">DatasetAccessExpression ::= &quot;dataset&quot; ( ( Identifier ( &quot;.&quot; Identifier )? )
237 | ( &quot;(&quot; Expression &quot;)&quot; ) )
238Identifier ::= &lt;IDENTIFIER&gt; | StringLiteral
239&lt;IDENTIFIER&gt; ::= &lt;LETTER&gt; (&lt;LETTER&gt; | &lt;DIGIT&gt; | &lt;SPECIALCHARS&gt;)*
240&lt;SPECIALCHARS&gt; ::= [&quot;$&quot;, &quot;_&quot;, &quot;-&quot;]
241</pre></div></div>
242
243<p>Querying Big Data is the main point of AsterixDB and AQL. Data in AsterixDB reside in datasets (collections of ADM objects), each of which in turn resides in some namespace known as a dataverse (data universe). Data access in a query expression is accomplished via a DatasetAccessExpression. Dataset access expressions are most commonly used in FLWOR expressions, where variables are bound to their contents.</p>
244<p>Note that the Identifier that identifies a dataset (or any other Identifier in AQL) can also be a StringLiteral. This is especially useful to avoid conficts with AQL keywords (e.g. &#x201c;dataset&#x201d;, &#x201c;null&#x201d;, or &#x201c;type&#x201d;).</p>
245<p>The following are three examples of legal dataset access expressions. The first one accesses a dataset called Customers in the dataverse called SalesDV. The second one accesses the Customers dataverse in whatever the current dataverse is. The third one does the same thing as the second but uses a slightly older AQL syntax.</p>
246<div class="section">
247<h5><a name="Examples"></a>Examples</h5>
248
249<div>
250<div>
251<pre class="source">dataset SalesDV.Customers
252dataset Customers
253dataset(&quot;Customers&quot;)
254</pre></div></div>
255</div></div>
256<div class="section">
257<h4><a name="Constructors"></a>Constructors</h4>
258
259<div>
260<div>
261<pre class="source">ListConstructor ::= ( OrderedListConstructor | UnorderedListConstructor )
262OrderedListConstructor ::= &quot;[&quot; ( Expression ( &quot;,&quot; Expression )* )? &quot;]&quot;
263UnorderedListConstructor ::= &quot;{{&quot; ( Expression ( &quot;,&quot; Expression )* )? &quot;}}&quot;
264ObjectConstructor ::= &quot;{&quot; ( FieldBinding ( &quot;,&quot; FieldBinding )* )? &quot;}&quot;
265FieldBinding ::= Expression &quot;:&quot; Expression
266</pre></div></div>
267
268<p>A major feature of AQL is its ability to construct new ADM data instances. This is accomplished using its constructors for each of the major ADM complex object structures, namely lists (ordered or unordered) and objects. Ordered lists are like JSON arrays, while unordered lists have bag (multiset) semantics. Objects are built from attributes that are field-name/field-value pairs, again like JSON. (See the AsterixDB Data Model document for more details on each.)</p>
269<p>The following examples illustrate how to construct a new ordered list with 3 items, a new unordered list with 4 items, and a new object with 2 fields, respectively. List elements can be homogeneous (as in the first example), which is the common case, or they may be heterogeneous (as in the second example). The data values and field name values used to construct lists and objects in constructors are all simply AQL expressions. Thus the list elements, field names, and field values used in constructors can be simple literals (as in these three examples) or they can come from query variable references or even arbitrarily complex AQL expressions.</p>
270<div class="section">
271<h5><a name="Examples"></a>Examples</h5>
272
273<div>
274<div>
275<pre class="source">[ &quot;a&quot;, &quot;b&quot;, &quot;c&quot; ]
276
277{{ 42, &quot;forty-two&quot;, &quot;AsterixDB!&quot;, 3.14f }}
278
279{
280 &quot;project name&quot;: &quot;AsterixDB&quot;
281 &quot;project members&quot;: {{ &quot;vinayakb&quot;, &quot;dtabass&quot;, &quot;chenli&quot; }}
282}
283</pre></div></div>
284</div>
285<div class="section">
286<h5><a name="Note"></a>Note</h5>
287<p>When constructing nested objects there needs to be a space between the closing braces to avoid confusion with the <tt>}}</tt> token that ends an unordered list constructor: <tt>{ &quot;a&quot; : { &quot;b&quot; : &quot;c&quot; }}</tt> will fail to parse while <tt>{ &quot;a&quot; : { &quot;b&quot; : &quot;c&quot; } }</tt> will work.</p></div></div></div>
288<div class="section">
289<h3><a name="Path_Expressions"></a>Path Expressions</h3>
290
291<div>
292<div>
293<pre class="source">ValueExpr ::= PrimaryExpr ( Field | Index )*
294Field ::= &quot;.&quot; Identifier
295Index ::= &quot;[&quot; ( Expression | &quot;?&quot; ) &quot;]&quot;
296</pre></div></div>
297
298<p>Components of complex types in ADM are accessed via path expressions. Path access can be applied to the result of an AQL expression that yields an instance of such a type, e.g., a object or list instance. For objects, path access is based on field names. For ordered lists, path access is based on (zero-based) array-style indexing. AQL also supports an &#x201c;I&#x2019;m feeling lucky&#x201d; style index accessor, [?], for selecting an arbitrary element from an ordered list. Attempts to access non-existent fields or list elements produce a null (i.e., missing information) result as opposed to signaling a runtime error.</p>
299<p>The following examples illustrate field access for a object, index-based element access for an ordered list, and also a composition thereof.</p>
300<div class="section">
301<div class="section">
302<h5><a name="Examples"></a>Examples</h5>
303
304<div>
305<div>
306<pre class="source">({&quot;list&quot;: [ &quot;a&quot;, &quot;b&quot;, &quot;c&quot;]}).list
307
308([&quot;a&quot;, &quot;b&quot;, &quot;c&quot;])[2]
309
310({ &quot;list&quot;: [ &quot;a&quot;, &quot;b&quot;, &quot;c&quot;]}).list[2]
311</pre></div></div>
312</div></div></div>
313<div class="section">
314<h3><a name="Logical_Expressions"></a>Logical Expressions</h3>
315
316<div>
317<div>
318<pre class="source">OperatorExpr ::= AndExpr ( &quot;or&quot; AndExpr )*
319AndExpr ::= RelExpr ( &quot;and&quot; RelExpr )*
320</pre></div></div>
321
322<p>As in most languages, boolean expressions can be built up from smaller expressions by combining them with the logical connectives and/or. Legal boolean values in AQL are true, false, and null. (Nulls in AQL are treated much like SQL treats its unknown truth value in boolean expressions.)</p>
323<p>The following is an example of a conjuctive range predicate in AQL. It will yield true if $a is bound to 4, null if $a is bound to null, and false otherwise.</p>
324<div class="section">
325<div class="section">
326<h5><a name="Example"></a>Example</h5>
327
328<div>
329<div>
330<pre class="source">$a &gt; 3 and $a &lt; 5
331</pre></div></div>
332</div></div></div>
333<div class="section">
334<h3><a name="Comparison_Expressions"></a>Comparison Expressions</h3>
335
336<div>
337<div>
338<pre class="source">RelExpr ::= AddExpr ( ( &quot;&lt;&quot; | &quot;&gt;&quot; | &quot;&lt;=&quot; | &quot;&gt;=&quot; | &quot;=&quot; | &quot;!=&quot; | &quot;~=&quot; ) AddExpr )?
339</pre></div></div>
340
341<p>AQL has the usual list of suspects, plus one, for comparing pairs of atomic values. The &#x201c;plus one&#x201d; is the last operator listed above, which is the &#x201c;roughly equal&#x201d; operator provided for similarity queries. (See the separate document on <a href="similarity.html">AsterixDB Similarity Queries</a> for more details on similarity matching.)</p>
342<p>An example comparison expression (which yields the boolean value true) is shown below.</p>
343<div class="section">
344<div class="section">
345<h5><a name="Example"></a>Example</h5>
346
347<div>
348<div>
349<pre class="source">5 &gt; 3
350</pre></div></div>
351</div></div></div>
352<div class="section">
353<h3><a name="Arithmetic_Expressions"></a>Arithmetic Expressions</h3>
354
355<div>
356<div>
357<pre class="source">AddExpr ::= MultExpr ( ( &quot;+&quot; | &quot;-&quot; ) MultExpr )*
358MultExpr ::= UnaryExpr ( ( &quot;*&quot; | &quot;/&quot; | &quot;div&quot; | &quot;%&quot; | &quot;mod&quot; | &quot;^&quot; ) UnaryExpr )*
359UnaryExpr ::= ( ( &quot;+&quot; | &quot;-&quot; ) )? ValueExpr
360</pre></div></div>
361
362<p>AQL also supports the usual cast of characters for arithmetic expressions. The example below evaluates to 25.</p>
363<div class="section">
364<div class="section">
365<h5><a name="Example"></a>Example</h5>
366
367<div>
368<div>
369<pre class="source">3 ^ 2 + 4 ^ 2
370</pre></div></div>
371</div></div></div>
372<div class="section">
373<h3><a name="FLWOR_Expression"></a>FLWOR Expression</h3>
374
375<div>
376<div>
377<pre class="source">FLWOR ::= ( ForClause | LetClause ) ( Clause )* (&quot;return&quot;|&quot;select&quot;) Expression
378Clause ::= ForClause | LetClause | WhereClause | OrderbyClause
379 | GroupClause | LimitClause | DistinctClause
380ForClause ::= (&quot;for&quot;|&quot;from&quot;) Variable ( &quot;at&quot; Variable )? &quot;in&quot; ( Expression )
381LetClause ::= (&quot;let&quot;|&quot;with&quot;) Variable &quot;:=&quot; Expression
382WhereClause ::= &quot;where&quot; Expression
383OrderbyClause ::= &quot;order&quot; &quot;by&quot; Expression ( ( &quot;asc&quot; ) | ( &quot;desc&quot; ) )?
384 ( &quot;,&quot; Expression ( ( &quot;asc&quot; ) | ( &quot;desc&quot; ) )? )*
385GroupClause ::= &quot;group&quot; &quot;by&quot; ( Variable &quot;:=&quot; )? Expression ( &quot;,&quot; ( Variable &quot;:=&quot; )? Expression )*
386 (&quot;with&quot;|&quot;keeping&quot;) VariableRef ( &quot;,&quot; VariableRef )*
387LimitClause ::= &quot;limit&quot; Expression ( &quot;offset&quot; Expression )?
388DistinctClause ::= &quot;distinct&quot; &quot;by&quot; Expression ( &quot;,&quot; Expression )*
389Variable ::= &lt;VARIABLE&gt;
390</pre></div></div>
391
392<p>The heart of AQL is the FLWOR (for-let-where-orderby-return) expression. The roots of this expression were borrowed from the expression of the same name in XQuery. A FLWOR expression starts with one or more clauses that establish variable bindings. A <tt>for</tt> clause binds a variable incrementally to each element of its associated expression; it includes an optional positional variable for counting/numbering the bindings. By default no ordering is implied or assumed by a <tt>for</tt> clause. A <tt>let</tt> clause binds a variable to the collection of elements computed by its associated expression.</p>
393<p>Following the initial <tt>for</tt> or <tt>let</tt> clause(s), a FLWOR expression may contain an arbitrary sequence of other clauses. The <tt>where</tt> clause in a FLWOR expression filters the preceding bindings via a boolean expression, much like a <tt>where</tt> clause does in a SQL query. The <tt>order by</tt> clause in a FLWOR expression induces an ordering on the data. The <tt>group by</tt> clause, discussed further below, forms groups based on its group by expressions, optionally naming the expressions&#x2019; values (which together form the grouping key for the expression). The <tt>with</tt> subclause of a <tt>group by</tt> clause specifies the variable(s) whose values should be grouped based on the grouping key(s); following the grouping clause, only the grouping key(s) and the variables named in the with subclause remain in scope, and the named grouping variables now contain lists formed from their input values. The <tt>limit</tt> clause caps the number of values returned, optionally starting its result count from a specified offset. (Web applications can use this feature for doing pagination.) The <tt>distinct</tt> clause is similar to the <tt>group-by</tt> clause, but it forms no groups; it serves only to eliminate duplicate values. As indicated by the grammar, the clauses in an AQL query can appear in any order. To interpret a query, one can think of data as flowing down through the query from the first clause to the <tt>return</tt> clause.</p>
394<p>The following example shows a FLWOR expression that selects and returns one user from the dataset FacebookUsers.</p>
395<div class="section">
396<div class="section">
397<h5><a name="Example"></a>Example</h5>
398
399<div>
400<div>
401<pre class="source">for $user in dataset FacebookUsers
402where $user.id = 8
403return $user
404</pre></div></div>
405
406<p>The next example shows a FLWOR expression that joins two datasets, FacebookUsers and FacebookMessages, returning user/message pairs. The results contain one object per pair, with result objects containing the user&#x2019;s name and an entire message.</p></div>
407<div class="section">
408<h5><a name="Example"></a>Example</h5>
409
410<div>
411<div>
412<pre class="source">for $user in dataset FacebookUsers
413for $message in dataset FacebookMessages
414where $message.author-id = $user.id
415return
416 {
417 &quot;uname&quot;: $user.name,
418 &quot;message&quot;: $message.message
419 };
420</pre></div></div>
421
422<p>In the next example, a <tt>let</tt> clause is used to bind a variable to all of a user&#x2019;s FacebookMessages. The query returns one object per user, with result objects containing the user&#x2019;s name and the set of all messages by that user.</p></div>
423<div class="section">
424<h5><a name="Example"></a>Example</h5>
425
426<div>
427<div>
428<pre class="source">for $user in dataset FacebookUsers
429let $messages :=
430 for $message in dataset FacebookMessages
431 where $message.author-id = $user.id
432 return $message.message
433return
434 {
435 &quot;uname&quot;: $user.name,
436 &quot;messages&quot;: $messages
437 };
438</pre></div></div>
439
440<p>The following example returns all TwitterUsers ordered by their followers count (most followers first) and language. When ordering <tt>null</tt> is treated as being smaller than any other value if <tt>null</tt>s are encountered in the ordering key(s).</p></div>
441<div class="section">
442<h5><a name="Example"></a>Example</h5>
443
444<div>
445<div>
446<pre class="source"> for $user in dataset TwitterUsers
447 order by $user.followers_count desc, $user.lang asc
448 return $user
449</pre></div></div>
450
451<p>The next example illustrates the use of the <tt>group by</tt> clause in AQL. After the <tt>group by</tt> clause in the query, only variables that are either in the <tt>group by</tt> list or in the <tt>with</tt> list are in scope. The variables in the clause&#x2019;s <tt>with</tt> list will each contain a collection of items following the <tt>group by</tt> clause; the collected items are the values that the source variable was bound to in the tuples that formed the group. For grouping <tt>null</tt> is handled as a single value.</p></div>
452<div class="section">
453<h5><a name="Example"></a>Example</h5>
454
455<div>
456<div>
457<pre class="source"> for $x in dataset FacebookMessages
458 let $messages := $x.message
459 group by $loc := $x.sender-location with $messages
460 return
461 {
462 &quot;location&quot; : $loc,
463 &quot;message&quot; : $messages
464 }
465</pre></div></div>
466
467<p>The use of the <tt>limit</tt> clause is illustrated in the next example.</p></div>
468<div class="section">
469<h5><a name="Example"></a>Example</h5>
470
471<div>
472<div>
473<pre class="source"> for $user in dataset TwitterUsers
474 order by $user.followers_count desc
475 limit 2
476 return $user
477</pre></div></div>
478
479<p>The final example shows how AQL&#x2019;s <tt>distinct by</tt> clause works. Each variable in scope before the distinct clause is also in scope after the <tt>distinct by</tt> clause. This clause works similarly to <tt>group by</tt>, but for each variable that contains more than one value after the <tt>distinct by</tt> clause, one value is picked nondeterministically. (If the variable is in the <tt>distinct by</tt> list, then its value will be deterministic.) Nulls are treated as a single value when they occur in a grouping field.</p></div>
480<div class="section">
481<h5><a name="Example"></a>Example</h5>
482
483<div>
484<div>
485<pre class="source"> for $x in dataset FacebookMessages
486 distinct by $x.sender-location
487 return
488 {
489 &quot;location&quot; : $x.sender-location,
490 &quot;message&quot; : $x.message
491 }
492</pre></div></div>
493
494<p>In order to allow SQL fans to write queries in their favored ways, AQL provides synonyms: <i>from</i> for <i>for</i>, <i>select</i> for <i>return</i>, <i>with</i> for <i>let</i>, and <i>keeping</i> for <i>with</i> in the group by clause. The following query is such an example.</p></div>
495<div class="section">
496<h5><a name="Example"></a>Example</h5>
497
498<div>
499<div>
500<pre class="source"> from $x in dataset FacebookMessages
501 with $messages := $x.message
502 group by $loc := $x.sender-location keeping $messages
503 select
504 {
505 &quot;location&quot; : $loc,
506 &quot;message&quot; : $messages
507 }
508</pre></div></div>
509</div></div></div>
510<div class="section">
511<h3><a name="Conditional_Expression"></a>Conditional Expression</h3>
512
513<div>
514<div>
515<pre class="source">IfThenElse ::= &quot;if&quot; &quot;(&quot; Expression &quot;)&quot; &quot;then&quot; Expression &quot;else&quot; Expression
516</pre></div></div>
517
518<p>A conditional expression is useful for choosing between two alternative values based on a boolean condition. If its first (<tt>if</tt>) expression is true, its second (<tt>then</tt>) expression&#x2019;s value is returned, and otherwise its third (<tt>else</tt>) expression is returned.</p>
519<p>The following example illustrates the form of a conditional expression.</p>
520<div class="section">
521<div class="section">
522<h5><a name="Example"></a>Example</h5>
523
524<div>
525<div>
526<pre class="source">if (2 &lt; 3) then &quot;yes&quot; else &quot;no&quot;
527</pre></div></div>
528</div></div></div>
529<div class="section">
530<h3><a name="Quantified_Expressions"></a>Quantified Expressions</h3>
531
532<div>
533<div>
534<pre class="source">QuantifiedExpression ::= ( ( &quot;some&quot; ) | ( &quot;every&quot; ) ) Variable &quot;in&quot; Expression
535 ( &quot;,&quot; Variable &quot;in&quot; Expression )* &quot;satisfies&quot; Expression
536</pre></div></div>
537
538<p>Quantified expressions are used for expressing existential or universal predicates involving the elements of a collection.</p>
539<p>The following pair of examples illustrate the use of a quantified expression to test that every (or some) element in the set [1, 2, 3] of integers is less than three. The first example yields <tt>false</tt> and second example yields <tt>true</tt>.</p>
540<p>It is useful to note that if the set were instead the empty set, the first expression would yield <tt>true</tt> (&#x201c;every&#x201d; value in an empty set satisfies the condition) while the second expression would yield <tt>false</tt> (since there isn&#x2019;t &#x201c;some&#x201d; value, as there are no values in the set, that satisfies the condition).</p>
541<div class="section">
542<div class="section">
543<h5><a name="Examples"></a>Examples</h5>
544
545<div>
546<div>
547<pre class="source">every $x in [ 1, 2, 3 ] satisfies $x &lt; 3
548some $x in [ 1, 2, 3 ] satisfies $x &lt; 3
549</pre></div></div>
550</div></div></div></div>
551<div class="section">
552<h2><a name="a3._Statements_.5BBack_to_TOC.5D"></a><a name="Statements" id="Statements">3. Statements</a> <font size="4"><a href="#toc">[Back to TOC]</a></font></h2>
553
554<div>
555<div>
556<pre class="source">Statement ::= ( SingleStatement ( &quot;;&quot; )? )* &lt;EOF&gt;
557SingleStatement ::= DataverseDeclaration
558 | FunctionDeclaration
559 | CreateStatement
560 | DropStatement
561 | LoadStatement
562 | SetStatement
563 | InsertStatement
564 | DeleteStatement
565 | UpsertStatement
566 | Query
567</pre></div></div>
568
569<p>In addition to expresssions for queries, AQL supports a variety of statements for data definition and manipulation purposes as well as controlling the context to be used in evaluating AQL expressions. AQL supports object-level ACID transactions that begin and terminate implicitly for each object inserted, deleted, upserted, or searched while a given AQL statement is being executed.</p>
570<p>This section details the statements supported in the AQL language.</p>
571<div class="section">
572<h3><a name="Declarations"></a>Declarations</h3>
573
574<div>
575<div>
576<pre class="source">DataverseDeclaration ::= &quot;use&quot; &quot;dataverse&quot; Identifier
577</pre></div></div>
578
579<p>The world of data in an AsterixDB cluster is organized into data namespaces called dataverses. To set the default dataverse for a series of statements, the use dataverse statement is provided.</p>
580<p>As an example, the following statement sets the default dataverse to be TinySocial.</p>
581<div class="section">
582<div class="section">
583<h5><a name="Example"></a>Example</h5>
584
585<div>
586<div>
587<pre class="source">use dataverse TinySocial;
588</pre></div></div>
589
590<p>The set statement in AQL is used to control aspects of the expression evalation context for queries.</p>
591
592<div>
593<div>
594<pre class="source">SetStatement ::= &quot;set&quot; Identifier StringLiteral
595</pre></div></div>
596
597<p>As an example, the following set statements request that Jaccard similarity with a similarity threshold 0.6 be used for set similarity matching when the ~= operator is used in a query expression.</p></div>
598<div class="section">
599<h5><a name="Example"></a>Example</h5>
600
601<div>
602<div>
603<pre class="source">set simfunction &quot;jaccard&quot;;
604set simthreshold &quot;0.6f&quot;;
605</pre></div></div>
606
607<p>When writing a complex AQL query, it can sometimes be helpful to define one or more auxilliary functions that each address a sub-piece of the overall query. The declare function statement supports the creation of such helper functions.</p>
608
609<div>
610<div>
611<pre class="source">FunctionDeclaration ::= &quot;declare&quot; &quot;function&quot; Identifier ParameterList &quot;{&quot; Expression &quot;}&quot;
612ParameterList ::= &quot;(&quot; ( &lt;VARIABLE&gt; ( &quot;,&quot; &lt;VARIABLE&gt; )* )? &quot;)&quot;
613</pre></div></div>
614
615<p>The following is a very simple example of a temporary AQL function definition.</p></div>
616<div class="section">
617<h5><a name="Example"></a>Example</h5>
618
619<div>
620<div>
621<pre class="source">declare function add($a, $b) {
622 $a + $b
623};
624</pre></div></div>
625</div></div></div>
626<div class="section">
627<h3><a name="Lifecycle_Management_Statements"></a>Lifecycle Management Statements</h3>
628
629<div>
630<div>
631<pre class="source">CreateStatement ::= &quot;create&quot; ( DataverseSpecification
632 | TypeSpecification
633 | DatasetSpecification
634 | IndexSpecification
635 | FunctionSpecification )
636
637QualifiedName ::= Identifier ( &quot;.&quot; Identifier )?
638DoubleQualifiedName ::= Identifier &quot;.&quot; Identifier ( &quot;.&quot; Identifier )?
639</pre></div></div>
640
641<p>The create statement in AQL is used for creating persistent artifacts in the context of dataverses. It can be used to create new dataverses, datatypes, datasets, indexes, and user-defined AQL functions.</p>
642<div class="section">
643<h4><a name="Dataverses"></a>Dataverses</h4>
644
645<div>
646<div>
647<pre class="source">DataverseSpecification ::= &quot;dataverse&quot; Identifier IfNotExists ( &quot;with format&quot; StringLiteral )?
648</pre></div></div>
649
650<p>The create dataverse statement is used to create new dataverses. To ease the authoring of reusable AQL scripts, its optional IfNotExists clause allows creation to be requested either unconditionally or only if the the dataverse does not already exist. If this clause is absent, an error will be returned if the specified dataverse already exists. The <tt>with format</tt> clause is a placeholder for future functionality that can safely be ignored.</p>
651<p>The following example creates a dataverse named TinySocial.</p>
652<div class="section">
653<h5><a name="Example"></a>Example</h5>
654
655<div>
656<div>
657<pre class="source">create dataverse TinySocial;
658</pre></div></div>
659</div></div>
660<div class="section">
661<h4><a name="Types"></a>Types</h4>
662
663<div>
664<div>
665<pre class="source">TypeSpecification ::= &quot;type&quot; FunctionOrTypeName IfNotExists &quot;as&quot; TypeExpr
666FunctionOrTypeName ::= QualifiedName
667IfNotExists ::= ( &quot;if not exists&quot; )?
668TypeExpr ::= ObjectTypeDef | TypeReference | OrderedListTypeDef | UnorderedListTypeDef
669ObjectTypeDef ::= ( &quot;closed&quot; | &quot;open&quot; )? &quot;{&quot; ( ObjectField ( &quot;,&quot; ObjectField )* )? &quot;}&quot;
670ObjectField ::= Identifier &quot;:&quot; ( TypeExpr ) ( &quot;?&quot; )?
671NestedField ::= Identifier ( &quot;.&quot; Identifier )*
672IndexField ::= NestedField ( &quot;:&quot; TypeReference )?
673TypeReference ::= Identifier
674OrderedListTypeDef ::= &quot;[&quot; ( TypeExpr ) &quot;]&quot;
675UnorderedListTypeDef ::= &quot;{{&quot; ( TypeExpr ) &quot;}}&quot;
676</pre></div></div>
677
678<p>The create type statement is used to create a new named ADM datatype. This type can then be used to create datasets or utilized when defining one or more other ADM datatypes. Much more information about the Asterix Data Model (ADM) is available in the <a href="datamodel.html">data model reference guide</a> to ADM. A new type can be a object type, a renaming of another type, an ordered list type, or an unordered list type. A object type can be defined as being either open or closed. Instances of a closed object type are not permitted to contain fields other than those specified in the create type statement. Instances of an open object type may carry additional fields, and open is the default for a new type (if neither option is specified).</p>
679<p>The following example creates a new ADM object type called FacebookUser type. Since it is closed, its instances will contain only what is specified in the type definition. The first four fields are traditional typed name/value pairs. The friend-ids field is an unordered list of 32-bit integers. The employment field is an ordered list of instances of another named object type, EmploymentType.</p>
680<div class="section">
681<h5><a name="Example"></a>Example</h5>
682
683<div>
684<div>
685<pre class="source">create type FacebookUserType as closed {
686 &quot;id&quot; : int32,
687 &quot;alias&quot; : string,
688 &quot;name&quot; : string,
689 &quot;user-since&quot; : datetime,
690 &quot;friend-ids&quot; : {{ int32 }},
691 &quot;employment&quot; : [ EmploymentType ]
692}
693</pre></div></div>
694
695<p>The next example creates a new ADM object type called FbUserType. Note that the type of the id field is UUID. You need to use this field type if you want to have this field be an autogenerated-PK field. Refer to the Datasets section later for more details.</p></div>
696<div class="section">
697<h5><a name="Example"></a>Example</h5>
698
699<div>
700<div>
701<pre class="source">create type FbUserType as closed {
702 &quot;id&quot; : uuid,
703 &quot;alias&quot; : string,
704 &quot;name&quot; : string
705}
706</pre></div></div>
707</div></div>
708<div class="section">
709<h4><a name="Datasets"></a>Datasets</h4>
710
711<div>
712<div>
713<pre class="source">DatasetSpecification ::= &quot;internal&quot;? &quot;dataset&quot; QualifiedName &quot;(&quot; QualifiedName &quot;)&quot; IfNotExists
714 PrimaryKey ( &quot;on&quot; Identifier )? ( &quot;hints&quot; Properties )?
715 ( &quot;using&quot; &quot;compaction&quot; &quot;policy&quot; CompactionPolicy ( Configuration )? )?
716 ( &quot;with filter on&quot; Identifier )?
717 | &quot;external&quot; &quot;dataset&quot; QualifiedName &quot;(&quot; QualifiedName &quot;)&quot; IfNotExists
718 &quot;using&quot; AdapterName Configuration ( &quot;hints&quot; Properties )?
719 ( &quot;using&quot; &quot;compaction&quot; &quot;policy&quot; CompactionPolicy ( Configuration )? )?
720AdapterName ::= Identifier
721Configuration ::= &quot;(&quot; ( KeyValuePair ( &quot;,&quot; KeyValuePair )* )? &quot;)&quot;
722KeyValuePair ::= &quot;(&quot; StringLiteral &quot;=&quot; StringLiteral &quot;)&quot;
723Properties ::= ( &quot;(&quot; Property ( &quot;,&quot; Property )* &quot;)&quot; )?
724Property ::= Identifier &quot;=&quot; ( StringLiteral | IntegerLiteral )
725FunctionSignature ::= FunctionOrTypeName &quot;@&quot; IntegerLiteral
726PrimaryKey ::= &quot;primary&quot; &quot;key&quot; NestedField ( &quot;,&quot; NestedField )* ( &quot;autogenerated &quot;)?
727CompactionPolicy ::= Identifier
728PrimaryKey ::= &quot;primary&quot; &quot;key&quot; Identifier ( &quot;,&quot; Identifier )* ( &quot;autogenerated &quot;)?
729</pre></div></div>
730
731<p>The create dataset statement is used to create a new dataset. Datasets are named, unordered collections of ADM object instances; they are where data lives persistently and are the targets for queries in AsterixDB. Datasets are typed, and AsterixDB will ensure that their contents conform to their type definitions. An Internal dataset (the default) is a dataset that is stored in and managed by AsterixDB. It must have a specified unique primary key that can be used to partition data across nodes of an AsterixDB cluster. The primary key is also used in secondary indexes to uniquely identify the indexed primary data objects. Random primary key (UUID) values can be auto-generated by declaring the field to be UUID and putting &#x201c;autogenerated&#x201d; after the &#x201c;primary key&#x201d; identifier. In this case, values for the auto-generated PK field should not be provided by the user since it will be auto-generated by AsterixDB. Optionally, a filter can be created on a field to further optimize range queries with predicates on the filter&#x2019;s field. (Refer to <a href="filters.html">Filter-Based LSM Index Acceleration</a> for more information about filters.)</p>
732<p>An External dataset is stored outside of AsterixDB (currently datasets in HDFS or on the local filesystem(s) of the cluster&#x2019;s nodes are supported). External dataset support allows AQL queries to treat external data as though it were stored in AsterixDB, making it possible to query &#x201c;legacy&#x201d; file data (e.g., Hive data) without having to physically import it into AsterixDB. For an external dataset, an appropriate adapter must be selected to handle the nature of the desired external data. (See the <a href="externaldata.html">guide to external data</a> for more information on the available adapters.)</p>
733<p>When creating a dataset, it is possible to choose a merge policy that controls which of the underlaying LSM storage components to be merged. Currently, AsterixDB provides four different merge policies that can be configured per dataset: no-merge, constant, prefix, and correlated-prefix. The no-merge policy simply never merges disk components. While the constant policy merges disk components when the number of components reaches some constant number k, which can be configured by the user. The prefix policy relies on component sizes and the number of components to decide which components to merge. Specifically, it works by first trying to identify the smallest ordered (oldest to newest) sequence of components such that the sequence does not contain a single component that exceeds some threshold size M and that either the sum of the component&#x2019;s sizes exceeds M or the number of components in the sequence exceeds another threshold C. If such a sequence of components exists, then each of the components in the sequence are merged together to form a single component. Finally, the correlated-prefix is similar to the prefix policy but it delegates the decision of merging the disk components of all the indexes in a dataset to the primary index. When the policy decides that the primary index needs to be merged (using the same decision criteria as for the prefix policy), then it will issue successive merge requests on behalf of all other indexes associated with the same dataset. The default policy for AsterixDB is the prefix policy except when there is a filter on a dataset, where the preferred policy for filters is the correlated-prefix.</p>
734<p>The following example creates an internal dataset for storing FacefookUserType objects. It specifies that their id field is their primary key.</p>
735<div class="section">
736<h5><a name="Example"></a>Example</h5>
737
738<div>
739<div>
740<pre class="source">create internal dataset FacebookUsers(FacebookUserType) primary key id;
741</pre></div></div>
742
743<p>The following example creates an internal dataset for storing FbUserType objects. It specifies that their id field is their primary key. It also specifies that the id field is an auto-generated field, meaning that a randomly generated UUID value will be assigned to each object by the system. (A user should therefore not proivde a value for this field.) Note that the id field should be UUID.</p></div>
744<div class="section">
745<h5><a name="Example"></a>Example</h5>
746
747<div>
748<div>
749<pre class="source">create internal dataset FbMsgs(FbUserType) primary key id autogenerated;
750</pre></div></div>
751
752<p>The next example creates an external dataset for storing LineitemType objects. The choice of the <tt>hdfs</tt> adapter means that its data will reside in HDFS. The create statement provides parameters used by the hdfs adapter: the URL and path needed to locate the data in HDFS and a description of the data format.</p></div>
753<div class="section">
754<h5><a name="Example"></a>Example</h5>
755
756<div>
757<div>
758<pre class="source">create external dataset Lineitem('LineitemType) using hdfs (
759 (&quot;hdfs&quot;=&quot;hdfs://HOST:PORT&quot;),
760 (&quot;path&quot;=&quot;HDFS_PATH&quot;),
761 (&quot;input-format&quot;=&quot;text-input-format&quot;),
762 (&quot;format&quot;=&quot;delimited-text&quot;),
763 (&quot;delimiter&quot;=&quot;|&quot;));
764</pre></div></div>
765</div></div>
766<div class="section">
767<h4><a name="Indices"></a>Indices</h4>
768
769<div>
770<div>
771<pre class="source">IndexSpecification ::= &quot;index&quot; Identifier IfNotExists &quot;on&quot; QualifiedName
772 &quot;(&quot; ( IndexField ) ( &quot;,&quot; IndexField )* &quot;)&quot; ( &quot;type&quot; IndexType )? ( &quot;enforced&quot; )?
773IndexType ::= &quot;btree&quot;
774 | &quot;rtree&quot;
775 | &quot;keyword&quot;
776 | &quot;ngram&quot; &quot;(&quot; IntegerLiteral &quot;)&quot;
777 | &quot;fulltext&quot;
778</pre></div></div>
779
780<p>The create index statement creates a secondary index on one or more fields of a specified dataset. Supported index types include <tt>btree</tt> for totally ordered datatypes, <tt>rtree</tt> for spatial data, and <tt>keyword</tt>, <tt>ngram</tt>, and <tt>fulltext</tt> for textual (string) data. An index can be created on a nested field (or fields) by providing a valid path expression as an index field identifier. An index field is not required to be part of the datatype associated with a dataset if that datatype is declared as open and the field&#x2019;s type is provided along with its type and the <tt>enforced</tt> keyword is specified in the end of index definition. <tt>Enforcing</tt> an open field will introduce a check that will make sure that the actual type of an indexed field (if the field exists in the object) always matches this specified (open) field type.</p>
781<p>The following example creates a btree index called fbAuthorIdx on the author-id field of the FacebookMessages dataset. This index can be useful for accelerating exact-match queries, range search queries, and joins involving the author-id field.</p>
782<div class="section">
783<h5><a name="Example"></a>Example</h5>
784
785<div>
786<div>
787<pre class="source">create index fbAuthorIdx on FacebookMessages(author-id) type btree;
788</pre></div></div>
789
790<p>The following example creates an open btree index called fbSendTimeIdx on the open send-time field of the FacebookMessages dataset having datetime type. This index can be useful for accelerating exact-match queries, range search queries, and joins involving the send-time field.</p></div>
791<div class="section">
792<h5><a name="Example"></a>Example</h5>
793
794<div>
795<div>
796<pre class="source">create index fbSendTimeIdx on FacebookMessages(send-time:datetime) type btree enforced;
797</pre></div></div>
798
799<p>The following example creates a btree index called twUserScrNameIdx on the screen-name field, which is a nested field of the user field in the TweetMessages dataset. This index can be useful for accelerating exact-match queries, range search queries, and joins involving the screen-name field.</p></div>
800<div class="section">
801<h5><a name="Example"></a>Example</h5>
802
803<div>
804<div>
805<pre class="source">create index twUserScrNameIdx on TweetMessages(user.screen-name) type btree;
806</pre></div></div>
807
808<p>The following example creates an rtree index called fbSenderLocIdx on the sender-location field of the FacebookMessages dataset. This index can be useful for accelerating queries that use the <a href="functions.html#spatial-intersect"><tt>spatial-intersect</tt> function</a> in a predicate involving the sender-location field.</p></div>
809<div class="section">
810<h5><a name="Example"></a>Example</h5>
811
812<div>
813<div>
814<pre class="source">create index fbSenderLocIndex on FacebookMessages(sender-location) type rtree;
815</pre></div></div>
816
817<p>The following example creates a 3-gram index called fbUserIdx on the name field of the FacebookUsers dataset. This index can be used to accelerate some similarity or substring maching queries on the name field. For details refer to the <a href="similarity.html#NGram_Index">document on similarity queries</a>.</p></div>
818<div class="section">
819<h5><a name="Example"></a>Example</h5>
820
821<div>
822<div>
823<pre class="source">create index fbUserIdx on FacebookUsers(name) type ngram(3);
824</pre></div></div>
825
826<p>The following example creates a keyword index called fbMessageIdx on the message field of the FacebookMessages dataset. This keyword index can be used to optimize queries with token-based similarity predicates on the message field. For details refer to the <a href="similarity.html#Keyword_Index">document on similarity queries</a>.</p></div>
827<div class="section">
828<h5><a name="Example"></a>Example</h5>
829
830<div>
831<div>
832<pre class="source">create index fbMessageIdx on FacebookMessages(message) type keyword;
833</pre></div></div>
834
835<p>The following example creates a full-text index called fbMessageIdx on the message field of the FacebookMessages dataset. This full-text index can be used to optimize queries with full-text search predicates on the message field. For details refer to the <a href="fulltext.html#toc">document on full-text queries</a>.</p></div>
836<div class="section">
837<h5><a name="Example"></a>Example</h5>
838
839<div>
840<div>
841<pre class="source">create index fbMessageIdx on FacebookMessages(message) type fulltext;
842</pre></div></div>
843</div></div>
844<div class="section">
845<h4><a name="Functions"></a>Functions</h4>
846<p>The create function statement creates a named function that can then be used and reused in AQL queries. The body of a function can be any AQL expression involving the function&#x2019;s parameters.</p>
847
848<div>
849<div>
850<pre class="source">FunctionSpecification ::= &quot;function&quot; FunctionOrTypeName IfNotExists ParameterList &quot;{&quot; Expression &quot;}&quot;
851</pre></div></div>
852
853<p>The following is a very simple example of a create function statement. It differs from the declare function example shown previously in that it results in a function that is persistently registered by name in the specified dataverse.</p>
854<div class="section">
855<h5><a name="Example"></a>Example</h5>
856
857<div>
858<div>
859<pre class="source">create function add($a, $b) {
860 $a + $b
861};
862</pre></div></div>
863</div></div>
864<div class="section">
865<h4><a name="Removal"></a>Removal</h4>
866
867<div>
868<div>
869<pre class="source">DropStatement ::= &quot;drop&quot; ( &quot;dataverse&quot; Identifier IfExists
870 | &quot;type&quot; FunctionOrTypeName IfExists
871 | &quot;dataset&quot; QualifiedName IfExists
872 | &quot;index&quot; DoubleQualifiedName IfExists
873 | &quot;function&quot; FunctionSignature IfExists )
874IfExists ::= ( &quot;if&quot; &quot;exists&quot; )?
875</pre></div></div>
876
877<p>The drop statement in AQL is the inverse of the create statement. It can be used to drop dataverses, datatypes, datasets, indexes, and functions.</p>
878<p>The following examples illustrate uses of the drop statement.</p>
879<div class="section">
880<h5><a name="Example"></a>Example</h5>
881
882<div>
883<div>
884<pre class="source">drop dataset FacebookUsers if exists;
885
886drop index FacebookUsers.fbSenderLocIndex;
887
888drop type FacebookUserType;
889
890drop dataverse TinySocial;
891
892drop function add;
893</pre></div></div>
894</div></div></div>
895<div class="section">
896<h3><a name="Import.2FExport_Statements"></a>Import/Export Statements</h3>
897
898<div>
899<div>
900<pre class="source">LoadStatement ::= &quot;load&quot; &quot;dataset&quot; QualifiedName &quot;using&quot; AdapterName Configuration ( &quot;pre-sorted&quot; )?
901</pre></div></div>
902
903<p>The load statement is used to initially populate a dataset via bulk loading of data from an external file. An appropriate adapter must be selected to handle the nature of the desired external data. The load statement accepts the same adapters and the same parameters as external datasets. (See the <a href="externaldata.html">guide to external data</a> for more information on the available adapters.) If a dataset has an auto-generated primary key field, a file to be imported should not include that field in it.</p>
904<p>The following example shows how to bulk load the FacebookUsers dataset from an external file containing data that has been prepared in ADM format.</p>
905<div class="section">
906<div class="section">
907<h5><a name="Example"></a>Example</h5>
908
909<div>
910<div>
911<pre class="source">load dataset FacebookUsers using localfs
912((&quot;path&quot;=&quot;localhost:///Users/zuck/AsterixDB/load/fbu.adm&quot;),(&quot;format&quot;=&quot;adm&quot;));
913</pre></div></div>
914</div></div></div>
915<div class="section">
916<h3><a name="Modification_Statements"></a>Modification Statements</h3>
917<div class="section">
918<h4><a name="Insert"></a>Insert</h4>
919
920<div>
921<div>
922<pre class="source">InsertStatement ::= &quot;insert&quot; &quot;into&quot; &quot;dataset&quot; QualifiedName ( &quot;as&quot; Variable )? Query ( &quot;returning&quot; Query )?
923</pre></div></div>
924
925<p>The AQL insert statement is used to insert data into a dataset. The data to be inserted comes from an AQL query expression. The expression can be as simple as a constant expression, or in general it can be any legal AQL query. Inserts in AsterixDB are processed transactionally, with the scope of each insert transaction being the insertion of a single object plus its affiliated secondary index entries (if any). If the query part of an insert returns a single object, then the insert statement itself will be a single, atomic transaction. If the query part returns multiple objects, then each object inserted will be handled independently as a tranaction. If a dataset has an auto-generated primary key field, an insert statement should not include a value for that field in it. (The system will automatically extend the provided object with this additional field and a corresponding value.). The optional &#x201c;as Variable&#x201d; provides a variable binding for the inserted objects, which can be used in the &#x201c;returning&#x201d; clause. The optional &#x201c;returning Query&#x201d; allows users to run simple queries/functions on the objects returned by the insert. This query cannot refer to any datasets.</p>
926<p>The following example illustrates a query-based insertion.</p>
927<div class="section">
928<h5><a name="Example"></a>Example</h5>
929
930<div>
931<div>
932<pre class="source">insert into dataset UsersCopy as $inserted (for $user in dataset FacebookUsers return $user ) returning $inserted.screen-name
933</pre></div></div>
934</div></div>
935<div class="section">
936<h4><a name="Delete"></a>Delete</h4>
937
938<div>
939<div>
940<pre class="source">DeleteStatement ::= &quot;delete&quot; Variable &quot;from&quot; &quot;dataset&quot; QualifiedName ( &quot;where&quot; Expression )?
941</pre></div></div>
942
943<p>The AQL delete statement is used to delete data from a target dataset. The data to be deleted is identified by a boolean expression involving the variable bound to the target dataset in the delete statement. Deletes in AsterixDB are processed transactionally, with the scope of each delete transaction being the deletion of a single object plus its affiliated secondary index entries (if any). If the boolean expression for a delete identifies a single object, then the delete statement itself will be a single, atomic transaction. If the expression identifies multiple objects, then each object deleted will be handled independently as a transaction.</p>
944<p>The following example illustrates a single-object deletion.</p>
945<div class="section">
946<h5><a name="Example"></a>Example</h5>
947
948<div>
949<div>
950<pre class="source">delete $user from dataset FacebookUsers where $user.id = 8;
951</pre></div></div>
952</div></div>
953<div class="section">
954<h4><a name="Upsert"></a>Upsert</h4>
955
956<div>
957<div>
958<pre class="source">UpsertStatement ::= &quot;upsert&quot; &quot;into&quot; &quot;dataset&quot; QualifiedName Query
959</pre></div></div>
960
961<p>The AQL upsert statement is used to couple delete (if found) with insert data into a dataset. The data to be upserted comes from an AQL query expression. The expression can be as simple as a constant expression, or in general it can be any legal AQL query. Upserts in AsterixDB are processed transactionally, with the scope of each upsert transaction being the upsertion (deletion if found + insertion) of a single object plus its affiliated secondary index entries (if any). If the query part of an upsert returns a single object, then the upsert statement itself will be a single, atomic transaction. If the query part returns multiple objects, then each object upserted will be handled independently as a tranaction.</p>
962<p>The following example illustrates a query-based upsertion.</p>
963<div class="section">
964<h5><a name="Example"></a>Example</h5>
965
966<div>
967<div>
968<pre class="source">upsert into dataset Users (for $user in dataset FacebookUsers return $user)
969</pre></div></div>
970
971<p>We close this guide to AQL with one final example of a query expression.</p></div>
972<div class="section">
973<h5><a name="Example"></a>Example</h5>
974
975<div>
976<div>
977<pre class="source">for $praise in {{ &quot;great&quot;, &quot;brilliant&quot;, &quot;awesome&quot; }}
978return
979 string-concat([&quot;AsterixDB is &quot;, $praise])
980</pre></div></div></div></div></div></div>
981 </div>
982 </div>
983 </div>
984 <hr/>
985 <footer>
986 <div class="container-fluid">
987 <div class="row-fluid">
988<div class="row-fluid">Apache AsterixDB, AsterixDB, Apache, the Apache
989 feather logo, and the Apache AsterixDB project logo are either
990 registered trademarks or trademarks of The Apache Software
991 Foundation in the United States and other countries.
992 All other marks mentioned may be trademarks or registered
993 trademarks of their respective owners.
994 </div>
995 </div>
996 </div>
997 </footer>
998 </body>
999</html>