ES|QL REST API

ES|QL REST API

Overview

The ES|QL query API accepts an ES|QL query string in the query parameter, runs it, and returns the results. For example:

  1. resp = client.esql.query(
  2. format="txt",
  3. query="FROM library | KEEP author, name, page_count, release_date | SORT page_count DESC | LIMIT 5",
  4. )
  5. print(resp)
  1. const response = await client.esql.query({
  2. format: "txt",
  3. query:
  4. "FROM library | KEEP author, name, page_count, release_date | SORT page_count DESC | LIMIT 5",
  5. });
  6. console.log(response);
  1. POST /_query?format=txt
  2. {
  3. "query": "FROM library | KEEP author, name, page_count, release_date | SORT page_count DESC | LIMIT 5"
  4. }

Which returns:

  1. author | name | page_count | release_date
  2. -----------------+--------------------+---------------+------------------------
  3. Peter F. Hamilton|Pandora's Star |768 |2004-03-02T00:00:00.000Z
  4. Vernor Vinge |A Fire Upon the Deep|613 |1992-06-01T00:00:00.000Z
  5. Frank Herbert |Dune |604 |1965-06-01T00:00:00.000Z
  6. Alastair Reynolds|Revelation Space |585 |2000-03-15T00:00:00.000Z
  7. James S.A. Corey |Leviathan Wakes |561 |2011-06-02T00:00:00.000Z

Kibana Console

