2017-12-13 10:19:31 -05:00
|
|
|
[[sql-rest]]
|
|
|
|
== SQL REST API
|
|
|
|
|
|
|
|
The SQL REST API accepts SQL in a JSON document, executes it,
|
|
|
|
and returns the results. For example:
|
|
|
|
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql
|
|
|
|
{
|
|
|
|
"query": "SELECT * FROM library ORDER BY page_count DESC LIMIT 5"
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[setup:library]
|
|
|
|
|
|
|
|
Which returns:
|
|
|
|
|
|
|
|
[source,text]
|
|
|
|
--------------------------------------------------
|
|
|
|
author | name | page_count | release_date
|
2018-01-08 14:52:27 -05:00
|
|
|
-----------------+--------------------+---------------+------------------------
|
|
|
|
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
|
2017-12-13 10:19:31 -05:00
|
|
|
--------------------------------------------------
|
2017-12-18 20:57:50 -05:00
|
|
|
// TESTRESPONSE[s/\|/\\|/ s/\+/\\+/]
|
2017-12-13 10:19:31 -05:00
|
|
|
// TESTRESPONSE[_cat]
|
|
|
|
|
|
|
|
You can also choose to get results in a structured format by adding the `format` parameter. Currently supported formats:
|
|
|
|
- text (default)
|
|
|
|
- json
|
|
|
|
- smile
|
|
|
|
- yaml
|
|
|
|
- cbor
|
|
|
|
|
|
|
|
Alternatively you can set the Accept HTTP header to the appropriate media format.
|
|
|
|
All formats above are supported, the GET parameter takes precedence over the header.
|
|
|
|
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql?format=json
|
|
|
|
{
|
|
|
|
"query": "SELECT * FROM library ORDER BY page_count DESC",
|
|
|
|
"fetch_size": 5
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[setup:library]
|
|
|
|
|
|
|
|
Which returns:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
{
|
|
|
|
"columns": [
|
2017-12-28 11:10:18 -05:00
|
|
|
{"name": "author", "type": "text", "jdbc_type": 12, "display_size": 0},
|
|
|
|
{"name": "name", "type": "text", "jdbc_type": 12, "display_size": 0},
|
|
|
|
{"name": "page_count", "type": "short", "jdbc_type": 5, "display_size": 6},
|
|
|
|
{"name": "release_date", "type": "date", "jdbc_type": 93, "display_size": 20}
|
2017-12-13 10:19:31 -05:00
|
|
|
],
|
|
|
|
"rows": [
|
2018-01-08 14:52:27 -05:00
|
|
|
["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"]
|
2017-12-13 10:19:31 -05:00
|
|
|
],
|
|
|
|
"cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl+v///w8="
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWWWdrRlVfSS1TbDYtcW9lc1FJNmlYdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl\+v\/\/\/w8=/$body.cursor/]
|
|
|
|
|
|
|
|
You can continue to the next page by sending back the `cursor` field. In
|
|
|
|
case of text format the cursor is returned as `Cursor` http header.
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql?format=json
|
|
|
|
{
|
|
|
|
"cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f///w8="
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[continued]
|
|
|
|
// TEST[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f\/\/\/w8=/$body.cursor/]
|
|
|
|
|
|
|
|
Which looks like:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
{
|
|
|
|
"rows" : [
|
2018-01-08 14:52:27 -05:00
|
|
|
["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"]
|
2017-12-13 10:19:31 -05:00
|
|
|
],
|
|
|
|
"cursor" : "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWODRMaXBUaVlRN21iTlRyWHZWYUdrdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl9f///w8="
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// TESTRESPONSE[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWODRMaXBUaVlRN21iTlRyWHZWYUdrdw==:BAFmBmF1dGhvcgFmBG5hbWUBZgpwYWdlX2NvdW50AWYMcmVsZWFzZV9kYXRl9f\/\/\/w8=/$body.cursor/]
|
|
|
|
|
|
|
|
Note that the `column` object is only part of the first page.
|
|
|
|
|
|
|
|
You've reached the last page when there is no `cursor` returned
|
|
|
|
in the results. Like Elasticsearch's <<search-request-scroll,scroll>>,
|
|
|
|
SQL may keep state in Elasticsearch to support the cursor. Unlike
|
|
|
|
scroll, receiving the last page is enough to guarantee that the
|
|
|
|
Elasticsearch state is cleared.
|
|
|
|
|
|
|
|
To clear the state earlier, you can use the clear cursor command:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql/close
|
|
|
|
{
|
|
|
|
"cursor": "sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f///w8="
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[continued]
|
|
|
|
// TEST[s/sDXF1ZXJ5QW5kRmV0Y2gBAAAAAAAAAAEWYUpOYklQMHhRUEtld3RsNnFtYU1hQQ==:BAFmBGRhdGUBZgVsaWtlcwFzB21lc3NhZ2UBZgR1c2Vy9f\/\/\/w8=/$body.cursor/]
|
|
|
|
|
|
|
|
Which will like return the
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
{
|
|
|
|
"succeeded" : true
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// TESTRESPONSE
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
[[sql-rest-filtering]]
|
|
|
|
|
|
|
|
You can filter the results that SQL will run on using a standard
|
|
|
|
Elasticsearch query DSL by specifying the query in the filter
|
|
|
|
parameter.
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql
|
|
|
|
{
|
|
|
|
"query": "SELECT * FROM library ORDER BY page_count DESC",
|
|
|
|
"filter": {
|
|
|
|
"range": {
|
|
|
|
"page_count": {
|
|
|
|
"gte" : 100,
|
|
|
|
"lte" : 200
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"fetch_size": 5
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[setup:library]
|
|
|
|
|
|
|
|
Which returns:
|
|
|
|
|
|
|
|
[source,text]
|
|
|
|
--------------------------------------------------
|
|
|
|
author | name | page_count | release_date
|
2018-01-08 14:52:27 -05:00
|
|
|
---------------+------------------------------------+---------------+------------------------
|
|
|
|
Douglas Adams |The Hitchhiker's Guide to the Galaxy|180 |1979-10-12T00:00:00.000Z
|
2017-12-13 10:19:31 -05:00
|
|
|
--------------------------------------------------
|
2017-12-18 20:57:50 -05:00
|
|
|
// TESTRESPONSE[s/\|/\\|/ s/\+/\\+/]
|
2017-12-13 10:19:31 -05:00
|
|
|
// TESTRESPONSE[_cat]
|
|
|
|
|
|
|
|
[[sql-rest-fields]]
|
|
|
|
In addition to the `query` and `cursor` fields, the request can
|
|
|
|
contain `fetch_size` and `time_zone`. `fetch_size` is a hint for how
|
|
|
|
many results to return in each page. SQL might chose to return more
|
|
|
|
or fewer results though. `time_zone` is the time zone to use for date
|
|
|
|
functions and date parsing. `time_zone` defaults to `utc` and can take
|
|
|
|
any values documented
|
|
|
|
http://www.joda.org/joda-time/apidocs/org/joda/time/DateTimeZone.html[here].
|
2018-01-03 16:40:50 -05:00
|
|
|
|
|
|
|
|
|
|
|
[[sql-rest-metadata]]
|
|
|
|
|
|
|
|
SQL can only work with a subset of Elasticsearch indices. To get the list of
|
|
|
|
indices that are available to the user execute:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql/tables
|
|
|
|
{
|
|
|
|
"table_pattern": "lib*"
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[setup:library]
|
|
|
|
|
|
|
|
Which returns:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
{
|
|
|
|
"tables": [ "library" ]
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// TESTRESPONSE
|
|
|
|
|
|
|
|
|
|
|
|
To get the list of fields that are supported in SQL execute:
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
POST /_xpack/sql/columns
|
|
|
|
{
|
|
|
|
"table_pattern": "library",
|
|
|
|
"column_pattern": ""
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// CONSOLE
|
|
|
|
// TEST[continued]
|
|
|
|
|
|
|
|
[source,js]
|
|
|
|
--------------------------------------------------
|
|
|
|
{
|
|
|
|
"columns": [
|
|
|
|
{"table": "library", "name": "author", "type": "text", "jdbc_type": 12, "display_size": 0},
|
|
|
|
{"table": "library", "name": "name", "type": "text", "jdbc_type": 12, "display_size": 0},
|
|
|
|
{"table": "library", "name": "page_count", "type": "short", "jdbc_type": 5, "display_size": 6},
|
|
|
|
{"table": "library", "name": "release_date", "type": "date", "jdbc_type": 93, "display_size": 20}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
--------------------------------------------------
|
|
|
|
// TESTRESPONSE
|