From ba201ecce490d810e246b055dfb982f34f972d7d Mon Sep 17 00:00:00 2001 From: Luca Cavanna Date: Mon, 30 Oct 2023 17:11:03 +0100 Subject: [PATCH] Disable inter-segment concurrency when sorting by field The way multiple slices execute when sorting by field is not optimized and can lead to a slowdown. We will dig deeper and potentially re-enable once we have addressed that. When sorting by _doc or _shard_doc, and potentially by _timestamp, there will be no gain with inter-segment concurrency. Relates to #101230 --- .../search/builder/SearchSourceBuilder.java | 1 + .../search/sort/ScoreSortBuilder.java | 5 +++ .../search/sort/ScriptSortBuilder.java | 5 --- .../search/sort/SortBuilder.java | 2 +- .../search/SearchServiceTests.java | 40 +++++++++++++------ .../builder/SearchSourceBuilderTests.java | 28 +++++++++++++ 6 files changed, 63 insertions(+), 18 deletions(-) diff --git a/server/src/main/java/org/elasticsearch/search/builder/SearchSourceBuilder.java b/server/src/main/java/org/elasticsearch/search/builder/SearchSourceBuilder.java index 2d9d6d1d8d75d..40d46a71405dd 100644 --- a/server/src/main/java/org/elasticsearch/search/builder/SearchSourceBuilder.java +++ b/server/src/main/java/org/elasticsearch/search/builder/SearchSourceBuilder.java @@ -2107,6 +2107,7 @@ public boolean supportsParallelCollection() { if (profile) return false; if (sorts != null) { + // the implicit sorting is by _score, which supports parallel collection for (SortBuilder sortBuilder : sorts) { if (sortBuilder.supportsParallelCollection() == false) return false; } diff --git a/server/src/main/java/org/elasticsearch/search/sort/ScoreSortBuilder.java b/server/src/main/java/org/elasticsearch/search/sort/ScoreSortBuilder.java index 3751186357ff6..88eaadcec5136 100644 --- a/server/src/main/java/org/elasticsearch/search/sort/ScoreSortBuilder.java +++ b/server/src/main/java/org/elasticsearch/search/sort/ScoreSortBuilder.java @@ -170,4 +170,9 @@ public TransportVersion getMinimalSupportedVersion() { public ScoreSortBuilder rewrite(QueryRewriteContext ctx) throws IOException { return this; } + + @Override + public boolean supportsParallelCollection() { + return true; + } } diff --git a/server/src/main/java/org/elasticsearch/search/sort/ScriptSortBuilder.java b/server/src/main/java/org/elasticsearch/search/sort/ScriptSortBuilder.java index e6d1eec98ff94..4ac7348a6c4a4 100644 --- a/server/src/main/java/org/elasticsearch/search/sort/ScriptSortBuilder.java +++ b/server/src/main/java/org/elasticsearch/search/sort/ScriptSortBuilder.java @@ -502,9 +502,4 @@ public ScriptSortBuilder rewrite(QueryRewriteContext ctx) throws IOException { } return new ScriptSortBuilder(this).setNestedSort(rewrite); } - - @Override - public boolean supportsParallelCollection() { - return false; - } } diff --git a/server/src/main/java/org/elasticsearch/search/sort/SortBuilder.java b/server/src/main/java/org/elasticsearch/search/sort/SortBuilder.java index 3901ac281caf1..7ef595b51ca9b 100644 --- a/server/src/main/java/org/elasticsearch/search/sort/SortBuilder.java +++ b/server/src/main/java/org/elasticsearch/search/sort/SortBuilder.java @@ -280,6 +280,6 @@ public String toString() { } public boolean supportsParallelCollection() { - return true; + return false; } } diff --git a/server/src/test/java/org/elasticsearch/search/SearchServiceTests.java b/server/src/test/java/org/elasticsearch/search/SearchServiceTests.java index ae462d9115abf..1fc7ddc42d5d0 100644 --- a/server/src/test/java/org/elasticsearch/search/SearchServiceTests.java +++ b/server/src/test/java/org/elasticsearch/search/SearchServiceTests.java @@ -103,6 +103,8 @@ import org.elasticsearch.search.query.NonCountingTermQuery; import org.elasticsearch.search.query.QuerySearchRequest; import org.elasticsearch.search.query.QuerySearchResult; +import org.elasticsearch.search.sort.FieldSortBuilder; +import org.elasticsearch.search.sort.SortBuilders; import org.elasticsearch.search.suggest.SuggestBuilder; import org.elasticsearch.tasks.TaskCancelHelper; import org.elasticsearch.tasks.TaskCancelledException; @@ -1960,7 +1962,7 @@ public void testEnableSearchWorkerThreads() throws IOException { public void testDetermineMaximumNumberOfSlices() { IndexService indexService = createIndex("index", Settings.EMPTY); IndexShard indexShard = indexService.getShard(0); - ShardSearchRequest request = new ShardSearchRequest( + ShardSearchRequest parallelReq = new ShardSearchRequest( OriginalIndices.NONE, new SearchRequest().allowPartialSearchResults(randomBoolean()), indexShard.shardId(), @@ -1971,6 +1973,18 @@ public void testDetermineMaximumNumberOfSlices() { System.currentTimeMillis(), null ); + ShardSearchRequest singleSliceReq = new ShardSearchRequest( + OriginalIndices.NONE, + new SearchRequest().allowPartialSearchResults(randomBoolean()) + .source(new SearchSourceBuilder().sort(SortBuilders.fieldSort(FieldSortBuilder.DOC_FIELD_NAME))), + indexShard.shardId(), + 0, + indexService.numberOfShards(), + AliasFilter.EMPTY, + 1f, + System.currentTimeMillis(), + null + ); int executorPoolSize = randomIntBetween(1, 100); ExecutorService threadPoolExecutor = EsExecutors.newFixed( "test", @@ -1984,10 +1998,12 @@ public void testDetermineMaximumNumberOfSlices() { SearchService service = getInstanceFromNode(SearchService.class); { - assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.DFS)); - assertEquals(1, service.determineMaximumNumberOfSlices(null, request, ResultsType.DFS)); - assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.QUERY)); - assertEquals(1, service.determineMaximumNumberOfSlices(notThreadPoolExecutor, request, ResultsType.DFS)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.DFS)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, singleSliceReq, ResultsType.DFS)); + assertEquals(1, service.determineMaximumNumberOfSlices(null, parallelReq, ResultsType.DFS)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.QUERY)); + assertEquals(1, service.determineMaximumNumberOfSlices(threadPoolExecutor, singleSliceReq, ResultsType.QUERY)); + assertEquals(1, service.determineMaximumNumberOfSlices(notThreadPoolExecutor, parallelReq, ResultsType.DFS)); } try { ClusterUpdateSettingsResponse response = client().admin() @@ -1997,11 +2013,11 @@ public void testDetermineMaximumNumberOfSlices() { .get(); assertTrue(response.isAcknowledged()); { - assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.DFS)); - assertEquals(1, service.determineMaximumNumberOfSlices(null, request, ResultsType.DFS)); - assertEquals(1, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.QUERY)); - assertEquals(1, service.determineMaximumNumberOfSlices(null, request, ResultsType.QUERY)); - assertEquals(1, service.determineMaximumNumberOfSlices(notThreadPoolExecutor, request, ResultsType.DFS)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.DFS)); + assertEquals(1, service.determineMaximumNumberOfSlices(null, parallelReq, ResultsType.DFS)); + assertEquals(1, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.QUERY)); + assertEquals(1, service.determineMaximumNumberOfSlices(null, parallelReq, ResultsType.QUERY)); + assertEquals(1, service.determineMaximumNumberOfSlices(notThreadPoolExecutor, parallelReq, ResultsType.DFS)); } } finally { // reset original default setting @@ -2011,8 +2027,8 @@ public void testDetermineMaximumNumberOfSlices() { .setPersistentSettings(Settings.builder().putNull(QUERY_PHASE_PARALLEL_COLLECTION_ENABLED.getKey()).build()) .get(); { - assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.DFS)); - assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, request, ResultsType.QUERY)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.DFS)); + assertEquals(executorPoolSize, service.determineMaximumNumberOfSlices(threadPoolExecutor, parallelReq, ResultsType.QUERY)); } } } diff --git a/server/src/test/java/org/elasticsearch/search/builder/SearchSourceBuilderTests.java b/server/src/test/java/org/elasticsearch/search/builder/SearchSourceBuilderTests.java index 85f6d0a718e48..d5dd265fb1ea0 100644 --- a/server/src/test/java/org/elasticsearch/search/builder/SearchSourceBuilderTests.java +++ b/server/src/test/java/org/elasticsearch/search/builder/SearchSourceBuilderTests.java @@ -942,6 +942,34 @@ public void testSupportsParallelCollection() { ); assertFalse(searchSourceBuilder.supportsParallelCollection()); } + { + SearchSourceBuilder searchSourceBuilder = newSearchSourceBuilder.get(); + searchSourceBuilder.sort(SortBuilders.scoreSort().order(randomFrom(SortOrder.values()))); + assertTrue(searchSourceBuilder.supportsParallelCollection()); + searchSourceBuilder.sort(SortBuilders.fieldSort("field")); + assertFalse(searchSourceBuilder.supportsParallelCollection()); + } + { + SearchSourceBuilder searchSourceBuilder = newSearchSourceBuilder.get(); + searchSourceBuilder.sort(SortBuilders.scoreSort().order(randomFrom(SortOrder.values()))); + assertTrue(searchSourceBuilder.supportsParallelCollection()); + searchSourceBuilder.sort(SortBuilders.geoDistanceSort("field", 0, 0)); + assertFalse(searchSourceBuilder.supportsParallelCollection()); + } + { + SearchSourceBuilder searchSourceBuilder = newSearchSourceBuilder.get(); + searchSourceBuilder.sort(SortBuilders.scoreSort().order(randomFrom(SortOrder.values()))); + assertTrue(searchSourceBuilder.supportsParallelCollection()); + searchSourceBuilder.sort(SortBuilders.pitTiebreaker()); + assertFalse(searchSourceBuilder.supportsParallelCollection()); + } + { + SearchSourceBuilder searchSourceBuilder = newSearchSourceBuilder.get(); + searchSourceBuilder.sort(SortBuilders.scoreSort().order(randomFrom(SortOrder.values()))); + assertTrue(searchSourceBuilder.supportsParallelCollection()); + searchSourceBuilder.sort(SortBuilders.fieldSort(FieldSortBuilder.DOC_FIELD_NAME)); + assertFalse(searchSourceBuilder.supportsParallelCollection()); + } { SearchSourceBuilder searchSourceBuilder = newSearchSourceBuilder.get(); searchSourceBuilder.profile(true);