If you are using Kibana Console (which is highly recommended), take advantage of the triple quotes """ when creating the query. This not only automatically escapes double quotes (") inside the query string but also supports multi-line requests:

  1. resp = client.esql.query(
  2. format="txt",
  3. query="\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  4. )
  5. print(resp)
  1. const response = await client.esql.query({
  2. format: "txt",
  3. query:
  4. "\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  5. });
  6. console.log(response);
  1. POST /_query?format=txt
  2. {
  3. "query": """
  4. FROM library
  5. | KEEP author, name, page_count, release_date
  6. | SORT page_count DESC
  7. | LIMIT 5
  8. """
  9. }

Response formats

ES|QL can return the data in the following human readable and binary formats. You can set the format by specifying the format parameter in the URL or by setting the Accept or Content-Type HTTP header.

The URL parameter takes precedence over the HTTP headers. If neither is specified then the response is returned in the same format as the request.

format

HTTP header

Description

Human readable

csv

text/csv

Comma-separated values

json

application/json

JSON (JavaScript Object Notation) human-readable format

tsv

text/tab-separated-values

Tab-separated values

txt

text/plain

CLI-like representation

yaml

application/yaml

YAML (YAML Ain’t Markup Language) human-readable format

Binary

cbor

application/cbor

Concise Binary Object Representation

smile

application/smile

Smile binary data format similar to CBOR

arrow

application/vnd.apache.arrow.stream

Experimental. Apache Arrow dataframes, IPC streaming format

The csv format accepts a formatting URL query attribute, delimiter, which indicates which character should be used to separate the CSV values. It defaults to comma (,) and cannot take any of the following values: double quote ("), carriage-return (\r) and new-line (\n). The tab (\t) can also not be used. Use the tsv format instead.

Filtering using Elasticsearch Query DSL

Specify a Query DSL query in the filter parameter to filter the set of documents that an ES|QL query runs on.

  1. resp = client.esql.query(
  2. format="txt",
  3. query="\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  4. filter={
  5. "range": {
  6. "page_count": {
  7. "gte": 100,
  8. "lte": 200
  9. }
  10. }
  11. },
  12. )
  13. print(resp)
  1. const response = await client.esql.query({
  2. format: "txt",
  3. query:
  4. "\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  5. filter: {
  6. range: {
  7. page_count: {
  8. gte: 100,
  9. lte: 200,
  10. },
  11. },
  12. },
  13. });
  14. console.log(response);
  1. POST /_query?format=txt
  2. {
  3. "query": """
  4. FROM library
  5. | KEEP author, name, page_count, release_date
  6. | SORT page_count DESC
  7. | LIMIT 5
  8. """,
  9. "filter": {
  10. "range": {
  11. "page_count": {
  12. "gte": 100,
  13. "lte": 200
  14. }
  15. }
  16. }
  17. }

Which returns:

  1. author | name | page_count | release_date
  2. ---------------+------------------------------------+---------------+------------------------
  3. Douglas Adams |The Hitchhiker's Guide to the Galaxy|180 |1979-10-12T00:00:00.000Z

Columnar results

By default, ES|QL returns results as rows. For example, FROM returns each individual document as one row. For the json, yaml, cbor and smile formats, ES|QL can return the results in a columnar fashion where one row represents all the values of a certain column in the results.

  1. resp = client.esql.query(
  2. format="json",
  3. query="\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  4. columnar=True,
  5. )
  6. print(resp)
  1. const response = await client.esql.query({
  2. format: "json",
  3. query:
  4. "\n FROM library\n | KEEP author, name, page_count, release_date\n | SORT page_count DESC\n | LIMIT 5\n ",
  5. columnar: true,
  6. });
  7. console.log(response);
  1. POST /_query?format=json
  2. {
  3. "query": """
  4. FROM library
  5. | KEEP author, name, page_count, release_date
  6. | SORT page_count DESC
  7. | LIMIT 5
  8. """,
  9. "columnar": true
  10. }

Which returns:

  1. {
  2. "took": 28,
  3. "columns": [
  4. {"name": "author", "type": "text"},
  5. {"name": "name", "type": "text"},
  6. {"name": "page_count", "type": "integer"},
  7. {"name": "release_date", "type": "date"}
  8. ],
  9. "values": [
  10. ["Peter F. Hamilton", "Vernor Vinge", "Frank Herbert", "Alastair Reynolds", "James S.A. Corey"],
  11. ["Pandora's Star", "A Fire Upon the Deep", "Dune", "Revelation Space", "Leviathan Wakes"],
  12. [768, 613, 604, 585, 561],
  13. ["2004-03-02T00:00:00.000Z", "1992-06-01T00:00:00.000Z", "1965-06-01T00:00:00.000Z", "2000-03-15T00:00:00.000Z", "2011-06-02T00:00:00.000Z"]
  14. ]
  15. }

Returning localized results

Use the locale parameter in the request body to return results (especially dates) formatted per the conventions of the locale. If locale is not specified, defaults to en-US (English). Refer to JDK Supported Locales.

Syntax: the locale parameter accepts language tags in the (case-insensitive) format xy and xy-XY.

For example, to return a month name in French:

  1. resp = client.esql.query(
  2. locale="fr-FR",
  3. query="\n ROW birth_date_string = \"2023-01-15T00:00:00.000Z\"\n | EVAL birth_date = date_parse(birth_date_string)\n | EVAL month_of_birth = DATE_FORMAT(\"MMMM\",birth_date)\n | LIMIT 5\n ",
  4. )
  5. print(resp)
  1. const response = await client.esql.query({
  2. locale: "fr-FR",
  3. query:
  4. '\n ROW birth_date_string = "2023-01-15T00:00:00.000Z"\n | EVAL birth_date = date_parse(birth_date_string)\n | EVAL month_of_birth = DATE_FORMAT("MMMM",birth_date)\n | LIMIT 5\n ',
  5. });
  6. console.log(response);
  1. POST /_query
  2. {
  3. "locale": "fr-FR",
  4. "query": """
  5. ROW birth_date_string = "2023-01-15T00:00:00.000Z"
  6. | EVAL birth_date = date_parse(birth_date_string)
  7. | EVAL month_of_birth = DATE_FORMAT("MMMM",birth_date)
  8. | LIMIT 5
  9. """
  10. }

Passing parameters to a query

Values, for example for a condition, can be passed to a query “inline”, by integrating the value in the query string itself:

  1. resp = client.esql.query(
  2. query="\n FROM library\n | EVAL year = DATE_EXTRACT(\"year\", release_date)\n | WHERE page_count > 300 AND author == \"Frank Herbert\"\n | STATS count = COUNT(*) by year\n | WHERE count > 0\n | LIMIT 5\n ",
  3. )
  4. print(resp)
  1. const response = await client.esql.query({
  2. query:
  3. '\n FROM library\n | EVAL year = DATE_EXTRACT("year", release_date)\n | WHERE page_count > 300 AND author == "Frank Herbert"\n | STATS count = COUNT(*) by year\n | WHERE count > 0\n | LIMIT 5\n ',
  4. });
  5. console.log(response);
  1. POST /_query
  2. {
  3. "query": """
  4. FROM library
  5. | EVAL year = DATE_EXTRACT("year", release_date)
  6. | WHERE page_count > 300 AND author == "Frank Herbert"
  7. | STATS count = COUNT(*) by year
  8. | WHERE count > 0
  9. | LIMIT 5
  10. """
  11. }

To avoid any attempts of hacking or code injection, extract the values in a separate list of parameters. Use question mark placeholders (?) in the query string for each of the parameters:

  1. resp = client.esql.query(
  2. query="\n FROM library\n | EVAL year = DATE_EXTRACT(\"year\", release_date)\n | WHERE page_count > ? AND author == ?\n | STATS count = COUNT(*) by year\n | WHERE count > ?\n | LIMIT 5\n ",
  3. params=[
  4. 300,
  5. "Frank Herbert",
  6. 0
  7. ],
  8. )
  9. print(resp)
  1. const response = await client.esql.query({
  2. query:
  3. '\n FROM library\n | EVAL year = DATE_EXTRACT("year", release_date)\n | WHERE page_count > ? AND author == ?\n | STATS count = COUNT(*) by year\n | WHERE count > ?\n | LIMIT 5\n ',
  4. params: [300, "Frank Herbert", 0],
  5. });
  6. console.log(response);
  1. POST /_query
  2. {
  3. "query": """
  4. FROM library
  5. | EVAL year = DATE_EXTRACT("year", release_date)
  6. | WHERE page_count > ? AND author == ?
  7. | STATS count = COUNT(*) by year
  8. | WHERE count > ?
  9. | LIMIT 5
  10. """,
  11. "params": [300, "Frank Herbert", 0]
  12. }

The parameters can be named parameters or positional parameters.

Named parameters use question mark placeholders (?) followed by a string.

  1. resp = client.esql.query(
  2. query="\n FROM library\n | EVAL year = DATE_EXTRACT(\"year\", release_date)\n | WHERE page_count > ?page_count AND author == ?author\n | STATS count = COUNT(*) by year\n | WHERE count > ?count\n | LIMIT 5\n ",
  3. params=[
  4. {
  5. "page_count": 300
  6. },
  7. {
  8. "author": "Frank Herbert"
  9. },
  10. {
  11. "count": 0
  12. }
  13. ],
  14. )
  15. print(resp)
  1. const response = await client.esql.query({
  2. query:
  3. '\n FROM library\n | EVAL year = DATE_EXTRACT("year", release_date)\n | WHERE page_count > ?page_count AND author == ?author\n | STATS count = COUNT(*) by year\n | WHERE count > ?count\n | LIMIT 5\n ',
  4. params: [
  5. {
  6. page_count: 300,
  7. },
  8. {
  9. author: "Frank Herbert",
  10. },
  11. {
  12. count: 0,
  13. },
  14. ],
  15. });
  16. console.log(response);
  1. POST /_query
  2. {
  3. "query": """
  4. FROM library
  5. | EVAL year = DATE_EXTRACT("year", release_date)
  6. | WHERE page_count > ?page_count AND author == ?author
  7. | STATS count = COUNT(*) by year
  8. | WHERE count > ?count
  9. | LIMIT 5
  10. """,
  11. "params": [{"page_count" : 300}, {"author" : "Frank Herbert"}, {"count" : 0}]
  12. }

Positional parameters use question mark placeholders (?) followed by an integer.

  1. resp = client.esql.query(
  2. query="\n FROM library\n | EVAL year = DATE_EXTRACT(\"year\", release_date)\n | WHERE page_count > ?1 AND author == ?2\n | STATS count = COUNT(*) by year\n | WHERE count > ?3\n | LIMIT 5\n ",
  3. params=[
  4. 300,
  5. "Frank Herbert",
  6. 0
  7. ],
  8. )
  9. print(resp)
  1. const response = await client.esql.query({
  2. query:
  3. '\n FROM library\n | EVAL year = DATE_EXTRACT("year", release_date)\n | WHERE page_count > ?1 AND author == ?2\n | STATS count = COUNT(*) by year\n | WHERE count > ?3\n | LIMIT 5\n ',
  4. params: [300, "Frank Herbert", 0],
  5. });
  6. console.log(response);
  1. POST /_query
  2. {
  3. "query": """
  4. FROM library
  5. | EVAL year = DATE_EXTRACT("year", release_date)
  6. | WHERE page_count > ?1 AND author == ?2
  7. | STATS count = COUNT(*) by year
  8. | WHERE count > ?3
  9. | LIMIT 5
  10. """,
  11. "params": [300, "Frank Herbert", 0]
  12. }

Running an async ES|QL query

The ES|QL async query API lets you asynchronously execute a query request, monitor its progress, and retrieve results when they become available.

Executing an ES|QL query is commonly quite fast, however queries across large data sets or frozen data can take some time. To avoid long waits, run an async ES|QL query.

Queries initiated by the async query API may return results or not. The wait_for_completion_timeout property determines how long to wait for the results. If the results are not available by this time, a query id is returned which can be later used to retrieve the results. For example:

  1. resp = client.perform_request(
  2. "POST",
  3. "/_query/async",
  4. headers={"Content-Type": "application/json"},
  5. body={
  6. "query": "\n FROM library\n | EVAL year = DATE_TRUNC(1 YEARS, release_date)\n | STATS MAX(page_count) BY year\n | SORT year\n | LIMIT 5\n ",
  7. "wait_for_completion_timeout": "2s"
  8. },
  9. )
  10. print(resp)
  1. const response = await client.transport.request({
  2. method: "POST",
  3. path: "/_query/async",
  4. body: {
  5. query:
  6. "\n FROM library\n | EVAL year = DATE_TRUNC(1 YEARS, release_date)\n | STATS MAX(page_count) BY year\n | SORT year\n | LIMIT 5\n ",
  7. wait_for_completion_timeout: "2s",
  8. },
  9. });
  10. console.log(response);
  1. POST /_query/async
  2. {
  3. "query": """
  4. FROM library
  5. | EVAL year = DATE_TRUNC(1 YEARS, release_date)
  6. | STATS MAX(page_count) BY year
  7. | SORT year
  8. | LIMIT 5
  9. """,
  10. "wait_for_completion_timeout": "2s"
  11. }

If the results are not available within the given timeout period, 2 seconds in this case, no results are returned but rather a response that includes:

  • A query ID
  • An is_running value of true, indicating the query is ongoing

The query continues to run in the background without blocking other requests.

  1. {
  2. "id": "FmNJRUZ1YWZCU3dHY1BIOUhaenVSRkEaaXFlZ3h4c1RTWFNocDdnY2FSaERnUTozNDE=",
  3. "is_running": true
  4. }

To check the progress of an async query, use the ES|QL async query get API with the query ID. Specify how long you’d like to wait for complete results in the wait_for_completion_timeout parameter.

  1. resp = client.perform_request(
  2. "GET",
  3. "/_query/async/FmNJRUZ1YWZCU3dHY1BIOUhaenVSRkEaaXFlZ3h4c1RTWFNocDdnY2FSaERnUTozNDE=",
  4. params={
  5. "wait_for_completion_timeout": "30s"
  6. },
  7. )
  8. print(resp)
  1. response = client.esql.async_query_get(
  2. id: 'FmNJRUZ1YWZCU3dHY1BIOUhaenVSRkEaaXFlZ3h4c1RTWFNocDdnY2FSaERnUTozNDE=',
  3. wait_for_completion_timeout: '30s'
  4. )
  5. puts response
  1. const response = await client.transport.request({
  2. method: "GET",
  3. path: "/_query/async/FmNJRUZ1YWZCU3dHY1BIOUhaenVSRkEaaXFlZ3h4c1RTWFNocDdnY2FSaERnUTozNDE=",
  4. querystring: {
  5. wait_for_completion_timeout: "30s",
  6. },
  7. });
  8. console.log(response);
  1. GET /_query/async/FmNJRUZ1YWZCU3dHY1BIOUhaenVSRkEaaXFlZ3h4c1RTWFNocDdnY2FSaERnUTozNDE=?wait_for_completion_timeout=30s

If the response’s is_running value is false, the query has finished and the results are returned, along with the took time for the query.

  1. {
  2. "is_running": false,
  3. "took": 48,
  4. "columns": ...
  5. }

Use the ES|QL async query delete API to delete an async query before the keep_alive period ends. If the query is still running, Elasticsearch cancels it.

  1. DELETE /_query/async/FmdMX2pIang3UWhLRU5QS0lqdlppYncaMUpYQ05oSkpTc3kwZ21EdC1tbFJXQToxOTI=