| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735 | [role="xpack"][testenv="basic"][[sql-rest]]== SQL REST API* <<sql-rest-overview>>* <<sql-rest-format>>* <<sql-pagination>>* <<sql-rest-filtering>>* <<sql-rest-columnar>>* <<sql-rest-params>>* <<sql-runtime-fields>>* <<sql-async>>[[sql-rest-overview]]=== OverviewThe <<sql-search-api,SQL search API>> accepts SQL in a JSON document, executesit, and returns the results. For example:[source,console]--------------------------------------------------POST /_sql?format=txt{  "query": "SELECT * FROM library ORDER BY page_count DESC LIMIT 5"}--------------------------------------------------// TEST[setup:library]Which returns:[source,text]--------------------------------------------------     author      |        name        |  page_count   | release_date-----------------+--------------------+---------------+------------------------Peter F. Hamilton|Pandora's Star      |768            |2004-03-02T00:00:00.000ZVernor Vinge     |A Fire Upon the Deep|613            |1992-06-01T00:00:00.000ZFrank Herbert    |Dune                |604            |1965-06-01T00:00:00.000ZAlastair Reynolds|Revelation Space    |585            |2000-03-15T00:00:00.000ZJames S.A. Corey |Leviathan Wakes     |561            |2011-06-02T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[s/\|/\\|/ s/\+/\\+/]// TESTRESPONSE[non_json][[sql-kibana-console]][TIP].Using Kibana Console====If you are using {kibana-ref}/console-kibana.html[Kibana Console](which is highly recommended), take advantage of thetriple quotes `"""` when creating the query. This not only automatically escapes doublequotes (`"`) inside the query string but also support multi-line as shown below:image:images/sql/rest/console-triple-quotes.png[]====[[sql-rest-format]]=== Response Data FormatsWhile the textual format is nice for humans, computers prefer somethingmore structured.{es-sql} can return the data in the following formats which can be seteither through the `format` property in the URL or by setting the `Accept` HTTP header:NOTE: The URL parameter takes precedence over the `Accept` HTTP header.If neither is specified then the response is returned in the same format as the request.[cols="^m,^4m,^8"]|===s|formats|`Accept` HTTP headers|Description3+h| Human Readable|csv|text/csv|{wikipedia}/Comma-separated_values[Comma-separated values]|json|application/json|https://www.json.org/[JSON] (JavaScript Object Notation) human-readable format|tsv|text/tab-separated-values|{wikipedia}/Tab-separated_values[Tab-separated values]|txt|text/plain|CLI-like representation|yaml|application/yaml|{wikipedia}/YAML[YAML] (YAML Ain't Markup Language) human-readable format3+h| Binary Formats|cbor|application/cbor|https://cbor.io/[Concise Binary Object Representation]|smile|application/smile|{wikipedia}/Smile_(data_interchange_format)[Smile] binary data format similar to CBOR|===The `CSV` format accepts a formatting URL query attribute, `delimiter`, which indicates which character should be used to separate the CSVvalues. 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, the `tsv` format needs to be used instead.Here are some examples for the human readable formats:==== CSV[source,console]--------------------------------------------------POST /_sql?format=csv{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]which returns:[source,text]--------------------------------------------------author,name,page_count,release_datePeter F. Hamilton,Pandora's Star,768,2004-03-02T00:00:00.000ZVernor Vinge,A Fire Upon the Deep,613,1992-06-01T00:00:00.000ZFrank Herbert,Dune,604,1965-06-01T00:00:00.000ZAlastair Reynolds,Revelation Space,585,2000-03-15T00:00:00.000ZJames S.A. Corey,Leviathan Wakes,561,2011-06-02T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[non_json]or:[source,console]--------------------------------------------------POST /_sql?format=csv&delimiter=%3b{    "query": "SELECT * FROM library ORDER BY page_count DESC",    "fetch_size": 5}--------------------------------------------------// TEST[setup:library]which returns:[source,text]--------------------------------------------------author;name;page_count;release_datePeter F. Hamilton;Pandora's Star;768;2004-03-02T00:00:00.000ZVernor Vinge;A Fire Upon the Deep;613;1992-06-01T00:00:00.000ZFrank Herbert;Dune;604;1965-06-01T00:00:00.000ZAlastair Reynolds;Revelation Space;585;2000-03-15T00:00:00.000ZJames S.A. Corey;Leviathan Wakes;561;2011-06-02T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[non_json]==== JSON[source,console]--------------------------------------------------POST /_sql?format=json{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]Which returns:[source,console-result]--------------------------------------------------{  "columns": [    {"name": "author",       "type": "text"},    {"name": "name",         "type": "text"},    {"name": "page_count",   "type": "short"},    {"name": "release_date", "type": "datetime"}  ],  "rows": [    ["Peter F. Hamilton",  "Pandora's Star",       768, "2004-03-02T00:00:00.000Z"],    ["Vernor Vinge",       "A Fire Upon the Deep", 613, "1992-06-01T00:00:00.000Z"],    ["Frank Herbert",      "Dune",                 604, "1965-06-01T00:00:00.000Z"],    ["Alastair Reynolds",  "Revelation Space",     585, "2000-03-15T00:00:00.000Z"],    ["James S.A. Corey",   "Leviathan Wakes",      561, "2011-06-02T00:00:00.000Z"]  ],  "cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl+v///w8="}--------------------------------------------------// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl\+v\/\/\/w8=/$body.cursor/]==== TSV[source,console]--------------------------------------------------POST /_sql?format=tsv{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]Which returns:[source,text]--------------------------------------------------author	name	page_count	release_datePeter F. Hamilton	Pandora's Star	768	2004-03-02T00:00:00.000ZVernor Vinge	A Fire Upon the Deep	613	1992-06-01T00:00:00.000ZFrank Herbert	Dune	604	1965-06-01T00:00:00.000ZAlastair Reynolds	Revelation Space	585	2000-03-15T00:00:00.000ZJames S.A. Corey	Leviathan Wakes	561	2011-06-02T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[s/\t/ /]// TESTRESPONSE[non_json]==== TXT[source,console]--------------------------------------------------POST /_sql?format=txt{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]Which returns:[source,text]--------------------------------------------------     author      |        name        |  page_count   |      release_date-----------------+--------------------+---------------+------------------------Peter F. Hamilton|Pandora's Star      |768            |2004-03-02T00:00:00.000ZVernor Vinge     |A Fire Upon the Deep|613            |1992-06-01T00:00:00.000ZFrank Herbert    |Dune                |604            |1965-06-01T00:00:00.000ZAlastair Reynolds|Revelation Space    |585            |2000-03-15T00:00:00.000ZJames S.A. Corey |Leviathan Wakes     |561            |2011-06-02T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[s/\|/\\|/ s/\+/\\+/]// TESTRESPONSE[non_json]==== YAML[source,console]--------------------------------------------------POST /_sql?format=yaml{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]Which returns:[source,yaml]--------------------------------------------------columns:- name: "author"  type: "text"- name: "name"  type: "text"- name: "page_count"  type: "short"- name: "release_date"  type: "datetime"rows:- - "Peter F. Hamilton"  - "Pandora's Star"  - 768  - "2004-03-02T00:00:00.000Z"- - "Vernor Vinge"  - "A Fire Upon the Deep"  - 613  - "1992-06-01T00:00:00.000Z"- - "Frank Herbert"  - "Dune"  - 604  - "1965-06-01T00:00:00.000Z"- - "Alastair Reynolds"  - "Revelation Space"  - 585  - "2000-03-15T00:00:00.000Z"- - "James S.A. Corey"  - "Leviathan Wakes"  - 561  - "2011-06-02T00:00:00.000Z"cursor: "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl+v///w8="--------------------------------------------------// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl\+v\/\/\/w8=/$body.cursor/][[sql-pagination]]=== Paginating through a large responseUsing the example from the <<sql-rest-format,previous section>>, one cancontinue to the next page by sending back the cursor field. In the case of CSV, TSV and TXTformats, the cursor is returned in the `Cursor` HTTP header.[source,console]--------------------------------------------------POST /_sql?format=json{  "cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f///w8="}--------------------------------------------------// TEST[continued]// TEST[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f\/\/\/w8=/$body.cursor/]Which looks like:[source,console-result]--------------------------------------------------{  "rows" : [    ["Dan Simmons",        "Hyperion",             482,  "1989-05-26T00:00:00.000Z"],    ["Iain M. Banks",      "Consider Phlebas",     471,  "1987-04-23T00:00:00.000Z"],    ["Neal Stephenson",    "Snow Crash",           470,  "1992-06-01T00:00:00.000Z"],    ["Frank Herbert",      "God Emperor of Dune",  454,  "1981-05-28T00:00:00.000Z"],    ["Frank Herbert",      "Children of Dune",     408,  "1976-04-21T00:00:00.000Z"]  ],  "cursor" : "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWODRMaXBUaVlRN21iTlRyWHZWYUdrdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl9f///w8="}--------------------------------------------------// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWODRMaXBUaVlRN21iTlRyWHZWYUdrdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl9f\/\/\/w8=/$body.cursor/]Note that the `columns` object is only part of the first page.You've reached the last page when there is no `cursor` returnedin the results. Like Elasticsearch's <<scroll-search-results,scroll>>,SQL may keep state in Elasticsearch to support the cursor. Unlikescroll, receiving the last page is enough to guarantee that theElasticsearch state is cleared.To clear the state earlier, use the <<clear-sql-cursor-api,clear cursor API>>:[source,console]--------------------------------------------------POST /_sql/close{  "cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f///w8="}--------------------------------------------------// TEST[continued]// TEST[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f\/\/\/w8=/$body.cursor/]Which will like return the[source,console-result]--------------------------------------------------{  "succeeded" : true}--------------------------------------------------[[sql-rest-filtering]]=== Filtering using {es} Query DSLOne can filter the results that SQL will run on using a standard{es} Query DSL by specifying the query in the filterparameter.[source,console]--------------------------------------------------POST /_sql?format=txt{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "filter": {    "range": {      "page_count": {        "gte" : 100,        "lte" : 200      }    }  },  "fetch_size": 5}--------------------------------------------------// TEST[setup:library]Which returns:[source,text]--------------------------------------------------    author     |                name                |  page_count   | release_date---------------+------------------------------------+---------------+------------------------Douglas Adams  |The Hitchhiker's Guide to the Galaxy|180            |1979-10-12T00:00:00.000Z--------------------------------------------------// TESTRESPONSE[s/\|/\\|/ s/\+/\\+/]// TESTRESPONSE[non_json][TIP]=================A useful and less obvious usage for standard Query DSL filtering is to search documents by a specific <<search-routing, routing key>>.Because {es-sql} does not support a `routing` parameter, one can specify a <<mapping-routing-field, `terms` filter for the `_routing` field>> instead:[source,console]--------------------------------------------------POST /_sql?format=txt{  "query": "SELECT * FROM library",  "filter": {    "terms": {      "_routing": ["abc"]    }  }}--------------------------------------------------// TEST[setup:library]=================[[sql-rest-columnar]]=== Columnar resultsThe most well known way of displaying the results of an SQL query result in general is the one where eachindividual record/document represents one line/row. For certain formats, {es-sql} can return the resultsin a columnar fashion: one row represents all the values of a certain column from the current page of results.The following formats can be returned in columnar orientation: `json`, `yaml`, `cbor` and `smile`.[source,console]--------------------------------------------------POST /_sql?format=json{  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5,  "columnar": true}--------------------------------------------------// TEST[setup:library]Which returns:[source,console-result]--------------------------------------------------{  "columns": [    {"name": "author", "type": "text"},    {"name": "name", "type": "text"},    {"name": "page_count", "type": "short"},    {"name": "release_date", "type": "datetime"}  ],  "values": [    ["Peter F. Hamilton", "Vernor Vinge", "Frank Herbert", "Alastair Reynolds", "James S.A. Corey"],    ["Pandora's Star", "A Fire Upon the Deep", "Dune", "Revelation Space", "Leviathan Wakes"],    [768, 613, 604, 585, 561],    ["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"]  ],  "cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl+v///w8="}--------------------------------------------------// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl\+v\/\/\/w8=/$body.cursor/]Any subsequent calls using a `cursor` still have to contain the `columnar` parameter to preserve the orientation,meaning the initial query will not _remember_ the columnar option.[source,console]--------------------------------------------------POST /_sql?format=json{  "cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl+v///w8=",  "columnar": true}--------------------------------------------------// TEST[continued]// TEST[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl\+v\/\/\/w8=/$body.cursor/]Which looks like:[source,console-result]--------------------------------------------------{  "values": [    ["Dan Simmons", "Iain M. Banks", "Neal Stephenson", "Frank Herbert", "Frank Herbert"],    ["Hyperion", "Consider Phlebas", "Snow Crash", "God Emperor of Dune", "Children of Dune"],    [482, 471, 470, 454, 408],    ["1989-05-26T00:00:00.000Z", "1987-04-23T00:00:00.000Z", "1992-06-01T00:00:00.000Z", "1981-05-28T00:00:00.000Z", "1976-04-21T00:00:00.000Z"]  ],  "cursor": "46ToAwFzQERYRjFaWEo1UVc1a1JtVjBZMmdCQUFBQUFBQUFBQUVXWjBaNlFXbzNOV0pVY21Wa1NUZDJhV2t3V2xwblp3PT3/////DwQBZgZhdXRob3IBBHRleHQAAAFmBG5hbWUBBHRleHQAAAFmCnBhZ2VfY291bnQBBGxvbmcBAAFmDHJlbGVhc2VfZGF0ZQEIZGF0ZXRpbWUBAAEP"}--------------------------------------------------// TESTRESPONSE[s/46ToAwFzQERYRjFaWEo1UVc1a1JtVjBZMmdCQUFBQUFBQUFBQUVXWjBaNlFXbzNOV0pVY21Wa1NUZDJhV2t3V2xwblp3PT3\/\/\/\/\/DwQBZgZhdXRob3IBBHRleHQAAAFmBG5hbWUBBHRleHQAAAFmCnBhZ2VfY291bnQBBGxvbmcBAAFmDHJlbGVhc2VfZGF0ZQEIZGF0ZXRpbWUBAAEP/$body.cursor/][[sql-rest-params]]=== Passing parameters to a queryUsing values in a query condition, for example, or in a `HAVING` statement can be done "inline",by integrating the value in the query string itself:[source,console]--------------------------------------------------POST /_sql?format=txt{	"query": "SELECT YEAR(release_date) AS year FROM library WHERE page_count > 300 AND author = 'Frank Herbert' GROUP BY year HAVING COUNT(*) > 0"}--------------------------------------------------// TEST[setup:library]or it can be done by extracting the values in a separate list of parameters and using question mark placeholders (`?`) in the query string:[source,console]--------------------------------------------------POST /_sql?format=txt{	"query": "SELECT YEAR(release_date) AS year FROM library WHERE page_count > ? AND author = ? GROUP BY year HAVING COUNT(*) > ?",	"params": [300, "Frank Herbert", 0]}--------------------------------------------------// TEST[setup:library][IMPORTANT]The recommended way of passing values to a query is with question mark placeholders, to avoid any attempts of hacking or SQL injection.[[sql-runtime-fields]]=== Use runtime fieldsUse the `runtime_mappings` parameter to extract and create <<runtime,runtimefields>>, or columns, from existing ones during a search.The following search creates a `release_day_of_week` runtime field from`release_date` and returns it in the response.[source,console]----POST _sql?format=txt{  "runtime_mappings": {    "release_day_of_week": {      "type": "keyword",      "script": """        emit(doc['release_date'].value.dayOfWeekEnum.toString())      """    }  },  "query": """    SELECT * FROM library WHERE page_count > 300 AND author = 'Frank Herbert'  """}----// TEST[setup:library]The API returns:[source,txt]----    author     |     name      |  page_count   |      release_date      |release_day_of_week---------------+---------------+---------------+------------------------+-------------------Frank Herbert  |Dune           |604            |1965-06-01T00:00:00.000Z|TUESDAY----// TESTRESPONSE[non_json][[sql-async]]=== Run an async SQL searchBy default, SQL searches are synchronous. They wait for complete results beforereturning a response. However, results can take longer for searches across largedata sets or <<data-tiers,frozen data>>.To avoid long waits, run an async SQL search. Set `wait_for_completion_timeout`to a duration you’d like to wait for synchronous results.[source,console]----POST _sql?format=json{  "wait_for_completion_timeout": "2s",  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}----// TEST[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TEST[setup:library]// TEST[s/"wait_for_completion_timeout": "2s"/"wait_for_completion_timeout": "0"/]If the search doesn’t finish within this period, the search becomes async. TheAPI returns:* An `id` for the search.* An `is_partial` value of `true`, indicating the search results are incomplete.* An `is_running` value of `true`, indicating the search is still running in thebackground.For CSV, TSV, and TXT responses, the API returns these values in the respective`Async-ID`, `Async-partial`, and `Async-running` HTTP headers instead.[source,console-result]----{  "id": "FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=",  "is_partial": true,  "is_running": true,  "rows": [ ]}----// TESTRESPONSE[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TESTRESPONSE[s/FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=/$body.id/]// TESTRESPONSE[s/"is_partial": true/"is_partial": $body.is_partial/]// TESTRESPONSE[s/"is_running": true/"is_running": $body.is_running/]To check the progress of an async search, use the search ID with the<<get-async-sql-search-status-api,get async SQL search status API>>.[source,console]----GET _sql/async/status/FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=----// TEST[skip: no access to search ID]If `is_running` and `is_partial` are `false`, the async search has finished withcomplete results.[source,console-result]----{  "id": "FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=",  "is_running": false,  "is_partial": false,  "expiration_time_in_millis": 1611690295000,  "completion_status": 200}----// TESTRESPONSE[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TESTRESPONSE[s/FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=/$body.id/]// TESTRESPONSE[s/"expiration_time_in_millis": 1611690295000/"expiration_time_in_millis": $body.expiration_time_in_millis/]To get the results, use the search ID with the <<get-async-sql-search-api,getasync SQL search API>>. If the search is still running, specify how long you’dlike to wait using `wait_for_completion_timeout`. You can also specify theresponse `format`.[source,console]----GET _sql/async/FnR0TDhyWUVmUmVtWXRWZER4MXZiNFEad2F5UDk2ZVdTVHV1S0xDUy00SklUdzozMTU=?wait_for_completion_timeout=2s&format=json----// TEST[skip: no access to search ID][discrete][[sql-async-retention]]==== Change the search retention periodBy default, {es} stores async SQL searches for five days. After this period,{es} deletes the search and its results, even if the search is still running. Tochange this retention period, use the `keep_alive` parameter.[source,console]----POST _sql?format=json{  "keep_alive": "2d",  "wait_for_completion_timeout": "2s",  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}----// TEST[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TEST[setup:library]You can use the get async SQL search API's `keep_alive` parameter to laterchange the retention period. The new period starts after the request runs.[source,console]----GET _sql/async/FmdMX2pIang3UWhLRU5QS0lqdlppYncaMUpYQ05oSkpTc3kwZ21EdC1tbFJXQToxOTI=?keep_alive=5d&wait_for_completion_timeout=2s&format=json----// TEST[skip: no access to search ID]Use the <<delete-async-sql-search-api,delete async SQL search API>> to delete anasync search before the `keep_alive` period ends. If the search is stillrunning, {es} cancels it.[source,console]----DELETE _sql/async/delete/FmdMX2pIang3UWhLRU5QS0lqdlppYncaMUpYQ05oSkpTc3kwZ21EdC1tbFJXQToxOTI=----// TEST[skip: no access to search ID][discrete][[sql-store-searches]]==== Store synchronous SQL searchesBy default, {es} only stores async SQL searches. To save a synchronous search,specify `wait_for_completion_timeout` and set `keep_on_completion` to `true`.[source,console]----POST _sql?format=json{  "keep_on_completion": true,  "wait_for_completion_timeout": "2s",  "query": "SELECT * FROM library ORDER BY page_count DESC",  "fetch_size": 5}----// TEST[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TEST[setup:library]If `is_partial` and `is_running` are `false`, the search was synchronous andreturned complete results.[source,console-result]----{  "id": "Fnc5UllQdUVWU0NxRFNMbWxNYXplaFEaMUpYQ05oSkpTc3kwZ21EdC1tbFJXQTo0NzA=",  "is_partial": false,  "is_running": false,  "rows": ...,  "columns": ...,  "cursor": ...}----// TESTRESPONSE[skip:waiting on https://github.com/elastic/elasticsearch/issues/75069]// TESTRESPONSE[s/Fnc5UllQdUVWU0NxRFNMbWxNYXplaFEaMUpYQ05oSkpTc3kwZ21EdC1tbFJXQTo0NzA=/$body.id/]// TESTRESPONSE[s/"rows": \.\.\./"rows": $body.rows/]// TESTRESPONSE[s/"columns": \.\.\./"columns": $body.columns/]// TESTRESPONSE[s/"cursor": \.\.\./"cursor": $body.cursor/]You can get the same results later using the search ID with the<<get-async-sql-search-api,get async SQL search API>>.Saved synchronous searches are still subject to the `keep_alive` retentionperiod. When this period ends, {es} deletes the search results. You can alsodelete saved searches using the <<delete-async-sql-search-api,delete async SQLsearch API>>.
 |