diff --git a/site/content/3.11/get-started/how-to-interact-with-arangodb.md b/site/content/3.11/get-started/how-to-interact-with-arangodb.md index ab0efa927c..896de02759 100644 --- a/site/content/3.11/get-started/how-to-interact-with-arangodb.md +++ b/site/content/3.11/get-started/how-to-interact-with-arangodb.md @@ -41,21 +41,21 @@ programming language, and do all the talking to the server. Integrations combine a third-party technology with ArangoDB and can be seen as a translation layer that takes over the low-level communication with the server. -### REST API +### HTTP REST API -Under the hood, all interactions with the server make use of its REST API. -A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer) -API is an application programming interface based on the HTTP protocol that -powers the world wide web. +Under the hood, all interactions with the server make use of its RESTful HTTP API. +A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer)-based +API is an application programming interface using the HTTP protocol, the +protocol that powers the world wide web. All requests from the outside to the server need to made against the respective endpoints of this API to perform actions. This includes the web interface, _arangosh_, as well as the drivers and integrations for different programming languages and environments. They all provide a convenient way to work with ArangoDB, but you -may use the low-level REST API directly as needed. +may use the low-level HTTP API directly as needed. -See the [HTTP](../develop/http-api/_index.md) documentation to learn more about the API, how requests -are handled and what endpoints are available. +See the [HTTP API](../develop/http-api/_index.md) documentation to learn more +about the API, how requests are handled, and what endpoints are available. ## Set Up and Deploy ArangoDB diff --git a/site/content/3.11/get-started/start-using-aql.md b/site/content/3.11/get-started/start-using-aql.md deleted file mode 100644 index 5e4ade2c4c..0000000000 --- a/site/content/3.11/get-started/start-using-aql.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Start using AQL -menuTitle: Start using AQL -weight: 45 -description: >- - You can execute AQL queries in different ways, from the easy-to-use - web interface to the raw HTTP API ---- -## How to invoke AQL - -AQL queries can be executed using: - -- the web interface -- the `db` object (either in arangosh or in a Foxx service) -- the raw HTTP API -- through drivers and integrations as an abstraction over the HTTP API - -There are always calls to the server's API under the hood, but the web interface, -the `db` object, drivers, and integrations abstract away the low-level -communication details and are thus easier to use. - -The ArangoDB Web Interface has a [specific tab for AQL queries execution](../aql/how-to-invoke-aql/with-the-web-interface.md). - -You can run [AQL queries from the ArangoDB Shell](../aql/how-to-invoke-aql/with-arangosh.md) -with the [`_query()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and -[`_createStatement()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) methods -of the [`db` object](../develop/javascript-api/@arangodb/db-object.md). This chapter -also describes how to use bind parameters, statistics, counting and cursors with -arangosh. - -If you are using Foxx, see [how to write database queries](../develop/foxx-microservices/getting-started.md#writing-database-queries) -for examples including tagged template strings. - -If you want to run AQL queries from your application via the HTTP REST API, -see the full API description at [HTTP interface for AQL queries](../develop/http-api/queries/aql-queries.md). - -See the respective [driver](../develop/drivers/_index.md) or -[integration](../develop/integrations/_index.md) for its support of AQL queries. - -## Learn the query language - -See the [AQL documentation](../aql/_index.md) for the full language reference -as well as examples. - -For a tutorial, sign up for the [ArangoDB University](https://university.arangodb.com/) -to get access to the **AQL Fundamentals** course. diff --git a/site/content/3.11/get-started/start-using-aql/_index.md b/site/content/3.11/get-started/start-using-aql/_index.md new file mode 100644 index 0000000000..42366abd3c --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/_index.md @@ -0,0 +1,156 @@ +--- +title: Start using AQL +menuTitle: Start using AQL +weight: 52 +description: >- + Learn how to run your first queries written in ArangoDB's Query Language + and how to go from there, using a Game of Thrones dataset +--- +This is an introduction to ArangoDB's query language AQL, built around a small +dataset of characters from the novel and fantasy drama television series +Game of Thrones (as of season 1). It includes character traits in two languages, +some family relations, and last but not least a small set of filming locations, +which makes for an interesting mix of data to work with. + +There is no need to import the data before you start. It is provided as part +of the AQL queries in this tutorial. You can interact with ArangoDB using its +built-in web interface to manage collections and execute the queries with ease, +but you may also use a different interface. + +## How to run AQL queries + +{{< tabs "interfaces" >}} + +{{< tab "Web interface" >}} +ArangoDB's web interface has a **Queries** section for +[executing AQL queries](../../aql/how-to-invoke-aql/with-the-web-interface.md). + +1. If necessary, [switch to the database](../../concepts/data-structure/databases.md#set-the-database-context) + that you want to run queries in. +2. Click **QUERIES** in the main navigation. +3. Enter an AQL query in the code editor, e.g. `RETURN CONCAT("Hello, ", @name)`. +4. Specify any needed bind parameters in the panel on the right-hand side, + e.g. set `name` to a value of `AQL`. +5. Click the **Execute** button or hit `Ctrl`/`Cmd` + `Return`. +{{< /tab >}} + +{{< tab "arangosh" >}} +You can run AQL queries from the ArangoDB Shell ([arangosh](../../components/tools/arangodb-shell/_index.md)) +with the [`db._query()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and +[`db._createStatement()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) +methods of the [`db` object](../../develop/javascript-api/@arangodb/db-object.md). + +```js +--- +name: arangosh_execute_query_bindvars +description: '' +--- +db._query(`RETURN CONCAT("Hello, ", @name)`, { name: "AQL" }).toArray(); +// -- or -- +var name = "AQL"; +db._query(aql`RETURN CONCAT("Hello, ", ${name})`).toArray(); +``` +See [`db._query()`](../../develop/javascript-api/@arangodb/db-object.md#db_queryquerystring--bindvars--mainoptions--suboptions) +in the _JavaScript API_ for details. + +If you use Foxx, see [how to write database queries](../../develop/foxx-microservices/getting-started.md#writing-database-queries) +for examples including tagged template strings. +{{< /tab >}} + +{{< tab "cURL" >}} +You can use a tool like [cURL](https://curl.se/) to run AQL queries from a +command-line, directly using the HTTP REST API of ArangoDB. + +The response bodies are generally compact JSON (without any line breaks and +indentation). You can format them with the [jq](https://jqlang.github.io/jq/) +tool for better readability if you have it installed: + +```sh +curl -d '{"query":"RETURN CONCAT(\"Hello, \", @name)","bindVars":{"name":"AQL"}}' http://localhost:8529/_api/cursor | jq +``` + +See the [`POST /_db/{database-name}/_api/cursor`](../../develop/http-api/queries/aql-queries.md#create-a-cursor) +endpoint in the _HTTP API_ for details. +{{< /tab >}} + +{{< tab "JavaScript" >}} +```js +import { Database, aql } from "arangojs"; +const db = new Database(); + +const name = "AQL"; +const cursor = await db.query(aql`RETURN CONCAT("Hello, ", ${name})`); +const result = cursor.all(); +console.log(result); +``` + +See [`Database.query()`](https://arangodb.github.io/arangojs/latest/classes/databases.Database.html#query) +in the _arangojs_ documentation for details. +{{< /tab >}} + +{{< tab "Go" >}} +```go +ctx := context.Background() +query := `RETURN CONCAT("Hello, ", @name)` +options := arangodb.QueryOptions{ + BindVars: map[string]interface{}{ + "name": "AQL", + }, +} +cursor, err := db.Query(ctx, query, &options) +if err != nil { + log.Fatalf("Failed to run query:\n%v\n", err) +} else { + defer cursor.Close() + var str string + for cursor.HasMore() { + _, err := cursor.ReadDocument(ctx, &str) + if err != nil { + log.Fatalf("Failed to read cursor:\n%v\n", err) + } else { + fmt.Println(str) + } + } +} +``` + +See [`DatabaseQuery.Query()`](https://pkg.go.dev/github.com/arangodb/go-driver/v2/arangodb#DatabaseQuery) +in the _go-driver_ v2 documentation for details. +{{< /tab >}} + +{{< tab "Java" >}} +```java +String query = "RETURN CONCAT(\"Hello, \", @name)"; +Map bindVars = Collections.singletonMap("name", "AQL"); +ArangoCursor cursor = db.query(query, String.class, bindVars); +cursor.forEach(result -> System.out.println(result)); +``` + +See [`ArangoDatabase.query()`](https://www.javadoc.io/doc/com.arangodb/arangodb-java-driver/latest/com/arangodb/ArangoDatabase.html#query%28java.lang.String,java.lang.Class,java.util.Map%29) +in the _arangodb-java-driver_ documentation for details. +{{< /tab >}} + +{{< tab "Python" >}} +```py +query = "RETURN CONCAT('Hello, ', @name)" +bind_vars = { "name": "AQL" } +cursor = db.aql.execute(query, bind_vars=bind_vars) +for result in cursor: + print(result) +``` + +See [`AQL.execute()`](https://docs.python-arango.com/en/main/specs.html#arango.aql.AQL.execute) +in the _python-arango_ documentation for details. +{{< /tab >}} + +{{< /tabs >}} + +## Learn the query language + +The following pages guide you through important query constructs for storing +and retrieving data, covering basic as well as some advanced features. + +Afterwards, you can read the [AQL documentation](../../aql/_index.md) for the +full language reference and query examples. + +{{< comment >}}TODO: Advanced data manipulation: attributes, projections, calculations... Aggregation: Grouping techniques{{< /comment >}} diff --git a/site/content/3.11/get-started/start-using-aql/crud.md b/site/content/3.11/get-started/start-using-aql/crud.md new file mode 100644 index 0000000000..84ce06ce9f --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/crud.md @@ -0,0 +1,371 @@ +--- +title: AQL CRUD operations +menuTitle: CRUD operations +weight: 5 +description: >- + Learn how to **c**reate, **r**ead, **u**pdate, and **d**elete documents with + the ArangoDB Query Language +--- +## Create documents + +Before you can insert documents with AQL, you need a place to put them in – a +collection. You can [manage collections](../../concepts/data-structure/collections.md#collections-api) +via different interfaces including the web interface, arangosh, or a driver. +It is not possible to do so with AQL, however. + +1. In the web interface, click **COLLECTIONS** in the main navigation. +2. Click the first card labeled **Add Collection**. +3. Enter `Characters` as the **Name**. +4. Leave the **Type** set to the default value of **Document**. +5. Click the **Save** button. + +The new collection should appear in the list. Next, click **QUERIES** in the +main navigation. To create the first document for the collection with AQL, +use the following AQL query, which you can paste into the query textbox and +run by clicking the **Execute** button: + +```aql +INSERT { + _key: "test", + name: "First", + surname: "Test", + alive: false, + age: 123, + traits: [ "X", "Y" ] +} INTO Characters RETURN NEW +``` + +The syntax is `INSERT document INTO collectionName`. The document is an object +like you may know it from JavaScript or JSON, which is comprised of attribute +key and value pairs. The quotes around the attribute keys are optional in AQL. +Keys are always character sequences (strings), whereas attribute values can +have [different types](../../aql/fundamentals/data-types.md): + +- `null` +- boolean (`true`, `false`) +- number (integer and floating point values) +- string +- array +- object + +The name and surname of the character document you inserted are both string +values. The alive state uses a boolean value. Age is a numeric value. +The traits are an array of strings. The entire document is an object. + +The `RETURN NEW` part is optional and makes the query return the document +including any system attributes that may get added by ArangoDB. If you don't +specify the `_key` attribute then a document key is automatically generated. + +Next, add several characters with a single query: + +```aql +FOR d IN @data + INSERT d INTO Characters +``` + +The `FOR` loop iterates over `@data`, which is a placeholder in the query for +binding the list of characters in JSON format. + +In the web interface, there is a panel for **Bind Variables** on the right-hand +side of the query editor. When you enter a placeholder like `@data` in the editor, +a row appears in the side panel to specify the value for the placeholder. +Paste the following text into the field for the `data` bind variable: + +```json +[ + { "_key": "ned", "name": "Ned", "surname": "Stark", "alive": true, "age": 41, "traits": ["A","H","C","N","P"] }, + { "_key": "robert", "name": "Robert", "surname": "Baratheon", "alive": false, "traits": ["A","H","C"] }, + { "_key": "jaime", "name": "Jaime", "surname": "Lannister", "alive": true, "age": 36, "traits": ["A","F","B"] }, + { "_key": "catelyn", "name": "Catelyn", "surname": "Stark", "alive": false, "age": 40, "traits": ["D","H","C"] }, + { "_key": "cersei", "name": "Cersei", "surname": "Lannister", "alive": true, "age": 36, "traits": ["H","E","F"] }, + { "_key": "daenerys", "name": "Daenerys", "surname": "Targaryen", "alive": true, "age": 16, "traits": ["D","H","C"] }, + { "_key": "jorah", "name": "Jorah", "surname": "Mormont", "alive": false, "traits": ["A","B","C","F"] }, + { "_key": "petyr", "name": "Petyr", "surname": "Baelish", "alive": false, "traits": ["E","G","F"] }, + { "_key": "viserys", "name": "Viserys", "surname": "Targaryen", "alive": false, "traits": ["O","L","N"] }, + { "_key": "jon", "name": "Jon", "surname": "Snow", "alive": true, "age": 16, "traits": ["A","B","C","F"] }, + { "_key": "sansa", "name": "Sansa", "surname": "Stark", "alive": true, "age": 13, "traits": ["D","I","J"] }, + { "_key": "arya", "name": "Arya", "surname": "Stark", "alive": true, "age": 11, "traits": ["C","K","L"] }, + { "_key": "robb", "name": "Robb", "surname": "Stark", "alive": false, "traits": ["A","B","C","K"] }, + { "_key": "theon", "name": "Theon", "surname": "Greyjoy", "alive": true, "age": 16, "traits": ["E","R","K"] }, + { "_key": "bran", "name": "Bran", "surname": "Stark", "alive": true, "age": 10, "traits": ["L","J"] }, + { "_key": "joffrey", "name": "Joffrey", "surname": "Baratheon", "alive": false, "age": 19, "traits": ["I","L","O"] }, + { "_key": "sandor", "name": "Sandor", "surname": "Clegane", "alive": true, "traits": ["A","P","K","F"] }, + { "_key": "tyrion", "name": "Tyrion", "surname": "Lannister", "alive": true, "age": 32, "traits": ["F","K","M","N"] }, + { "_key": "khal", "name": "Khal", "surname": "Drogo", "alive": false, "traits": ["A","C","O","P"] }, + { "_key": "tywin", "name": "Tywin", "surname": "Lannister", "alive": false, "traits": ["O","M","H","F"] }, + { "_key": "davos", "name": "Davos", "surname": "Seaworth", "alive": true, "age": 49, "traits": ["C","K","P","F"] }, + { "_key": "samwell", "name": "Samwell", "surname": "Tarly", "alive": true, "age": 17, "traits": ["C","L","I"] }, + { "_key": "stannis", "name": "Stannis", "surname": "Baratheon", "alive": false, "traits": ["H","O","P","M"] }, + { "_key": "melisandre", "name": "Melisandre", "alive": true, "traits": ["G","E","H"] }, + { "_key": "margaery", "name": "Margaery", "surname": "Tyrell", "alive": false, "traits": ["M","D","B"] }, + { "_key": "jeor", "name": "Jeor", "surname": "Mormont", "alive": false, "traits": ["C","H","M","P"] }, + { "_key": "bronn", "name": "Bronn", "alive": true, "traits": ["K","E","C"] }, + { "_key": "varys", "name": "Varys", "alive": true, "traits": ["M","F","N","E"] }, + { "_key": "shae", "name": "Shae", "alive": false, "traits": ["M","D","G"] }, + { "_key": "talisa", "name": "Talisa", "surname": "Maegyr", "alive": false, "traits": ["D","C","B"] }, + { "_key": "gendry", "name": "Gendry", "alive": false, "traits": ["K","C","A"] }, + { "_key": "ygritte", "name": "Ygritte", "alive": false, "traits": ["A","P","K"] }, + { "_key": "tormund", "name": "Tormund", "surname": "Giantsbane", "alive": true, "traits": ["C","P","A","I"] }, + { "_key": "gilly", "name": "Gilly", "alive": true, "traits": ["L","J"] }, + { "_key": "brienne", "name": "Brienne", "surname": "Tarth", "alive": true, "age": 32, "traits": ["P","C","A","K"] }, + { "_key": "ramsay", "name": "Ramsay", "surname": "Bolton", "alive": true, "traits": ["E","O","G","A"] }, + { "_key": "ellaria", "name": "Ellaria", "surname": "Sand", "alive": true, "traits": ["P","O","A","E"] }, + { "_key": "daario", "name": "Daario", "surname": "Naharis", "alive": true, "traits": ["K","P","A"] }, + { "_key": "missandei", "name": "Missandei", "alive": true, "traits": ["D","L","C","M"] }, + { "_key": "tommen", "name": "Tommen", "surname": "Baratheon", "alive": true, "traits": ["I","L","B"] }, + { "_key": "jaqen", "name": "Jaqen", "surname": "H'ghar", "alive": true, "traits": ["H","F","K"] }, + { "_key": "roose", "name": "Roose", "surname": "Bolton", "alive": true, "traits": ["H","E","F","A"] }, + { "_key": "high-sparrow", "name": "The High Sparrow", "alive": true, "traits": ["H","M","F","O"] } +] +``` + +The data is an array of objects, like `[ {...}, {...}, ... ]`. + +`FOR variableName IN expression` is used to iterate over each element of the +array. In each loop, one element is assigned to the variable `d` (`FOR d IN @data`). +This variable is then used in the `INSERT` statement instead of a literal +object definition. What it does is basically the following: + +```aql +// Invalid query + +INSERT { + "_key": "robert", + "name": "Robert", + "surname": "Baratheon", + "alive": false, + "traits": ["A","H","C"] +} INTO Characters + +INSERT { + "_key": "jaime", + "name": "Jaime", + "surname": "Lannister", + "alive": true, + "age": 36, + "traits": ["A","F","B"] +} INTO Characters + +... +``` + +{{< info >}} +AQL does not permit multiple `INSERT` operations that target the same +collection in a single query. However, you can use a `FOR` loop like in the +above query to insert multiple documents into a collection using a single +`INSERT` operation. +{{< /info >}} + +## Read documents + +There are a couple of documents in the `Characters` collection by now. You can +retrieve them all using a `FOR` loop again. This time, however, it is for +going through all documents in the collection instead of an array: + +```aql +FOR c IN Characters + RETURN c +``` + +The syntax of the loop is `FOR variableName IN collectionName`. For each +document in the collection, `c` is assigned a document, which is then returned +as per the loop body. The query returns all characters you previously stored. + +Among them should be `Ned Stark`, similar to this example: + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + ... +] +``` + +The document features the attributes you stored, plus a few more added by +the database system. Each document needs a unique `_key`, which identifies it +within a collection. The `_id` is a computed property, a concatenation of the +collection name, a forward slash `/` and the document key. It uniquely identifies +a document within a database. `_rev` is a revision ID managed by the system. + +Document keys can be provided by the user upon document creation, or a unique +value is assigned automatically. It can not be changed later. All three system +attributes starting with an underscore `_` are read-only. + +You can use either the document key or the document ID to retrieve a specific +document with the help of an AQL function `DOCUMENT()`: + +```aql +RETURN DOCUMENT("Characters", "ned") +// --- or --- +RETURN DOCUMENT("Characters/ned") +``` + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + } +] +``` + +The `DOCUMENT()` function also allows to fetch multiple documents at once: + +```aql +RETURN DOCUMENT("Characters", ["ned", "catelyn"]) +// --- or --- +RETURN DOCUMENT(["Characters/ned", "Characters/catelyn"]) +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"] + } + ] +] +``` + +See the [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) +documentation for more details. + +The `DOCUMENT()` does not let you match documents based on the value of arbitrary +document attributes. It is also not ideal to use the function if the documents +you want to look up are all in the same collection for performance reasons. + +You can replace the call of the `DOCUMENT()` function with the powerful +combination of a `FOR` loop and a `FILTER` operation: + +```aql +FOR c IN Characters + FILTER c._key IN ["ned", "catelyn"] + RETURN c +``` + +This approach enables you to find documents using arbitrary conditions by +changing the filter criteria, but more about this later. + +## Update documents + +According to our `Ned Stark` document, he is alive. When we get to know that he +died, we need to change the `alive` attribute. Modify the existing document: + +```aql +UPDATE "ned" WITH { alive: false } IN Characters +``` + +The syntax is `UPDATE documentKey WITH object IN collectionName`. It updates the +specified document with the attributes listed (or adds them if they don't exist), +but leaves the rest untouched. To replace the entire document content, you may +use `REPLACE` instead of `UPDATE`: + +```aql +REPLACE "ned" WITH { + name: "Ned", + surname: "Stark", + alive: false, + age: 41, + traits: ["A","H","C","N","P"] +} IN Characters +``` + +This also works in a loop, to add a new attribute to all documents for instance: + +```aql +FOR c IN Characters + UPDATE c WITH { season: 1 } IN Characters +``` + +A variable is used instead of a literal document key, to update each document. +The query adds an attribute `season` to the documents' top-level. You can +inspect the result by re-running the query that returns all documents in +collection: + +```aql +FOR c IN Characters + RETURN c +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"], + "season": 1 + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"], + "season": 1 + }, + { + ... + } + ] +] +``` + +## Delete documents + +To fully remove documents from a collection, there is the `REMOVE` operation. +It works similar to the other modification operations, yet without a `WITH` clause: + +```aql +REMOVE "test" IN Characters +``` + +It can also be used in a loop body to effectively truncate a collection +(but less efficient than the dedicated feature to truncate a collection): + +```aql +FOR c IN Characters + REMOVE c IN Characters +``` + +Before you continue with the next chapter, re-run the query that +[creates the character documents](#create-documents) from above to get the data back. diff --git a/site/content/3.11/get-started/start-using-aql/dataset.md b/site/content/3.11/get-started/start-using-aql/dataset.md new file mode 100644 index 0000000000..ff1780550f --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/dataset.md @@ -0,0 +1,99 @@ +--- +title: Game of Thrones example dataset +menuTitle: Dataset +weight: 3 +--- +## Characters + +The dataset features 43 characters with their name, surname, age, alive status +and trait references. Each character also has a document key derived from the +character's name. The surname and age properties are not always present. + +| _key | name | surname | alive | age | traits | +|--------------|------------|------------|-------|-----|---------------| +| ned | Ned | Stark | true | 41 | A, H, C, N, P | +| robert | Robert | Baratheon | false | | A, H, C | +| jaime | Jaime | Lannister | true | 36 | A, F, B | +| catelyn | Catelyn | Stark | false | 40 | D, H, C | +| cersei | Cersei | Lannister | true | 36 | H, E, F | +| daenerys | Daenerys | Targaryen | true | 16 | D, H, C | +| jorah | Jorah | Mormont | false | | A, B, C, F | +| petyr | Petyr | Baelish | false | | E, G, F | +| viserys | Viserys | Targaryen | false | | O, L, N | +| jon | Jon | Snow | true | 16 | A, B, C, F | +| sansa | Sansa | Stark | true | 13 | D, I, J | +| arya | Arya | Stark | true | 11 | C, K, L | +| robb | Robb | Stark | false | | A, B, C, K | +| theon | Theon | Greyjoy | true | 16 | E, R, K | +| bran | Bran | Stark | true | 10 | L, J | +| joffrey | Joffrey | Baratheon | false | 19 | I, L, O | +| sandor | Sandor | Clegane | true | | A, P, K, F | +| tyrion | Tyrion | Lannister | true | 32 | F, K, M, N | +| khal | Khal | Drogo | false | | A, C, O, P | +| tywin | Tywin | Lannister | false | | O, M, H, F | +| davos | Davos | Seaworth | true | 49 | C, K, P, F | +| samwell | Samwell | Tarly | true | 17 | C, L, I | +| stannis | Stannis | Baratheon | false | | H, O, P, M | +| melisandre | Melisandre | | true | | G, E, H | +| margaery | Margaery | Tyrell | false | | M, D, B | +| jeor | Jeor | Mormont | false | | C, H, M, P | +| bronn | Bronn | | true | | K, E, C | +| varys | Varys | | true | | M, F, N, E | +| shae | Shae | | false | | M, D, G | +| talisa | Talisa | Maegyr | false | | D, C, B | +| gendry | Gendry | | false | | K, C, A | +| ygritte | Ygritte | | false | | A, P, K | +| tormund | Tormund | Giantsbane | true | | C, P, A, I | +| gilly | Gilly | | true | | L, J | +| brienne | Brienne | Tarth | true | 32 | P, C, A, K | +| ramsay | Ramsay | Bolton | true | | E, O, G, A | +| ellaria | Ellaria | Sand | true | | P, O, A, E | +| daario | Daario | Naharis | true | | K, P, A | +| missandei | Missandei | | true | | D, L, C, M | +| tommen | Tommen | Baratheon | true | | I, L, B | +| jaqen | Jaqen | H'ghar | true | | H, F, K | +| roose | Roose | Bolton | true | | H, E, F, A | +| high-sparrow | The High Sparrow | | true | | H, M, F, O | + +## Traits + +There are 18 unique traits. Each trait has a random letter as document key. +The trait labels come in English and German. + +| _key | en | de | +|------|-------------|---------------| +| A | strong | stark | +| B | polite | freundlich | +| C | loyal | loyal | +| D | beautiful | schön | +| E | sneaky | hinterlistig | +| F | experienced | erfahren | +| G | corrupt | korrupt | +| H | powerful | einflussreich | +| I | naive | naiv | +| J | unmarried | unverheiratet | +| K | skillful | geschickt | +| L | young | jung | +| M | smart | klug | +| N | rational | rational | +| O | ruthless | skrupellos | +| P | brave | mutig | +| Q | mighty | mächtig | +| R | weak | schwach | + +## Locations + +This small collection of 8 filming locations comes with two attributes, +`name` and `coordinates`. The coordinate pairs are modeled as number arrays, +comprised of a latitude and a longitude value each. + +| name | coordinates | +|-----------------|-----------------------| +| Dragonstone | 55.167801, -6.815096 | +| King's Landing | 42.639752, 18.110189 | +| The Red Keep | 35.896447, 14.446442 | +| Yunkai | 31.046642, -7.129532 | +| Astapor | 31.509740, -9.774249 | +| Winterfell | 54.368321, -5.581312 | +| Vaes Dothrak | 54.167760, -6.096125 | +| Beyond the wall | 64.265473, -21.094093 | diff --git a/site/content/3.11/get-started/start-using-aql/filter.md b/site/content/3.11/get-started/start-using-aql/filter.md new file mode 100644 index 0000000000..0be386c435 --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/filter.md @@ -0,0 +1,137 @@ +--- +title: Match documents with `FILTER` +menuTitle: Match documents +weight: 10 +--- +So far, you either looked up a single document, or returned the entire character +collection. For the lookup, you used the `DOCUMENT()` function, which means you +can only find documents by their key or ID. + +To find documents that fulfill certain criteria more complex than key equality, +there is the `FILTER` operation in AQL, which enables you to formulate arbitrary +conditions for documents to match. + +## Equality condition + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + RETURN c +``` + +The filter condition reads like: "the `name` attribute of a character document +must be equal to the string `Ned`". If the condition applies, character +document gets returned. This works with any attribute likewise: + +```aql +FOR c IN Characters + FILTER c.surname == "Stark" + RETURN c +``` + +## Range conditions + +Strict equality is one possible condition you can state. There are plenty of +other conditions you can formulate, however. For example, you could ask for all +adult characters: + +```aql +FOR c IN Characters + FILTER c.age >= 13 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tyrion", + "Samwell", + "Ned", + "Catelyn", + "Cersei", + "Jon", + "Sansa", + "Brienne", + "Theon", + "Davos", + "Jaime", + "Daenerys" +] +``` + +The operator `>=` stands for *greater-or-equal*, so every character of age 13 +or older is returned (only their name in the example). You can return names +and age of all characters younger than 13 by changing the operator to +*less-than* and using the object syntax to define a subset of attributes to +return: + +```aql +FOR c IN Characters + FILTER c.age < 13 + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Tommen", "age": null }, + { "name": "Arya", "age": 11 }, + { "name": "Roose", "age": null }, + ... +] +``` + +You may notice that it returns name and age of 30 characters, most with an +age of `null`. The reason for this is, that `null` is the fallback value if +an attribute is requested by the query, but no such attribute exists in the +document, and the `null` is compares to numbers as lower (see +[Type and value order](../../aql/fundamentals/type-and-value-order.md)). Hence, it +accidentally fulfills the age criterion `c.age < 13` (`null < 13`). + +## Multiple conditions + +To not let documents pass the filter without an age attribute, you can add a +second criterion: + +```aql +FOR c IN Characters + FILTER c.age < 13 + FILTER c.age != null + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Arya", "age": 11 }, + { "name": "Bran", "age": 10 } +] +``` + +This could equally be written with a boolean `AND` operator as: + +```aql +FOR c IN Characters + FILTER c.age < 13 AND c.age != null + RETURN { name: c.name, age: c.age } +``` + +And the second condition could as well be `c.age > null`. + +## Alternative conditions + +If you want documents to fulfill one or another condition, possibly for +different attributes as well, use `OR`: + +```aql +FOR c IN Characters + FILTER c.name == "Jon" OR c.name == "Joffrey" + RETURN { name: c.name, surname: c.surname } +``` + +```json +[ + { "name": "Joffrey", "surname": "Baratheon" }, + { "name": "Jon", "surname": "Snow" } +] +``` + +See more details about [Filter operations](../../aql/high-level-operations/filter.md). diff --git a/site/content/3.11/get-started/start-using-aql/geo.md b/site/content/3.11/get-started/start-using-aql/geo.md new file mode 100644 index 0000000000..9f7d008620 --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/geo.md @@ -0,0 +1,144 @@ +--- +title: Geospatial queries +menuTitle: Geospatial queries +weight: 30 +--- +Geospatial coordinates consisting of a latitude and longitude value +can be stored either as two separate attributes, or as a single +attribute in the form of an array with both numeric values. +ArangoDB can index such coordinates for fast geospatial queries. + +## Locations data + +Insert some filming locations into a new collection called `Locations`, +which you need to create first, and then run below AQL query: + +```aql +LET places = [ + { "name": "Dragonstone", "coordinates": [ 55.167801, -6.815096 ] }, + { "name": "King's Landing", "coordinates": [ 42.639752, 18.110189 ] }, + { "name": "The Red Keep", "coordinates": [ 35.896447, 14.446442 ] }, + { "name": "Yunkai", "coordinates": [ 31.046642, -7.129532 ] }, + { "name": "Astapor", "coordinates": [ 31.50974, -9.774249 ] }, + { "name": "Winterfell", "coordinates": [ 54.368321, -5.581312 ] }, + { "name": "Vaes Dothrak", "coordinates": [ 54.16776, -6.096125 ] }, + { "name": "Beyond the wall", "coordinates": [ 64.265473, -21.094093 ] } +] + +FOR place IN places + INSERT place INTO Locations + RETURN GEO_POINT(NEW.coordinates[1], NEW.coordinates[0]) +``` + +The last line of the query returns the locations as GeoJSON Points to make the +web interface render a map with markers to give you a visualization of where +the filming locations are. Note that the coordinate order is longitude, than +latitude with GeoJSON, whereas the dataset uses latitude, longitude. + +## Geospatial index + +To query based on coordinates, a [geo index](../../index-and-search/indexing/working-with-indexes/geo-spatial-indexes.md) +is required. It determines which fields contain the latitude and longitude +values. + +1. Click **COLLECTIONS** in the main navigation. +2. Click the card of the **Locations** collection. +3. Switch to the **Indexes** tab at the top. +4. Click the **Add Index** button. +5. Change the **Type** to **Geo Index**. +6. Enter `coordinates` into **Fields**. +7. Click the **Create** button to confirm. + +## Find nearby locations + +A `FOR` loop is used again, with a subsequent `SORT` operation based on the +`DISTANCE()` between a stored coordinate pair and a coordinate pair given in a query. +This pattern is recognized by the query optimizer. A geo index will be used to +accelerate such queries if one is available. + +The default sorting direction is ascending, so a query finds the coordinates +closest to the reference point first (lowest distance). `LIMIT` can be used +to restrict the number of results to at most *n* matches. + +In below example, the limit is set to 3. The origin (the reference point) is +a coordinate pair somewhere downtown in Dublin, Ireland: + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + LIMIT 3 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91491.58596795711 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121425.66829502625 + }, + { + "name": "Dragonstone", + "latitude": 55.167801, + "longitude": -6.815096, + "distance": 205433.7784182078 + } +] +``` + +The query returns the location name, as well as the coordinates and the +calculated distance in meters. The coordinates are returned as two separate +attributes. You may return just the document with a simple `RETURN loc` instead +if you want. Or return the whole document with an added distance attribute using +`RETURN MERGE(loc, { distance })`. + +## Find locations within radius + +`LIMIT` can be swapped out with a `FILTER` that checks the distance, to find +locations within a given radius from a reference point. Remember that the unit +is meters. The example uses a radius of 200,000 meters (200 kilometers): + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + FILTER distance < 200 * 1000 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance: ROUND(distance / 1000) + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121 + } +] +``` + +The distances are converted to kilometers and rounded for readability. diff --git a/site/content/3.11/get-started/start-using-aql/graphs.md b/site/content/3.11/get-started/start-using-aql/graphs.md new file mode 100644 index 0000000000..704679a4dc --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/graphs.md @@ -0,0 +1,303 @@ +--- +title: Graphs and traversals +menuTitle: Graphs and traversals +weight: 25 +description: >- + How relations such as between parents and children can be modeled as graph + and how they can be queried +--- +Relations such as between parents and children can be modeled as graph. +In ArangoDB, two documents (a parent and a child character document) can be +linked by an edge document. Edge documents are stored in edge collections and +have two additional attributes: `_from` and `_to`. They reference any two +documents by their document IDs (`_id`). + +## ChildOf relations + +Our characters have the following relations between parents and children +(first names only for a better overview): + +``` + Robb -> Ned + Sansa -> Ned + Arya -> Ned + Bran -> Ned + Jon -> Ned + Robb -> Catelyn + Sansa -> Catelyn + Arya -> Catelyn + Bran -> Catelyn + Jaime -> Tywin + Cersei -> Tywin + Tyrion -> Tywin + Joffrey -> Jaime + Joffrey -> Cersei +``` + +Visualized as a graph: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +## Creating the edges + +To create the required edge documents to store these relations in the database, +you can run a query that combines joining and filtering to match up the right +character documents, then use their `_id` attribute to insert an edge into an +edge collection called `ChildOf`. + +1. Click **COLLECTIONS** in the main navigation. +2. Click the first card labelled **Add collection**. +3. Enter `ChildOf` as the **Name**. +4. Change the collection type to **Edge**. + +Then run the following query: + +```aql +LET relations = [ + { "parent": "ned", "child": "robb" }, + { "parent": "ned", "child": "sansa" }, + { "parent": "ned", "child": "arya" }, + { "parent": "ned", "child": "bran" }, + { "parent": "catelyn", "child": "robb" }, + { "parent": "catelyn", "child": "sansa" }, + { "parent": "catelyn", "child": "arya" }, + { "parent": "catelyn", "child": "bran" }, + { "parent": "ned", "child": "jon" }, + { "parent": "tywin", "child": "jaime" }, + { "parent": "tywin", "child": "cersei" }, + { "parent": "tywin", "child": "tyrion" }, + { "parent": "cersei", "child": "joffrey" }, + { "parent": "jaime", "child": "joffrey" } +] + +FOR rel in relations + INSERT { + _from: CONCAT("Characters/", rel.child), + _to: CONCAT("Characters/", rel.parent) + } INTO ChildOf + RETURN NEW +``` + +Breakdown of the query: + +- Assign the relations in form of an array of objects with a `parent` and + a `child` attribute each, both with the document key of the character. +- For each element in this array, assign a relation to a variable `rel` and + execute the subsequent instructions. + - Insert a new edge document into the ChildOf collection, with the edge going + from `child` to `parent`. The `_from` and `_to` edge attributes require + document IDs like `collection/key`. Therefore, the document keys derived + from the character names are prefixed with `Characters/` to obtain the IDs. + No other attributes are set for the edge in this example. + - Return the new edge document (optional) + +## Traverse to the parents + +Now that edges link character documents (vertices), it is a graph you can +query to find out who the parents are of another character – or in +graph terms, you want to start at a vertex and follow the edges to other +vertices in an [AQL graph traversal](../../aql/graphs/traversals.md): + +```aql +// Declare collection of start vertex (cluster only) +WITH Characters + +FOR v IN 1..1 OUTBOUND "Characters/bran" ChildOf + RETURN v.name +``` + +This `FOR` loop doesn't iterate over a collection or an array, it walks the +graph and iterates over the connected vertices it finds, with the vertex +document assigned to a variable (here: `v`). It can also emit the edges it +walked as well as the full path from start to end to +[another two variables](../../aql/graphs/traversals.md#syntax). + +In above query, the traversal is restricted to a minimum and maximum traversal +depth of 1 (how many steps to take from the start vertex), and to only follow +edges in `OUTBOUND` direction. Our edges point from child to parent, and the +parent is one step away from the child, thus it gives you the parents of the +child you start at. `"Characters/bran"` is that start vertex. + +To determine the ID of e.g. the Joffrey Baratheon document, you may iterate over +the collection of characters, filter by the name or other criteria, and return +the `_id` attribute: + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + RETURN c._id +``` + +```json +[ "Characters/joffrey" ] +``` + +You may also combine this query with the traversal directly, to easily change +the start vertex by adjusting the filter condition(s): + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +The start vertex is followed by `ChildOf`, which is our edge collection. The +example query returns only the name of each parent to keep the result short: + +```json +[ + "Jaime", + "Cersei" +] +``` + +For Robb, Sansa, Arya, and Bran as the starting point, the result is Catelyn and +Ned, and for Jon Snow it is only Ned. + +Be mindful of the `FILTER` criteria. If more than one character fulfills the +conditions, there will be multiple traversals. And with the query as it is, +all of the parent names would be combined into a single list: + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Tywin", + "Tywin", + "Tywin" +] +``` + +You can achieve a more useful output by returning the result of each traversal +as a separate list and set the minimum traversal depth to `0` to include the +start vertex. This lets you see the child's name as the first element of each +array, followed by the parent name(s) if this information is available. + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + RETURN (FOR v IN 0..1 OUTBOUND c ChildOf + RETURN v.name) +``` + +```json +[ + [ + "Jaime", + "Tywin" + ], + [ + "Cersei", + "Tywin" + ], + [ + "Tyrion", + "Tywin" + ], + [ + "Tywin" + ] +] +``` + +## Traverse to the children + +You can also walk from a parent in reverse edge direction (`INBOUND` that is) +to the children: + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + FOR v IN 1..1 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Robb", + "Sansa", + "Jon", + "Arya", + "Bran" +] +``` + +## Traverse to the grandchildren + +For the Lannister family, there are relations that span from parent to +grandchild. Change the traversal depth to return grandchildren, +which means to go exactly two steps: + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Joffrey", + "Joffrey" +] +``` + +It might be a bit unexpected, that Joffrey is returned twice. However, if you +look at the graph visualization, you can see that multiple paths lead from +Joffrey (bottom right) to Tywin: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +``` +Tywin <- Jaime <- Joffrey +Tywin <- Cersei <- Joffrey +``` + +As a quick fix, change the last line of the query to `RETURN DISTINCT v.name` +to return each value only once. However, there are +[traversal options](../../aql/graphs/traversals.md#syntax) including one to +suppress duplicate vertices early on for the entire traversal (which requires +breadth-first search): + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf OPTIONS { uniqueVertices: "global", order: "bfs" } + RETURN v.name +``` + +```json +[ + "Joffrey" +] +``` + +## Traverse with variable depth + +To return the parents and grandparents of Joffrey, you can walk edges in +`OUTBOUND` direction and adjust the traversal depth to go at least 1 step, +and 2 at most: + +```aql +FOR c IN Characters + FILTER c.name == "Joffrey" + FOR v IN 1..2 OUTBOUND c ChildOf + RETURN DISTINCT v.name +``` + +```json +[ + "Cersei", + "Tywin", + "Jaime" +] +``` + +With deeper family trees, it would only be a matter of changing the depth +values to query for great-grandchildren and similar relations. diff --git a/site/content/3.11/get-started/start-using-aql/joins.md b/site/content/3.11/get-started/start-using-aql/joins.md new file mode 100644 index 0000000000..abf30c9045 --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/joins.md @@ -0,0 +1,323 @@ +--- +title: References and joins +menuTitle: References and joins +weight: 20 +--- +## References to other documents + +The character data you imported has an attribute `traits` for each character, +which is an array of strings. It does not store character features directly, +however: + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"] +} +``` + +It is rather a list of letters without an apparent meaning. The idea here is +that `traits` is supposed to store documents keys of another collection, which +you can use to resolve the letters to labels such as "strong". The benefit of +using another collection for the actual traits is, that you can easily query +for all existing traits later on and store labels in multiple languages for +instance in a central place. If you would embed traits directly... + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": [ + { + "de": "stark", + "en": "strong" + }, + { + "de": "einflussreich", + "en": "powerful" + }, + { + "de": "loyal", + "en": "loyal" + }, + { + "de": "rational", + "en": "rational" + }, + { + "de": "mutig", + "en": "brave" + } + ] +} +``` + +... it becomes difficult to maintain traits. If you were to rename or +translate one of them, you would need to find all other character documents +with the same trait and perform the changes there too. If you only refer to a +trait in another collection, it is as easy as updating a single document. + +{{< comment >}}What if Trait doc is deleted? DOCUMENT() skips null{{< /comment >}} + +## Importing traits + +1. In the web interface, create a document collection called `Traits`. +2. Enter the following AQL query: + ```aql + FOR trait IN @data + INSERT trait INTO Traits + ``` +3. Set the following for the `data` bind variable: + ```json + [ + { "_key": "A", "en": "strong", "de": "stark" }, + { "_key": "B", "en": "polite", "de": "freundlich" }, + { "_key": "C", "en": "loyal", "de": "loyal" }, + { "_key": "D", "en": "beautiful", "de": "schön" }, + { "_key": "E", "en": "sneaky", "de": "hinterlistig" }, + { "_key": "F", "en": "experienced", "de": "erfahren" }, + { "_key": "G", "en": "corrupt", "de": "korrupt" }, + { "_key": "H", "en": "powerful", "de": "einflussreich" }, + { "_key": "I", "en": "naive", "de": "naiv" }, + { "_key": "J", "en": "unmarried", "de": "unverheiratet" }, + { "_key": "K", "en": "skillful", "de": "geschickt" }, + { "_key": "L", "en": "young", "de": "jung" }, + { "_key": "M", "en": "smart", "de": "klug" }, + { "_key": "N", "en": "rational", "de": "rational" }, + { "_key": "O", "en": "ruthless", "de": "skrupellos" }, + { "_key": "P", "en": "brave", "de": "mutig" }, + { "_key": "Q", "en": "mighty", "de": "mächtig" }, + { "_key": "R", "en": "weak", "de": "schwach" } + ] + ``` +4. Execute the query to import the trait data. + +## Resolving traits + +Start simple by returning only the traits attribute of each character: + +```aql +FOR c IN Characters + RETURN c.traits +``` + +```json +[ + ["A","H","C","N","P"], + ["D","H","C"], + ... +] +``` + +Also see the [Fundamentals of Objects / Documents](../../aql/fundamentals/data-types.md#objects--documents) +about attribute access. + +You can use the `traits` array together with the `DOCUMENT()` function to use +the elements as document keys and look them up in the `Traits` collection: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits) +``` + +```json +[ + [ + { + "_key": "A", + "_id": "Traits/A", + "_rev": "_V5oRUS2---", + "en": "strong", + "de": "stark" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + }, + { + "_key": "N", + "_id": "Traits/N", + "_rev": "_V5oRUT---D", + "en": "rational", + "de": "rational" + }, + { + "_key": "P", + "_id": "Traits/P", + "_rev": "_V5oRUTC---", + "en": "brave", + "de": "mutig" + } + ], + [ + { + "_key": "D", + "_id": "Traits/D", + "_rev": "_V5oRUS6--A", + "en": "beautiful", + "de": "schön" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + } + ], + ... +] +``` + +The [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) can be used +to look up a single or multiple documents via document identifiers. In our +example, you pass the collection name from which you want to fetch documents +as first argument (`"Traits"`) and an array of document keys (`_key` attribute) +as second argument. In return, you get an array of the full trait documents +for each character. + +This is a bit too much information, so only return English labels using +the [array expansion](../../aql/operators.md#array-expansion) notation: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits)[*].en +``` + +```json +[ + [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ], + [ + "beautiful", + "powerful", + "loyal" + ], + ... +] +``` + +## Merging characters and traits + +Great, you resolved the letters to meaningful traits! But you also need to know +to which character they belong. Thus, you need to merge both the character +document and the data from the trait documents: + +```aql +FOR c IN Characters + RETURN MERGE(c, { traits: DOCUMENT("Traits", c.traits)[*].en } ) +``` + +```json +[ + { + "_id": "Characters/ned", + "_key": "ned", + "_rev": "_V1bzsXa---", + "age": 41, + "alive": false, + "name": "Ned", + "surname": "Stark", + "traits": [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ] + }, + { + "_id": "Characters/catelyn", + "_key": "catelyn", + "_rev": "_V1bzsXa--B", + "age": 40, + "alive": false, + "name": "Catelyn", + "surname": "Stark", + "traits": [ + "beautiful", + "powerful", + "loyal" + ] + }, + ... +] +``` + +The `MERGE()` functions merges objects together. Because you used an object +`{ traits: ... }` which has the same attribute name `traits` as the original +character attribute, the latter got overwritten by the merge operation. + +## Join another way + +The `DOCUMENT()` function utilizes primary indexes to look up documents quickly. +It is limited to find documents via their identifiers however. For a use case +like in our example it is sufficient to accomplish a simple join. + +There is another, more flexible syntax for joins: nested `FOR` loops over +multiple collections, with a `FILTER` condition to match up attributes. +In case of the traits key array, there needs to be a third loop to iterate +over the keys: + +```aql +FOR c IN Characters + RETURN MERGE(c, { + traits: ( + FOR key IN c.traits + FOR t IN Traits + FILTER t._key == key + RETURN t.en + ) + }) +``` + +For each character, it loops over its `traits` attribute (e.g. `["D","H","C"]`) +and for each document reference in this array, it loops over the `Traits` +collections. There is a condition to match the document key with the key +reference. The inner `FOR` loop and the `FILTER` get transformed to a primary +index lookup in this case instead of building up a Cartesian product only to +filter away everything but a single match: Document keys within a collection +are unique, thus there can only be one match. + +Each written-out, English trait is returned and all the traits are then merged +with the character document. The result is identical to the query using +`DOCUMENT()`. However, this approach with a nested `FOR` loop and a `FILTER` +is not limited to primary keys. You can do this with any other attribute as well. +For an efficient lookup, make sure you add a persistent index for this attribute. +If its values are unique, then also set the index option to unique. + +Another advantage of the `FOR` loop approach is the performance compared to +calling the `DOCUMENT()` function: The query optimizer can optimize AQL queries +better that iterate over a collection and possibly filter by attributes and only +make use of a subset of the found documents. With the `DOCUMENT()` function, +there are individual lookups, potentially across all collections, and the full +documents need to be loaded regardless of which attributes are actually used. diff --git a/site/content/3.11/get-started/start-using-aql/sort-limit.md b/site/content/3.11/get-started/start-using-aql/sort-limit.md new file mode 100644 index 0000000000..553924c379 --- /dev/null +++ b/site/content/3.11/get-started/start-using-aql/sort-limit.md @@ -0,0 +1,183 @@ +--- +title: Sort and limit +menuTitle: Sort and limit +weight: 15 +--- +## Cap the result count with `LIMIT` + +It may not always be necessary to return all documents, that a `FOR` loop +would normally return. You can limit the amount of documents +with a `LIMIT` operation: + +```aql +FOR c IN Characters + LIMIT 5 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tommen", + "Tyrion", + "Roose", + "Tywin" +] +``` + +`LIMIT` is followed by a number for the maximum document count. There is a +second syntax however, which allows you to skip a certain amount of record +and return the next *n* documents: + +```aql +FOR c IN Characters + LIMIT 2, 5 + RETURN c.name +``` + +```json +[ + "Tyrion", + "Roose", + "Tywin", + "Samwell", + "Melisandre" +] +``` + +See how the second query skipped the first two names and returned the next +five (both results feature Tyrion, Roose and Tywin). + +## Sort by name with `SORT` + +The order in which matching records were returned by the queries shown until +here was basically random. To return them in a defined order, you can add a +`SORT()` operation. It can have a big impact on the result if combined with +a `LIMIT()`, because the result becomes predictable if you sort first. + +```aql +FOR c IN Characters + SORT c.name + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Arya", + "Bran", + "Brienne", + "Bronn", + "Catelyn", + "Cersei", + "Daario", + "Daenerys", + "Davos", + "Ellaria" +] +``` + +See how it sorted by name, then returned the ten alphabetically first coming +names. You can reverse the sort order with `DESC` like descending: + +```aql +FOR c IN Characters + SORT c.name DESC + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Ygritte", + "Viserys", + "Varys", + "Tywin", + "Tyrion", + "Tormund", + "Tommen", + "Theon", + "The High Sparrow", + "Talisa" +] +``` + +The first sort was ascending, which is the default order. Because it is the +default, it is not required to explicitly ask for `ASC` order. + +## Sort by multiple attributes + +Assume you want to sort by surname. Many of the characters share a surname. +The result order among characters with the same surname is undefined. You can +first sort by `surname`, then `name`, to determine the order: + +```aql +FOR c IN Characters + FILTER c.surname + SORT c.surname, c.name + LIMIT 10 + RETURN { + surname: c.surname, + name: c.name + } +``` + +```json +[ + { "surname": "Baelish", "name": "Petyr" }, + { "surname": "Baratheon", "name": "Joffrey" }, + { "surname": "Baratheon", "name": "Robert" }, + { "surname": "Baratheon", "name": "Stannis" }, + { "surname": "Baratheon", "name": "Tommen" }, + { "surname": "Bolton", "name": "Ramsay" }, + { "surname": "Bolton", "name": "Roose" }, + { "surname": "Clegane", "name": "Sandor" }, + { "surname": "Drogo", "name": "Khal" }, + { "surname": "Giantsbane", "name": "Tormund" } +] +``` + +Overall, the documents are sorted by last name. If the `surname` is the same +for two characters, the `name` values are compared and the result sorted. + +Note that a filter is applied before sorting, to only let documents through, +that actually feature a surname value (many don't have it and would cause +`null` values in the result). + +## Sort by age + +The order can also be determined by a numeric value, such as the age: + +```aql +FOR c IN Characters + FILTER c.age + SORT c.age + LIMIT 10 + RETURN { + name: c.name, + age: c.age + } +``` + +```json +[ + { "name": "Bran", "age": 10 }, + { "name": "Arya", "age": 11 }, + { "name": "Sansa", "age": 13 }, + { "name": "Jon", "age": 16 }, + { "name": "Theon", "age": 16 }, + { "name": "Daenerys", "age": 16 }, + { "name": "Samwell", "age": 17 }, + { "name": "Joffrey", "age": 19 }, + { "name": "Tyrion", "age": 32 }, + { "name": "Brienne", "age": 32 } +] +``` + +A filter is applied to avoid documents without age attribute. The remaining +documents are sorted by age in ascending order, and the name and age of the +ten youngest characters are returned. + +See the [SORT operation](../../aql/high-level-operations/sort.md) and +[LIMIT operation](../../aql/high-level-operations/limit.md) documentation for +more details. diff --git a/site/content/3.12/aql/how-to-invoke-aql/_index.md b/site/content/3.12/aql/how-to-invoke-aql/_index.md index 206ffc4a73..b44a6ed75a 100644 --- a/site/content/3.12/aql/how-to-invoke-aql/_index.md +++ b/site/content/3.12/aql/how-to-invoke-aql/_index.md @@ -2,17 +2,22 @@ title: How to execute AQL queries menuTitle: How to invoke AQL weight: 5 -description: '' +description: >- + You can execute AQL queries in different ways, from the easy-to-use + web interface to the raw HTTP REST API --- -AQL queries can be invoked in the following ways: +You can execute AQL queries using different interfaces: -- Via the web interface -- Using the `db` object of the JavaScript API, for example, in arangosh or in a Foxx service -- Via the raw REST HTTP API +- The web interface +- The `db` object of the JavaScript API (either in arangosh or in a Foxx service) +- The raw HTTP REST API +- Through a [driver](../../develop/drivers/_index.md) or + [integration](../../develop/integrations/_index.md) as an abstraction over the + HTTP REST API -There are always calls to the server's HTTP API under the hood, but the web interface -and the `db` object abstract away the low-level communication details and are -thus easier to use. +There are always calls to the server's API under the hood, but the web interface, +arangosh, drivers, and integrations abstract away the low-level +communication details and are thus easier to use. The ArangoDB web interface has a specific section for [**Queries**](with-the-web-interface.md). diff --git a/site/content/3.12/components/tools/arangoimport/examples-json.md b/site/content/3.12/components/tools/arangoimport/examples-json.md index d38ed3bccd..e7df8314d1 100644 --- a/site/content/3.12/components/tools/arangoimport/examples-json.md +++ b/site/content/3.12/components/tools/arangoimport/examples-json.md @@ -88,7 +88,7 @@ which allows any valid JSON value on a line. An input with JSON objects in an array, optionally pretty printed, can be easily converted into JSONL with one JSON object per line using the -[**jq** command line tool](http://stedolan.github.io/jq/): +[**jq** command-line tool](https://jqlang.github.io/jq/): ``` jq -c ".[]" inputFile.json > outputFile.jsonl diff --git a/site/content/3.12/develop/http-api/administration.md b/site/content/3.12/develop/http-api/administration.md index 0fe1d0e940..d1b1baa3ed 100644 --- a/site/content/3.12/develop/http-api/administration.md +++ b/site/content/3.12/develop/http-api/administration.md @@ -1030,7 +1030,7 @@ var url = "/_admin/license"; var response = logCurlRequest('GET', url); assert(response.code === 200); -assertTypeOf("string", response.parsedBody.license); +assertTypeOf("object", response.parsedBody.diskUsage); logJsonResponse(response); ``` diff --git a/site/content/3.12/get-started/how-to-interact-with-arangodb.md b/site/content/3.12/get-started/how-to-interact-with-arangodb.md index 204aa5a9db..e92080622d 100644 --- a/site/content/3.12/get-started/how-to-interact-with-arangodb.md +++ b/site/content/3.12/get-started/how-to-interact-with-arangodb.md @@ -41,21 +41,21 @@ programming language, and do all the talking to the server. Integrations combine a third-party technology with ArangoDB and can be seen as a translation layer that takes over the low-level communication with the server. -### REST API +### HTTP REST API -Under the hood, all interactions with the server make use of its REST API. -A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer) -API is an application programming interface based on the HTTP protocol that -powers the world wide web. +Under the hood, all interactions with the server make use of its RESTful HTTP API. +A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer)-based +API is an application programming interface using the HTTP protocol, the +protocol that powers the world wide web. All requests from the outside to the server need to made against the respective endpoints of this API to perform actions. This includes the web interface, _arangosh_, as well as the drivers and integrations for different programming languages and environments. They all provide a convenient way to work with ArangoDB, but you -may use the low-level REST API directly as needed. +may use the low-level HTTP API directly as needed. -See the [HTTP](../develop/http-api/_index.md) documentation to learn more about the API, how requests -are handled and what endpoints are available. +See the [HTTP API](../develop/http-api/_index.md) documentation to learn more +about the API, how requests are handled, and what endpoints are available. ## Set Up and Deploy ArangoDB diff --git a/site/content/3.12/get-started/start-using-aql.md b/site/content/3.12/get-started/start-using-aql.md deleted file mode 100644 index 5e4ade2c4c..0000000000 --- a/site/content/3.12/get-started/start-using-aql.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Start using AQL -menuTitle: Start using AQL -weight: 45 -description: >- - You can execute AQL queries in different ways, from the easy-to-use - web interface to the raw HTTP API ---- -## How to invoke AQL - -AQL queries can be executed using: - -- the web interface -- the `db` object (either in arangosh or in a Foxx service) -- the raw HTTP API -- through drivers and integrations as an abstraction over the HTTP API - -There are always calls to the server's API under the hood, but the web interface, -the `db` object, drivers, and integrations abstract away the low-level -communication details and are thus easier to use. - -The ArangoDB Web Interface has a [specific tab for AQL queries execution](../aql/how-to-invoke-aql/with-the-web-interface.md). - -You can run [AQL queries from the ArangoDB Shell](../aql/how-to-invoke-aql/with-arangosh.md) -with the [`_query()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and -[`_createStatement()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) methods -of the [`db` object](../develop/javascript-api/@arangodb/db-object.md). This chapter -also describes how to use bind parameters, statistics, counting and cursors with -arangosh. - -If you are using Foxx, see [how to write database queries](../develop/foxx-microservices/getting-started.md#writing-database-queries) -for examples including tagged template strings. - -If you want to run AQL queries from your application via the HTTP REST API, -see the full API description at [HTTP interface for AQL queries](../develop/http-api/queries/aql-queries.md). - -See the respective [driver](../develop/drivers/_index.md) or -[integration](../develop/integrations/_index.md) for its support of AQL queries. - -## Learn the query language - -See the [AQL documentation](../aql/_index.md) for the full language reference -as well as examples. - -For a tutorial, sign up for the [ArangoDB University](https://university.arangodb.com/) -to get access to the **AQL Fundamentals** course. diff --git a/site/content/3.12/get-started/start-using-aql/_index.md b/site/content/3.12/get-started/start-using-aql/_index.md new file mode 100644 index 0000000000..94012b4a7d --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/_index.md @@ -0,0 +1,157 @@ +--- +title: Start using AQL +menuTitle: Start using AQL +weight: 52 +description: >- + Learn how to run your first queries written in ArangoDB's Query Language + and how to go from there, using a Game of Thrones dataset +--- +This is an introduction to ArangoDB's query language AQL, built around a small +dataset of characters from the novel and fantasy drama television series +Game of Thrones (as of season 1). It includes character traits in two languages, +some family relations, and last but not least a small set of filming locations, +which makes for an interesting mix of data to work with. + +There is no need to import the data before you start. It is provided as part +of the AQL queries in this tutorial. You can interact with ArangoDB using its +built-in web interface to manage collections and execute the queries with ease, +but you may also use a different interface. + +## How to run AQL queries + +{{< tabs "interfaces" >}} + +{{< tab "Web interface" >}} +ArangoDB's web interface has a **Queries** section for +[executing AQL queries](../../aql/how-to-invoke-aql/with-the-web-interface.md). + +1. If necessary, [switch to the database](../../concepts/data-structure/databases.md#set-the-database-context) + that you want to run queries in. +2. Click **Queries** in the main navigation. +3. Enter an AQL query in the code editor, e.g. `RETURN CONCAT("Hello, ", @name)`. +4. Specify any needed bind parameters in the panel on the right-hand side on the + **Bind Variables** tab, e.g. set `name` to a value of `AQL`. +5. Optionally set query options on the **Options** tab. +6. Click the **Execute** button or hit `Ctrl`/`Cmd` + `Return`. +{{< /tab >}} + +{{< tab "arangosh" >}} +You can run AQL queries from the ArangoDB Shell ([arangosh](../../components/tools/arangodb-shell/_index.md)) +with the [`db._query()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and +[`db._createStatement()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) +methods of the [`db` object](../../develop/javascript-api/@arangodb/db-object.md). + +```js +--- +name: arangosh_execute_query_bindvars +description: '' +--- +db._query(`RETURN CONCAT("Hello, ", @name)`, { name: "AQL" }).toArray(); +// -- or -- +var name = "AQL"; +db._query(aql`RETURN CONCAT("Hello, ", ${name})`).toArray(); +``` +See [`db._query()`](../../develop/javascript-api/@arangodb/db-object.md#db_queryquerystring--bindvars--mainoptions--suboptions) +in the _JavaScript API_ for details. + +If you use Foxx, see [how to write database queries](../../develop/foxx-microservices/getting-started.md#writing-database-queries) +for examples including tagged template strings. +{{< /tab >}} + +{{< tab "cURL" >}} +You can use a tool like [cURL](https://curl.se/) to run AQL queries from a +command-line, directly using the HTTP REST API of ArangoDB. + +The response bodies are generally compact JSON (without any line breaks and +indentation). You can format them with the [jq](https://jqlang.github.io/jq/) +tool for better readability if you have it installed: + +```sh +curl -d '{"query":"RETURN CONCAT(\"Hello, \", @name)","bindVars":{"name":"AQL"}}' http://localhost:8529/_api/cursor | jq +``` + +See the [`POST /_db/{database-name}/_api/cursor`](../../develop/http-api/queries/aql-queries.md#create-a-cursor) +endpoint in the _HTTP API_ for details. +{{< /tab >}} + +{{< tab "JavaScript" >}} +```js +import { Database, aql } from "arangojs"; +const db = new Database(); + +const name = "AQL"; +const cursor = await db.query(aql`RETURN CONCAT("Hello, ", ${name})`); +const result = cursor.all(); +console.log(result); +``` + +See [`Database.query()`](https://arangodb.github.io/arangojs/latest/classes/databases.Database.html#query) +in the _arangojs_ documentation for details. +{{< /tab >}} + +{{< tab "Go" >}} +```go +ctx := context.Background() +query := `RETURN CONCAT("Hello, ", @name)` +options := arangodb.QueryOptions{ + BindVars: map[string]interface{}{ + "name": "AQL", + }, +} +cursor, err := db.Query(ctx, query, &options) +if err != nil { + log.Fatalf("Failed to run query:\n%v\n", err) +} else { + defer cursor.Close() + var str string + for cursor.HasMore() { + _, err := cursor.ReadDocument(ctx, &str) + if err != nil { + log.Fatalf("Failed to read cursor:\n%v\n", err) + } else { + fmt.Println(str) + } + } +} +``` + +See [`DatabaseQuery.Query()`](https://pkg.go.dev/github.com/arangodb/go-driver/v2/arangodb#DatabaseQuery) +in the _go-driver_ v2 documentation for details. +{{< /tab >}} + +{{< tab "Java" >}} +```java +String query = "RETURN CONCAT(\"Hello, \", @name)"; +Map bindVars = Collections.singletonMap("name", "AQL"); +ArangoCursor cursor = db.query(query, String.class, bindVars); +cursor.forEach(result -> System.out.println(result)); +``` + +See [`ArangoDatabase.query()`](https://www.javadoc.io/doc/com.arangodb/arangodb-java-driver/latest/com/arangodb/ArangoDatabase.html#query%28java.lang.String,java.lang.Class,java.util.Map%29) +in the _arangodb-java-driver_ documentation for details. +{{< /tab >}} + +{{< tab "Python" >}} +```py +query = "RETURN CONCAT('Hello, ', @name)" +bind_vars = { "name": "AQL" } +cursor = db.aql.execute(query, bind_vars=bind_vars) +for result in cursor: + print(result) +``` + +See [`AQL.execute()`](https://docs.python-arango.com/en/main/specs.html#arango.aql.AQL.execute) +in the _python-arango_ documentation for details. +{{< /tab >}} + +{{< /tabs >}} + +## Learn the query language + +The following pages guide you through important query constructs for storing +and retrieving data, covering basic as well as some advanced features. + +Afterwards, you can read the [AQL documentation](../../aql/_index.md) for the +full language reference and query examples. + +{{< comment >}}TODO: Advanced data manipulation: attributes, projections, calculations... Aggregation: Grouping techniques{{< /comment >}} diff --git a/site/content/3.12/get-started/start-using-aql/crud.md b/site/content/3.12/get-started/start-using-aql/crud.md new file mode 100644 index 0000000000..48a0a2a560 --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/crud.md @@ -0,0 +1,371 @@ +--- +title: AQL CRUD operations +menuTitle: CRUD operations +weight: 5 +description: >- + Learn how to **c**reate, **r**ead, **u**pdate, and **d**elete documents with + the ArangoDB Query Language +--- +## Create documents + +Before you can insert documents with AQL, you need a place to put them in – a +collection. You can [manage collections](../../concepts/data-structure/collections.md#collections-api) +via different interfaces including the web interface, arangosh, or a driver. +It is not possible to do so with AQL, however. + +1. In the web interface, click **Collections** in the main navigation. +2. Click the **Add Collection** button. +3. Enter `Characters` as the **Name**. +4. Leave the **Type** set to the default value of **Document**. +5. Click the **Create** button. + +The new collection should appear in the list. Next, click **Queries** in the +main navigation. To create the first document for the collection with AQL, +use the following AQL query, which you can paste into the query textbox and +run by clicking the **Execute** button: + +```aql +INSERT { + _key: "test", + name: "First", + surname: "Test", + alive: false, + age: 123, + traits: [ "X", "Y" ] +} INTO Characters RETURN NEW +``` + +The syntax is `INSERT document INTO collectionName`. The document is an object +like you may know it from JavaScript or JSON, which is comprised of attribute +key and value pairs. The quotes around the attribute keys are optional in AQL. +Keys are always character sequences (strings), whereas attribute values can +have [different types](../../aql/fundamentals/data-types.md): + +- `null` +- boolean (`true`, `false`) +- number (integer and floating point values) +- string +- array +- object + +The name and surname of the character document you inserted are both string +values. The alive state uses a boolean value. Age is a numeric value. +The traits are an array of strings. The entire document is an object. + +The `RETURN NEW` part is optional and makes the query return the document +including any system attributes that may get added by ArangoDB. If you don't +specify the `_key` attribute then a document key is automatically generated. + +Next, add several characters with a single query: + +```aql +FOR d IN @data + INSERT d INTO Characters +``` + +The `FOR` loop iterates over `@data`, which is a placeholder in the query for +binding the list of characters in JSON format. + +In the web interface, there is a tab called **Bind Variables** on the right-hand +side of the query editor. When you enter a placeholder like `@data` in the editor, +a row appears in the **Bind Variables** tab to specify the value for the placeholder. +Paste the following text into the field for the `data` bind variable: + +```json +[ + { "_key": "ned", "name": "Ned", "surname": "Stark", "alive": true, "age": 41, "traits": ["A","H","C","N","P"] }, + { "_key": "robert", "name": "Robert", "surname": "Baratheon", "alive": false, "traits": ["A","H","C"] }, + { "_key": "jaime", "name": "Jaime", "surname": "Lannister", "alive": true, "age": 36, "traits": ["A","F","B"] }, + { "_key": "catelyn", "name": "Catelyn", "surname": "Stark", "alive": false, "age": 40, "traits": ["D","H","C"] }, + { "_key": "cersei", "name": "Cersei", "surname": "Lannister", "alive": true, "age": 36, "traits": ["H","E","F"] }, + { "_key": "daenerys", "name": "Daenerys", "surname": "Targaryen", "alive": true, "age": 16, "traits": ["D","H","C"] }, + { "_key": "jorah", "name": "Jorah", "surname": "Mormont", "alive": false, "traits": ["A","B","C","F"] }, + { "_key": "petyr", "name": "Petyr", "surname": "Baelish", "alive": false, "traits": ["E","G","F"] }, + { "_key": "viserys", "name": "Viserys", "surname": "Targaryen", "alive": false, "traits": ["O","L","N"] }, + { "_key": "jon", "name": "Jon", "surname": "Snow", "alive": true, "age": 16, "traits": ["A","B","C","F"] }, + { "_key": "sansa", "name": "Sansa", "surname": "Stark", "alive": true, "age": 13, "traits": ["D","I","J"] }, + { "_key": "arya", "name": "Arya", "surname": "Stark", "alive": true, "age": 11, "traits": ["C","K","L"] }, + { "_key": "robb", "name": "Robb", "surname": "Stark", "alive": false, "traits": ["A","B","C","K"] }, + { "_key": "theon", "name": "Theon", "surname": "Greyjoy", "alive": true, "age": 16, "traits": ["E","R","K"] }, + { "_key": "bran", "name": "Bran", "surname": "Stark", "alive": true, "age": 10, "traits": ["L","J"] }, + { "_key": "joffrey", "name": "Joffrey", "surname": "Baratheon", "alive": false, "age": 19, "traits": ["I","L","O"] }, + { "_key": "sandor", "name": "Sandor", "surname": "Clegane", "alive": true, "traits": ["A","P","K","F"] }, + { "_key": "tyrion", "name": "Tyrion", "surname": "Lannister", "alive": true, "age": 32, "traits": ["F","K","M","N"] }, + { "_key": "khal", "name": "Khal", "surname": "Drogo", "alive": false, "traits": ["A","C","O","P"] }, + { "_key": "tywin", "name": "Tywin", "surname": "Lannister", "alive": false, "traits": ["O","M","H","F"] }, + { "_key": "davos", "name": "Davos", "surname": "Seaworth", "alive": true, "age": 49, "traits": ["C","K","P","F"] }, + { "_key": "samwell", "name": "Samwell", "surname": "Tarly", "alive": true, "age": 17, "traits": ["C","L","I"] }, + { "_key": "stannis", "name": "Stannis", "surname": "Baratheon", "alive": false, "traits": ["H","O","P","M"] }, + { "_key": "melisandre", "name": "Melisandre", "alive": true, "traits": ["G","E","H"] }, + { "_key": "margaery", "name": "Margaery", "surname": "Tyrell", "alive": false, "traits": ["M","D","B"] }, + { "_key": "jeor", "name": "Jeor", "surname": "Mormont", "alive": false, "traits": ["C","H","M","P"] }, + { "_key": "bronn", "name": "Bronn", "alive": true, "traits": ["K","E","C"] }, + { "_key": "varys", "name": "Varys", "alive": true, "traits": ["M","F","N","E"] }, + { "_key": "shae", "name": "Shae", "alive": false, "traits": ["M","D","G"] }, + { "_key": "talisa", "name": "Talisa", "surname": "Maegyr", "alive": false, "traits": ["D","C","B"] }, + { "_key": "gendry", "name": "Gendry", "alive": false, "traits": ["K","C","A"] }, + { "_key": "ygritte", "name": "Ygritte", "alive": false, "traits": ["A","P","K"] }, + { "_key": "tormund", "name": "Tormund", "surname": "Giantsbane", "alive": true, "traits": ["C","P","A","I"] }, + { "_key": "gilly", "name": "Gilly", "alive": true, "traits": ["L","J"] }, + { "_key": "brienne", "name": "Brienne", "surname": "Tarth", "alive": true, "age": 32, "traits": ["P","C","A","K"] }, + { "_key": "ramsay", "name": "Ramsay", "surname": "Bolton", "alive": true, "traits": ["E","O","G","A"] }, + { "_key": "ellaria", "name": "Ellaria", "surname": "Sand", "alive": true, "traits": ["P","O","A","E"] }, + { "_key": "daario", "name": "Daario", "surname": "Naharis", "alive": true, "traits": ["K","P","A"] }, + { "_key": "missandei", "name": "Missandei", "alive": true, "traits": ["D","L","C","M"] }, + { "_key": "tommen", "name": "Tommen", "surname": "Baratheon", "alive": true, "traits": ["I","L","B"] }, + { "_key": "jaqen", "name": "Jaqen", "surname": "H'ghar", "alive": true, "traits": ["H","F","K"] }, + { "_key": "roose", "name": "Roose", "surname": "Bolton", "alive": true, "traits": ["H","E","F","A"] }, + { "_key": "high-sparrow", "name": "The High Sparrow", "alive": true, "traits": ["H","M","F","O"] } +] +``` + +The data is an array of objects, like `[ {...}, {...}, ... ]`. + +`FOR variableName IN expression` is used to iterate over each element of the +array. In each loop, one element is assigned to the variable `d` (`FOR d IN @data`). +This variable is then used in the `INSERT` statement instead of a literal +object definition. What it does is basically the following: + +```aql +// Invalid query + +INSERT { + "_key": "robert", + "name": "Robert", + "surname": "Baratheon", + "alive": false, + "traits": ["A","H","C"] +} INTO Characters + +INSERT { + "_key": "jaime", + "name": "Jaime", + "surname": "Lannister", + "alive": true, + "age": 36, + "traits": ["A","F","B"] +} INTO Characters + +... +``` + +{{< info >}} +AQL does not permit multiple `INSERT` operations that target the same +collection in a single query. However, you can use a `FOR` loop like in the +above query to insert multiple documents into a collection using a single +`INSERT` operation. +{{< /info >}} + +## Read documents + +There are a couple of documents in the `Characters` collection by now. You can +retrieve them all using a `FOR` loop again. This time, however, it is for +going through all documents in the collection instead of an array: + +```aql +FOR c IN Characters + RETURN c +``` + +The syntax of the loop is `FOR variableName IN collectionName`. For each +document in the collection, `c` is assigned a document, which is then returned +as per the loop body. The query returns all characters you previously stored. + +Among them should be `Ned Stark`, similar to this example: + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + ... +] +``` + +The document features the attributes you stored, plus a few more added by +the database system. Each document needs a unique `_key`, which identifies it +within a collection. The `_id` is a computed property, a concatenation of the +collection name, a forward slash `/` and the document key. It uniquely identifies +a document within a database. `_rev` is a revision ID managed by the system. + +Document keys can be provided by the user upon document creation, or a unique +value is assigned automatically. It can not be changed later. All three system +attributes starting with an underscore `_` are read-only. + +You can use either the document key or the document ID to retrieve a specific +document with the help of an AQL function `DOCUMENT()`: + +```aql +RETURN DOCUMENT("Characters", "ned") +// --- or --- +RETURN DOCUMENT("Characters/ned") +``` + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + } +] +``` + +The `DOCUMENT()` function also allows to fetch multiple documents at once: + +```aql +RETURN DOCUMENT("Characters", ["ned", "catelyn"]) +// --- or --- +RETURN DOCUMENT(["Characters/ned", "Characters/catelyn"]) +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"] + } + ] +] +``` + +See the [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) +documentation for more details. + +The `DOCUMENT()` does not let you match documents based on the value of arbitrary +document attributes. It is also not ideal to use the function if the documents +you want to look up are all in the same collection for performance reasons. + +You can replace the call of the `DOCUMENT()` function with the powerful +combination of a `FOR` loop and a `FILTER` operation: + +```aql +FOR c IN Characters + FILTER c._key IN ["ned", "catelyn"] + RETURN c +``` + +This approach enables you to find documents using arbitrary conditions by +changing the filter criteria, but more about this later. + +## Update documents + +According to our `Ned Stark` document, he is alive. When we get to know that he +died, we need to change the `alive` attribute. Modify the existing document: + +```aql +UPDATE "ned" WITH { alive: false } IN Characters +``` + +The syntax is `UPDATE documentKey WITH object IN collectionName`. It updates the +specified document with the attributes listed (or adds them if they don't exist), +but leaves the rest untouched. To replace the entire document content, you may +use `REPLACE` instead of `UPDATE`: + +```aql +REPLACE "ned" WITH { + name: "Ned", + surname: "Stark", + alive: false, + age: 41, + traits: ["A","H","C","N","P"] +} IN Characters +``` + +This also works in a loop, to add a new attribute to all documents for instance: + +```aql +FOR c IN Characters + UPDATE c WITH { season: 1 } IN Characters +``` + +A variable is used instead of a literal document key, to update each document. +The query adds an attribute `season` to the documents' top-level. You can +inspect the result by re-running the query that returns all documents in +collection: + +```aql +FOR c IN Characters + RETURN c +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"], + "season": 1 + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"], + "season": 1 + }, + { + ... + } + ] +] +``` + +## Delete documents + +To fully remove documents from a collection, there is the `REMOVE` operation. +It works similar to the other modification operations, yet without a `WITH` clause: + +```aql +REMOVE "test" IN Characters +``` + +It can also be used in a loop body to effectively truncate a collection +(but less efficient than the dedicated feature to truncate a collection): + +```aql +FOR c IN Characters + REMOVE c IN Characters +``` + +Before you continue with the next chapter, re-run the query that +[creates the character documents](#create-documents) from above to get the data back. diff --git a/site/content/3.12/get-started/start-using-aql/dataset.md b/site/content/3.12/get-started/start-using-aql/dataset.md new file mode 100644 index 0000000000..ff1780550f --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/dataset.md @@ -0,0 +1,99 @@ +--- +title: Game of Thrones example dataset +menuTitle: Dataset +weight: 3 +--- +## Characters + +The dataset features 43 characters with their name, surname, age, alive status +and trait references. Each character also has a document key derived from the +character's name. The surname and age properties are not always present. + +| _key | name | surname | alive | age | traits | +|--------------|------------|------------|-------|-----|---------------| +| ned | Ned | Stark | true | 41 | A, H, C, N, P | +| robert | Robert | Baratheon | false | | A, H, C | +| jaime | Jaime | Lannister | true | 36 | A, F, B | +| catelyn | Catelyn | Stark | false | 40 | D, H, C | +| cersei | Cersei | Lannister | true | 36 | H, E, F | +| daenerys | Daenerys | Targaryen | true | 16 | D, H, C | +| jorah | Jorah | Mormont | false | | A, B, C, F | +| petyr | Petyr | Baelish | false | | E, G, F | +| viserys | Viserys | Targaryen | false | | O, L, N | +| jon | Jon | Snow | true | 16 | A, B, C, F | +| sansa | Sansa | Stark | true | 13 | D, I, J | +| arya | Arya | Stark | true | 11 | C, K, L | +| robb | Robb | Stark | false | | A, B, C, K | +| theon | Theon | Greyjoy | true | 16 | E, R, K | +| bran | Bran | Stark | true | 10 | L, J | +| joffrey | Joffrey | Baratheon | false | 19 | I, L, O | +| sandor | Sandor | Clegane | true | | A, P, K, F | +| tyrion | Tyrion | Lannister | true | 32 | F, K, M, N | +| khal | Khal | Drogo | false | | A, C, O, P | +| tywin | Tywin | Lannister | false | | O, M, H, F | +| davos | Davos | Seaworth | true | 49 | C, K, P, F | +| samwell | Samwell | Tarly | true | 17 | C, L, I | +| stannis | Stannis | Baratheon | false | | H, O, P, M | +| melisandre | Melisandre | | true | | G, E, H | +| margaery | Margaery | Tyrell | false | | M, D, B | +| jeor | Jeor | Mormont | false | | C, H, M, P | +| bronn | Bronn | | true | | K, E, C | +| varys | Varys | | true | | M, F, N, E | +| shae | Shae | | false | | M, D, G | +| talisa | Talisa | Maegyr | false | | D, C, B | +| gendry | Gendry | | false | | K, C, A | +| ygritte | Ygritte | | false | | A, P, K | +| tormund | Tormund | Giantsbane | true | | C, P, A, I | +| gilly | Gilly | | true | | L, J | +| brienne | Brienne | Tarth | true | 32 | P, C, A, K | +| ramsay | Ramsay | Bolton | true | | E, O, G, A | +| ellaria | Ellaria | Sand | true | | P, O, A, E | +| daario | Daario | Naharis | true | | K, P, A | +| missandei | Missandei | | true | | D, L, C, M | +| tommen | Tommen | Baratheon | true | | I, L, B | +| jaqen | Jaqen | H'ghar | true | | H, F, K | +| roose | Roose | Bolton | true | | H, E, F, A | +| high-sparrow | The High Sparrow | | true | | H, M, F, O | + +## Traits + +There are 18 unique traits. Each trait has a random letter as document key. +The trait labels come in English and German. + +| _key | en | de | +|------|-------------|---------------| +| A | strong | stark | +| B | polite | freundlich | +| C | loyal | loyal | +| D | beautiful | schön | +| E | sneaky | hinterlistig | +| F | experienced | erfahren | +| G | corrupt | korrupt | +| H | powerful | einflussreich | +| I | naive | naiv | +| J | unmarried | unverheiratet | +| K | skillful | geschickt | +| L | young | jung | +| M | smart | klug | +| N | rational | rational | +| O | ruthless | skrupellos | +| P | brave | mutig | +| Q | mighty | mächtig | +| R | weak | schwach | + +## Locations + +This small collection of 8 filming locations comes with two attributes, +`name` and `coordinates`. The coordinate pairs are modeled as number arrays, +comprised of a latitude and a longitude value each. + +| name | coordinates | +|-----------------|-----------------------| +| Dragonstone | 55.167801, -6.815096 | +| King's Landing | 42.639752, 18.110189 | +| The Red Keep | 35.896447, 14.446442 | +| Yunkai | 31.046642, -7.129532 | +| Astapor | 31.509740, -9.774249 | +| Winterfell | 54.368321, -5.581312 | +| Vaes Dothrak | 54.167760, -6.096125 | +| Beyond the wall | 64.265473, -21.094093 | diff --git a/site/content/3.12/get-started/start-using-aql/filter.md b/site/content/3.12/get-started/start-using-aql/filter.md new file mode 100644 index 0000000000..0be386c435 --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/filter.md @@ -0,0 +1,137 @@ +--- +title: Match documents with `FILTER` +menuTitle: Match documents +weight: 10 +--- +So far, you either looked up a single document, or returned the entire character +collection. For the lookup, you used the `DOCUMENT()` function, which means you +can only find documents by their key or ID. + +To find documents that fulfill certain criteria more complex than key equality, +there is the `FILTER` operation in AQL, which enables you to formulate arbitrary +conditions for documents to match. + +## Equality condition + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + RETURN c +``` + +The filter condition reads like: "the `name` attribute of a character document +must be equal to the string `Ned`". If the condition applies, character +document gets returned. This works with any attribute likewise: + +```aql +FOR c IN Characters + FILTER c.surname == "Stark" + RETURN c +``` + +## Range conditions + +Strict equality is one possible condition you can state. There are plenty of +other conditions you can formulate, however. For example, you could ask for all +adult characters: + +```aql +FOR c IN Characters + FILTER c.age >= 13 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tyrion", + "Samwell", + "Ned", + "Catelyn", + "Cersei", + "Jon", + "Sansa", + "Brienne", + "Theon", + "Davos", + "Jaime", + "Daenerys" +] +``` + +The operator `>=` stands for *greater-or-equal*, so every character of age 13 +or older is returned (only their name in the example). You can return names +and age of all characters younger than 13 by changing the operator to +*less-than* and using the object syntax to define a subset of attributes to +return: + +```aql +FOR c IN Characters + FILTER c.age < 13 + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Tommen", "age": null }, + { "name": "Arya", "age": 11 }, + { "name": "Roose", "age": null }, + ... +] +``` + +You may notice that it returns name and age of 30 characters, most with an +age of `null`. The reason for this is, that `null` is the fallback value if +an attribute is requested by the query, but no such attribute exists in the +document, and the `null` is compares to numbers as lower (see +[Type and value order](../../aql/fundamentals/type-and-value-order.md)). Hence, it +accidentally fulfills the age criterion `c.age < 13` (`null < 13`). + +## Multiple conditions + +To not let documents pass the filter without an age attribute, you can add a +second criterion: + +```aql +FOR c IN Characters + FILTER c.age < 13 + FILTER c.age != null + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Arya", "age": 11 }, + { "name": "Bran", "age": 10 } +] +``` + +This could equally be written with a boolean `AND` operator as: + +```aql +FOR c IN Characters + FILTER c.age < 13 AND c.age != null + RETURN { name: c.name, age: c.age } +``` + +And the second condition could as well be `c.age > null`. + +## Alternative conditions + +If you want documents to fulfill one or another condition, possibly for +different attributes as well, use `OR`: + +```aql +FOR c IN Characters + FILTER c.name == "Jon" OR c.name == "Joffrey" + RETURN { name: c.name, surname: c.surname } +``` + +```json +[ + { "name": "Joffrey", "surname": "Baratheon" }, + { "name": "Jon", "surname": "Snow" } +] +``` + +See more details about [Filter operations](../../aql/high-level-operations/filter.md). diff --git a/site/content/3.12/get-started/start-using-aql/geo.md b/site/content/3.12/get-started/start-using-aql/geo.md new file mode 100644 index 0000000000..bc19acbdef --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/geo.md @@ -0,0 +1,144 @@ +--- +title: Geospatial queries +menuTitle: Geospatial queries +weight: 30 +--- +Geospatial coordinates consisting of a latitude and longitude value +can be stored either as two separate attributes, or as a single +attribute in the form of an array with both numeric values. +ArangoDB can index such coordinates for fast geospatial queries. + +## Locations data + +Insert some filming locations into a new collection called `Locations`, +which you need to create first, and then run below AQL query: + +```aql +LET places = [ + { "name": "Dragonstone", "coordinates": [ 55.167801, -6.815096 ] }, + { "name": "King's Landing", "coordinates": [ 42.639752, 18.110189 ] }, + { "name": "The Red Keep", "coordinates": [ 35.896447, 14.446442 ] }, + { "name": "Yunkai", "coordinates": [ 31.046642, -7.129532 ] }, + { "name": "Astapor", "coordinates": [ 31.50974, -9.774249 ] }, + { "name": "Winterfell", "coordinates": [ 54.368321, -5.581312 ] }, + { "name": "Vaes Dothrak", "coordinates": [ 54.16776, -6.096125 ] }, + { "name": "Beyond the wall", "coordinates": [ 64.265473, -21.094093 ] } +] + +FOR place IN places + INSERT place INTO Locations + RETURN GEO_POINT(NEW.coordinates[1], NEW.coordinates[0]) +``` + +The last line of the query returns the locations as GeoJSON Points to make the +web interface render a map with markers to give you a visualization of where +the filming locations are. Note that the coordinate order is longitude, than +latitude with GeoJSON, whereas the dataset uses latitude, longitude. + +## Geospatial index + +To query based on coordinates, a [geo index](../../index-and-search/indexing/working-with-indexes/geo-spatial-indexes.md) +is required. It determines which fields contain the latitude and longitude +values. + +1. Click **Collections** in the main navigation. +2. Click the name or row of the **Locations** collection. +3. Switch to the **Indexes** tab at the top. +4. Click the **Add Index** button. +5. Change the **Type** to **Geo Index**. +6. Enter `coordinates` into **Fields**. +7. Click the **Create** button to confirm. + +## Find nearby locations + +A `FOR` loop is used again, with a subsequent `SORT` operation based on the +`DISTANCE()` between a stored coordinate pair and a coordinate pair given in a query. +This pattern is recognized by the query optimizer. A geo index will be used to +accelerate such queries if one is available. + +The default sorting direction is ascending, so a query finds the coordinates +closest to the reference point first (lowest distance). `LIMIT` can be used +to restrict the number of results to at most *n* matches. + +In below example, the limit is set to 3. The origin (the reference point) is +a coordinate pair somewhere downtown in Dublin, Ireland: + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + LIMIT 3 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91491.58596795711 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121425.66829502625 + }, + { + "name": "Dragonstone", + "latitude": 55.167801, + "longitude": -6.815096, + "distance": 205433.7784182078 + } +] +``` + +The query returns the location name, as well as the coordinates and the +calculated distance in meters. The coordinates are returned as two separate +attributes. You may return just the document with a simple `RETURN loc` instead +if you want. Or return the whole document with an added distance attribute using +`RETURN MERGE(loc, { distance })`. + +## Find locations within radius + +`LIMIT` can be swapped out with a `FILTER` that checks the distance, to find +locations within a given radius from a reference point. Remember that the unit +is meters. The example uses a radius of 200,000 meters (200 kilometers): + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + FILTER distance < 200 * 1000 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance: ROUND(distance / 1000) + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121 + } +] +``` + +The distances are converted to kilometers and rounded for readability. diff --git a/site/content/3.12/get-started/start-using-aql/graphs.md b/site/content/3.12/get-started/start-using-aql/graphs.md new file mode 100644 index 0000000000..d896826f35 --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/graphs.md @@ -0,0 +1,303 @@ +--- +title: Graphs and traversals +menuTitle: Graphs and traversals +weight: 25 +description: >- + How relations such as between parents and children can be modeled as graph + and how they can be queried +--- +Relations such as between parents and children can be modeled as graph. +In ArangoDB, two documents (a parent and a child character document) can be +linked by an edge document. Edge documents are stored in edge collections and +have two additional attributes: `_from` and `_to`. They reference any two +documents by their document IDs (`_id`). + +## ChildOf relations + +Our characters have the following relations between parents and children +(first names only for a better overview): + +``` + Robb -> Ned + Sansa -> Ned + Arya -> Ned + Bran -> Ned + Jon -> Ned + Robb -> Catelyn + Sansa -> Catelyn + Arya -> Catelyn + Bran -> Catelyn + Jaime -> Tywin + Cersei -> Tywin + Tyrion -> Tywin + Joffrey -> Jaime + Joffrey -> Cersei +``` + +Visualized as a graph: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +## Creating the edges + +To create the required edge documents to store these relations in the database, +you can run a query that combines joining and filtering to match up the right +character documents, then use their `_id` attribute to insert an edge into an +edge collection called `ChildOf`. + +1. Click **Collections** in the main navigation. +2. Click the **Add collection** button. +3. Enter `ChildOf` as the **Name**. +4. Change the collection type to **Edge**. + +Then run the following query: + +```aql +LET relations = [ + { "parent": "ned", "child": "robb" }, + { "parent": "ned", "child": "sansa" }, + { "parent": "ned", "child": "arya" }, + { "parent": "ned", "child": "bran" }, + { "parent": "catelyn", "child": "robb" }, + { "parent": "catelyn", "child": "sansa" }, + { "parent": "catelyn", "child": "arya" }, + { "parent": "catelyn", "child": "bran" }, + { "parent": "ned", "child": "jon" }, + { "parent": "tywin", "child": "jaime" }, + { "parent": "tywin", "child": "cersei" }, + { "parent": "tywin", "child": "tyrion" }, + { "parent": "cersei", "child": "joffrey" }, + { "parent": "jaime", "child": "joffrey" } +] + +FOR rel in relations + INSERT { + _from: CONCAT("Characters/", rel.child), + _to: CONCAT("Characters/", rel.parent) + } INTO ChildOf + RETURN NEW +``` + +Breakdown of the query: + +- Assign the relations in form of an array of objects with a `parent` and + a `child` attribute each, both with the document key of the character. +- For each element in this array, assign a relation to a variable `rel` and + execute the subsequent instructions. + - Insert a new edge document into the ChildOf collection, with the edge going + from `child` to `parent`. The `_from` and `_to` edge attributes require + document IDs like `collection/key`. Therefore, the document keys derived + from the character names are prefixed with `Characters/` to obtain the IDs. + No other attributes are set for the edge in this example. + - Return the new edge document (optional) + +## Traverse to the parents + +Now that edges link character documents (vertices), it is a graph you can +query to find out who the parents are of another character – or in +graph terms, you want to start at a vertex and follow the edges to other +vertices in an [AQL graph traversal](../../aql/graphs/traversals.md): + +```aql +// Declare collection of start vertex (cluster only) +WITH Characters + +FOR v IN 1..1 OUTBOUND "Characters/bran" ChildOf + RETURN v.name +``` + +This `FOR` loop doesn't iterate over a collection or an array, it walks the +graph and iterates over the connected vertices it finds, with the vertex +document assigned to a variable (here: `v`). It can also emit the edges it +walked as well as the full path from start to end to +[another two variables](../../aql/graphs/traversals.md#syntax). + +In above query, the traversal is restricted to a minimum and maximum traversal +depth of 1 (how many steps to take from the start vertex), and to only follow +edges in `OUTBOUND` direction. Our edges point from child to parent, and the +parent is one step away from the child, thus it gives you the parents of the +child you start at. `"Characters/bran"` is that start vertex. + +To determine the ID of e.g. the Joffrey Baratheon document, you may iterate over +the collection of characters, filter by the name or other criteria, and return +the `_id` attribute: + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + RETURN c._id +``` + +```json +[ "Characters/joffrey" ] +``` + +You may also combine this query with the traversal directly, to easily change +the start vertex by adjusting the filter condition(s): + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +The start vertex is followed by `ChildOf`, which is our edge collection. The +example query returns only the name of each parent to keep the result short: + +```json +[ + "Jaime", + "Cersei" +] +``` + +For Robb, Sansa, Arya, and Bran as the starting point, the result is Catelyn and +Ned, and for Jon Snow it is only Ned. + +Be mindful of the `FILTER` criteria. If more than one character fulfills the +conditions, there will be multiple traversals. And with the query as it is, +all of the parent names would be combined into a single list: + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Tywin", + "Tywin", + "Tywin" +] +``` + +You can achieve a more useful output by returning the result of each traversal +as a separate list and set the minimum traversal depth to `0` to include the +start vertex. This lets you see the child's name as the first element of each +array, followed by the parent name(s) if this information is available. + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + RETURN (FOR v IN 0..1 OUTBOUND c ChildOf + RETURN v.name) +``` + +```json +[ + [ + "Jaime", + "Tywin" + ], + [ + "Cersei", + "Tywin" + ], + [ + "Tyrion", + "Tywin" + ], + [ + "Tywin" + ] +] +``` + +## Traverse to the children + +You can also walk from a parent in reverse edge direction (`INBOUND` that is) +to the children: + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + FOR v IN 1..1 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Robb", + "Sansa", + "Jon", + "Arya", + "Bran" +] +``` + +## Traverse to the grandchildren + +For the Lannister family, there are relations that span from parent to +grandchild. Change the traversal depth to return grandchildren, +which means to go exactly two steps: + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Joffrey", + "Joffrey" +] +``` + +It might be a bit unexpected, that Joffrey is returned twice. However, if you +look at the graph visualization, you can see that multiple paths lead from +Joffrey (bottom right) to Tywin: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +``` +Tywin <- Jaime <- Joffrey +Tywin <- Cersei <- Joffrey +``` + +As a quick fix, change the last line of the query to `RETURN DISTINCT v.name` +to return each value only once. However, there are +[traversal options](../../aql/graphs/traversals.md#syntax) including one to +suppress duplicate vertices early on for the entire traversal (which requires +breadth-first search): + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf OPTIONS { uniqueVertices: "global", order: "bfs" } + RETURN v.name +``` + +```json +[ + "Joffrey" +] +``` + +## Traverse with variable depth + +To return the parents and grandparents of Joffrey, you can walk edges in +`OUTBOUND` direction and adjust the traversal depth to go at least 1 step, +and 2 at most: + +```aql +FOR c IN Characters + FILTER c.name == "Joffrey" + FOR v IN 1..2 OUTBOUND c ChildOf + RETURN DISTINCT v.name +``` + +```json +[ + "Cersei", + "Tywin", + "Jaime" +] +``` + +With deeper family trees, it would only be a matter of changing the depth +values to query for great-grandchildren and similar relations. diff --git a/site/content/3.12/get-started/start-using-aql/joins.md b/site/content/3.12/get-started/start-using-aql/joins.md new file mode 100644 index 0000000000..abf30c9045 --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/joins.md @@ -0,0 +1,323 @@ +--- +title: References and joins +menuTitle: References and joins +weight: 20 +--- +## References to other documents + +The character data you imported has an attribute `traits` for each character, +which is an array of strings. It does not store character features directly, +however: + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"] +} +``` + +It is rather a list of letters without an apparent meaning. The idea here is +that `traits` is supposed to store documents keys of another collection, which +you can use to resolve the letters to labels such as "strong". The benefit of +using another collection for the actual traits is, that you can easily query +for all existing traits later on and store labels in multiple languages for +instance in a central place. If you would embed traits directly... + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": [ + { + "de": "stark", + "en": "strong" + }, + { + "de": "einflussreich", + "en": "powerful" + }, + { + "de": "loyal", + "en": "loyal" + }, + { + "de": "rational", + "en": "rational" + }, + { + "de": "mutig", + "en": "brave" + } + ] +} +``` + +... it becomes difficult to maintain traits. If you were to rename or +translate one of them, you would need to find all other character documents +with the same trait and perform the changes there too. If you only refer to a +trait in another collection, it is as easy as updating a single document. + +{{< comment >}}What if Trait doc is deleted? DOCUMENT() skips null{{< /comment >}} + +## Importing traits + +1. In the web interface, create a document collection called `Traits`. +2. Enter the following AQL query: + ```aql + FOR trait IN @data + INSERT trait INTO Traits + ``` +3. Set the following for the `data` bind variable: + ```json + [ + { "_key": "A", "en": "strong", "de": "stark" }, + { "_key": "B", "en": "polite", "de": "freundlich" }, + { "_key": "C", "en": "loyal", "de": "loyal" }, + { "_key": "D", "en": "beautiful", "de": "schön" }, + { "_key": "E", "en": "sneaky", "de": "hinterlistig" }, + { "_key": "F", "en": "experienced", "de": "erfahren" }, + { "_key": "G", "en": "corrupt", "de": "korrupt" }, + { "_key": "H", "en": "powerful", "de": "einflussreich" }, + { "_key": "I", "en": "naive", "de": "naiv" }, + { "_key": "J", "en": "unmarried", "de": "unverheiratet" }, + { "_key": "K", "en": "skillful", "de": "geschickt" }, + { "_key": "L", "en": "young", "de": "jung" }, + { "_key": "M", "en": "smart", "de": "klug" }, + { "_key": "N", "en": "rational", "de": "rational" }, + { "_key": "O", "en": "ruthless", "de": "skrupellos" }, + { "_key": "P", "en": "brave", "de": "mutig" }, + { "_key": "Q", "en": "mighty", "de": "mächtig" }, + { "_key": "R", "en": "weak", "de": "schwach" } + ] + ``` +4. Execute the query to import the trait data. + +## Resolving traits + +Start simple by returning only the traits attribute of each character: + +```aql +FOR c IN Characters + RETURN c.traits +``` + +```json +[ + ["A","H","C","N","P"], + ["D","H","C"], + ... +] +``` + +Also see the [Fundamentals of Objects / Documents](../../aql/fundamentals/data-types.md#objects--documents) +about attribute access. + +You can use the `traits` array together with the `DOCUMENT()` function to use +the elements as document keys and look them up in the `Traits` collection: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits) +``` + +```json +[ + [ + { + "_key": "A", + "_id": "Traits/A", + "_rev": "_V5oRUS2---", + "en": "strong", + "de": "stark" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + }, + { + "_key": "N", + "_id": "Traits/N", + "_rev": "_V5oRUT---D", + "en": "rational", + "de": "rational" + }, + { + "_key": "P", + "_id": "Traits/P", + "_rev": "_V5oRUTC---", + "en": "brave", + "de": "mutig" + } + ], + [ + { + "_key": "D", + "_id": "Traits/D", + "_rev": "_V5oRUS6--A", + "en": "beautiful", + "de": "schön" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + } + ], + ... +] +``` + +The [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) can be used +to look up a single or multiple documents via document identifiers. In our +example, you pass the collection name from which you want to fetch documents +as first argument (`"Traits"`) and an array of document keys (`_key` attribute) +as second argument. In return, you get an array of the full trait documents +for each character. + +This is a bit too much information, so only return English labels using +the [array expansion](../../aql/operators.md#array-expansion) notation: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits)[*].en +``` + +```json +[ + [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ], + [ + "beautiful", + "powerful", + "loyal" + ], + ... +] +``` + +## Merging characters and traits + +Great, you resolved the letters to meaningful traits! But you also need to know +to which character they belong. Thus, you need to merge both the character +document and the data from the trait documents: + +```aql +FOR c IN Characters + RETURN MERGE(c, { traits: DOCUMENT("Traits", c.traits)[*].en } ) +``` + +```json +[ + { + "_id": "Characters/ned", + "_key": "ned", + "_rev": "_V1bzsXa---", + "age": 41, + "alive": false, + "name": "Ned", + "surname": "Stark", + "traits": [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ] + }, + { + "_id": "Characters/catelyn", + "_key": "catelyn", + "_rev": "_V1bzsXa--B", + "age": 40, + "alive": false, + "name": "Catelyn", + "surname": "Stark", + "traits": [ + "beautiful", + "powerful", + "loyal" + ] + }, + ... +] +``` + +The `MERGE()` functions merges objects together. Because you used an object +`{ traits: ... }` which has the same attribute name `traits` as the original +character attribute, the latter got overwritten by the merge operation. + +## Join another way + +The `DOCUMENT()` function utilizes primary indexes to look up documents quickly. +It is limited to find documents via their identifiers however. For a use case +like in our example it is sufficient to accomplish a simple join. + +There is another, more flexible syntax for joins: nested `FOR` loops over +multiple collections, with a `FILTER` condition to match up attributes. +In case of the traits key array, there needs to be a third loop to iterate +over the keys: + +```aql +FOR c IN Characters + RETURN MERGE(c, { + traits: ( + FOR key IN c.traits + FOR t IN Traits + FILTER t._key == key + RETURN t.en + ) + }) +``` + +For each character, it loops over its `traits` attribute (e.g. `["D","H","C"]`) +and for each document reference in this array, it loops over the `Traits` +collections. There is a condition to match the document key with the key +reference. The inner `FOR` loop and the `FILTER` get transformed to a primary +index lookup in this case instead of building up a Cartesian product only to +filter away everything but a single match: Document keys within a collection +are unique, thus there can only be one match. + +Each written-out, English trait is returned and all the traits are then merged +with the character document. The result is identical to the query using +`DOCUMENT()`. However, this approach with a nested `FOR` loop and a `FILTER` +is not limited to primary keys. You can do this with any other attribute as well. +For an efficient lookup, make sure you add a persistent index for this attribute. +If its values are unique, then also set the index option to unique. + +Another advantage of the `FOR` loop approach is the performance compared to +calling the `DOCUMENT()` function: The query optimizer can optimize AQL queries +better that iterate over a collection and possibly filter by attributes and only +make use of a subset of the found documents. With the `DOCUMENT()` function, +there are individual lookups, potentially across all collections, and the full +documents need to be loaded regardless of which attributes are actually used. diff --git a/site/content/3.12/get-started/start-using-aql/sort-limit.md b/site/content/3.12/get-started/start-using-aql/sort-limit.md new file mode 100644 index 0000000000..553924c379 --- /dev/null +++ b/site/content/3.12/get-started/start-using-aql/sort-limit.md @@ -0,0 +1,183 @@ +--- +title: Sort and limit +menuTitle: Sort and limit +weight: 15 +--- +## Cap the result count with `LIMIT` + +It may not always be necessary to return all documents, that a `FOR` loop +would normally return. You can limit the amount of documents +with a `LIMIT` operation: + +```aql +FOR c IN Characters + LIMIT 5 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tommen", + "Tyrion", + "Roose", + "Tywin" +] +``` + +`LIMIT` is followed by a number for the maximum document count. There is a +second syntax however, which allows you to skip a certain amount of record +and return the next *n* documents: + +```aql +FOR c IN Characters + LIMIT 2, 5 + RETURN c.name +``` + +```json +[ + "Tyrion", + "Roose", + "Tywin", + "Samwell", + "Melisandre" +] +``` + +See how the second query skipped the first two names and returned the next +five (both results feature Tyrion, Roose and Tywin). + +## Sort by name with `SORT` + +The order in which matching records were returned by the queries shown until +here was basically random. To return them in a defined order, you can add a +`SORT()` operation. It can have a big impact on the result if combined with +a `LIMIT()`, because the result becomes predictable if you sort first. + +```aql +FOR c IN Characters + SORT c.name + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Arya", + "Bran", + "Brienne", + "Bronn", + "Catelyn", + "Cersei", + "Daario", + "Daenerys", + "Davos", + "Ellaria" +] +``` + +See how it sorted by name, then returned the ten alphabetically first coming +names. You can reverse the sort order with `DESC` like descending: + +```aql +FOR c IN Characters + SORT c.name DESC + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Ygritte", + "Viserys", + "Varys", + "Tywin", + "Tyrion", + "Tormund", + "Tommen", + "Theon", + "The High Sparrow", + "Talisa" +] +``` + +The first sort was ascending, which is the default order. Because it is the +default, it is not required to explicitly ask for `ASC` order. + +## Sort by multiple attributes + +Assume you want to sort by surname. Many of the characters share a surname. +The result order among characters with the same surname is undefined. You can +first sort by `surname`, then `name`, to determine the order: + +```aql +FOR c IN Characters + FILTER c.surname + SORT c.surname, c.name + LIMIT 10 + RETURN { + surname: c.surname, + name: c.name + } +``` + +```json +[ + { "surname": "Baelish", "name": "Petyr" }, + { "surname": "Baratheon", "name": "Joffrey" }, + { "surname": "Baratheon", "name": "Robert" }, + { "surname": "Baratheon", "name": "Stannis" }, + { "surname": "Baratheon", "name": "Tommen" }, + { "surname": "Bolton", "name": "Ramsay" }, + { "surname": "Bolton", "name": "Roose" }, + { "surname": "Clegane", "name": "Sandor" }, + { "surname": "Drogo", "name": "Khal" }, + { "surname": "Giantsbane", "name": "Tormund" } +] +``` + +Overall, the documents are sorted by last name. If the `surname` is the same +for two characters, the `name` values are compared and the result sorted. + +Note that a filter is applied before sorting, to only let documents through, +that actually feature a surname value (many don't have it and would cause +`null` values in the result). + +## Sort by age + +The order can also be determined by a numeric value, such as the age: + +```aql +FOR c IN Characters + FILTER c.age + SORT c.age + LIMIT 10 + RETURN { + name: c.name, + age: c.age + } +``` + +```json +[ + { "name": "Bran", "age": 10 }, + { "name": "Arya", "age": 11 }, + { "name": "Sansa", "age": 13 }, + { "name": "Jon", "age": 16 }, + { "name": "Theon", "age": 16 }, + { "name": "Daenerys", "age": 16 }, + { "name": "Samwell", "age": 17 }, + { "name": "Joffrey", "age": 19 }, + { "name": "Tyrion", "age": 32 }, + { "name": "Brienne", "age": 32 } +] +``` + +A filter is applied to avoid documents without age attribute. The remaining +documents are sorted by age in ascending order, and the name and age of the +ten youngest characters are returned. + +See the [SORT operation](../../aql/high-level-operations/sort.md) and +[LIMIT operation](../../aql/high-level-operations/limit.md) documentation for +more details. diff --git a/site/content/3.12/operations/troubleshooting/query-debug-packages.md b/site/content/3.12/operations/troubleshooting/query-debug-packages.md index 2ca88e0612..08308ebdcf 100644 --- a/site/content/3.12/operations/troubleshooting/query-debug-packages.md +++ b/site/content/3.12/operations/troubleshooting/query-debug-packages.md @@ -61,7 +61,7 @@ for details. ## Inspect a query debug package with *arangosh* The debug package JSON is compactly formatted. To get a more readable output, -you can use a tool for pretty-printing like [`jq`](https://stedolan.github.io/jq/), +you can use a tool for pretty-printing like [`jq`](https://jqlang.github.io/jq/), or use the `inspectDump()` method of the explainer module for formatting. ```js diff --git a/site/content/3.13/develop/http-api/administration.md b/site/content/3.13/develop/http-api/administration.md index 0fe1d0e940..d1b1baa3ed 100644 --- a/site/content/3.13/develop/http-api/administration.md +++ b/site/content/3.13/develop/http-api/administration.md @@ -1030,7 +1030,7 @@ var url = "/_admin/license"; var response = logCurlRequest('GET', url); assert(response.code === 200); -assertTypeOf("string", response.parsedBody.license); +assertTypeOf("object", response.parsedBody.diskUsage); logJsonResponse(response); ``` diff --git a/site/content/3.13/get-started/how-to-interact-with-arangodb.md b/site/content/3.13/get-started/how-to-interact-with-arangodb.md index 204aa5a9db..e92080622d 100644 --- a/site/content/3.13/get-started/how-to-interact-with-arangodb.md +++ b/site/content/3.13/get-started/how-to-interact-with-arangodb.md @@ -41,21 +41,21 @@ programming language, and do all the talking to the server. Integrations combine a third-party technology with ArangoDB and can be seen as a translation layer that takes over the low-level communication with the server. -### REST API +### HTTP REST API -Under the hood, all interactions with the server make use of its REST API. -A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer) -API is an application programming interface based on the HTTP protocol that -powers the world wide web. +Under the hood, all interactions with the server make use of its RESTful HTTP API. +A [REST](https://en.wikipedia.org/wiki/Representational_state_transfer)-based +API is an application programming interface using the HTTP protocol, the +protocol that powers the world wide web. All requests from the outside to the server need to made against the respective endpoints of this API to perform actions. This includes the web interface, _arangosh_, as well as the drivers and integrations for different programming languages and environments. They all provide a convenient way to work with ArangoDB, but you -may use the low-level REST API directly as needed. +may use the low-level HTTP API directly as needed. -See the [HTTP](../develop/http-api/_index.md) documentation to learn more about the API, how requests -are handled and what endpoints are available. +See the [HTTP API](../develop/http-api/_index.md) documentation to learn more +about the API, how requests are handled, and what endpoints are available. ## Set Up and Deploy ArangoDB diff --git a/site/content/3.13/get-started/start-using-aql.md b/site/content/3.13/get-started/start-using-aql.md deleted file mode 100644 index 5e4ade2c4c..0000000000 --- a/site/content/3.13/get-started/start-using-aql.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Start using AQL -menuTitle: Start using AQL -weight: 45 -description: >- - You can execute AQL queries in different ways, from the easy-to-use - web interface to the raw HTTP API ---- -## How to invoke AQL - -AQL queries can be executed using: - -- the web interface -- the `db` object (either in arangosh or in a Foxx service) -- the raw HTTP API -- through drivers and integrations as an abstraction over the HTTP API - -There are always calls to the server's API under the hood, but the web interface, -the `db` object, drivers, and integrations abstract away the low-level -communication details and are thus easier to use. - -The ArangoDB Web Interface has a [specific tab for AQL queries execution](../aql/how-to-invoke-aql/with-the-web-interface.md). - -You can run [AQL queries from the ArangoDB Shell](../aql/how-to-invoke-aql/with-arangosh.md) -with the [`_query()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and -[`_createStatement()`](../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) methods -of the [`db` object](../develop/javascript-api/@arangodb/db-object.md). This chapter -also describes how to use bind parameters, statistics, counting and cursors with -arangosh. - -If you are using Foxx, see [how to write database queries](../develop/foxx-microservices/getting-started.md#writing-database-queries) -for examples including tagged template strings. - -If you want to run AQL queries from your application via the HTTP REST API, -see the full API description at [HTTP interface for AQL queries](../develop/http-api/queries/aql-queries.md). - -See the respective [driver](../develop/drivers/_index.md) or -[integration](../develop/integrations/_index.md) for its support of AQL queries. - -## Learn the query language - -See the [AQL documentation](../aql/_index.md) for the full language reference -as well as examples. - -For a tutorial, sign up for the [ArangoDB University](https://university.arangodb.com/) -to get access to the **AQL Fundamentals** course. diff --git a/site/content/3.13/get-started/start-using-aql/_index.md b/site/content/3.13/get-started/start-using-aql/_index.md new file mode 100644 index 0000000000..94012b4a7d --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/_index.md @@ -0,0 +1,157 @@ +--- +title: Start using AQL +menuTitle: Start using AQL +weight: 52 +description: >- + Learn how to run your first queries written in ArangoDB's Query Language + and how to go from there, using a Game of Thrones dataset +--- +This is an introduction to ArangoDB's query language AQL, built around a small +dataset of characters from the novel and fantasy drama television series +Game of Thrones (as of season 1). It includes character traits in two languages, +some family relations, and last but not least a small set of filming locations, +which makes for an interesting mix of data to work with. + +There is no need to import the data before you start. It is provided as part +of the AQL queries in this tutorial. You can interact with ArangoDB using its +built-in web interface to manage collections and execute the queries with ease, +but you may also use a different interface. + +## How to run AQL queries + +{{< tabs "interfaces" >}} + +{{< tab "Web interface" >}} +ArangoDB's web interface has a **Queries** section for +[executing AQL queries](../../aql/how-to-invoke-aql/with-the-web-interface.md). + +1. If necessary, [switch to the database](../../concepts/data-structure/databases.md#set-the-database-context) + that you want to run queries in. +2. Click **Queries** in the main navigation. +3. Enter an AQL query in the code editor, e.g. `RETURN CONCAT("Hello, ", @name)`. +4. Specify any needed bind parameters in the panel on the right-hand side on the + **Bind Variables** tab, e.g. set `name` to a value of `AQL`. +5. Optionally set query options on the **Options** tab. +6. Click the **Execute** button or hit `Ctrl`/`Cmd` + `Return`. +{{< /tab >}} + +{{< tab "arangosh" >}} +You can run AQL queries from the ArangoDB Shell ([arangosh](../../components/tools/arangodb-shell/_index.md)) +with the [`db._query()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_query) and +[`db._createStatement()`](../../aql/how-to-invoke-aql/with-arangosh.md#with-db_createstatement-arangostatement) +methods of the [`db` object](../../develop/javascript-api/@arangodb/db-object.md). + +```js +--- +name: arangosh_execute_query_bindvars +description: '' +--- +db._query(`RETURN CONCAT("Hello, ", @name)`, { name: "AQL" }).toArray(); +// -- or -- +var name = "AQL"; +db._query(aql`RETURN CONCAT("Hello, ", ${name})`).toArray(); +``` +See [`db._query()`](../../develop/javascript-api/@arangodb/db-object.md#db_queryquerystring--bindvars--mainoptions--suboptions) +in the _JavaScript API_ for details. + +If you use Foxx, see [how to write database queries](../../develop/foxx-microservices/getting-started.md#writing-database-queries) +for examples including tagged template strings. +{{< /tab >}} + +{{< tab "cURL" >}} +You can use a tool like [cURL](https://curl.se/) to run AQL queries from a +command-line, directly using the HTTP REST API of ArangoDB. + +The response bodies are generally compact JSON (without any line breaks and +indentation). You can format them with the [jq](https://jqlang.github.io/jq/) +tool for better readability if you have it installed: + +```sh +curl -d '{"query":"RETURN CONCAT(\"Hello, \", @name)","bindVars":{"name":"AQL"}}' http://localhost:8529/_api/cursor | jq +``` + +See the [`POST /_db/{database-name}/_api/cursor`](../../develop/http-api/queries/aql-queries.md#create-a-cursor) +endpoint in the _HTTP API_ for details. +{{< /tab >}} + +{{< tab "JavaScript" >}} +```js +import { Database, aql } from "arangojs"; +const db = new Database(); + +const name = "AQL"; +const cursor = await db.query(aql`RETURN CONCAT("Hello, ", ${name})`); +const result = cursor.all(); +console.log(result); +``` + +See [`Database.query()`](https://arangodb.github.io/arangojs/latest/classes/databases.Database.html#query) +in the _arangojs_ documentation for details. +{{< /tab >}} + +{{< tab "Go" >}} +```go +ctx := context.Background() +query := `RETURN CONCAT("Hello, ", @name)` +options := arangodb.QueryOptions{ + BindVars: map[string]interface{}{ + "name": "AQL", + }, +} +cursor, err := db.Query(ctx, query, &options) +if err != nil { + log.Fatalf("Failed to run query:\n%v\n", err) +} else { + defer cursor.Close() + var str string + for cursor.HasMore() { + _, err := cursor.ReadDocument(ctx, &str) + if err != nil { + log.Fatalf("Failed to read cursor:\n%v\n", err) + } else { + fmt.Println(str) + } + } +} +``` + +See [`DatabaseQuery.Query()`](https://pkg.go.dev/github.com/arangodb/go-driver/v2/arangodb#DatabaseQuery) +in the _go-driver_ v2 documentation for details. +{{< /tab >}} + +{{< tab "Java" >}} +```java +String query = "RETURN CONCAT(\"Hello, \", @name)"; +Map bindVars = Collections.singletonMap("name", "AQL"); +ArangoCursor cursor = db.query(query, String.class, bindVars); +cursor.forEach(result -> System.out.println(result)); +``` + +See [`ArangoDatabase.query()`](https://www.javadoc.io/doc/com.arangodb/arangodb-java-driver/latest/com/arangodb/ArangoDatabase.html#query%28java.lang.String,java.lang.Class,java.util.Map%29) +in the _arangodb-java-driver_ documentation for details. +{{< /tab >}} + +{{< tab "Python" >}} +```py +query = "RETURN CONCAT('Hello, ', @name)" +bind_vars = { "name": "AQL" } +cursor = db.aql.execute(query, bind_vars=bind_vars) +for result in cursor: + print(result) +``` + +See [`AQL.execute()`](https://docs.python-arango.com/en/main/specs.html#arango.aql.AQL.execute) +in the _python-arango_ documentation for details. +{{< /tab >}} + +{{< /tabs >}} + +## Learn the query language + +The following pages guide you through important query constructs for storing +and retrieving data, covering basic as well as some advanced features. + +Afterwards, you can read the [AQL documentation](../../aql/_index.md) for the +full language reference and query examples. + +{{< comment >}}TODO: Advanced data manipulation: attributes, projections, calculations... Aggregation: Grouping techniques{{< /comment >}} diff --git a/site/content/3.13/get-started/start-using-aql/crud.md b/site/content/3.13/get-started/start-using-aql/crud.md new file mode 100644 index 0000000000..48a0a2a560 --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/crud.md @@ -0,0 +1,371 @@ +--- +title: AQL CRUD operations +menuTitle: CRUD operations +weight: 5 +description: >- + Learn how to **c**reate, **r**ead, **u**pdate, and **d**elete documents with + the ArangoDB Query Language +--- +## Create documents + +Before you can insert documents with AQL, you need a place to put them in – a +collection. You can [manage collections](../../concepts/data-structure/collections.md#collections-api) +via different interfaces including the web interface, arangosh, or a driver. +It is not possible to do so with AQL, however. + +1. In the web interface, click **Collections** in the main navigation. +2. Click the **Add Collection** button. +3. Enter `Characters` as the **Name**. +4. Leave the **Type** set to the default value of **Document**. +5. Click the **Create** button. + +The new collection should appear in the list. Next, click **Queries** in the +main navigation. To create the first document for the collection with AQL, +use the following AQL query, which you can paste into the query textbox and +run by clicking the **Execute** button: + +```aql +INSERT { + _key: "test", + name: "First", + surname: "Test", + alive: false, + age: 123, + traits: [ "X", "Y" ] +} INTO Characters RETURN NEW +``` + +The syntax is `INSERT document INTO collectionName`. The document is an object +like you may know it from JavaScript or JSON, which is comprised of attribute +key and value pairs. The quotes around the attribute keys are optional in AQL. +Keys are always character sequences (strings), whereas attribute values can +have [different types](../../aql/fundamentals/data-types.md): + +- `null` +- boolean (`true`, `false`) +- number (integer and floating point values) +- string +- array +- object + +The name and surname of the character document you inserted are both string +values. The alive state uses a boolean value. Age is a numeric value. +The traits are an array of strings. The entire document is an object. + +The `RETURN NEW` part is optional and makes the query return the document +including any system attributes that may get added by ArangoDB. If you don't +specify the `_key` attribute then a document key is automatically generated. + +Next, add several characters with a single query: + +```aql +FOR d IN @data + INSERT d INTO Characters +``` + +The `FOR` loop iterates over `@data`, which is a placeholder in the query for +binding the list of characters in JSON format. + +In the web interface, there is a tab called **Bind Variables** on the right-hand +side of the query editor. When you enter a placeholder like `@data` in the editor, +a row appears in the **Bind Variables** tab to specify the value for the placeholder. +Paste the following text into the field for the `data` bind variable: + +```json +[ + { "_key": "ned", "name": "Ned", "surname": "Stark", "alive": true, "age": 41, "traits": ["A","H","C","N","P"] }, + { "_key": "robert", "name": "Robert", "surname": "Baratheon", "alive": false, "traits": ["A","H","C"] }, + { "_key": "jaime", "name": "Jaime", "surname": "Lannister", "alive": true, "age": 36, "traits": ["A","F","B"] }, + { "_key": "catelyn", "name": "Catelyn", "surname": "Stark", "alive": false, "age": 40, "traits": ["D","H","C"] }, + { "_key": "cersei", "name": "Cersei", "surname": "Lannister", "alive": true, "age": 36, "traits": ["H","E","F"] }, + { "_key": "daenerys", "name": "Daenerys", "surname": "Targaryen", "alive": true, "age": 16, "traits": ["D","H","C"] }, + { "_key": "jorah", "name": "Jorah", "surname": "Mormont", "alive": false, "traits": ["A","B","C","F"] }, + { "_key": "petyr", "name": "Petyr", "surname": "Baelish", "alive": false, "traits": ["E","G","F"] }, + { "_key": "viserys", "name": "Viserys", "surname": "Targaryen", "alive": false, "traits": ["O","L","N"] }, + { "_key": "jon", "name": "Jon", "surname": "Snow", "alive": true, "age": 16, "traits": ["A","B","C","F"] }, + { "_key": "sansa", "name": "Sansa", "surname": "Stark", "alive": true, "age": 13, "traits": ["D","I","J"] }, + { "_key": "arya", "name": "Arya", "surname": "Stark", "alive": true, "age": 11, "traits": ["C","K","L"] }, + { "_key": "robb", "name": "Robb", "surname": "Stark", "alive": false, "traits": ["A","B","C","K"] }, + { "_key": "theon", "name": "Theon", "surname": "Greyjoy", "alive": true, "age": 16, "traits": ["E","R","K"] }, + { "_key": "bran", "name": "Bran", "surname": "Stark", "alive": true, "age": 10, "traits": ["L","J"] }, + { "_key": "joffrey", "name": "Joffrey", "surname": "Baratheon", "alive": false, "age": 19, "traits": ["I","L","O"] }, + { "_key": "sandor", "name": "Sandor", "surname": "Clegane", "alive": true, "traits": ["A","P","K","F"] }, + { "_key": "tyrion", "name": "Tyrion", "surname": "Lannister", "alive": true, "age": 32, "traits": ["F","K","M","N"] }, + { "_key": "khal", "name": "Khal", "surname": "Drogo", "alive": false, "traits": ["A","C","O","P"] }, + { "_key": "tywin", "name": "Tywin", "surname": "Lannister", "alive": false, "traits": ["O","M","H","F"] }, + { "_key": "davos", "name": "Davos", "surname": "Seaworth", "alive": true, "age": 49, "traits": ["C","K","P","F"] }, + { "_key": "samwell", "name": "Samwell", "surname": "Tarly", "alive": true, "age": 17, "traits": ["C","L","I"] }, + { "_key": "stannis", "name": "Stannis", "surname": "Baratheon", "alive": false, "traits": ["H","O","P","M"] }, + { "_key": "melisandre", "name": "Melisandre", "alive": true, "traits": ["G","E","H"] }, + { "_key": "margaery", "name": "Margaery", "surname": "Tyrell", "alive": false, "traits": ["M","D","B"] }, + { "_key": "jeor", "name": "Jeor", "surname": "Mormont", "alive": false, "traits": ["C","H","M","P"] }, + { "_key": "bronn", "name": "Bronn", "alive": true, "traits": ["K","E","C"] }, + { "_key": "varys", "name": "Varys", "alive": true, "traits": ["M","F","N","E"] }, + { "_key": "shae", "name": "Shae", "alive": false, "traits": ["M","D","G"] }, + { "_key": "talisa", "name": "Talisa", "surname": "Maegyr", "alive": false, "traits": ["D","C","B"] }, + { "_key": "gendry", "name": "Gendry", "alive": false, "traits": ["K","C","A"] }, + { "_key": "ygritte", "name": "Ygritte", "alive": false, "traits": ["A","P","K"] }, + { "_key": "tormund", "name": "Tormund", "surname": "Giantsbane", "alive": true, "traits": ["C","P","A","I"] }, + { "_key": "gilly", "name": "Gilly", "alive": true, "traits": ["L","J"] }, + { "_key": "brienne", "name": "Brienne", "surname": "Tarth", "alive": true, "age": 32, "traits": ["P","C","A","K"] }, + { "_key": "ramsay", "name": "Ramsay", "surname": "Bolton", "alive": true, "traits": ["E","O","G","A"] }, + { "_key": "ellaria", "name": "Ellaria", "surname": "Sand", "alive": true, "traits": ["P","O","A","E"] }, + { "_key": "daario", "name": "Daario", "surname": "Naharis", "alive": true, "traits": ["K","P","A"] }, + { "_key": "missandei", "name": "Missandei", "alive": true, "traits": ["D","L","C","M"] }, + { "_key": "tommen", "name": "Tommen", "surname": "Baratheon", "alive": true, "traits": ["I","L","B"] }, + { "_key": "jaqen", "name": "Jaqen", "surname": "H'ghar", "alive": true, "traits": ["H","F","K"] }, + { "_key": "roose", "name": "Roose", "surname": "Bolton", "alive": true, "traits": ["H","E","F","A"] }, + { "_key": "high-sparrow", "name": "The High Sparrow", "alive": true, "traits": ["H","M","F","O"] } +] +``` + +The data is an array of objects, like `[ {...}, {...}, ... ]`. + +`FOR variableName IN expression` is used to iterate over each element of the +array. In each loop, one element is assigned to the variable `d` (`FOR d IN @data`). +This variable is then used in the `INSERT` statement instead of a literal +object definition. What it does is basically the following: + +```aql +// Invalid query + +INSERT { + "_key": "robert", + "name": "Robert", + "surname": "Baratheon", + "alive": false, + "traits": ["A","H","C"] +} INTO Characters + +INSERT { + "_key": "jaime", + "name": "Jaime", + "surname": "Lannister", + "alive": true, + "age": 36, + "traits": ["A","F","B"] +} INTO Characters + +... +``` + +{{< info >}} +AQL does not permit multiple `INSERT` operations that target the same +collection in a single query. However, you can use a `FOR` loop like in the +above query to insert multiple documents into a collection using a single +`INSERT` operation. +{{< /info >}} + +## Read documents + +There are a couple of documents in the `Characters` collection by now. You can +retrieve them all using a `FOR` loop again. This time, however, it is for +going through all documents in the collection instead of an array: + +```aql +FOR c IN Characters + RETURN c +``` + +The syntax of the loop is `FOR variableName IN collectionName`. For each +document in the collection, `c` is assigned a document, which is then returned +as per the loop body. The query returns all characters you previously stored. + +Among them should be `Ned Stark`, similar to this example: + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + ... +] +``` + +The document features the attributes you stored, plus a few more added by +the database system. Each document needs a unique `_key`, which identifies it +within a collection. The `_id` is a computed property, a concatenation of the +collection name, a forward slash `/` and the document key. It uniquely identifies +a document within a database. `_rev` is a revision ID managed by the system. + +Document keys can be provided by the user upon document creation, or a unique +value is assigned automatically. It can not be changed later. All three system +attributes starting with an underscore `_` are read-only. + +You can use either the document key or the document ID to retrieve a specific +document with the help of an AQL function `DOCUMENT()`: + +```aql +RETURN DOCUMENT("Characters", "ned") +// --- or --- +RETURN DOCUMENT("Characters/ned") +``` + +```json +[ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + } +] +``` + +The `DOCUMENT()` function also allows to fetch multiple documents at once: + +```aql +RETURN DOCUMENT("Characters", ["ned", "catelyn"]) +// --- or --- +RETURN DOCUMENT(["Characters/ned", "Characters/catelyn"]) +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": true, + "age": 41, + "traits": ["A","H","C","N","P"] + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"] + } + ] +] +``` + +See the [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) +documentation for more details. + +The `DOCUMENT()` does not let you match documents based on the value of arbitrary +document attributes. It is also not ideal to use the function if the documents +you want to look up are all in the same collection for performance reasons. + +You can replace the call of the `DOCUMENT()` function with the powerful +combination of a `FOR` loop and a `FILTER` operation: + +```aql +FOR c IN Characters + FILTER c._key IN ["ned", "catelyn"] + RETURN c +``` + +This approach enables you to find documents using arbitrary conditions by +changing the filter criteria, but more about this later. + +## Update documents + +According to our `Ned Stark` document, he is alive. When we get to know that he +died, we need to change the `alive` attribute. Modify the existing document: + +```aql +UPDATE "ned" WITH { alive: false } IN Characters +``` + +The syntax is `UPDATE documentKey WITH object IN collectionName`. It updates the +specified document with the attributes listed (or adds them if they don't exist), +but leaves the rest untouched. To replace the entire document content, you may +use `REPLACE` instead of `UPDATE`: + +```aql +REPLACE "ned" WITH { + name: "Ned", + surname: "Stark", + alive: false, + age: 41, + traits: ["A","H","C","N","P"] +} IN Characters +``` + +This also works in a loop, to add a new attribute to all documents for instance: + +```aql +FOR c IN Characters + UPDATE c WITH { season: 1 } IN Characters +``` + +A variable is used instead of a literal document key, to update each document. +The query adds an attribute `season` to the documents' top-level. You can +inspect the result by re-running the query that returns all documents in +collection: + +```aql +FOR c IN Characters + RETURN c +``` + +```json +[ + [ + { + "_key": "ned", + "_id": "Characters/ned", + "_rev": "_V1bzsXa---", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"], + "season": 1 + }, + { + "_key": "catelyn", + "_id": "Characters/catelyn", + "_rev": "_V1bzsXa--B", + "name": "Catelyn", + "surname": "Stark", + "alive": false, + "age": 40, + "traits": ["D","H","C"], + "season": 1 + }, + { + ... + } + ] +] +``` + +## Delete documents + +To fully remove documents from a collection, there is the `REMOVE` operation. +It works similar to the other modification operations, yet without a `WITH` clause: + +```aql +REMOVE "test" IN Characters +``` + +It can also be used in a loop body to effectively truncate a collection +(but less efficient than the dedicated feature to truncate a collection): + +```aql +FOR c IN Characters + REMOVE c IN Characters +``` + +Before you continue with the next chapter, re-run the query that +[creates the character documents](#create-documents) from above to get the data back. diff --git a/site/content/3.13/get-started/start-using-aql/dataset.md b/site/content/3.13/get-started/start-using-aql/dataset.md new file mode 100644 index 0000000000..ff1780550f --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/dataset.md @@ -0,0 +1,99 @@ +--- +title: Game of Thrones example dataset +menuTitle: Dataset +weight: 3 +--- +## Characters + +The dataset features 43 characters with their name, surname, age, alive status +and trait references. Each character also has a document key derived from the +character's name. The surname and age properties are not always present. + +| _key | name | surname | alive | age | traits | +|--------------|------------|------------|-------|-----|---------------| +| ned | Ned | Stark | true | 41 | A, H, C, N, P | +| robert | Robert | Baratheon | false | | A, H, C | +| jaime | Jaime | Lannister | true | 36 | A, F, B | +| catelyn | Catelyn | Stark | false | 40 | D, H, C | +| cersei | Cersei | Lannister | true | 36 | H, E, F | +| daenerys | Daenerys | Targaryen | true | 16 | D, H, C | +| jorah | Jorah | Mormont | false | | A, B, C, F | +| petyr | Petyr | Baelish | false | | E, G, F | +| viserys | Viserys | Targaryen | false | | O, L, N | +| jon | Jon | Snow | true | 16 | A, B, C, F | +| sansa | Sansa | Stark | true | 13 | D, I, J | +| arya | Arya | Stark | true | 11 | C, K, L | +| robb | Robb | Stark | false | | A, B, C, K | +| theon | Theon | Greyjoy | true | 16 | E, R, K | +| bran | Bran | Stark | true | 10 | L, J | +| joffrey | Joffrey | Baratheon | false | 19 | I, L, O | +| sandor | Sandor | Clegane | true | | A, P, K, F | +| tyrion | Tyrion | Lannister | true | 32 | F, K, M, N | +| khal | Khal | Drogo | false | | A, C, O, P | +| tywin | Tywin | Lannister | false | | O, M, H, F | +| davos | Davos | Seaworth | true | 49 | C, K, P, F | +| samwell | Samwell | Tarly | true | 17 | C, L, I | +| stannis | Stannis | Baratheon | false | | H, O, P, M | +| melisandre | Melisandre | | true | | G, E, H | +| margaery | Margaery | Tyrell | false | | M, D, B | +| jeor | Jeor | Mormont | false | | C, H, M, P | +| bronn | Bronn | | true | | K, E, C | +| varys | Varys | | true | | M, F, N, E | +| shae | Shae | | false | | M, D, G | +| talisa | Talisa | Maegyr | false | | D, C, B | +| gendry | Gendry | | false | | K, C, A | +| ygritte | Ygritte | | false | | A, P, K | +| tormund | Tormund | Giantsbane | true | | C, P, A, I | +| gilly | Gilly | | true | | L, J | +| brienne | Brienne | Tarth | true | 32 | P, C, A, K | +| ramsay | Ramsay | Bolton | true | | E, O, G, A | +| ellaria | Ellaria | Sand | true | | P, O, A, E | +| daario | Daario | Naharis | true | | K, P, A | +| missandei | Missandei | | true | | D, L, C, M | +| tommen | Tommen | Baratheon | true | | I, L, B | +| jaqen | Jaqen | H'ghar | true | | H, F, K | +| roose | Roose | Bolton | true | | H, E, F, A | +| high-sparrow | The High Sparrow | | true | | H, M, F, O | + +## Traits + +There are 18 unique traits. Each trait has a random letter as document key. +The trait labels come in English and German. + +| _key | en | de | +|------|-------------|---------------| +| A | strong | stark | +| B | polite | freundlich | +| C | loyal | loyal | +| D | beautiful | schön | +| E | sneaky | hinterlistig | +| F | experienced | erfahren | +| G | corrupt | korrupt | +| H | powerful | einflussreich | +| I | naive | naiv | +| J | unmarried | unverheiratet | +| K | skillful | geschickt | +| L | young | jung | +| M | smart | klug | +| N | rational | rational | +| O | ruthless | skrupellos | +| P | brave | mutig | +| Q | mighty | mächtig | +| R | weak | schwach | + +## Locations + +This small collection of 8 filming locations comes with two attributes, +`name` and `coordinates`. The coordinate pairs are modeled as number arrays, +comprised of a latitude and a longitude value each. + +| name | coordinates | +|-----------------|-----------------------| +| Dragonstone | 55.167801, -6.815096 | +| King's Landing | 42.639752, 18.110189 | +| The Red Keep | 35.896447, 14.446442 | +| Yunkai | 31.046642, -7.129532 | +| Astapor | 31.509740, -9.774249 | +| Winterfell | 54.368321, -5.581312 | +| Vaes Dothrak | 54.167760, -6.096125 | +| Beyond the wall | 64.265473, -21.094093 | diff --git a/site/content/3.13/get-started/start-using-aql/filter.md b/site/content/3.13/get-started/start-using-aql/filter.md new file mode 100644 index 0000000000..0be386c435 --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/filter.md @@ -0,0 +1,137 @@ +--- +title: Match documents with `FILTER` +menuTitle: Match documents +weight: 10 +--- +So far, you either looked up a single document, or returned the entire character +collection. For the lookup, you used the `DOCUMENT()` function, which means you +can only find documents by their key or ID. + +To find documents that fulfill certain criteria more complex than key equality, +there is the `FILTER` operation in AQL, which enables you to formulate arbitrary +conditions for documents to match. + +## Equality condition + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + RETURN c +``` + +The filter condition reads like: "the `name` attribute of a character document +must be equal to the string `Ned`". If the condition applies, character +document gets returned. This works with any attribute likewise: + +```aql +FOR c IN Characters + FILTER c.surname == "Stark" + RETURN c +``` + +## Range conditions + +Strict equality is one possible condition you can state. There are plenty of +other conditions you can formulate, however. For example, you could ask for all +adult characters: + +```aql +FOR c IN Characters + FILTER c.age >= 13 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tyrion", + "Samwell", + "Ned", + "Catelyn", + "Cersei", + "Jon", + "Sansa", + "Brienne", + "Theon", + "Davos", + "Jaime", + "Daenerys" +] +``` + +The operator `>=` stands for *greater-or-equal*, so every character of age 13 +or older is returned (only their name in the example). You can return names +and age of all characters younger than 13 by changing the operator to +*less-than* and using the object syntax to define a subset of attributes to +return: + +```aql +FOR c IN Characters + FILTER c.age < 13 + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Tommen", "age": null }, + { "name": "Arya", "age": 11 }, + { "name": "Roose", "age": null }, + ... +] +``` + +You may notice that it returns name and age of 30 characters, most with an +age of `null`. The reason for this is, that `null` is the fallback value if +an attribute is requested by the query, but no such attribute exists in the +document, and the `null` is compares to numbers as lower (see +[Type and value order](../../aql/fundamentals/type-and-value-order.md)). Hence, it +accidentally fulfills the age criterion `c.age < 13` (`null < 13`). + +## Multiple conditions + +To not let documents pass the filter without an age attribute, you can add a +second criterion: + +```aql +FOR c IN Characters + FILTER c.age < 13 + FILTER c.age != null + RETURN { name: c.name, age: c.age } +``` + +```json +[ + { "name": "Arya", "age": 11 }, + { "name": "Bran", "age": 10 } +] +``` + +This could equally be written with a boolean `AND` operator as: + +```aql +FOR c IN Characters + FILTER c.age < 13 AND c.age != null + RETURN { name: c.name, age: c.age } +``` + +And the second condition could as well be `c.age > null`. + +## Alternative conditions + +If you want documents to fulfill one or another condition, possibly for +different attributes as well, use `OR`: + +```aql +FOR c IN Characters + FILTER c.name == "Jon" OR c.name == "Joffrey" + RETURN { name: c.name, surname: c.surname } +``` + +```json +[ + { "name": "Joffrey", "surname": "Baratheon" }, + { "name": "Jon", "surname": "Snow" } +] +``` + +See more details about [Filter operations](../../aql/high-level-operations/filter.md). diff --git a/site/content/3.13/get-started/start-using-aql/geo.md b/site/content/3.13/get-started/start-using-aql/geo.md new file mode 100644 index 0000000000..bc19acbdef --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/geo.md @@ -0,0 +1,144 @@ +--- +title: Geospatial queries +menuTitle: Geospatial queries +weight: 30 +--- +Geospatial coordinates consisting of a latitude and longitude value +can be stored either as two separate attributes, or as a single +attribute in the form of an array with both numeric values. +ArangoDB can index such coordinates for fast geospatial queries. + +## Locations data + +Insert some filming locations into a new collection called `Locations`, +which you need to create first, and then run below AQL query: + +```aql +LET places = [ + { "name": "Dragonstone", "coordinates": [ 55.167801, -6.815096 ] }, + { "name": "King's Landing", "coordinates": [ 42.639752, 18.110189 ] }, + { "name": "The Red Keep", "coordinates": [ 35.896447, 14.446442 ] }, + { "name": "Yunkai", "coordinates": [ 31.046642, -7.129532 ] }, + { "name": "Astapor", "coordinates": [ 31.50974, -9.774249 ] }, + { "name": "Winterfell", "coordinates": [ 54.368321, -5.581312 ] }, + { "name": "Vaes Dothrak", "coordinates": [ 54.16776, -6.096125 ] }, + { "name": "Beyond the wall", "coordinates": [ 64.265473, -21.094093 ] } +] + +FOR place IN places + INSERT place INTO Locations + RETURN GEO_POINT(NEW.coordinates[1], NEW.coordinates[0]) +``` + +The last line of the query returns the locations as GeoJSON Points to make the +web interface render a map with markers to give you a visualization of where +the filming locations are. Note that the coordinate order is longitude, than +latitude with GeoJSON, whereas the dataset uses latitude, longitude. + +## Geospatial index + +To query based on coordinates, a [geo index](../../index-and-search/indexing/working-with-indexes/geo-spatial-indexes.md) +is required. It determines which fields contain the latitude and longitude +values. + +1. Click **Collections** in the main navigation. +2. Click the name or row of the **Locations** collection. +3. Switch to the **Indexes** tab at the top. +4. Click the **Add Index** button. +5. Change the **Type** to **Geo Index**. +6. Enter `coordinates` into **Fields**. +7. Click the **Create** button to confirm. + +## Find nearby locations + +A `FOR` loop is used again, with a subsequent `SORT` operation based on the +`DISTANCE()` between a stored coordinate pair and a coordinate pair given in a query. +This pattern is recognized by the query optimizer. A geo index will be used to +accelerate such queries if one is available. + +The default sorting direction is ascending, so a query finds the coordinates +closest to the reference point first (lowest distance). `LIMIT` can be used +to restrict the number of results to at most *n* matches. + +In below example, the limit is set to 3. The origin (the reference point) is +a coordinate pair somewhere downtown in Dublin, Ireland: + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + LIMIT 3 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91491.58596795711 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121425.66829502625 + }, + { + "name": "Dragonstone", + "latitude": 55.167801, + "longitude": -6.815096, + "distance": 205433.7784182078 + } +] +``` + +The query returns the location name, as well as the coordinates and the +calculated distance in meters. The coordinates are returned as two separate +attributes. You may return just the document with a simple `RETURN loc` instead +if you want. Or return the whole document with an added distance attribute using +`RETURN MERGE(loc, { distance })`. + +## Find locations within radius + +`LIMIT` can be swapped out with a `FILTER` that checks the distance, to find +locations within a given radius from a reference point. Remember that the unit +is meters. The example uses a radius of 200,000 meters (200 kilometers): + +```aql +FOR loc IN Locations + LET distance = DISTANCE(loc.coordinates[0], loc.coordinates[1], 53.35, -6.25) + SORT distance + FILTER distance < 200 * 1000 + RETURN { + name: loc.name, + latitude: loc.coordinates[0], + longitude: loc.coordinates[1], + distance: ROUND(distance / 1000) + } +``` + +```json +[ + { + "name": "Vaes Dothrak", + "latitude": 54.16776, + "longitude": -6.096125, + "distance": 91 + }, + { + "name": "Winterfell", + "latitude": 54.368321, + "longitude": -5.581312, + "distance": 121 + } +] +``` + +The distances are converted to kilometers and rounded for readability. diff --git a/site/content/3.13/get-started/start-using-aql/graphs.md b/site/content/3.13/get-started/start-using-aql/graphs.md new file mode 100644 index 0000000000..d896826f35 --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/graphs.md @@ -0,0 +1,303 @@ +--- +title: Graphs and traversals +menuTitle: Graphs and traversals +weight: 25 +description: >- + How relations such as between parents and children can be modeled as graph + and how they can be queried +--- +Relations such as between parents and children can be modeled as graph. +In ArangoDB, two documents (a parent and a child character document) can be +linked by an edge document. Edge documents are stored in edge collections and +have two additional attributes: `_from` and `_to`. They reference any two +documents by their document IDs (`_id`). + +## ChildOf relations + +Our characters have the following relations between parents and children +(first names only for a better overview): + +``` + Robb -> Ned + Sansa -> Ned + Arya -> Ned + Bran -> Ned + Jon -> Ned + Robb -> Catelyn + Sansa -> Catelyn + Arya -> Catelyn + Bran -> Catelyn + Jaime -> Tywin + Cersei -> Tywin + Tyrion -> Tywin + Joffrey -> Jaime + Joffrey -> Cersei +``` + +Visualized as a graph: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +## Creating the edges + +To create the required edge documents to store these relations in the database, +you can run a query that combines joining and filtering to match up the right +character documents, then use their `_id` attribute to insert an edge into an +edge collection called `ChildOf`. + +1. Click **Collections** in the main navigation. +2. Click the **Add collection** button. +3. Enter `ChildOf` as the **Name**. +4. Change the collection type to **Edge**. + +Then run the following query: + +```aql +LET relations = [ + { "parent": "ned", "child": "robb" }, + { "parent": "ned", "child": "sansa" }, + { "parent": "ned", "child": "arya" }, + { "parent": "ned", "child": "bran" }, + { "parent": "catelyn", "child": "robb" }, + { "parent": "catelyn", "child": "sansa" }, + { "parent": "catelyn", "child": "arya" }, + { "parent": "catelyn", "child": "bran" }, + { "parent": "ned", "child": "jon" }, + { "parent": "tywin", "child": "jaime" }, + { "parent": "tywin", "child": "cersei" }, + { "parent": "tywin", "child": "tyrion" }, + { "parent": "cersei", "child": "joffrey" }, + { "parent": "jaime", "child": "joffrey" } +] + +FOR rel in relations + INSERT { + _from: CONCAT("Characters/", rel.child), + _to: CONCAT("Characters/", rel.parent) + } INTO ChildOf + RETURN NEW +``` + +Breakdown of the query: + +- Assign the relations in form of an array of objects with a `parent` and + a `child` attribute each, both with the document key of the character. +- For each element in this array, assign a relation to a variable `rel` and + execute the subsequent instructions. + - Insert a new edge document into the ChildOf collection, with the edge going + from `child` to `parent`. The `_from` and `_to` edge attributes require + document IDs like `collection/key`. Therefore, the document keys derived + from the character names are prefixed with `Characters/` to obtain the IDs. + No other attributes are set for the edge in this example. + - Return the new edge document (optional) + +## Traverse to the parents + +Now that edges link character documents (vertices), it is a graph you can +query to find out who the parents are of another character – or in +graph terms, you want to start at a vertex and follow the edges to other +vertices in an [AQL graph traversal](../../aql/graphs/traversals.md): + +```aql +// Declare collection of start vertex (cluster only) +WITH Characters + +FOR v IN 1..1 OUTBOUND "Characters/bran" ChildOf + RETURN v.name +``` + +This `FOR` loop doesn't iterate over a collection or an array, it walks the +graph and iterates over the connected vertices it finds, with the vertex +document assigned to a variable (here: `v`). It can also emit the edges it +walked as well as the full path from start to end to +[another two variables](../../aql/graphs/traversals.md#syntax). + +In above query, the traversal is restricted to a minimum and maximum traversal +depth of 1 (how many steps to take from the start vertex), and to only follow +edges in `OUTBOUND` direction. Our edges point from child to parent, and the +parent is one step away from the child, thus it gives you the parents of the +child you start at. `"Characters/bran"` is that start vertex. + +To determine the ID of e.g. the Joffrey Baratheon document, you may iterate over +the collection of characters, filter by the name or other criteria, and return +the `_id` attribute: + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + RETURN c._id +``` + +```json +[ "Characters/joffrey" ] +``` + +You may also combine this query with the traversal directly, to easily change +the start vertex by adjusting the filter condition(s): + +```aql +FOR c IN Characters + FILTER c.surname == "Baratheon" AND c.age != null + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +The start vertex is followed by `ChildOf`, which is our edge collection. The +example query returns only the name of each parent to keep the result short: + +```json +[ + "Jaime", + "Cersei" +] +``` + +For Robb, Sansa, Arya, and Bran as the starting point, the result is Catelyn and +Ned, and for Jon Snow it is only Ned. + +Be mindful of the `FILTER` criteria. If more than one character fulfills the +conditions, there will be multiple traversals. And with the query as it is, +all of the parent names would be combined into a single list: + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + FOR v IN 1..1 OUTBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Tywin", + "Tywin", + "Tywin" +] +``` + +You can achieve a more useful output by returning the result of each traversal +as a separate list and set the minimum traversal depth to `0` to include the +start vertex. This lets you see the child's name as the first element of each +array, followed by the parent name(s) if this information is available. + +```aql +FOR c IN Characters + FILTER c.surname == "Lannister" + RETURN (FOR v IN 0..1 OUTBOUND c ChildOf + RETURN v.name) +``` + +```json +[ + [ + "Jaime", + "Tywin" + ], + [ + "Cersei", + "Tywin" + ], + [ + "Tyrion", + "Tywin" + ], + [ + "Tywin" + ] +] +``` + +## Traverse to the children + +You can also walk from a parent in reverse edge direction (`INBOUND` that is) +to the children: + +```aql +FOR c IN Characters + FILTER c.name == "Ned" + FOR v IN 1..1 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Robb", + "Sansa", + "Jon", + "Arya", + "Bran" +] +``` + +## Traverse to the grandchildren + +For the Lannister family, there are relations that span from parent to +grandchild. Change the traversal depth to return grandchildren, +which means to go exactly two steps: + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf + RETURN v.name +``` + +```json +[ + "Joffrey", + "Joffrey" +] +``` + +It might be a bit unexpected, that Joffrey is returned twice. However, if you +look at the graph visualization, you can see that multiple paths lead from +Joffrey (bottom right) to Tywin: + +![ChildOf graph visualization](../../../images/ChildOf_Graph.png) + +``` +Tywin <- Jaime <- Joffrey +Tywin <- Cersei <- Joffrey +``` + +As a quick fix, change the last line of the query to `RETURN DISTINCT v.name` +to return each value only once. However, there are +[traversal options](../../aql/graphs/traversals.md#syntax) including one to +suppress duplicate vertices early on for the entire traversal (which requires +breadth-first search): + +```aql +FOR c IN Characters + FILTER c.name == "Tywin" + FOR v IN 2..2 INBOUND c ChildOf OPTIONS { uniqueVertices: "global", order: "bfs" } + RETURN v.name +``` + +```json +[ + "Joffrey" +] +``` + +## Traverse with variable depth + +To return the parents and grandparents of Joffrey, you can walk edges in +`OUTBOUND` direction and adjust the traversal depth to go at least 1 step, +and 2 at most: + +```aql +FOR c IN Characters + FILTER c.name == "Joffrey" + FOR v IN 1..2 OUTBOUND c ChildOf + RETURN DISTINCT v.name +``` + +```json +[ + "Cersei", + "Tywin", + "Jaime" +] +``` + +With deeper family trees, it would only be a matter of changing the depth +values to query for great-grandchildren and similar relations. diff --git a/site/content/3.13/get-started/start-using-aql/joins.md b/site/content/3.13/get-started/start-using-aql/joins.md new file mode 100644 index 0000000000..abf30c9045 --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/joins.md @@ -0,0 +1,323 @@ +--- +title: References and joins +menuTitle: References and joins +weight: 20 +--- +## References to other documents + +The character data you imported has an attribute `traits` for each character, +which is an array of strings. It does not store character features directly, +however: + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": ["A","H","C","N","P"] +} +``` + +It is rather a list of letters without an apparent meaning. The idea here is +that `traits` is supposed to store documents keys of another collection, which +you can use to resolve the letters to labels such as "strong". The benefit of +using another collection for the actual traits is, that you can easily query +for all existing traits later on and store labels in multiple languages for +instance in a central place. If you would embed traits directly... + +```json +{ + "_key": "ned", + "name": "Ned", + "surname": "Stark", + "alive": false, + "age": 41, + "traits": [ + { + "de": "stark", + "en": "strong" + }, + { + "de": "einflussreich", + "en": "powerful" + }, + { + "de": "loyal", + "en": "loyal" + }, + { + "de": "rational", + "en": "rational" + }, + { + "de": "mutig", + "en": "brave" + } + ] +} +``` + +... it becomes difficult to maintain traits. If you were to rename or +translate one of them, you would need to find all other character documents +with the same trait and perform the changes there too. If you only refer to a +trait in another collection, it is as easy as updating a single document. + +{{< comment >}}What if Trait doc is deleted? DOCUMENT() skips null{{< /comment >}} + +## Importing traits + +1. In the web interface, create a document collection called `Traits`. +2. Enter the following AQL query: + ```aql + FOR trait IN @data + INSERT trait INTO Traits + ``` +3. Set the following for the `data` bind variable: + ```json + [ + { "_key": "A", "en": "strong", "de": "stark" }, + { "_key": "B", "en": "polite", "de": "freundlich" }, + { "_key": "C", "en": "loyal", "de": "loyal" }, + { "_key": "D", "en": "beautiful", "de": "schön" }, + { "_key": "E", "en": "sneaky", "de": "hinterlistig" }, + { "_key": "F", "en": "experienced", "de": "erfahren" }, + { "_key": "G", "en": "corrupt", "de": "korrupt" }, + { "_key": "H", "en": "powerful", "de": "einflussreich" }, + { "_key": "I", "en": "naive", "de": "naiv" }, + { "_key": "J", "en": "unmarried", "de": "unverheiratet" }, + { "_key": "K", "en": "skillful", "de": "geschickt" }, + { "_key": "L", "en": "young", "de": "jung" }, + { "_key": "M", "en": "smart", "de": "klug" }, + { "_key": "N", "en": "rational", "de": "rational" }, + { "_key": "O", "en": "ruthless", "de": "skrupellos" }, + { "_key": "P", "en": "brave", "de": "mutig" }, + { "_key": "Q", "en": "mighty", "de": "mächtig" }, + { "_key": "R", "en": "weak", "de": "schwach" } + ] + ``` +4. Execute the query to import the trait data. + +## Resolving traits + +Start simple by returning only the traits attribute of each character: + +```aql +FOR c IN Characters + RETURN c.traits +``` + +```json +[ + ["A","H","C","N","P"], + ["D","H","C"], + ... +] +``` + +Also see the [Fundamentals of Objects / Documents](../../aql/fundamentals/data-types.md#objects--documents) +about attribute access. + +You can use the `traits` array together with the `DOCUMENT()` function to use +the elements as document keys and look them up in the `Traits` collection: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits) +``` + +```json +[ + [ + { + "_key": "A", + "_id": "Traits/A", + "_rev": "_V5oRUS2---", + "en": "strong", + "de": "stark" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + }, + { + "_key": "N", + "_id": "Traits/N", + "_rev": "_V5oRUT---D", + "en": "rational", + "de": "rational" + }, + { + "_key": "P", + "_id": "Traits/P", + "_rev": "_V5oRUTC---", + "en": "brave", + "de": "mutig" + } + ], + [ + { + "_key": "D", + "_id": "Traits/D", + "_rev": "_V5oRUS6--A", + "en": "beautiful", + "de": "schön" + }, + { + "_key": "H", + "_id": "Traits/H", + "_rev": "_V5oRUS6--E", + "en": "powerful", + "de": "einflussreich" + }, + { + "_key": "C", + "_id": "Traits/C", + "_rev": "_V5oRUS6--_", + "en": "loyal", + "de": "loyal" + } + ], + ... +] +``` + +The [`DOCUMENT()` function](../../aql/functions/miscellaneous.md#document) can be used +to look up a single or multiple documents via document identifiers. In our +example, you pass the collection name from which you want to fetch documents +as first argument (`"Traits"`) and an array of document keys (`_key` attribute) +as second argument. In return, you get an array of the full trait documents +for each character. + +This is a bit too much information, so only return English labels using +the [array expansion](../../aql/operators.md#array-expansion) notation: + +```aql +FOR c IN Characters + RETURN DOCUMENT("Traits", c.traits)[*].en +``` + +```json +[ + [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ], + [ + "beautiful", + "powerful", + "loyal" + ], + ... +] +``` + +## Merging characters and traits + +Great, you resolved the letters to meaningful traits! But you also need to know +to which character they belong. Thus, you need to merge both the character +document and the data from the trait documents: + +```aql +FOR c IN Characters + RETURN MERGE(c, { traits: DOCUMENT("Traits", c.traits)[*].en } ) +``` + +```json +[ + { + "_id": "Characters/ned", + "_key": "ned", + "_rev": "_V1bzsXa---", + "age": 41, + "alive": false, + "name": "Ned", + "surname": "Stark", + "traits": [ + "strong", + "powerful", + "loyal", + "rational", + "brave" + ] + }, + { + "_id": "Characters/catelyn", + "_key": "catelyn", + "_rev": "_V1bzsXa--B", + "age": 40, + "alive": false, + "name": "Catelyn", + "surname": "Stark", + "traits": [ + "beautiful", + "powerful", + "loyal" + ] + }, + ... +] +``` + +The `MERGE()` functions merges objects together. Because you used an object +`{ traits: ... }` which has the same attribute name `traits` as the original +character attribute, the latter got overwritten by the merge operation. + +## Join another way + +The `DOCUMENT()` function utilizes primary indexes to look up documents quickly. +It is limited to find documents via their identifiers however. For a use case +like in our example it is sufficient to accomplish a simple join. + +There is another, more flexible syntax for joins: nested `FOR` loops over +multiple collections, with a `FILTER` condition to match up attributes. +In case of the traits key array, there needs to be a third loop to iterate +over the keys: + +```aql +FOR c IN Characters + RETURN MERGE(c, { + traits: ( + FOR key IN c.traits + FOR t IN Traits + FILTER t._key == key + RETURN t.en + ) + }) +``` + +For each character, it loops over its `traits` attribute (e.g. `["D","H","C"]`) +and for each document reference in this array, it loops over the `Traits` +collections. There is a condition to match the document key with the key +reference. The inner `FOR` loop and the `FILTER` get transformed to a primary +index lookup in this case instead of building up a Cartesian product only to +filter away everything but a single match: Document keys within a collection +are unique, thus there can only be one match. + +Each written-out, English trait is returned and all the traits are then merged +with the character document. The result is identical to the query using +`DOCUMENT()`. However, this approach with a nested `FOR` loop and a `FILTER` +is not limited to primary keys. You can do this with any other attribute as well. +For an efficient lookup, make sure you add a persistent index for this attribute. +If its values are unique, then also set the index option to unique. + +Another advantage of the `FOR` loop approach is the performance compared to +calling the `DOCUMENT()` function: The query optimizer can optimize AQL queries +better that iterate over a collection and possibly filter by attributes and only +make use of a subset of the found documents. With the `DOCUMENT()` function, +there are individual lookups, potentially across all collections, and the full +documents need to be loaded regardless of which attributes are actually used. diff --git a/site/content/3.13/get-started/start-using-aql/sort-limit.md b/site/content/3.13/get-started/start-using-aql/sort-limit.md new file mode 100644 index 0000000000..553924c379 --- /dev/null +++ b/site/content/3.13/get-started/start-using-aql/sort-limit.md @@ -0,0 +1,183 @@ +--- +title: Sort and limit +menuTitle: Sort and limit +weight: 15 +--- +## Cap the result count with `LIMIT` + +It may not always be necessary to return all documents, that a `FOR` loop +would normally return. You can limit the amount of documents +with a `LIMIT` operation: + +```aql +FOR c IN Characters + LIMIT 5 + RETURN c.name +``` + +```json +[ + "Joffrey", + "Tommen", + "Tyrion", + "Roose", + "Tywin" +] +``` + +`LIMIT` is followed by a number for the maximum document count. There is a +second syntax however, which allows you to skip a certain amount of record +and return the next *n* documents: + +```aql +FOR c IN Characters + LIMIT 2, 5 + RETURN c.name +``` + +```json +[ + "Tyrion", + "Roose", + "Tywin", + "Samwell", + "Melisandre" +] +``` + +See how the second query skipped the first two names and returned the next +five (both results feature Tyrion, Roose and Tywin). + +## Sort by name with `SORT` + +The order in which matching records were returned by the queries shown until +here was basically random. To return them in a defined order, you can add a +`SORT()` operation. It can have a big impact on the result if combined with +a `LIMIT()`, because the result becomes predictable if you sort first. + +```aql +FOR c IN Characters + SORT c.name + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Arya", + "Bran", + "Brienne", + "Bronn", + "Catelyn", + "Cersei", + "Daario", + "Daenerys", + "Davos", + "Ellaria" +] +``` + +See how it sorted by name, then returned the ten alphabetically first coming +names. You can reverse the sort order with `DESC` like descending: + +```aql +FOR c IN Characters + SORT c.name DESC + LIMIT 10 + RETURN c.name +``` + +```json +[ + "Ygritte", + "Viserys", + "Varys", + "Tywin", + "Tyrion", + "Tormund", + "Tommen", + "Theon", + "The High Sparrow", + "Talisa" +] +``` + +The first sort was ascending, which is the default order. Because it is the +default, it is not required to explicitly ask for `ASC` order. + +## Sort by multiple attributes + +Assume you want to sort by surname. Many of the characters share a surname. +The result order among characters with the same surname is undefined. You can +first sort by `surname`, then `name`, to determine the order: + +```aql +FOR c IN Characters + FILTER c.surname + SORT c.surname, c.name + LIMIT 10 + RETURN { + surname: c.surname, + name: c.name + } +``` + +```json +[ + { "surname": "Baelish", "name": "Petyr" }, + { "surname": "Baratheon", "name": "Joffrey" }, + { "surname": "Baratheon", "name": "Robert" }, + { "surname": "Baratheon", "name": "Stannis" }, + { "surname": "Baratheon", "name": "Tommen" }, + { "surname": "Bolton", "name": "Ramsay" }, + { "surname": "Bolton", "name": "Roose" }, + { "surname": "Clegane", "name": "Sandor" }, + { "surname": "Drogo", "name": "Khal" }, + { "surname": "Giantsbane", "name": "Tormund" } +] +``` + +Overall, the documents are sorted by last name. If the `surname` is the same +for two characters, the `name` values are compared and the result sorted. + +Note that a filter is applied before sorting, to only let documents through, +that actually feature a surname value (many don't have it and would cause +`null` values in the result). + +## Sort by age + +The order can also be determined by a numeric value, such as the age: + +```aql +FOR c IN Characters + FILTER c.age + SORT c.age + LIMIT 10 + RETURN { + name: c.name, + age: c.age + } +``` + +```json +[ + { "name": "Bran", "age": 10 }, + { "name": "Arya", "age": 11 }, + { "name": "Sansa", "age": 13 }, + { "name": "Jon", "age": 16 }, + { "name": "Theon", "age": 16 }, + { "name": "Daenerys", "age": 16 }, + { "name": "Samwell", "age": 17 }, + { "name": "Joffrey", "age": 19 }, + { "name": "Tyrion", "age": 32 }, + { "name": "Brienne", "age": 32 } +] +``` + +A filter is applied to avoid documents without age attribute. The remaining +documents are sorted by age in ascending order, and the name and age of the +ten youngest characters are returned. + +See the [SORT operation](../../aql/high-level-operations/sort.md) and +[LIMIT operation](../../aql/high-level-operations/limit.md) documentation for +more details. diff --git a/site/content/images/ChildOf_Graph.png b/site/content/images/ChildOf_Graph.png new file mode 100644 index 0000000000..c489285967 Binary files /dev/null and b/site/content/images/ChildOf_Graph.png differ