[Docs] Adding aggregation sections to high level client docs (#25707)
This adds a section about how to add aggregations to the SearchSourceBuilder and how to retrieve them from a SearchRepsonse to the documentation for the high level rest client.
This commit is contained in:
parent
4f0dc5bf32
commit
f809a12493
|
@ -40,6 +40,14 @@ import org.elasticsearch.rest.RestStatus;
|
||||||
import org.elasticsearch.search.Scroll;
|
import org.elasticsearch.search.Scroll;
|
||||||
import org.elasticsearch.search.SearchHit;
|
import org.elasticsearch.search.SearchHit;
|
||||||
import org.elasticsearch.search.SearchHits;
|
import org.elasticsearch.search.SearchHits;
|
||||||
|
import org.elasticsearch.search.aggregations.Aggregation;
|
||||||
|
import org.elasticsearch.search.aggregations.AggregationBuilders;
|
||||||
|
import org.elasticsearch.search.aggregations.Aggregations;
|
||||||
|
import org.elasticsearch.search.aggregations.bucket.range.Range;
|
||||||
|
import org.elasticsearch.search.aggregations.bucket.terms.Terms;
|
||||||
|
import org.elasticsearch.search.aggregations.bucket.terms.Terms.Bucket;
|
||||||
|
import org.elasticsearch.search.aggregations.bucket.terms.TermsAggregationBuilder;
|
||||||
|
import org.elasticsearch.search.aggregations.metrics.avg.Avg;
|
||||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||||
import org.elasticsearch.search.sort.ScoreSortBuilder;
|
import org.elasticsearch.search.sort.ScoreSortBuilder;
|
||||||
import org.elasticsearch.search.sort.SortOrder;
|
import org.elasticsearch.search.sort.SortOrder;
|
||||||
|
@ -85,7 +93,7 @@ public class SearchDocumentationIT extends ESRestHighLevelClientTestCase {
|
||||||
request.add(new IndexRequest("posts", "doc", "3")
|
request.add(new IndexRequest("posts", "doc", "3")
|
||||||
.source(XContentType.JSON, "title", "The Future of Federated Search in Elasticsearch", "user",
|
.source(XContentType.JSON, "title", "The Future of Federated Search in Elasticsearch", "user",
|
||||||
Arrays.asList("kimchy", "tanguy"), "innerObject", Collections.singletonMap("key", "value")));
|
Arrays.asList("kimchy", "tanguy"), "innerObject", Collections.singletonMap("key", "value")));
|
||||||
request.setRefreshPolicy(WriteRequest.RefreshPolicy.WAIT_UNTIL);
|
request.setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE);
|
||||||
BulkResponse bulkResponse = client.bulk(request);
|
BulkResponse bulkResponse = client.bulk(request);
|
||||||
assertSame(bulkResponse.status(), RestStatus.OK);
|
assertSame(bulkResponse.status(), RestStatus.OK);
|
||||||
assertFalse(bulkResponse.hasFailures());
|
assertFalse(bulkResponse.hasFailures());
|
||||||
|
@ -99,8 +107,8 @@ public class SearchDocumentationIT extends ESRestHighLevelClientTestCase {
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
// tag::search-request-indices-types
|
// tag::search-request-indices-types
|
||||||
SearchRequest searchRequest = new SearchRequest("posts");
|
SearchRequest searchRequest = new SearchRequest("posts"); // <1>
|
||||||
searchRequest.types("doc");
|
searchRequest.types("doc"); // <2>
|
||||||
// end::search-request-indices-types
|
// end::search-request-indices-types
|
||||||
// tag::search-request-routing
|
// tag::search-request-routing
|
||||||
searchRequest.routing("routing"); // <1>
|
searchRequest.routing("routing"); // <1>
|
||||||
|
@ -199,6 +207,81 @@ public class SearchDocumentationIT extends ESRestHighLevelClientTestCase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings({ "unused", "unchecked" })
|
||||||
|
public void testSearchRequestAggregations() throws IOException {
|
||||||
|
RestHighLevelClient client = highLevelClient();
|
||||||
|
{
|
||||||
|
BulkRequest request = new BulkRequest();
|
||||||
|
request.add(new IndexRequest("posts", "doc", "1")
|
||||||
|
.source(XContentType.JSON, "company", "Elastic", "age", 20));
|
||||||
|
request.add(new IndexRequest("posts", "doc", "2")
|
||||||
|
.source(XContentType.JSON, "company", "Elastic", "age", 30));
|
||||||
|
request.add(new IndexRequest("posts", "doc", "3")
|
||||||
|
.source(XContentType.JSON, "company", "Elastic", "age", 40));
|
||||||
|
request.setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE);
|
||||||
|
BulkResponse bulkResponse = client.bulk(request);
|
||||||
|
assertSame(bulkResponse.status(), RestStatus.OK);
|
||||||
|
assertFalse(bulkResponse.hasFailures());
|
||||||
|
}
|
||||||
|
{
|
||||||
|
SearchRequest searchRequest = new SearchRequest();
|
||||||
|
// tag::search-request-aggregations
|
||||||
|
SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();
|
||||||
|
TermsAggregationBuilder aggregation = AggregationBuilders.terms("by_company")
|
||||||
|
.field("company.keyword");
|
||||||
|
aggregation.subAggregation(AggregationBuilders.avg("average_age")
|
||||||
|
.field("age"));
|
||||||
|
searchSourceBuilder.aggregation(aggregation);
|
||||||
|
// end::search-request-aggregations
|
||||||
|
searchSourceBuilder.query(QueryBuilders.matchAllQuery());
|
||||||
|
searchRequest.source(searchSourceBuilder);
|
||||||
|
SearchResponse searchResponse = client.search(searchRequest);
|
||||||
|
{
|
||||||
|
// tag::search-request-aggregations-get
|
||||||
|
Aggregations aggregations = searchResponse.getAggregations();
|
||||||
|
Terms byCompanyAggregation = aggregations.get("by_company"); // <1>
|
||||||
|
Bucket elasticBucket = byCompanyAggregation.getBucketByKey("Elastic"); // <2>
|
||||||
|
Avg averageAge = elasticBucket.getAggregations().get("average_age"); // <3>
|
||||||
|
double avg = averageAge.getValue();
|
||||||
|
// end::search-request-aggregations-get
|
||||||
|
|
||||||
|
try {
|
||||||
|
// tag::search-request-aggregations-get-wrongCast
|
||||||
|
Range range = aggregations.get("by_company"); // <1>
|
||||||
|
// end::search-request-aggregations-get-wrongCast
|
||||||
|
} catch (ClassCastException ex) {
|
||||||
|
assertEquals("org.elasticsearch.search.aggregations.bucket.terms.ParsedStringTerms"
|
||||||
|
+ " cannot be cast to org.elasticsearch.search.aggregations.bucket.range.Range", ex.getMessage());
|
||||||
|
}
|
||||||
|
assertEquals(3, elasticBucket.getDocCount());
|
||||||
|
assertEquals(30, avg, 0.0);
|
||||||
|
}
|
||||||
|
Aggregations aggregations = searchResponse.getAggregations();
|
||||||
|
{
|
||||||
|
// tag::search-request-aggregations-asMap
|
||||||
|
Map<String, Aggregation> aggregationMap = aggregations.getAsMap();
|
||||||
|
Terms companyAggregation = (Terms) aggregationMap.get("by_company");
|
||||||
|
// end::search-request-aggregations-asMap
|
||||||
|
}
|
||||||
|
{
|
||||||
|
// tag::search-request-aggregations-asList
|
||||||
|
List<Aggregation> aggregationList = aggregations.asList();
|
||||||
|
// end::search-request-aggregations-asList
|
||||||
|
}
|
||||||
|
{
|
||||||
|
// tag::search-request-aggregations-iterator
|
||||||
|
for (Aggregation agg : aggregations) {
|
||||||
|
String type = agg.getType();
|
||||||
|
if (type.equals(TermsAggregationBuilder.NAME)) {
|
||||||
|
Bucket elasticBucket = ((Terms) agg).getBucketByKey("Elastic");
|
||||||
|
long numberOfDocs = elasticBucket.getDocCount();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// end::search-request-aggregations-iterator
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void testScroll() throws IOException {
|
public void testScroll() throws IOException {
|
||||||
RestHighLevelClient client = highLevelClient();
|
RestHighLevelClient client = highLevelClient();
|
||||||
{
|
{
|
||||||
|
@ -209,7 +292,7 @@ public class SearchDocumentationIT extends ESRestHighLevelClientTestCase {
|
||||||
.source(XContentType.JSON, "title", "Current status and upcoming changes in Elasticsearch"));
|
.source(XContentType.JSON, "title", "Current status and upcoming changes in Elasticsearch"));
|
||||||
request.add(new IndexRequest("posts", "doc", "3")
|
request.add(new IndexRequest("posts", "doc", "3")
|
||||||
.source(XContentType.JSON, "title", "The Future of Federated Search in Elasticsearch"));
|
.source(XContentType.JSON, "title", "The Future of Federated Search in Elasticsearch"));
|
||||||
request.setRefreshPolicy(WriteRequest.RefreshPolicy.WAIT_UNTIL);
|
request.setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE);
|
||||||
BulkResponse bulkResponse = client.bulk(request);
|
BulkResponse bulkResponse = client.bulk(request);
|
||||||
assertSame(bulkResponse.status(), RestStatus.OK);
|
assertSame(bulkResponse.status(), RestStatus.OK);
|
||||||
assertFalse(bulkResponse.hasFailures());
|
assertFalse(bulkResponse.hasFailures());
|
||||||
|
|
|
@ -8,7 +8,7 @@ The `SearchRequest` is used for any operation that has to do with searching
|
||||||
documents, aggregations, suggestions and also offers ways of requesting
|
documents, aggregations, suggestions and also offers ways of requesting
|
||||||
highlighting on the resulting documents.
|
highlighting on the resulting documents.
|
||||||
|
|
||||||
In its most basic form, a query can be added to the request like this:
|
In its most basic form, we can add a query to the request:
|
||||||
|
|
||||||
["source","java",subs="attributes,callouts,macros"]
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
@ -16,21 +16,19 @@ include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-basic]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
|
||||||
<1> Creates the `SeachRequest`. Without arguments this runs against all indices.
|
<1> Creates the `SeachRequest`. Without arguments this runs against all indices.
|
||||||
<2> Most parameters of the search can be added to the `SearchSourceBuilder`
|
<2> Most search parameters are added to the `SearchSourceBuilder`. It offers setters for everything that goes into the search request body.
|
||||||
which contains everything that
|
|
||||||
in the Rest API would be placed in the search request body.
|
|
||||||
<3> Add a `match_all` query to the `SearchSourceBuilder`.
|
<3> Add a `match_all` query to the `SearchSourceBuilder`.
|
||||||
|
|
||||||
==== Optional arguments
|
===== Optional arguments
|
||||||
|
|
||||||
Lets first look at some of the optional argument of a `SearchRequest`.
|
Let's first look at some of the optional arguments of a `SearchRequest`:
|
||||||
First of all, the request can be restricted to one or more indices using the
|
|
||||||
constructor or to on or more types using a setter:
|
|
||||||
|
|
||||||
["source","java",subs="attributes,callouts,macros"]
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-indices-types]
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-indices-types]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
<1> Restricts the request to an index
|
||||||
|
<2> Limits the request to a type
|
||||||
|
|
||||||
There are a couple of other interesting optional parameters:
|
There are a couple of other interesting optional parameters:
|
||||||
|
|
||||||
|
@ -52,9 +50,9 @@ how wildcard expressions are expanded
|
||||||
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-preference]
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-preference]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
<1> Use the preference parameter e.g. to execute the search to prefer local
|
<1> Use the preference parameter e.g. to execute the search to prefer local
|
||||||
shards. The The default is to randomize across shards.
|
shards. The default is to randomize across shards.
|
||||||
|
|
||||||
==== Using the SearchSourceBuilder
|
===== Using the SearchSourceBuilder
|
||||||
|
|
||||||
Most options controlling the search behavior can be set on the
|
Most options controlling the search behavior can be set on the
|
||||||
`SearchSourceBuilder`,
|
`SearchSourceBuilder`,
|
||||||
|
@ -84,6 +82,19 @@ After this, the `SearchSourceBuilder` only needs to be added to the
|
||||||
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-source-setter]
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-source-setter]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
|
||||||
|
===== Requesting Aggregations
|
||||||
|
|
||||||
|
Aggregations can be added to the search by first creating the appropriate
|
||||||
|
`AggregationBuilder` and then setting it on the `SearchSourceBuilder`. In the
|
||||||
|
following example we create a `terms` aggregation on company names with a
|
||||||
|
sub-aggregation on the average age of employees in the company:
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations]
|
||||||
|
--------------------------------------------------
|
||||||
|
|
||||||
|
We will later see how to <<java-rest-high-retrieve-aggs,access aggregations>> in the `SearchResponse`.
|
||||||
|
|
||||||
[[java-rest-high-document-search-sync]]
|
[[java-rest-high-document-search-sync]]
|
||||||
==== Synchronous Execution
|
==== Synchronous Execution
|
||||||
|
@ -135,6 +146,9 @@ failures can also be handled by iterating over an array off
|
||||||
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-response-2]
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-response-2]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
|
||||||
|
[[java-rest-high-retrieve-searchHits]]
|
||||||
|
===== Retrieving SearchHits
|
||||||
|
|
||||||
To get access to the returned documents, we need to first get the `SearchHits`
|
To get access to the returned documents, we need to first get the `SearchHits`
|
||||||
contained in the response:
|
contained in the response:
|
||||||
|
|
||||||
|
@ -152,7 +166,7 @@ include-tagged::{doc-tests}/SearchDocumentationIT.java[search-hits-info]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
|
||||||
Nested inside the `SearchHits` are the individual search results that can
|
Nested inside the `SearchHits` are the individual search results that can
|
||||||
be iterated over like this:
|
be iterated over:
|
||||||
|
|
||||||
|
|
||||||
["source","java",subs="attributes,callouts,macros"]
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
@ -172,9 +186,59 @@ Furthermore, it lets you get back the document source, either as a simple
|
||||||
JSON-String or as a map of key/value pairs. In this map, regular fields
|
JSON-String or as a map of key/value pairs. In this map, regular fields
|
||||||
are keyed by the field name and contain the field value. Multi-valued fields are
|
are keyed by the field name and contain the field value. Multi-valued fields are
|
||||||
returned as lists of objects, nested objects as another key/value map. These
|
returned as lists of objects, nested objects as another key/value map. These
|
||||||
cases need to be case accordingly:
|
cases need to be cast accordingly:
|
||||||
|
|
||||||
["source","java",subs="attributes,callouts,macros"]
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-hits-singleHit-source]
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-hits-singleHit-source]
|
||||||
--------------------------------------------------
|
--------------------------------------------------
|
||||||
|
|
||||||
|
[[java-rest-high-retrieve-aggs]]
|
||||||
|
===== Retrieving Aggregations
|
||||||
|
|
||||||
|
Aggregations can be retrieved from the `SearchResponse` by first getting the
|
||||||
|
root of the aggregation tree, the `Aggregations` object, and then getting the
|
||||||
|
aggregation by name.
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations-get]
|
||||||
|
--------------------------------------------------
|
||||||
|
<1> Get the `by_company` terms aggregation
|
||||||
|
<2> Get the buckets that is keyed with `Elastic`
|
||||||
|
<3> Get the `average_age` sub-aggregation from that bucket
|
||||||
|
|
||||||
|
Note that if you access aggregations by name, you need to specify the
|
||||||
|
aggregation interface according to the type of aggregation you requested,
|
||||||
|
otherwise a `ClassCastException` will be thrown:
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations-get-wrongCast]
|
||||||
|
--------------------------------------------------
|
||||||
|
<1> This will throw an exception because "by_company" is a `terms` aggregation
|
||||||
|
but we try to retrieve it as a `range` aggregation
|
||||||
|
|
||||||
|
It is also possible to access all aggregations as a map that is keyed by the
|
||||||
|
aggregation name. In this case, the cast to the proper aggregation interface
|
||||||
|
needs to happen explicitly:
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations-asMap]
|
||||||
|
--------------------------------------------------
|
||||||
|
|
||||||
|
There are also getters that return all top level aggregations as a list:
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations-asList]
|
||||||
|
--------------------------------------------------
|
||||||
|
|
||||||
|
And last but not least you can iterate over all aggregations and then e.g.
|
||||||
|
decide how to further process them based on their type:
|
||||||
|
|
||||||
|
["source","java",subs="attributes,callouts,macros"]
|
||||||
|
--------------------------------------------------
|
||||||
|
include-tagged::{doc-tests}/SearchDocumentationIT.java[search-request-aggregations-iterator]
|
||||||
|
--------------------------------------------------
|
Loading…
Reference in New Issue