From d662f36cf4818740fdcb03707d1207d8f2c45546 Mon Sep 17 00:00:00 2001 From: Igor Motov Date: Thu, 11 Feb 2021 15:46:53 -1000 Subject: [PATCH] [7.x] Revert "Remove aggregation's postCollect phase (#68942) This partially reverts #64016 and and adds #67839 and adds additional tests that would have caught issues with the changes in #64016. It's mostly Nik's code, I am just cleaning things up a bit. Co-authored-by: Nik Everett nik9000@gmail.com --- docs/reference/search/profile.asciidoc | 12 +- .../aggregations/ParentJoinAggregator.java | 14 +- .../rest-api-spec/test/50_order_by.yml | 238 ++++++++++++++++++ .../170_cardinality_metric.yml | 3 + .../test/search.aggregation/20_terms.yml | 133 ++++++++++ .../action/search/TransportSearchIT.java | 3 + .../aggregation/AggregationProfilerIT.java | 15 ++ .../aggregations/AdaptingAggregator.java | 5 + .../search/aggregations/AggregationPhase.java | 1 + .../search/aggregations/Aggregator.java | 2 +- .../search/aggregations/AggregatorBase.java | 23 ++ .../search/aggregations/BucketCollector.java | 10 + .../aggregations/MultiBucketCollector.java | 7 + .../bucket/BestBucketsDeferringCollector.java | 12 +- .../bucket/BucketsAggregator.java | 2 +- .../bucket/DeferringBucketCollector.java | 6 + .../bucket/composite/CompositeAggregator.java | 7 +- .../bucket/nested/NestedAggregator.java | 6 +- .../sampler/BestDocsDeferringCollector.java | 9 +- .../GlobalOrdinalsStringTermsAggregator.java | 3 +- .../metrics/CardinalityAggregator.java | 17 +- .../GlobalOrdCardinalityAggregator.java | 24 +- .../metrics/MetricsAggregator.java | 17 +- .../aggregation/AggregationTimingType.java | 1 + .../aggregation/ProfilingAggregator.java | 11 + .../MultiBucketCollectorTests.java | 9 + .../BestBucketsDeferringCollectorTests.java | 16 ++ .../bucket/GlobalAggregatorTests.java | 1 + .../geogrid/GeoGridAggregatorTestCase.java | 1 + .../BestDocsDeferringCollectorTests.java | 6 + .../terms/RareTermsAggregatorTests.java | 1 + .../bucket/terms/TermsAggregatorTests.java | 116 +++++++++ .../metrics/AvgAggregatorTests.java | 5 + .../HDRPercentilesAggregatorTests.java | 1 + .../metrics/MaxAggregatorTests.java | 9 + .../TDigestPercentilesAggregatorTests.java | 1 + .../metrics/WeightedAvgAggregatorTests.java | 1 + .../aggregations/AggregatorTestCase.java | 2 + ...eAggregatedPercentilesAggregatorTests.java | 1 + ...eAggregatedPercentilesAggregatorTests.java | 1 + .../StringStatsAggregatorTests.java | 1 + .../RollupResponseTranslationTests.java | 1 + .../xpack/rollup/job/IndexerUtilsTests.java | 6 + .../geogrid/GeoShapeGeoGridTestCase.java | 2 + 44 files changed, 703 insertions(+), 59 deletions(-) create mode 100644 modules/parent-join/src/yamlRestTest/resources/rest-api-spec/test/50_order_by.yml diff --git a/docs/reference/search/profile.asciidoc b/docs/reference/search/profile.asciidoc index 72e1bd79e94f5..548a6a71f7682 100644 --- a/docs/reference/search/profile.asciidoc +++ b/docs/reference/search/profile.asciidoc @@ -799,7 +799,9 @@ This yields the following aggregation profile output: "collect": 45786, "collect_count": 4, "build_leaf_collector": 18211, - "build_leaf_collector_count": 1 + "build_leaf_collector_count": 1, + "post_collection": 929, + "post_collection_count": 1 }, "debug": { "total_buckets": 1, @@ -820,7 +822,9 @@ This yields the following aggregation profile output: "collect": 69401, "collect_count": 4, "build_leaf_collector": 8150, - "build_leaf_collector_count": 1 + "build_leaf_collector_count": 1, + "post_collection": 1584, + "post_collection_count": 1 }, "children": [ { @@ -837,7 +841,9 @@ This yields the following aggregation profile output: "collect": 61611, "collect_count": 4, "build_leaf_collector": 5564, - "build_leaf_collector_count": 1 + "build_leaf_collector_count": 1, + "post_collection": 471, + "post_collection_count": 1 }, "debug": { "total_buckets": 1, diff --git a/modules/parent-join/src/main/java/org/elasticsearch/join/aggregations/ParentJoinAggregator.java b/modules/parent-join/src/main/java/org/elasticsearch/join/aggregations/ParentJoinAggregator.java index dc62c3f017b84..8439184ffca50 100644 --- a/modules/parent-join/src/main/java/org/elasticsearch/join/aggregations/ParentJoinAggregator.java +++ b/modules/parent-join/src/main/java/org/elasticsearch/join/aggregations/ParentJoinAggregator.java @@ -102,7 +102,12 @@ public void collect(int docId, long owningBucketOrd) throws IOException { } @Override - protected void prepareSubAggs(long[] bucketOrdsToCollect) throws IOException { + public void postCollection() throws IOException { + // Delaying until beforeBuildingBuckets + } + + @Override + protected void prepareSubAggs(long[] ordsToCollect) throws IOException { IndexReader indexReader = searcher().getIndexReader(); for (LeafReaderContext ctx : indexReader.leaves()) { Scorer childDocsScorer = outFilter.scorer(ctx); @@ -144,13 +149,14 @@ public int docID() { * structure that maps a primitive long to a list of primitive * longs. */ - for (long o: bucketOrdsToCollect) { - if (collectionStrategy.exists(o, globalOrdinal)) { - collectBucket(sub, docId, o); + for (long owningBucketOrd: ordsToCollect) { + if (collectionStrategy.exists(owningBucketOrd, globalOrdinal)) { + collectBucket(sub, docId, owningBucketOrd); } } } } + super.postCollection(); // Run post collection after collecting the sub-aggs } @Override diff --git a/modules/parent-join/src/yamlRestTest/resources/rest-api-spec/test/50_order_by.yml b/modules/parent-join/src/yamlRestTest/resources/rest-api-spec/test/50_order_by.yml new file mode 100644 index 0000000000000..f8b607de8c432 --- /dev/null +++ b/modules/parent-join/src/yamlRestTest/resources/rest-api-spec/test/50_order_by.yml @@ -0,0 +1,238 @@ +--- +"order by sub agg containing join": + - skip: + reason: "It was fixed it 7.11.2" + version: " - 7.11.2" + - do: + indices.create: + index: test_1 + body: + settings: + number_of_shards: 1 + number_of_replicas: 0 + mappings: + properties: + name: + type: keyword + join: + type: join + relations: + animal: feature + + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {"_id": "sheep"} } + { "name": "sheep", "join": {"name": "animal"} } + { "index": {"_id": "cow"} } + { "name": "cow", "join": {"name": "animal"} } + { "index": {"_id": "pig"} } + { "name": "pig", "join": {"name": "animal"} } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "danger", "number": 1 } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "fluffiness", "number": 10 } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "danger", "number": 3 } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "fluffiness", "number": 1 } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "danger", "number": 100 } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "fluffiness", "number": 1 } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + name: + terms: + size: 1 + shard_size: 1 + field: name + order: + - "children>max_number": desc + aggs: + children: + children: + type: feature + aggs: + max_number: + max: + field: number + - length: { aggregations.name.buckets: 1 } + - match: { aggregations.name.buckets.0.key: pig } + - match: { aggregations.name.buckets.0.doc_count: 1 } + - match: { aggregations.name.buckets.0.children.max_number.value: 100.0 } + +--- +"order by sub agg containing join and nested": + - skip: + reason: "It was fixed it 7.11.2" + version: " - 7.11.2" + - do: + indices.create: + index: test_1 + body: + settings: + number_of_shards: 1 + number_of_replicas: 0 + mappings: + properties: + name: + type: keyword + join: + type: join + relations: + animal: feature + nested: + type: nested + properties: + number: + type: long + + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {"_id": "sheep"} } + { "name": "sheep", "join": {"name": "animal"} } + { "index": {"_id": "cow"} } + { "name": "cow", "join": {"name": "animal"} } + { "index": {"_id": "pig"} } + { "name": "pig", "join": {"name": "animal"} } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "danger", "nested": [{"number": 1}] } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "fluffiness", "nested": [{"number": 10}] } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "danger", "nested": [{"number": 3}] } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "fluffiness", "nested": [{"number": 1}] } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "danger", "nested": [{"number": 100}, {"number": 1}] } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "fluffiness", "nested": [{"number": 1}] } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + name: + terms: + size: 1 + shard_size: 1 + field: name + order: + - "children>nested>max_number": desc + aggs: + children: + children: + type: feature + aggs: + nested: + nested: + path: nested + aggs: + max_number: + max: + field: nested.number + - length: { aggregations.name.buckets: 1 } + - match: { aggregations.name.buckets.0.key: pig } + - match: { aggregations.name.buckets.0.doc_count: 1 } + - match: { aggregations.name.buckets.0.children.nested.max_number.value: 100.0 } + +--- +"order by sub agg containing join and nested and filter": + - skip: + reason: "It was fixed it 7.11.2" + version: " - 7.11.2" + - do: + indices.create: + index: test_1 + body: + settings: + number_of_shards: 1 + number_of_replicas: 0 + mappings: + properties: + name: + type: keyword + join: + type: join + relations: + animal: feature + nested: + type: nested + properties: + code: + type: + keyword + number: + type: long + + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {"_id": "sheep"} } + { "name": "sheep", "join": {"name": "animal"} } + { "index": {"_id": "cow"} } + { "name": "cow", "join": {"name": "animal"} } + { "index": {"_id": "pig"} } + { "name": "pig", "join": {"name": "animal"} } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "danger", "nested": [{"code": "mighty", "number": 1}, {"code": "meek", "number": 100}] } + { "index": {"routing": "sheep"} } + { "join": {"name": "feature", "parent": "sheep"}, "tag": "fluffiness", "nested": [{"code": "mighty", "number": 10}, {"code": "meek", "number": 10}] } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "danger", "nested": [{"code": "mighty", "number": 3}, {"code": "meek", "number": 3}] } + { "index": {"routing": "cow"} } + { "join": {"name": "feature", "parent": "cow"}, "tag": "fluffiness", "nested": [{"code": "mighty", "number": 1}, {"code": "meek", "number": 1}] } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "danger", "nested": [{"code": "mighty", "number": 100}, {"code": "meek", "number": 1}] } + { "index": {"routing": "pig"} } + { "join": {"name": "feature", "parent": "pig"}, "tag": "fluffiness", "nested": [{"code": "mighty", "number": 1}, {"code": "meek", "number": 1}] } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + name: + terms: + size: 1 + shard_size: 1 + field: name + order: + - "children>nested>filter>max_number": desc + aggs: + children: + children: + type: feature + aggs: + nested: + nested: + path: nested + aggs: + filter: + filter: + match: + nested.code: mighty + aggs: + max_number: + max: + field: nested.number + - length: { aggregations.name.buckets: 1 } + - match: { aggregations.name.buckets.0.key: pig } + - match: { aggregations.name.buckets.0.doc_count: 1 } + - match: { aggregations.name.buckets.0.children.nested.filter.max_number.value: 100.0 } diff --git a/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/170_cardinality_metric.yml b/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/170_cardinality_metric.yml index 61d67ddf655fc..de5b9b9b014a0 100644 --- a/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/170_cardinality_metric.yml +++ b/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/170_cardinality_metric.yml @@ -232,6 +232,7 @@ setup: - gt: { profile.shards.0.aggregations.0.breakdown.build_leaf_collector: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.collect: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.build_aggregation: 0 } + - gt: { profile.shards.0.aggregations.0.breakdown.post_collection: 0 } - match: { profile.shards.0.aggregations.0.debug.empty_collectors_used: 0 } - gt: { profile.shards.0.aggregations.0.debug.numeric_collectors_used: 0 } - match: { profile.shards.0.aggregations.0.debug.ordinals_collectors_used: 0 } @@ -257,6 +258,7 @@ setup: - gt: { profile.shards.0.aggregations.0.breakdown.build_leaf_collector: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.collect: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.build_aggregation: 0 } + - gt: { profile.shards.0.aggregations.0.breakdown.post_collection: 0 } - match: { profile.shards.0.aggregations.0.debug.empty_collectors_used: 0 } - gt: { profile.shards.0.aggregations.0.debug.numeric_collectors_used: 0 } - match: { profile.shards.0.aggregations.0.debug.ordinals_collectors_used: 0 } @@ -283,3 +285,4 @@ setup: - gt: { profile.shards.0.aggregations.0.breakdown.build_leaf_collector: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.collect: 0 } - gt: { profile.shards.0.aggregations.0.breakdown.build_aggregation: 0 } + - gt: { profile.shards.0.aggregations.0.breakdown.post_collection: 0 } diff --git a/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/20_terms.yml b/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/20_terms.yml index 287c144f4e262..6703975eeaa00 100644 --- a/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/20_terms.yml +++ b/rest-api-spec/src/main/resources/rest-api-spec/test/search.aggregation/20_terms.yml @@ -22,6 +22,13 @@ setup: type: long date: type: date + nested: + type: nested + properties: + tag: + type: keyword + number: + type: long - do: indices.create: @@ -1128,3 +1135,129 @@ setup: - match: { aggregations.str_terms.buckets.2.key: pig } - match: { aggregations.str_terms.buckets.2.doc_count: 0 } - match: { aggregations.str_terms.buckets.2.max_number.value: null } + +--- +"order by sub agg": + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {} } + { "str": "sheep", "number": 1 } + { "index": {} } + { "str": "sheep", "number": 3 } + { "index": {} } + { "str": "cow", "number": 7 } + { "index": {} } + { "str": "pig", "number": 1 } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + str_terms: + terms: + size: 1 + shard_size: 1 + field: str + order: + - max_number: desc + aggs: + max_number: + max: + field: number + - length: { aggregations.str_terms.buckets: 1 } + - match: { aggregations.str_terms.buckets.0.key: cow } + - match: { aggregations.str_terms.buckets.0.doc_count: 1 } + - match: { aggregations.str_terms.buckets.0.max_number.value: 7.0 } + +--- +"order by sub agg containing nested": + - skip: + version: " - 7.11.2" + reason: "It was fixed it 7.11.2" + + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {} } + { "str": "sheep", "nested": [{"tag": "danger", "number": 1}, {"tag": "fluffiness", "number": 10}] } + { "index": {} } + { "str": "cow", "nested": [{"tag": "danger", "number": 3}, {"tag": "fluffiness", "number": 1}] } + { "index": {} } + { "str": "pig", "nested": [{"tag": "danger", "number": 100}, {"tag": "fluffiness", "number": 1}] } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + str_terms: + terms: + size: 1 + shard_size: 1 + field: str + order: + - "nested>max_number": desc + aggs: + nested: + nested: + path: nested + aggs: + max_number: + max: + field: nested.number + - length: { aggregations.str_terms.buckets: 1 } + - match: { aggregations.str_terms.buckets.0.key: pig } + - match: { aggregations.str_terms.buckets.0.doc_count: 1 } + - match: { aggregations.str_terms.buckets.0.nested.max_number.value: 100.0 } + +--- +"order by sub agg containing filter": + - do: + bulk: + index: test_1 + refresh: true + body: | + { "index": {} } + { "str": "sheep", "number": 1 } + { "index": {} } + { "str": "sheep", "number": 3 } + { "index": {} } + { "str": "cow", "number": 7 } + { "index": {} } + { "str": "pig", "number": 100 } + + - do: + search: + index: test_1 + body: + size: 0 + aggs: + str_terms: + terms: + size: 1 + shard_size: 1 + field: str + order: + - "filter>max_number": desc + aggs: + filter: + filter: + range: + number: + lt: 10 + aggs: + max_number: + max: + field: number + - length: { aggregations.str_terms.buckets: 1 } + - match: { aggregations.str_terms.buckets.0.key: cow } + - match: { aggregations.str_terms.buckets.0.doc_count: 1 } + - match: { aggregations.str_terms.buckets.0.filter.max_number.value: 7.0 } diff --git a/server/src/internalClusterTest/java/org/elasticsearch/action/search/TransportSearchIT.java b/server/src/internalClusterTest/java/org/elasticsearch/action/search/TransportSearchIT.java index b52b4d7dc6b72..90df861a977a9 100644 --- a/server/src/internalClusterTest/java/org/elasticsearch/action/search/TransportSearchIT.java +++ b/server/src/internalClusterTest/java/org/elasticsearch/action/search/TransportSearchIT.java @@ -584,6 +584,9 @@ public ScoreMode scoreMode() { @Override public void preCollection() throws IOException {} + @Override + public void postCollection() throws IOException {} + @Override public Aggregator[] subAggregators() { throw new UnsupportedOperationException(); diff --git a/server/src/internalClusterTest/java/org/elasticsearch/search/profile/aggregation/AggregationProfilerIT.java b/server/src/internalClusterTest/java/org/elasticsearch/search/profile/aggregation/AggregationProfilerIT.java index 3305283b8bb38..2758e6d30afaf 100644 --- a/server/src/internalClusterTest/java/org/elasticsearch/search/profile/aggregation/AggregationProfilerIT.java +++ b/server/src/internalClusterTest/java/org/elasticsearch/search/profile/aggregation/AggregationProfilerIT.java @@ -47,6 +47,7 @@ public class AggregationProfilerIT extends ESIntegTestCase { private static final String BUILD_LEAF_COLLECTOR = AggregationTimingType.BUILD_LEAF_COLLECTOR.toString(); private static final String COLLECT = AggregationTimingType.COLLECT.toString(); + private static final String POST_COLLECTION = AggregationTimingType.POST_COLLECTION.toString(); private static final String INITIALIZE = AggregationTimingType.INITIALIZE.toString(); private static final String BUILD_AGGREGATION = AggregationTimingType.BUILD_AGGREGATION.toString(); private static final String REDUCE = AggregationTimingType.REDUCE.toString(); @@ -54,11 +55,13 @@ public class AggregationProfilerIT extends ESIntegTestCase { INITIALIZE, BUILD_LEAF_COLLECTOR, COLLECT, + POST_COLLECTION, BUILD_AGGREGATION, REDUCE, INITIALIZE + "_count", BUILD_LEAF_COLLECTOR + "_count", COLLECT + "_count", + POST_COLLECTION + "_count", BUILD_AGGREGATION + "_count", REDUCE + "_count" ); @@ -322,6 +325,7 @@ public void testDiversifiedAggProfile() { assertThat(diversifyBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(diversifyBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(diversifyBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(diversifyBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(diversifyBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(diversifyBreakdown.get(REDUCE), equalTo(0L)); assertThat(diversifyAggResult.getDebugInfo(), equalTo( @@ -339,6 +343,7 @@ public void testDiversifiedAggProfile() { assertThat(diversifyBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(diversifyBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(diversifyBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(diversifyBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(maxBreakdown.get(REDUCE), equalTo(0L)); assertThat(maxAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -382,6 +387,7 @@ public void testComplexProfile() { assertThat(histoBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(histoBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(histoBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(histoBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(histoBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(histoBreakdown.get(REDUCE), equalTo(0L)); Map histoDebugInfo = histoAggResult.getDebugInfo(); @@ -403,6 +409,7 @@ public void testComplexProfile() { assertThat(tagsBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(tagsBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(tagsBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(tagsBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(tagsBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(tagsBreakdown.get(REDUCE), equalTo(0L)); assertRemapTermsDebugInfo(tagsAggResult); @@ -421,6 +428,7 @@ public void testComplexProfile() { assertThat(avgBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(avgBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(avgBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(avgBreakdown.get(REDUCE), equalTo(0L)); assertThat(avgAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -436,6 +444,7 @@ public void testComplexProfile() { assertThat(maxBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(maxBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(maxBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(maxBreakdown.get(REDUCE), equalTo(0L)); assertThat(maxAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -451,6 +460,7 @@ public void testComplexProfile() { assertThat(stringsBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(stringsBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(stringsBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(stringsBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(stringsBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(stringsBreakdown.get(REDUCE), equalTo(0L)); assertRemapTermsDebugInfo(stringsAggResult); @@ -469,6 +479,7 @@ public void testComplexProfile() { assertThat(avgBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(avgBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(avgBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(avgBreakdown.get(REDUCE), equalTo(0L)); assertThat(avgAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -484,6 +495,7 @@ public void testComplexProfile() { assertThat(maxBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(maxBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(maxBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(maxBreakdown.get(REDUCE), equalTo(0L)); assertThat(maxAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -500,6 +512,7 @@ public void testComplexProfile() { assertThat(tagsBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(tagsBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(tagsBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(tagsBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(tagsBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(tagsBreakdown.get(REDUCE), equalTo(0L)); assertRemapTermsDebugInfo(tagsAggResult); @@ -518,6 +531,7 @@ public void testComplexProfile() { assertThat(avgBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(avgBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(avgBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(avgBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(avgBreakdown.get(REDUCE), equalTo(0L)); assertThat(avgAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); @@ -533,6 +547,7 @@ public void testComplexProfile() { assertThat(maxBreakdown.get(INITIALIZE), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_LEAF_COLLECTOR), greaterThan(0L)); assertThat(maxBreakdown.get(COLLECT), greaterThan(0L)); + assertThat(maxBreakdown.get(POST_COLLECTION), greaterThan(0L)); assertThat(maxBreakdown.get(BUILD_AGGREGATION), greaterThan(0L)); assertThat(maxBreakdown.get(REDUCE), equalTo(0L)); assertThat(maxAggResult.getDebugInfo(), equalTo(org.elasticsearch.common.collect.Map.of())); diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/AdaptingAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/AdaptingAggregator.java index ec4c3d771f82b..14323d3be37f9 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/AdaptingAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/AdaptingAggregator.java @@ -84,6 +84,11 @@ public final void preCollection() throws IOException { delegate.preCollection(); } + @Override + public final void postCollection() throws IOException { + delegate.postCollection(); + } + @Override public final InternalAggregation[] buildAggregations(long[] owningBucketOrds) throws IOException { InternalAggregation[] delegateResults = delegate.buildAggregations(owningBucketOrds); diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/AggregationPhase.java b/server/src/main/java/org/elasticsearch/search/aggregations/AggregationPhase.java index da2e16081b74f..44c4fbabe30c8 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/AggregationPhase.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/AggregationPhase.java @@ -110,6 +110,7 @@ public void execute(SearchContext context) { } for (Aggregator aggregator : context.aggregations().aggregators()) { try { + aggregator.postCollection(); aggregations.add(aggregator.buildTopLevel()); } catch (IOException e) { throw new AggregationExecutionException("Failed to build aggregation [" + aggregator.name() + "]", e); diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/Aggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/Aggregator.java index 436fefe30e5bb..e6a773b5aae79 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/Aggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/Aggregator.java @@ -125,7 +125,7 @@ public interface BucketComparator { /** * Build the results of this aggregation. - * @param ordsToCollect the ordinals of the buckets that we want to + * @param ordsToCollect the ordinals of the buckets that we want to * collect from this aggregation * @return the results for each ordinal, in the same order as the array * of ordinals diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/AggregatorBase.java b/server/src/main/java/org/elasticsearch/search/aggregations/AggregatorBase.java index db5e7a5764f6f..bffd626a13cbc 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/AggregatorBase.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/AggregatorBase.java @@ -80,6 +80,10 @@ public void preCollection() throws IOException { badState(); } + @Override + public void postCollection() throws IOException { + badState(); + } @Override public ScoreMode scoreMode() { badState(); @@ -218,6 +222,19 @@ public Aggregator subAggregator(String aggName) { return subAggregatorbyName.get(aggName); } + /** + * Called after collection of all document is done. + *

