2014-04-18 13:51:59 -04:00
[[getting-started]]
2019-07-01 18:34:32 -04:00
= Getting started with {es}
2014-04-18 13:51:59 -04:00
[partintro]
--
2019-07-01 18:34:32 -04:00
Ready to take {es} for a test drive and see for yourself how you can use the
REST APIs to store, search, and analyze data?
2014-04-18 13:51:59 -04:00
2019-07-01 18:34:32 -04:00
Step through this getting started tutorial to:
2014-04-18 13:51:59 -04:00
2019-08-12 21:19:47 -04:00
. Get an {es} cluster up and running
2019-07-01 18:34:32 -04:00
. Index some sample documents
. Search for documents using the {es} query language
. Analyze the results using bucket and metrics aggregations
2014-04-18 13:51:59 -04:00
2019-07-01 18:34:32 -04:00
Need more context?
2017-07-05 06:30:19 -04:00
2019-07-01 18:34:32 -04:00
Check out the <<elasticsearch-intro,
2019-08-26 21:24:05 -04:00
{es} Introduction>> to learn the lingo and understand the basics of
2019-07-01 18:34:32 -04:00
how {es} works. If you're already familiar with {es} and want to see how it works
with the rest of the stack, you might want to jump to the
{stack-gs}/get-started-elastic-stack.html[Elastic Stack
Tutorial] to see how to set up a system monitoring solution with {es}, {kib},
2019-08-20 12:14:49 -04:00
{beats}, and {ls}.
2014-04-18 13:51:59 -04:00
2019-07-01 18:34:32 -04:00
TIP: The fastest way to get started with {es} is to
https://www.elastic.co/cloud/elasticsearch-service/signup[start a free 14-day
2019-08-26 21:24:05 -04:00
trial of {ess}] in the cloud.
2019-07-01 18:34:32 -04:00
--
2014-04-18 13:51:59 -04:00
2018-12-06 13:14:37 -05:00
[[getting-started-install]]
2019-08-12 21:19:47 -04:00
== Get {es} up and running
2019-08-26 21:24:05 -04:00
To take {es} for a test drive, you can create a
https://www.elastic.co/cloud/elasticsearch-service/signup[hosted deployment] on
the {ess} or set up a multi-node {es} cluster on your own
2019-08-12 21:19:47 -04:00
Linux, macOS, or Windows machine.
2014-04-18 13:51:59 -04:00
2017-06-19 00:52:47 -04:00
[float]
2019-08-12 21:19:47 -04:00
[[run-elasticsearch-local]]
=== Run {es} locally on Linux, macOS, or Windows
2017-06-19 00:52:47 -04:00
2019-08-26 21:24:05 -04:00
When you create a deployment on the {ess}, a master node and
two data nodes are provisioned automatically. By installing from the tar or zip
archive, you can start multiple instances of {es} locally to see how a multi-node
cluster behaves.
2017-06-19 00:52:47 -04:00
2019-08-12 21:19:47 -04:00
To run a three-node {es} cluster locally:
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
. Download the {es} archive for your OS:
2019-08-12 21:19:47 -04:00
+
Linux: https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-{version}-linux-x86_64.tar.gz[elasticsearch-{version}-linux-x86_64.tar.gz]
+
2014-07-23 05:54:53 -04:00
["source","sh",subs="attributes,callouts"]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-03-29 09:04:46 -04:00
curl -L -O https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-{version}-linux-x86_64.tar.gz
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-09-30 13:17:35 -04:00
// NOTCONSOLE
2019-08-12 21:19:47 -04:00
+
macOS: https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-{version}-darwin-x86_64.tar.gz[elasticsearch-{version}-darwin-x86_64.tar.gz]
+
["source","sh",subs="attributes,callouts"]
--------------------------------------------------
curl -L -O https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-{version}-darwin-x86_64.tar.gz
--------------------------------------------------
// NOTCONSOLE
+
Windows:
https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-{version}-windows-x86_64.zip[elasticsearch-{version}-windows-x86_64.zip]
. Extract the archive:
+
Linux:
+
2014-07-23 05:54:53 -04:00
["source","sh",subs="attributes,callouts"]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-03-29 09:04:46 -04:00
tar -xvf elasticsearch-{version}-linux-x86_64.tar.gz
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
+
macOS:
+
2014-07-23 05:54:53 -04:00
["source","sh",subs="attributes,callouts"]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
tar -xvf elasticsearch-{version}-darwin-x86_64.tar.gz
--------------------------------------------------
+
Windows PowerShell:
+
2019-08-21 04:47:50 -04:00
["source","powershell",subs="attributes,callouts"]
2019-08-12 21:19:47 -04:00
--------------------------------------------------
Expand-Archive elasticsearch-{version}-windows-x86_64.zip
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-26 21:24:05 -04:00
. Start {es} from the `bin` directory:
2019-08-12 21:19:47 -04:00
+
Linux and macOS:
+
["source","sh",subs="attributes,callouts"]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
cd elasticsearch-{version}/bin
2014-04-18 13:51:59 -04:00
./elasticsearch
--------------------------------------------------
2019-08-12 21:19:47 -04:00
+
Windows:
+
2019-08-21 04:47:50 -04:00
["source","powershell",subs="attributes,callouts"]
2017-06-19 00:52:47 -04:00
--------------------------------------------------
2019-08-21 04:47:50 -04:00
cd elasticsearch-{version}\bin
.\elasticsearch.bat
2017-06-19 00:52:47 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
+
You now have a single-node {es} cluster up and running!
2017-06-19 00:52:47 -04:00
2019-08-12 21:19:47 -04:00
. Start two more instances of {es} so you can see how a typical multi-node
cluster behaves. You need to specify unique data and log paths
for each node.
+
Linux and macOS:
+
["source","sh",subs="attributes,callouts"]
2017-06-19 00:52:47 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
./elasticsearch -Epath.data=data2 -Epath.logs=log2
./elasticsearch -Epath.data=data3 -Epath.logs=log3
2017-06-19 00:52:47 -04:00
--------------------------------------------------
2019-08-12 21:19:47 -04:00
+
Windows:
+
2019-08-21 04:47:50 -04:00
["source","powershell",subs="attributes,callouts"]
2017-06-19 00:52:47 -04:00
--------------------------------------------------
2019-08-21 04:47:50 -04:00
.\elasticsearch.bat -E path.data=data2 -E path.logs=log2
.\elasticsearch.bat -E path.data=data3 -E path.logs=log3
2019-08-12 21:19:47 -04:00
--------------------------------------------------
+
The additional nodes are assigned unique IDs. Because you're running all three
nodes locally, they automatically join the cluster with the first node.
2019-08-20 12:14:49 -04:00
. Use the cat health API to verify that your three-node cluster is up running.
The cat APIs return information about your cluster and indices in a
2019-08-12 21:19:47 -04:00
format that's easier to read than raw JSON.
+
You can interact directly with your cluster by submitting HTTP requests to
the {es} REST API. Most of the examples in this guide enable you to copy the
appropriate cURL command and submit the request to your local {es} instance from
the command line. If you have Kibana installed and running, you can also
open Kibana and submit requests through the Dev Console.
+
TIP: You'll want to check out the
https://www.elastic.co/guide/en/elasticsearch/client/index.html[{es} language
clients] when you're ready to start using {es} in your own applications.
+
[source,js]
--------------------------------------------------
GET /_cat/health?v
--------------------------------------------------
// CONSOLE
+
2019-08-20 12:14:49 -04:00
The response should indicate that the status of the `elasticsearch` cluster
is `green` and it has three nodes:
2019-08-12 21:19:47 -04:00
+
[source,txt]
--------------------------------------------------
epoch timestamp cluster status node.total node.data shards pri relo init unassign pending_tasks max_task_wait_time active_shards_percent
1565052807 00:53:27 elasticsearch green 3 3 6 3 0 0 0 0 - 100.0%
--------------------------------------------------
// TESTRESPONSE[s/1565052807 00:53:27 elasticsearch/\\d+ \\d+:\\d+:\\d+ integTest/]
// TESTRESPONSE[s/3 3 6 3/\\d+ \\d+ \\d+ \\d+/]
// TESTRESPONSE[s/0 0 -/0 \\d+ -/]
// TESTRESPONSE[non_json]
+
NOTE: The cluster status will remain yellow if you are only running a single
instance of {es}. A single node cluster is fully functional, but data
cannot be replicated to another node to provide resiliency. Replica shards must
be available for the cluster status to be green. If the cluster status is red,
some data is unavailable.
2017-06-19 00:52:47 -04:00
[float]
2019-08-12 21:19:47 -04:00
[[gs-other-install]]
=== Other installation options
Installing {es} from an archive file enables you to easily install and run
multiple instances locally so you can try things out. To run a single instance,
you can run {es} in a Docker container, install {es} using the DEB or RPM
packages on Linux, install using Homebrew on macOS, or install using the MSI
package installer on Windows. See <<install-elasticsearch>> for more information.
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
[[getting-started-index]]
2019-08-16 17:54:21 -04:00
== Index some documents
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
Once you have a cluster up and running, you're ready to index some data.
There are a variety of ingest options for {es}, but in the end they all
do the same thing: put JSON documents into an {es} index.
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
You can do this directly with a simple PUT request that specifies
the index you want to add the document, a unique document ID, and one or more
2019-08-14 13:59:17 -04:00
`"field": "value"` pairs in the request body:
2014-07-23 05:54:53 -04:00
2016-09-30 13:17:35 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-14 13:59:17 -04:00
PUT /customer/_doc/1
2014-07-23 05:54:53 -04:00
{
"name": "John Doe"
2016-09-30 13:17:35 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-09-30 13:17:35 -04:00
// CONSOLE
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
This request automatically creates the `customer` index if it doesn't already
2019-08-14 13:59:17 -04:00
exist, adds a new document that has an ID of `1`, and stores and
2019-08-20 12:14:49 -04:00
indexes the `name` field.
2019-08-14 13:59:17 -04:00
Since this is a new document, the response shows that the result of the
operation was that version 1 of the document was created:
2014-04-18 13:51:59 -04:00
Enforce that responses in docs are valid json (#26249)
All of the snippets in our docs marked with `// TESTRESPONSE` are
checked against the response from Elasticsearch but, due to the
way they are implemented they are actually parsed as YAML instead
of JSON. Luckilly, all valid JSON is valid YAML! Unfurtunately
that means that invalid JSON has snuck into the exmples!
This adds a step during the build to parse them as JSON and fail
the build if they don't parse.
But no! It isn't quite that simple. The displayed text of some of
these responses looks like:
```
{
...
"aggregations": {
"range": {
"buckets": [
{
"to": 1.4436576E12,
"to_as_string": "10-2015",
"doc_count": 7,
"key": "*-10-2015"
},
{
"from": 1.4436576E12,
"from_as_string": "10-2015",
"doc_count": 0,
"key": "10-2015-*"
}
]
}
}
}
```
Note the `...` which isn't valid json but we like it anyway and want
it in the output. We use substitution rules to convert the `...`
into the response we expect. That yields a response that looks like:
```
{
"took": $body.took,"timed_out": false,"_shards": $body._shards,"hits": $body.hits,
"aggregations": {
"range": {
"buckets": [
{
"to": 1.4436576E12,
"to_as_string": "10-2015",
"doc_count": 7,
"key": "*-10-2015"
},
{
"from": 1.4436576E12,
"from_as_string": "10-2015",
"doc_count": 0,
"key": "10-2015-*"
}
]
}
}
}
```
That is what the tests consume but it isn't valid JSON! Oh no! We don't
want to go update all the substitution rules because that'd be huge and,
ultimately, wouldn't buy much. So we quote the `$body.took` bits before
parsing the JSON.
Note the responses that we use for the `_cat` APIs are all converted into
regexes and there is no expectation that they are valid JSON.
Closes #26233
2017-08-17 09:02:10 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
{
"_index" : "customer",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2014-04-18 13:51:59 -04:00
"_id" : "1",
"_version" : 1,
2016-09-16 12:15:41 -04:00
"result" : "created",
"_shards" : {
"total" : 2,
2019-08-14 13:59:17 -04:00
"successful" : 2,
2016-09-16 12:15:41 -04:00
"failed" : 0
},
2019-08-14 13:59:17 -04:00
"_seq_no" : 26,
"_primary_term" : 4
2014-04-18 13:51:59 -04:00
}
--------------------------------------------------
2019-08-14 13:59:17 -04:00
// TESTRESPONSE[s/"_seq_no" : \d+/"_seq_no" : $body._seq_no/]
// TESTRESPONSE[s/"successful" : \d+/"successful" : $body._shards.successful/]
// TESTRESPONSE[s/"_primary_term" : \d+/"_primary_term" : $body._primary_term/]
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
The new document is available immediately from any node in the cluster.
You can retrieve it with a GET request that specifies its document ID:
2014-07-23 05:54:53 -04:00
2016-09-30 13:17:35 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-14 13:59:17 -04:00
GET /customer/_doc/1
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-09-30 13:17:35 -04:00
// CONSOLE
// TEST[continued]
2014-07-23 05:54:53 -04:00
2019-08-14 13:59:17 -04:00
The response indicates that a document with the specified ID was found
and shows the original source fields that were indexed.
2014-04-18 13:51:59 -04:00
2016-09-30 13:17:35 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
{
"_index" : "customer",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2014-04-18 13:51:59 -04:00
"_id" : "1",
"_version" : 1,
2019-08-14 13:59:17 -04:00
"_seq_no" : 26,
"_primary_term" : 4,
2016-05-16 10:29:06 -04:00
"found" : true,
2019-08-14 13:59:17 -04:00
"_source" : {
"name": "John Doe"
}
2016-09-30 13:17:35 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2019-08-14 13:59:17 -04:00
// TESTRESPONSE[s/"_seq_no" : \d+/"_seq_no" : $body._seq_no/ ]
// TESTRESPONSE[s/"_primary_term" : \d+/"_primary_term" : $body._primary_term/]
2014-07-23 05:54:53 -04:00
2014-04-18 13:51:59 -04:00
[float]
2018-12-06 13:14:37 -05:00
[[getting-started-batch-processing]]
2019-08-20 12:14:49 -04:00
=== Indexing documents in bulk
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
If you have a lot of documents to index, you can submit them in batches with
the {ref}/docs-bulk.html[bulk API]. Using bulk to batch document
operations is significantly faster than submitting requests individually as it minimizes network roundtrips.
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
The optimal batch size depends a number of factors: the document size and complexity, the indexing and search load, and the resources available to your cluster. A good place to start is with batches of 1,000 to 5,000 documents
and a total payload between 5MB and 15MB. From there, you can experiment
to find the sweet spot.
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
To get some data into {es} that you can start searching and analyzing:
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
. Download the https://github.com/elastic/elasticsearch/blob/master/docs/src/test/resources/accounts.json?raw=true[`accounts.json`] sample data set. The documents in this randomly-generated data set represent user accounts with the following information:
+
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
{
"account_number": 0,
"balance": 16623,
"firstname": "Bradshaw",
"lastname": "Mckenzie",
"age": 29,
"gender": "F",
"address": "244 Columbus Place",
"employer": "Euron",
"email": "bradshawmckenzie@euron.com",
"city": "Hobucken",
"state": "CO"
}
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// NOTCONSOLE
2014-04-18 13:51:59 -04:00
2019-08-20 12:14:49 -04:00
. Index the account data into the `bank` index with the following `_bulk` request:
+
2014-04-18 13:51:59 -04:00
[source,sh]
--------------------------------------------------
2019-01-23 03:46:28 -05:00
curl -H "Content-Type: application/json" -XPOST "localhost:9200/bank/_bulk?pretty&refresh" --data-binary "@accounts.json"
2018-01-10 12:00:12 -05:00
curl "localhost:9200/_cat/indices?v"
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// NOTCONSOLE
2019-08-20 12:14:49 -04:00
+
2016-10-02 23:16:21 -04:00
////
This replicates the above in a document-testing friendly way but isn't visible
in the docs:
2019-08-20 12:14:49 -04:00
+
2016-10-02 23:16:21 -04:00
[source,js]
--------------------------------------------------
GET /_cat/indices?v
--------------------------------------------------
// CONSOLE
// TEST[setup:bank]
////
2019-08-20 12:14:49 -04:00
+
The response indicates that 1,000 documents were indexed successfully.
+
Enforce that responses in docs are valid json (#26249)
All of the snippets in our docs marked with `// TESTRESPONSE` are
checked against the response from Elasticsearch but, due to the
way they are implemented they are actually parsed as YAML instead
of JSON. Luckilly, all valid JSON is valid YAML! Unfurtunately
that means that invalid JSON has snuck into the exmples!
This adds a step during the build to parse them as JSON and fail
the build if they don't parse.
But no! It isn't quite that simple. The displayed text of some of
these responses looks like:
```
{
...
"aggregations": {
"range": {
"buckets": [
{
"to": 1.4436576E12,
"to_as_string": "10-2015",
"doc_count": 7,
"key": "*-10-2015"
},
{
"from": 1.4436576E12,
"from_as_string": "10-2015",
"doc_count": 0,
"key": "10-2015-*"
}
]
}
}
}
```
Note the `...` which isn't valid json but we like it anyway and want
it in the output. We use substitution rules to convert the `...`
into the response we expect. That yields a response that looks like:
```
{
"took": $body.took,"timed_out": false,"_shards": $body._shards,"hits": $body.hits,
"aggregations": {
"range": {
"buckets": [
{
"to": 1.4436576E12,
"to_as_string": "10-2015",
"doc_count": 7,
"key": "*-10-2015"
},
{
"from": 1.4436576E12,
"from_as_string": "10-2015",
"doc_count": 0,
"key": "10-2015-*"
}
]
}
}
}
```
That is what the tests consume but it isn't valid JSON! Oh no! We don't
want to go update all the substitution rules because that'd be huge and,
ultimately, wouldn't buy much. So we quote the `$body.took` bits before
parsing the JSON.
Note the responses that we use for the `_cat` APIs are all converted into
regexes and there is no expectation that they are valid JSON.
Closes #26233
2017-08-17 09:02:10 -04:00
[source,txt]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
health status index uuid pri rep docs.count docs.deleted store.size pri.store.size
yellow open bank l7sSYV2cQXmu6_4rJWVIww 5 1 1000 0 128.6kb 128.6kb
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-09 13:46:24 -04:00
// TESTRESPONSE[s/128.6kb/\\d+(\\.\\d+)?[mk]?b/]
2019-06-10 09:33:32 -04:00
// TESTRESPONSE[s/l7sSYV2cQXmu6_4rJWVIww/.+/ non_json]
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
[[getting-started-search]]
2019-08-16 17:54:21 -04:00
== Start searching
2014-04-18 13:51:59 -04:00
2017-07-17 18:24:31 -04:00
Now let's start with some simple searches. There are two basic ways to run searches: one is by sending search parameters through the {ref}/search-uri-request.html[REST request URI] and the other by sending them through the {ref}/search-request-body.html[REST request body]. The request body method allows you to be more expressive and also to define your searches in a more readable JSON format. We'll try one example of the request URI method but for the remainder of this tutorial, we will exclusively be using the request body method.
2014-04-18 13:51:59 -04:00
The REST API for search is accessible from the `_search` endpoint. This example returns all documents in the bank index:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-12-12 09:00:27 -05:00
GET /bank/_search?q=*&sort=account_number:asc&pretty
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2016-12-12 09:00:27 -05:00
Let's first dissect the search call. We are searching (`_search` endpoint) in the bank index, and the `q=*` parameter instructs Elasticsearch to match all documents in the index. The `sort=account_number:asc` parameter indicates to sort the results using the `account_number` field of each document in an ascending order. The `pretty` parameter, again, just tells Elasticsearch to return pretty-printed JSON results.
2014-04-18 13:51:59 -04:00
And the response (partially shown):
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
{
"took" : 63,
"timed_out" : false,
"_shards" : {
"total" : 5,
"successful" : 5,
2017-07-12 16:19:20 -04:00
"skipped" : 0,
2014-04-18 13:51:59 -04:00
"failed" : 0
},
"hits" : {
2018-12-05 13:49:06 -05:00
"total" : {
"value": 1000,
"relation": "eq"
},
2016-10-02 23:16:21 -04:00
"max_score" : null,
2014-04-18 13:51:59 -04:00
"hits" : [ {
"_index" : "bank",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2016-10-02 23:16:21 -04:00
"_id" : "0",
"sort": [0],
"_score" : null,
"_source" : {"account_number":0,"balance":16623,"firstname":"Bradshaw","lastname":"Mckenzie","age":29,"gender":"F","address":"244 Columbus Place","employer":"Euron","email":"bradshawmckenzie@euron.com","city":"Hobucken","state":"CO"}
2014-04-18 13:51:59 -04:00
}, {
"_index" : "bank",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2016-10-02 23:16:21 -04:00
"_id" : "1",
"sort": [1],
"_score" : null,
"_source" : {"account_number":1,"balance":39225,"firstname":"Amber","lastname":"Duke","age":32,"gender":"M","address":"880 Holmes Lane","employer":"Pyrami","email":"amberduke@pyrami.com","city":"Brogan","state":"IL"}
}, ...
]
}
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// TESTRESPONSE[s/"took" : 63/"took" : $body.took/]
// TESTRESPONSE[s/\.\.\./$body.hits.hits.2, $body.hits.hits.3, $body.hits.hits.4, $body.hits.hits.5, $body.hits.hits.6, $body.hits.hits.7, $body.hits.hits.8, $body.hits.hits.9/]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
For example, the following request retrieves all documents in the `bank`
index sorted by account number:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
2016-10-02 23:16:21 -04:00
"query": { "match_all": {} },
"sort": [
{ "account_number": "asc" }
]
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2018-04-03 08:57:42 -04:00
The difference here is that instead of passing `q=*` in the URI, we provide a JSON-style query request body to the `_search` API. We'll discuss this JSON query in the next section.
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
////
Hidden response just so we can assert that it is indeed the same but don't have
to clutter the docs with it:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
{
2016-10-02 23:16:21 -04:00
"took" : 63,
2014-04-18 13:51:59 -04:00
"timed_out" : false,
"_shards" : {
"total" : 5,
"successful" : 5,
2017-07-12 16:19:20 -04:00
"skipped" : 0,
2014-04-18 13:51:59 -04:00
"failed" : 0
},
"hits" : {
2018-12-05 13:49:06 -05:00
"total" : {
"value": 1000,
"relation": "eq"
},
2016-10-02 23:16:21 -04:00
"max_score": null,
2014-04-18 13:51:59 -04:00
"hits" : [ {
"_index" : "bank",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2016-10-02 23:16:21 -04:00
"_id" : "0",
"sort": [0],
"_score": null,
"_source" : {"account_number":0,"balance":16623,"firstname":"Bradshaw","lastname":"Mckenzie","age":29,"gender":"F","address":"244 Columbus Place","employer":"Euron","email":"bradshawmckenzie@euron.com","city":"Hobucken","state":"CO"}
2014-04-18 13:51:59 -04:00
}, {
"_index" : "bank",
2017-12-14 11:47:53 -05:00
"_type" : "_doc",
2016-10-02 23:16:21 -04:00
"_id" : "1",
"sort": [1],
"_score": null,
"_source" : {"account_number":1,"balance":39225,"firstname":"Amber","lastname":"Duke","age":32,"gender":"M","address":"880 Holmes Lane","employer":"Pyrami","email":"amberduke@pyrami.com","city":"Brogan","state":"IL"}
}, ...
]
}
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// TESTRESPONSE[s/"took" : 63/"took" : $body.took/]
// TESTRESPONSE[s/\.\.\./$body.hits.hits.2, $body.hits.hits.3, $body.hits.hits.4, $body.hits.hits.5, $body.hits.hits.6, $body.hits.hits.7, $body.hits.hits.8, $body.hits.hits.9/]
////
2014-04-18 13:51:59 -04:00
It is important to understand that once you get your search results back, Elasticsearch is completely done with the request and does not maintain any kind of server-side resources or open cursors into your results. This is in stark contrast to many other platforms such as SQL wherein you may initially get a partial subset of your query results up-front and then you have to continuously go back to the server if you want to fetch (or page through) the rest of the results using some kind of stateful server-side cursor.
2019-08-14 13:59:17 -04:00
[float]
2018-12-06 13:14:37 -05:00
[[getting-started-query-lang]]
2019-08-16 17:54:21 -04:00
=== Introducing the Query Language
2014-04-18 13:51:59 -04:00
2017-07-17 18:24:31 -04:00
Elasticsearch provides a JSON-style domain-specific language that you can use to execute queries. This is referred to as the {ref}/query-dsl.html[Query DSL]. The query language is quite comprehensive and can be intimidating at first glance but the best way to actually learn it is to start with a few basic examples.
2014-04-18 13:51:59 -04:00
Going back to our last example, we executed this query:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": { "match_all": {} }
}
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
Dissecting the above, the `query` part tells us what our query definition is and the `match_all` part is simply the type of query that we want to run. The `match_all` query is simply a search for all documents in the specified index.
2016-10-02 23:16:21 -04:00
In addition to the `query` parameter, we also can pass other parameters to
influence the search results. In the example in the section above we passed in
`sort`, here we pass in `size`:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": { "match_all": {} },
"size": 1
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
Each search request is self-contained: {es} does not maintain any
state information across requests. To page through the search hits, specify
the `from` and `size` parameters in your request.
2014-04-18 13:51:59 -04:00
2018-01-25 05:35:50 -05:00
This example does a `match_all` and returns documents 10 through 19:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": { "match_all": {} },
"from": 10,
"size": 10
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
The `from` parameter (0-based) specifies which document index to start from and the `size` parameter specifies how many documents to return starting at the from parameter. This feature is useful when implementing paging of search results. Note that if `from` is not specified, it defaults to 0.
2019-08-26 21:24:05 -04:00
To search for specific terms within a field, you can use a `match` query.
For example, the following request searches the `address` field to find
customers whose addresses contain `mill` or `lane`:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": { "match": { "address": "mill lane" } }
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
This example is a variant of `match` (`match_phrase`) that returns all accounts containing the phrase "mill lane" in the address:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": { "match_phrase": { "address": "mill lane" } }
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2017-07-17 18:24:31 -04:00
Let's now introduce the {ref}/query-dsl-bool-query.html[`bool` query]. The `bool` query allows us to compose smaller queries into bigger queries using boolean logic.
2014-04-18 13:51:59 -04:00
This example composes two `match` queries and returns all accounts containing "mill" and "lane" in the address:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": {
"bool": {
"must": [
{ "match": { "address": "mill" } },
{ "match": { "address": "lane" } }
]
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
In the above example, the `bool must` clause specifies all the queries that must be true for a document to be considered a match.
In contrast, this example composes two `match` queries and returns all accounts containing "mill" or "lane" in the address:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": {
"bool": {
"should": [
{ "match": { "address": "mill" } },
{ "match": { "address": "lane" } }
]
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
In the above example, the `bool should` clause specifies a list of queries either of which must be true for a document to be considered a match.
This example composes two `match` queries and returns all accounts that contain neither "mill" nor "lane" in the address:
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": {
"bool": {
"must_not": [
{ "match": { "address": "mill" } },
{ "match": { "address": "lane" } }
]
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
In the above example, the `bool must_not` clause specifies a list of queries none of which must be true for a document to be considered a match.
We can combine `must`, `should`, and `must_not` clauses simultaneously inside a `bool` query. Furthermore, we can compose `bool` queries inside any of these `bool` clauses to mimic any complex multi-level boolean logic.
2017-01-20 12:59:43 -05:00
This example returns all accounts of anybody who is 40 years old but doesn't live in ID(aho):
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": {
"bool": {
"must": [
{ "match": { "age": "40" } }
],
"must_not": [
{ "match": { "state": "ID" } }
]
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
[float]
2018-12-06 13:14:37 -05:00
[[getting-started-filters]]
2019-08-16 17:54:21 -04:00
=== Executing filters
2014-04-18 13:51:59 -04:00
In the previous section, we skipped over a little detail called the document score (`_score` field in the search results). The score is a numeric value that is a relative measure of how well the document matches the search query that we specified. The higher the score, the more relevant the document is, the lower the score, the less relevant the document is.
2015-08-29 07:15:11 -04:00
But queries do not always need to produce scores, in particular when they are only used for "filtering" the document set. Elasticsearch detects these situations and automatically optimizes query execution in order not to compute useless scores.
2014-04-18 13:51:59 -04:00
2018-08-20 02:54:03 -04:00
The {ref}/query-dsl-bool-query.html[`bool` query] that we introduced in the previous section also supports `filter` clauses which allow us to use a query to restrict the documents that will be matched by other clauses, without changing how scores are computed. As an example, let's introduce the {ref}/query-dsl-range-query.html[`range` query], which allows us to filter documents by a range of values. This is generally used for numeric or date filtering.
2014-04-18 13:51:59 -04:00
2015-09-11 04:35:56 -04:00
This example uses a bool query to return all accounts with balances between 20000 and 30000, inclusive. In other words, we want to find accounts with a balance that is greater than or equal to 20000 and less than or equal to 30000.
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"query": {
2015-09-11 04:35:56 -04:00
"bool": {
"must": { "match_all": {} },
2014-04-18 13:51:59 -04:00
"filter": {
"range": {
"balance": {
"gte": 20000,
"lte": 30000
}
}
}
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2015-09-11 04:35:56 -04:00
Dissecting the above, the bool query contains a `match_all` query (the query part) and a `range` query (the filter part). We can substitute any other queries into the query and the filter parts. In the above case, the range query makes perfect sense since documents falling into the range all match "equally", i.e., no document is more relevant than another.
2014-04-18 13:51:59 -04:00
2015-09-11 04:35:56 -04:00
In addition to the `match_all`, `match`, `bool`, and `range` queries, there are a lot of other query types that are available and we won't go into them here. Since we already have a basic understanding of how they work, it shouldn't be too difficult to apply this knowledge in learning and experimenting with the other query types.
2014-04-18 13:51:59 -04:00
2018-12-06 13:14:37 -05:00
[[getting-started-aggregations]]
2019-08-16 17:54:21 -04:00
== Analyze results with aggregations
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
{es} aggregations enable you to get meta-information about your search results
and answer questions like, "How many account holders are in Texas?" or
"What's the average balance of accounts in Tennessee?" You can search
documents, filter hits, and use aggregations to analyze the results all in one
request.
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
For example, the following request uses a `terms` aggregation to group
all of the accounts in the `bank` index by state, and returns the ten states
with the most accounts in descending order:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"size": 0,
"aggs": {
"group_by_state": {
"terms": {
2016-04-22 19:34:15 -04:00
"field": "state.keyword"
2014-04-18 13:51:59 -04:00
}
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
The `buckets` in the response are the values of the `state` field. The
`doc_count` shows the number of accounts in each state. For example, you
can see that there are 27 accounts in `ID` (Idaho). Because the request
set `size=0`, the response only contains the aggregation results.
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
{
"took": 29,
"timed_out": false,
"_shards": {
"total": 5,
"successful": 5,
2017-07-12 16:19:20 -04:00
"skipped" : 0,
2016-10-02 23:16:21 -04:00
"failed": 0
},
2014-04-18 13:51:59 -04:00
"hits" : {
2018-12-05 13:49:06 -05:00
"total" : {
"value": 1000,
"relation": "eq"
},
2018-08-22 11:23:54 -04:00
"max_score" : null,
2014-04-18 13:51:59 -04:00
"hits" : [ ]
},
"aggregations" : {
"group_by_state" : {
2016-10-02 23:16:21 -04:00
"doc_count_error_upper_bound": 20,
"sum_other_doc_count": 770,
2014-04-18 13:51:59 -04:00
"buckets" : [ {
2016-10-02 23:16:21 -04:00
"key" : "ID",
"doc_count" : 27
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "TX",
"doc_count" : 27
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "AL",
"doc_count" : 25
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "MD",
"doc_count" : 25
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "TN",
"doc_count" : 23
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "MA",
"doc_count" : 21
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "NC",
"doc_count" : 21
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "ND",
"doc_count" : 21
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "ME",
"doc_count" : 20
2014-04-18 13:51:59 -04:00
}, {
2016-10-02 23:16:21 -04:00
"key" : "MO",
"doc_count" : 20
2014-04-18 13:51:59 -04:00
} ]
}
}
}
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// TESTRESPONSE[s/"took": 29/"took": $body.took/]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
You can combine aggregations to build more complex summaries of your data. For
example, the following request nests an `avg` aggregation within the previous
`group_by_state` aggregation to calculate the average account balances for
each state.
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"size": 0,
"aggs": {
"group_by_state": {
"terms": {
2016-04-22 19:34:15 -04:00
"field": "state.keyword"
2014-04-18 13:51:59 -04:00
},
"aggs": {
"average_balance": {
"avg": {
"field": "balance"
}
}
}
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
Instead of sorting the results by count, you could sort using the result of
the nested aggregation by specifying the order within the `terms` aggregation:
2014-04-18 13:51:59 -04:00
2016-10-02 23:16:21 -04:00
[source,js]
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
GET /bank/_search
2014-04-18 13:51:59 -04:00
{
"size": 0,
"aggs": {
"group_by_state": {
"terms": {
2016-04-22 19:34:15 -04:00
"field": "state.keyword",
2014-04-18 13:51:59 -04:00
"order": {
"average_balance": "desc"
}
},
"aggs": {
"average_balance": {
"avg": {
"field": "balance"
}
}
}
}
}
2016-10-02 23:16:21 -04:00
}
2014-04-18 13:51:59 -04:00
--------------------------------------------------
2016-10-02 23:16:21 -04:00
// CONSOLE
// TEST[continued]
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
In addition to basic bucketing and metrics aggregations like these, {es}
provides specialized aggregations for operating on multiple fields and
analyzing particular types of data such as dates, IP addresses, and geo
data. You can also feed the results of individual aggregations into pipeline
aggregations for further analysis.
2014-04-18 13:51:59 -04:00
2019-08-26 21:24:05 -04:00
The core analysis capabilities provided by aggregations enable advanced
features such as using machine learning to detect anomalies.
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
[[getting-started-next-steps]]
2019-08-16 17:54:21 -04:00
== Where to go from here
2019-08-14 13:59:17 -04:00
Now that you've set up a cluster, indexed some documents, and run some
searches and aggregations, you might want to:
* {stack-gs}/get-started-elastic-stack.html#install-kibana[Dive in to the Elastic
Stack Tutorial] to install Kibana, Logstash, and Beats and
set up a basic system monitoring solution.
* {kibana-ref}/add-sample-data.html[Load one of the sample data sets into Kibana]
to see how you can use {es} and Kibana together to visualize your data.
2014-04-18 13:51:59 -04:00
2019-08-14 13:59:17 -04:00
* Try out one of the Elastic search solutions:
** https://swiftype.com/documentation/site-search/crawler-quick-start[Site Search]
** https://swiftype.com/documentation/app-search/getting-started[App Search]
** https://swiftype.com/documentation/enterprise-search/getting-started[Enterprise Search]