+ * Warning: this is not final only to allow the parent join aggregator + * to delay this until building buckets. + */ + @Override + public void postCollection() throws IOException { + // post-collect this agg before subs to make it possible to buffer and then replay in postCollection() + doPostCollection(); + collectableSubAggregators.postCollection(); + } + /** Called upon release of the aggregator. */ @Override public void close() { @@ -231,6 +248,12 @@ public void close() { /** Release instance-specific data. */ protected void doClose() {} + /** + * Can be overridden by aggregator implementation to be called back when the collection phase ends. + */ + protected void doPostCollection() throws IOException { + } + protected final InternalAggregations buildEmptySubAggregations() { List aggs = new ArrayList<>(); for (Aggregator aggregator : subAggregators) { diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/BucketCollector.java b/server/src/main/java/org/elasticsearch/search/aggregations/BucketCollector.java index 69cea58d08b11..ac42deb67f3df 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/BucketCollector.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/BucketCollector.java @@ -31,6 +31,10 @@ public void preCollection() throws IOException { // no-op } @Override + public void postCollection() throws IOException { + // no-op + } + @Override public ScoreMode scoreMode() { return ScoreMode.COMPLETE_NO_SCORES; } @@ -43,4 +47,10 @@ public ScoreMode scoreMode() { * Pre collection callback. */ public abstract void preCollection() throws IOException; + + /** + * Post-collection callback. + */ + public abstract void postCollection() throws IOException; + } diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/MultiBucketCollector.java b/server/src/main/java/org/elasticsearch/search/aggregations/MultiBucketCollector.java index 9b668c791a65e..75201e6d5fe19 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/MultiBucketCollector.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/MultiBucketCollector.java @@ -115,6 +115,13 @@ public void preCollection() throws IOException { } } + @Override + public void postCollection() throws IOException { + for (BucketCollector collector : collectors) { + collector.postCollection(); + } + } + @Override public String toString() { return Arrays.toString(collectors); diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollector.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollector.java index 79a48af165805..008b877809c52 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollector.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollector.java @@ -62,6 +62,7 @@ static class Entry { private PackedLongValues.Builder docDeltasBuilder; private PackedLongValues.Builder bucketsBuilder; private LongHash selectedBuckets; + private boolean finished = false; /** * Sole constructor. @@ -134,12 +135,20 @@ public void preCollection() throws IOException { collector.preCollection(); } + @Override + public void postCollection() throws IOException { + finishLeaf(); + finished = true; + } + /** * Replay the wrapped collector, but only on a selection of buckets. */ @Override public void prepareSelectedBuckets(long... selectedBuckets) throws IOException { - finishLeaf(); + if (finished == false) { + throw new IllegalStateException("Cannot replay yet, collection is not finished: postCollect() has not been called"); + } if (this.selectedBuckets != null) { throw new IllegalStateException("Already been replayed"); } @@ -191,6 +200,7 @@ public void prepareSelectedBuckets(long... selectedBuckets) throws IOException { // continue with the following leaf } } + collector.postCollection(); } /** diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BucketsAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BucketsAggregator.java index f923b5275ed52..32017f3a61b13 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BucketsAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/BucketsAggregator.java @@ -140,7 +140,7 @@ public final long bucketDocCount(long bucketOrd) { /** * Hook to allow taking an action before building the sub agg results. */ - protected void prepareSubAggs(long[] bucketOrdsToCollect) throws IOException {} + protected void prepareSubAggs(long[] ordsToCollect) throws IOException {} /** * Build the results of the sub-aggregations of the buckets at each of diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/DeferringBucketCollector.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/DeferringBucketCollector.java index 3383625682352..1177239c43d89 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/DeferringBucketCollector.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/DeferringBucketCollector.java @@ -99,6 +99,12 @@ public void preCollection() throws IOException { "Deferred collectors cannot be collected directly. They must be collected through the recording wrapper."); } + @Override + public void postCollection() throws IOException { + throw new IllegalStateException( + "Deferred collectors cannot be collected directly. They must be collected through the recording wrapper."); + } + @Override public Aggregator resolveSortPath(PathElement next, Iterator path) { return in.resolveSortPath(next, path); diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/composite/CompositeAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/composite/CompositeAggregator.java index 4997cbd9ce30c..d71e60c3c5569 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/composite/CompositeAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/composite/CompositeAggregator.java @@ -123,11 +123,15 @@ protected void doPreCollection() throws IOException { collectableSubAggregators = BucketCollector.NO_OP_COLLECTOR; } + @Override + protected void doPostCollection() throws IOException { + finishLeaf(); + } + @Override public InternalAggregation[] buildAggregations(long[] owningBucketOrds) throws IOException { // Composite aggregator must be at the top of the aggregation tree assert owningBucketOrds.length == 1 && owningBucketOrds[0] == 0L; - finishLeaf(); if (deferredCollectors != NO_OP_COLLECTOR) { // Replay all documents that contain at least one top bucket (collected during the first pass). runDeferredCollections(); @@ -471,6 +475,7 @@ private void runDeferredCollections() throws IOException { collector.collect(docID); } } + deferredCollectors.postCollection(); } /** diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/nested/NestedAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/nested/NestedAggregator.java index a374647176968..dfc79dd3a37bb 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/nested/NestedAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/nested/NestedAggregator.java @@ -101,6 +101,11 @@ protected void preGetSubLeafCollectors(LeafReaderContext ctx) throws IOException processBufferedDocs(); } + @Override + protected void doPostCollection() throws IOException { + processBufferedDocs(); + } + private void processBufferedDocs() throws IOException { if (bufferingNestedLeafBucketCollector != null) { bufferingNestedLeafBucketCollector.processBufferedChildBuckets(); @@ -109,7 +114,6 @@ private void processBufferedDocs() throws IOException { @Override public InternalAggregation[] buildAggregations(long[] owningBucketOrds) throws IOException { - processBufferedDocs(); return buildAggregationsForSingleBucket(owningBucketOrds, (owningBucketOrd, subAggregationResults) -> new InternalNested(name, bucketDocCount(owningBucketOrd), subAggregationResults, metadata())); } diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollector.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollector.java index 62c0af767a7a6..e83d1054ae67f 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollector.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollector.java @@ -113,9 +113,15 @@ public void preCollection() throws IOException { deferred.preCollection(); } + @Override + public void postCollection() throws IOException { + runDeferredAggs(); + } + + @Override public void prepareSelectedBuckets(long... selectedBuckets) throws IOException { - runDeferredAggs(); // TODO should we only prepare the selected buckets?! + // no-op - deferred aggs processed in postCollection call } private void runDeferredAggs() throws IOException { @@ -149,6 +155,7 @@ private void runDeferredAggs() throws IOException { // done with allDocs now, reclaim some memory circuitBreakerConsumer.accept(-12L * shardSize); } + deferred.postCollection(); } class PerParentBucketSamples { diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/terms/GlobalOrdinalsStringTermsAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/terms/GlobalOrdinalsStringTermsAggregator.java index fcef5590f62ed..8a8bc52ae92f1 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/bucket/terms/GlobalOrdinalsStringTermsAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/bucket/terms/GlobalOrdinalsStringTermsAggregator.java @@ -326,12 +326,11 @@ public void collect(int doc, long owningBucketOrd) throws IOException { } @Override - public InternalAggregation[] buildAggregations(long[] owningBucketOrds) throws IOException { + protected void doPostCollection() throws IOException { if (mapping != null) { mapSegmentCountsToGlobalCounts(mapping); mapping = null; } - return super.buildAggregations(owningBucketOrds); } @Override diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/CardinalityAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/CardinalityAggregator.java index 77e59359dfffe..c259b9b4d41f7 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/CardinalityAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/CardinalityAggregator.java @@ -8,7 +8,9 @@ package org.elasticsearch.search.aggregations.metrics; -import com.carrotsearch.hppc.BitMixer; +import java.io.IOException; +import java.util.Map; +import java.util.function.BiConsumer; import org.apache.lucene.index.LeafReaderContext; import org.apache.lucene.index.SortedNumericDocValues; @@ -27,7 +29,6 @@ import org.elasticsearch.common.util.ObjectArray; import org.elasticsearch.index.fielddata.SortedBinaryDocValues; import org.elasticsearch.index.fielddata.SortedNumericDoubleValues; -import org.elasticsearch.search.aggregations.AggregationExecutionException; import org.elasticsearch.search.aggregations.Aggregator; import org.elasticsearch.search.aggregations.InternalAggregation; import org.elasticsearch.search.aggregations.LeafBucketCollector; @@ -35,9 +36,7 @@ import org.elasticsearch.search.aggregations.support.ValuesSource; import org.elasticsearch.search.aggregations.support.ValuesSourceConfig; -import java.io.IOException; -import java.util.Map; -import java.util.function.BiConsumer; +import com.carrotsearch.hppc.BitMixer; /** * An aggregator that computes approximate counts of unique values. @@ -136,18 +135,12 @@ private void postCollectLastCollector() throws IOException { } @Override - protected void beforeBuildingResults(long[] ordsToCollect) throws IOException { + protected void doPostCollection() throws IOException { postCollectLastCollector(); } @Override public double metric(long owningBucketOrd) { - try { - // Make sure all outstanding data has been synced down to the counts. - postCollectLastCollector(); - } catch (IOException e) { - throw new AggregationExecutionException("error collecting data in last segment", e); - } return counts == null ? 0 : counts.cardinality(owningBucketOrd); } diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/GlobalOrdCardinalityAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/GlobalOrdCardinalityAggregator.java index 6d273b9777cc5..982ae72d9bec1 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/GlobalOrdCardinalityAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/GlobalOrdCardinalityAggregator.java @@ -8,6 +8,8 @@ package org.elasticsearch.search.aggregations.metrics; +import java.io.IOException; +import java.util.Map; import org.apache.lucene.index.LeafReaderContext; import org.apache.lucene.index.SortedSetDocValues; @@ -20,16 +22,12 @@ import org.elasticsearch.common.util.BitArray; import org.elasticsearch.common.util.LongArray; import org.elasticsearch.common.util.ObjectArray; -import org.elasticsearch.search.aggregations.AggregationExecutionException; import org.elasticsearch.search.aggregations.Aggregator; import org.elasticsearch.search.aggregations.InternalAggregation; import org.elasticsearch.search.aggregations.LeafBucketCollector; import org.elasticsearch.search.aggregations.support.AggregationContext; import org.elasticsearch.search.aggregations.support.ValuesSource; -import java.io.IOException; -import java.util.Map; - /** * An aggregator that computes approximate counts of unique values * using global ords. @@ -91,15 +89,7 @@ public void collect(int doc, long bucketOrd) throws IOException { }; } - @Override - protected void beforeBuildingResults(long[] ordsToCollect) throws IOException { - buildCountIfNeeded(); - } - - private void buildCountIfNeeded() throws IOException { - if (counts != null) { - return; - } + protected void doPostCollection() throws IOException { counts = new HyperLogLogPlusPlusSparse(precision, bigArrays, visitedOrds.size()); try (LongArray hashes = bigArrays.newLongArray(maxOrd, false)) { try (BitArray allVisitedOrds = new BitArray(maxOrd, bigArrays)) { @@ -138,18 +128,12 @@ private void buildCountIfNeeded() throws IOException { @Override public double metric(long owningBucketOrd) { - try { - // Make sure all outstanding data has been synced down to the counts. - buildCountIfNeeded(); - } catch (IOException e) { - throw new AggregationExecutionException("error collecting data in last segment", e); - } return counts.cardinality(owningBucketOrd); } @Override public InternalAggregation buildAggregation(long owningBucketOrdinal) { - if (owningBucketOrdinal >= counts.maxOrd() || counts.cardinality(owningBucketOrdinal) == 0) { + if (counts == null || owningBucketOrdinal >= counts.maxOrd() || counts.cardinality(owningBucketOrdinal) == 0) { return buildEmptyAggregation(); } // We need to build a copy because the returned Aggregation needs remain usable after diff --git a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/MetricsAggregator.java b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/MetricsAggregator.java index d245379626b62..8b298394d34d3 100644 --- a/server/src/main/java/org/elasticsearch/search/aggregations/metrics/MetricsAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/aggregations/metrics/MetricsAggregator.java @@ -28,24 +28,17 @@ protected MetricsAggregator(String name, AggregationContext context, Aggregator */ } - /** - * Called once before any calls to {@link #buildAggregation(long)} so the - * Aggregator can finish up any work it has to do. - */ - protected void beforeBuildingResults(long[] ordsToCollect) throws IOException {} - /** * Build an aggregation for data that has been collected into * {@code owningBucketOrd}. */ - public abstract InternalAggregation buildAggregation(long ordToCollect) throws IOException; + public abstract InternalAggregation buildAggregation(long owningBucketOrd) throws IOException; @Override - public final InternalAggregation[] buildAggregations(long[] ordsToCollect) throws IOException { - beforeBuildingResults(ordsToCollect); - InternalAggregation[] results = new InternalAggregation[ordsToCollect.length]; - for (int ordIdx = 0; ordIdx < ordsToCollect.length; ordIdx++) { - results[ordIdx] = buildAggregation(ordsToCollect[ordIdx]); + public final InternalAggregation[] buildAggregations(long[] owningBucketOrds) throws IOException { + InternalAggregation[] results = new InternalAggregation[owningBucketOrds.length]; + for (int ordIdx = 0; ordIdx < owningBucketOrds.length; ordIdx++) { + results[ordIdx] = buildAggregation(owningBucketOrds[ordIdx]); } return results; } diff --git a/server/src/main/java/org/elasticsearch/search/profile/aggregation/AggregationTimingType.java b/server/src/main/java/org/elasticsearch/search/profile/aggregation/AggregationTimingType.java index e3a5909b170c2..da840c8630b5b 100644 --- a/server/src/main/java/org/elasticsearch/search/profile/aggregation/AggregationTimingType.java +++ b/server/src/main/java/org/elasticsearch/search/profile/aggregation/AggregationTimingType.java @@ -14,6 +14,7 @@ public enum AggregationTimingType { INITIALIZE, BUILD_LEAF_COLLECTOR, COLLECT, + POST_COLLECTION, BUILD_AGGREGATION, REDUCE; diff --git a/server/src/main/java/org/elasticsearch/search/profile/aggregation/ProfilingAggregator.java b/server/src/main/java/org/elasticsearch/search/profile/aggregation/ProfilingAggregator.java index 57fbc7a8926ee..02c5f2a382345 100644 --- a/server/src/main/java/org/elasticsearch/search/profile/aggregation/ProfilingAggregator.java +++ b/server/src/main/java/org/elasticsearch/search/profile/aggregation/ProfilingAggregator.java @@ -107,6 +107,17 @@ public void preCollection() throws IOException { profiler.pollLastElement(); } + @Override + public void postCollection() throws IOException { + Timer timer = profileBreakdown.getTimer(AggregationTimingType.POST_COLLECTION); + timer.start(); + try { + delegate.postCollection(); + } finally { + timer.stop(); + } + } + @Override public String toString() { return delegate.toString(); diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/MultiBucketCollectorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/MultiBucketCollectorTests.java index 9ff2f0c0ea358..05cd13918e792 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/MultiBucketCollectorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/MultiBucketCollectorTests.java @@ -81,6 +81,9 @@ public ScoreMode scoreMode() { @Override public void preCollection() {} + + @Override + public void postCollection() {} } private static class TotalHitCountBucketCollector extends BucketCollector { @@ -108,6 +111,9 @@ public ScoreMode scoreMode() { @Override public void preCollection() {} + @Override + public void postCollection() {} + int getTotalHits() { return count; } @@ -141,6 +147,9 @@ public ScoreMode scoreMode() { @Override public void preCollection() {} + + @Override + public void postCollection() {} } public void testCollectionTerminatedExceptionHandling() throws IOException { diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollectorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollectorTests.java index 167c7cb771ad0..c97e9c219ea8a 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollectorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/BestBucketsDeferringCollectorTests.java @@ -73,6 +73,7 @@ public ScoreMode scoreMode() { collector.setDeferredCollector(Collections.singleton(bla(deferredCollectedDocIds))); collector.preCollection(); indexSearcher.search(termQuery, collector); + collector.postCollection(); collector.prepareSelectedBuckets(0); assertEquals(topDocs.scoreDocs.length, deferredCollectedDocIds.size()); @@ -86,6 +87,7 @@ public ScoreMode scoreMode() { collector.setDeferredCollector(Collections.singleton(bla(deferredCollectedDocIds))); collector.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), collector); + collector.postCollection(); collector.prepareSelectedBuckets(0); assertEquals(topDocs.scoreDocs.length, deferredCollectedDocIds.size()); @@ -113,6 +115,11 @@ public void preCollection() throws IOException { } + @Override + public void postCollection() throws IOException { + + } + @Override public ScoreMode scoreMode() { return ScoreMode.COMPLETE_NO_SCORES; @@ -201,12 +208,16 @@ public ScoreMode scoreMode() { @Override public void preCollection() throws IOException {} + @Override + public void postCollection() throws IOException {} + @Override public LeafBucketCollector getLeafCollector(LeafReaderContext ctx) throws IOException { LeafBucketCollector delegate = deferringCollector.getLeafCollector(ctx); return leafCollector.apply(deferringCollector, delegate); } }); + deferringCollector.postCollection(); verify.accept(deferringCollector, finalCollector); } } @@ -232,5 +243,10 @@ public void collect(int doc, long owningBucketOrd) throws IOException { @Override public void preCollection() throws IOException {} + + @Override + public void postCollection() throws IOException { + + } } } diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/GlobalAggregatorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/GlobalAggregatorTests.java index 5bd2a4f047e87..b4eb8c4d66d1d 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/GlobalAggregatorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/GlobalAggregatorTests.java @@ -69,6 +69,7 @@ private void testCase(CheckedConsumer buildIndex GlobalAggregator aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); InternalGlobal result = (InternalGlobal) aggregator.buildTopLevel(); verify.accept(result, (InternalMin) result.getAggregations().asMap().get("in_global")); diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/geogrid/GeoGridAggregatorTestCase.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/geogrid/GeoGridAggregatorTestCase.java index e6212bad6a676..eb593ed1d77bb 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/geogrid/GeoGridAggregatorTestCase.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/geogrid/GeoGridAggregatorTestCase.java @@ -274,6 +274,7 @@ private void testCase(Query query, int precision, GeoBoundingBox geoBoundingBox, Aggregator aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(query, aggregator); + aggregator.postCollection(); verify.accept((InternalGeoGrid) aggregator.buildTopLevel()); indexReader.close(); diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollectorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollectorTests.java index cbbd849301099..eca76fa6ee752 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollectorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/sampler/BestDocsDeferringCollectorTests.java @@ -63,6 +63,7 @@ public void testReplay() throws Exception { collector.setDeferredCollector(Collections.singleton(testCollector(deferredCollectedDocIds))); collector.preCollection(); indexSearcher.search(termQuery, collector); + collector.postCollection(); collector.prepareSelectedBuckets(0); assertEquals(topDocs.scoreDocs.length, deferredCollectedDocIds.size()); @@ -91,6 +92,11 @@ public void preCollection() throws IOException { } + @Override + public void postCollection() throws IOException { + + } + @Override public ScoreMode scoreMode() { return ScoreMode.COMPLETE_NO_SCORES; diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/RareTermsAggregatorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/RareTermsAggregatorTests.java index f105260df5754..f7a18ea421173 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/RareTermsAggregatorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/RareTermsAggregatorTests.java @@ -261,6 +261,7 @@ public void testUnmapped() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType1, fieldType2); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); RareTerms result = (RareTerms) aggregator.buildTopLevel(); assertEquals("_name", result.getName()); assertEquals(0, result.getBuckets().size()); diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/TermsAggregatorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/TermsAggregatorTests.java index c7960d8b3e427..ea7dd5c905905 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/TermsAggregatorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/bucket/terms/TermsAggregatorTests.java @@ -76,7 +76,9 @@ import org.elasticsearch.search.aggregations.bucket.nested.NestedAggregationBuilder; import org.elasticsearch.search.aggregations.bucket.nested.NestedAggregatorTests; import org.elasticsearch.search.aggregations.metrics.CardinalityAggregationBuilder; +import org.elasticsearch.search.aggregations.metrics.InternalMax; import org.elasticsearch.search.aggregations.metrics.InternalTopHits; +import org.elasticsearch.search.aggregations.metrics.MaxAggregationBuilder; import org.elasticsearch.search.aggregations.metrics.TopHitsAggregationBuilder; import org.elasticsearch.search.aggregations.pipeline.BucketScriptPipelineAggregationBuilder; import org.elasticsearch.search.aggregations.pipeline.PipelineAggregator.PipelineTree; @@ -108,6 +110,7 @@ import static org.elasticsearch.index.mapper.SeqNoFieldMapper.PRIMARY_TERM_NAME; import static org.elasticsearch.search.aggregations.AggregationBuilders.terms; import static org.elasticsearch.search.aggregations.PipelineAggregatorBuilders.bucketScript; +import static org.hamcrest.Matchers.closeTo; import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.greaterThan; import static org.hamcrest.Matchers.instanceOf; @@ -252,6 +255,7 @@ public void testSimple() throws Exception { TermsAggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(5, result.getBuckets().size()); assertEquals("", result.getBuckets().get(0).getKeyAsString()); @@ -321,6 +325,7 @@ public void testStringIncludeExclude() throws Exception { TermsAggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(10, result.getBuckets().size()); assertEquals("val000", result.getBuckets().get(0).getKeyAsString()); @@ -356,6 +361,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(5, result.getBuckets().size()); assertEquals("val001", result.getBuckets().get(0).getKeyAsString()); @@ -380,6 +386,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(8, result.getBuckets().size()); assertEquals("val002", result.getBuckets().get(0).getKeyAsString()); @@ -409,6 +416,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals("val010", result.getBuckets().get(0).getKeyAsString()); @@ -426,6 +434,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals("val000", result.getBuckets().get(0).getKeyAsString()); @@ -444,6 +453,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals("val000", result.getBuckets().get(0).getKeyAsString()); @@ -461,6 +471,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals("val000", result.getBuckets().get(0).getKeyAsString()); @@ -478,6 +489,7 @@ public void testStringIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals("val001", result.getBuckets().get(0).getKeyAsString()); @@ -533,6 +545,7 @@ public void testNumericIncludeExclude() throws Exception { TermsAggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals(0L, result.getBuckets().get(0).getKey()); @@ -550,6 +563,7 @@ public void testNumericIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(4, result.getBuckets().size()); assertEquals(1L, result.getBuckets().get(0).getKey()); @@ -573,6 +587,7 @@ public void testNumericIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(2, result.getBuckets().size()); assertEquals(0.0, result.getBuckets().get(0).getKey()); @@ -590,6 +605,7 @@ public void testNumericIncludeExclude() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals(4, result.getBuckets().size()); assertEquals(1.0, result.getBuckets().get(0).getKey()); @@ -756,6 +772,7 @@ private void termsAggregator(ValueType valueType, MappedFieldType fieldType, Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(size, result.getBuckets().size()); for (int i = 0; i < size; i++) { @@ -782,6 +799,7 @@ private void termsAggregator(ValueType valueType, MappedFieldType fieldType, aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = ((Filter) reduce(aggregator, context.bigArrays())).getAggregations().get("_name2"); int expectedFilteredCounts = 0; for (Integer count : filteredCounts.values()) { @@ -855,6 +873,7 @@ private void termsAggregatorWithNestedMaxAgg(ValueType valueType, MappedFiel Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(size, result.getBuckets().size()); for (int i = 0; i < size; i++) { @@ -884,6 +903,7 @@ public void testEmpty() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(0, result.getBuckets().size()); @@ -893,6 +913,7 @@ public void testEmpty() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(0, result.getBuckets().size()); @@ -902,6 +923,7 @@ public void testEmpty() throws Exception { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(0, result.getBuckets().size()); @@ -925,6 +947,7 @@ public void testUnmapped() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(0, result.getBuckets().size()); @@ -961,6 +984,7 @@ public void testUnmappedWithMissing() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(1, result.getBuckets().size()); @@ -1033,6 +1057,7 @@ public void testIpField() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals("_name", result.getName()); assertEquals(1, result.getBuckets().size()); @@ -1081,6 +1106,7 @@ public void testNestedTermsAgg() throws Exception { Aggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms result = reduce(aggregator, context.bigArrays()); assertEquals(3, result.getBuckets().size()); assertEquals("a", result.getBuckets().get(0).getKeyAsString()); @@ -1260,6 +1286,74 @@ public void testWithNestedAggregations() throws IOException { } } + public void testHeisenpig() throws IOException { + try (Directory directory = newDirectory()) { + try (RandomIndexWriter indexWriter = new RandomIndexWriter(random(), directory)) { + String[] tags = new String[] {"danger", "fluffiness"}; + indexWriter.addDocuments(generateAnimalDocsWithNested("1", "sheep", tags, new int[] {1, 10})); + indexWriter.addDocuments(generateAnimalDocsWithNested("2", "cow", tags, new int[] {3, 1})); + indexWriter.addDocuments(generateAnimalDocsWithNested("3", "pig", tags, new int[] {100, 1})); + indexWriter.commit(); + NestedAggregationBuilder nested = new NestedAggregationBuilder("nested", "nested_object") + .subAggregation( + new MaxAggregationBuilder("max_number").field("number") + ); + TermsAggregationBuilder terms = new TermsAggregationBuilder("str_terms") + .field("str") + .subAggregation(nested) + .shardSize(10) + .size(10) + .order(BucketOrder.aggregation("nested>max_number", false)); + MappedFieldType nestedFieldType = new NumberFieldMapper.NumberFieldType("number", NumberFieldMapper.NumberType.LONG); + MappedFieldType fieldType = new KeywordFieldMapper.KeywordFieldType("str"); + try (IndexReader indexReader = wrapInMockESDirectoryReader(DirectoryReader.open(directory))) { + StringTerms result = searchAndReduce(newSearcher(indexReader, false, true), + // match root document only + new DocValuesFieldExistsQuery(PRIMARY_TERM_NAME), terms, fieldType, nestedFieldType); + assertThat(result.getBuckets().get(0).getKeyAsString(), equalTo("pig")); + assertThat(result.getBuckets().get(0).docCount, equalTo(1L)); + assertThat(((InternalMax) (((InternalNested)result.getBuckets().get(0).getAggregations().get("nested")) + .getAggregations().get("max_number"))).getValue(), closeTo(100.0, 0.00001)); + } + } + } + } + + public void testSortingWithNestedAggregations() throws IOException { + try (Directory directory = newDirectory()) { + try (RandomIndexWriter indexWriter = new RandomIndexWriter(random(), directory)) { + for (int i = 0; i < 12; i++) { + int[] nestedValues = new int[i]; + for (int j = 0; j < i; j++) { + nestedValues[j] = j; + } + indexWriter.addDocuments(generateDocsWithNested(Integer.toString(i), i % 4, nestedValues)); + } + indexWriter.commit(); + NestedAggregationBuilder nested = new NestedAggregationBuilder("nested", "nested_object") + .subAggregation( + new MaxAggregationBuilder("max_val").field("nested_value") + ); + TermsAggregationBuilder terms = new TermsAggregationBuilder("terms") + .field("value") + .subAggregation(nested) + .shardSize(1) + .size(1) + .order(BucketOrder.aggregation("nested>max_val", false)); + MappedFieldType nestedFieldType = new NumberFieldMapper.NumberFieldType("nested_value", NumberFieldMapper.NumberType.LONG); + MappedFieldType fieldType = new NumberFieldMapper.NumberFieldType("value", NumberFieldMapper.NumberType.LONG); + try (IndexReader indexReader = wrapInMockESDirectoryReader(DirectoryReader.open(directory))) { + LongTerms result = searchAndReduce(newSearcher(indexReader, false, true), + // match root document only + new DocValuesFieldExistsQuery(PRIMARY_TERM_NAME), terms, fieldType, nestedFieldType); + assertThat(result.getBuckets().get(0).term, equalTo(3L)); + assertThat(((InternalMax) (((InternalNested)result.getBuckets().get(0).getAggregations().get("nested")) + .getAggregations().get("max_val"))).getValue(), closeTo(10.0, 0.00001)); + } + } + } + } + public void testNumberToStringValueScript() throws IOException { MappedFieldType fieldType = new NumberFieldMapper.NumberFieldType("number", NumberFieldMapper.NumberType.INTEGER); @@ -1495,6 +1589,27 @@ private List generateDocsWithNested(String id, int value, int[] nested return documents; } + private List generateAnimalDocsWithNested(String id, String animal, String[] tags, int[] nestedValues) { + List documents = new ArrayList<>(); + + for (int i = 0; i < tags.length; i++) { + Document document = new Document(); + document.add(new Field(IdFieldMapper.NAME, Uid.encodeId(id), IdFieldMapper.Defaults.NESTED_FIELD_TYPE)); + + document.add(new Field(TypeFieldMapper.NAME, "__nested_object", TypeFieldMapper.Defaults.NESTED_FIELD_TYPE)); + document.add(new SortedDocValuesField("tag", new BytesRef(tags[i]))); + document.add(new SortedNumericDocValuesField("number", nestedValues[i])); + documents.add(document); + } + + Document document = new Document(); + document.add(new Field(IdFieldMapper.NAME, Uid.encodeId(id), IdFieldMapper.Defaults.FIELD_TYPE)); + document.add(new SortedDocValuesField("str", new BytesRef(animal))); + document.add(sequenceIDFields.primaryTerm); + documents.add(document); + + return documents; + } private IndexReader createIndexWithLongs() throws IOException { Directory directory = newDirectory(); @@ -1539,6 +1654,7 @@ private InternalAggregation buildInternalAggregation(TermsAggregationBuilder bui TermsAggregator aggregator = createAggregator(builder, searcher, fieldType); aggregator.preCollection(); searcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); return aggregator.buildTopLevel(); } diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/metrics/AvgAggregatorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/metrics/AvgAggregatorTests.java index 129846b91e7c5..1798e25e60397 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/metrics/AvgAggregatorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/metrics/AvgAggregatorTests.java @@ -276,6 +276,7 @@ public void testSingleValuedFieldPartiallyUnmapped() throws IOException { AvgAggregator aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); InternalAvg avg = (InternalAvg) aggregator.buildAggregation(0L); @@ -520,6 +521,7 @@ public void testOrderByEmptyAggregation() throws IOException { TermsAggregator aggregator = createAggregator(aggregationBuilder, indexSearcher, fieldType); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); Terms terms = (Terms) aggregator.buildTopLevel(); assertNotNull(terms); @@ -593,6 +595,7 @@ public void testCacheAggregation() throws IOException { AvgAggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); InternalAvg avg = (InternalAvg) aggregator.buildAggregation(0L); @@ -639,6 +642,7 @@ public void testScriptCaching() throws IOException { AvgAggregator aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); InternalAvg avg = (InternalAvg) aggregator.buildAggregation(0L); @@ -657,6 +661,7 @@ public void testScriptCaching() throws IOException { aggregator = createAggregator(aggregationBuilder, context); aggregator.preCollection(); indexSearcher.search(new MatchAllDocsQuery(), aggregator); + aggregator.postCollection(); avg = (InternalAvg) aggregator.buildAggregation(0L); diff --git a/server/src/test/java/org/elasticsearch/search/aggregations/metrics/HDRPercentilesAggregatorTests.java b/server/src/test/java/org/elasticsearch/search/aggregations/metrics/HDRPercentilesAggregatorTests.java index 20191858a7b6e..c3a882e923a07 100644 --- a/server/src/test/java/org/elasticsearch/search/aggregations/metrics/HDRPercentilesAggregatorTests.java +++ b/server/src/test/java/org/elasticsearch/search/aggregations/metrics/HDRPercentilesAggregatorTests.java @@ -204,6 +204,7 @@ private void testCase(Query query, CheckedConsumer A searchAndReduc a.preCollection(); Weight weight = subSearcher.createWeight(rewritten, ScoreMode.COMPLETE, 1f); subSearcher.search(weight, a); + a.postCollection(); aggs.add(a.buildTopLevel()); } } else { root.preCollection(); searcher.search(rewritten, root); + root.postCollection(); aggs.add(root.buildTopLevel()); } diff --git a/x-pack/plugin/analytics/src/test/java/org/elasticsearch/xpack/analytics/aggregations/metrics/HDRPreAggregatedPercentilesAggregatorTests.java b/x-pack/plugin/analytics/src/test/java/org/elasticsearch/xpack/analytics/aggregations/metrics/HDRPreAggregatedPercentilesAggregatorTests.java index 091123c89323b..b5099c11daac1 100644 --- a/x-pack/plugin/analytics/src/test/java/org/elasticsearch/xpack/analytics/aggregations/metrics/HDRPreAggregatedPercentilesAggregatorTests.java +++ b/x-pack/plugin/analytics/src/test/java/org/elasticsearch/xpack/analytics/aggregations/metrics/HDRPreAggregatedPercentilesAggregatorTests.java @@ -155,6 +155,7 @@ private void testCase(Query query, CheckedConsumer) aggregator.buildTopLevel()); indexReader.close();