Skip to content

Commit 85f8a64

Browse files
authored
Use approximation to advance matched queries (#120133) (#120146)
This PR resolves a regression introduced in #94564 by ensuring that the approximation is used when advancing matched query clauses. Utilizing the two-phase iterator to validate matches guarantees that we do not attempt to find the next document fulfilling the two-phase criteria beyond the current document. This fix prevents scenarios where matching a document in the second phase significantly increases query complexity, especially in cases involving restrictive second-pass filters. Closes #120130
1 parent a047477 commit 85f8a64

File tree

2 files changed

+24
-8
lines changed

2 files changed

+24
-8
lines changed

docs/changelog/120133.yaml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
pr: 120133
2+
summary: Use approximation to advance matched queries
3+
area: Search
4+
type: bug
5+
issues:
6+
- 120130

server/src/main/java/org/elasticsearch/search/fetch/subphase/MatchedQueriesPhase.java

+18-8
Original file line numberDiff line numberDiff line change
@@ -9,10 +9,12 @@
99
package org.elasticsearch.search.fetch.subphase;
1010

1111
import org.apache.lucene.index.LeafReaderContext;
12+
import org.apache.lucene.search.DocIdSetIterator;
1213
import org.apache.lucene.search.Query;
1314
import org.apache.lucene.search.ScoreMode;
1415
import org.apache.lucene.search.Scorer;
1516
import org.apache.lucene.search.ScorerSupplier;
17+
import org.apache.lucene.search.TwoPhaseIterator;
1618
import org.apache.lucene.search.Weight;
1719
import org.elasticsearch.index.query.ParsedQuery;
1820
import org.elasticsearch.search.fetch.FetchContext;
@@ -52,8 +54,9 @@ public FetchSubPhaseProcessor getProcessor(FetchContext context) throws IOExcept
5254
);
5355
}
5456
return new FetchSubPhaseProcessor() {
57+
record ScorerAndIterator(Scorer scorer, DocIdSetIterator approximation, TwoPhaseIterator twoPhase) {}
5558

56-
final Map<String, Scorer> matchingIterators = new HashMap<>();
59+
final Map<String, ScorerAndIterator> matchingIterators = new HashMap<>();
5760

5861
@Override
5962
public void setNextReader(LeafReaderContext readerContext) throws IOException {
@@ -63,7 +66,14 @@ public void setNextReader(LeafReaderContext readerContext) throws IOException {
6366
if (ss != null) {
6467
Scorer scorer = ss.get(0L);
6568
if (scorer != null) {
66-
matchingIterators.put(entry.getKey(), scorer);
69+
final TwoPhaseIterator twoPhase = scorer.twoPhaseIterator();
70+
final DocIdSetIterator iterator;
71+
if (twoPhase == null) {
72+
iterator = scorer.iterator();
73+
} else {
74+
iterator = twoPhase.approximation();
75+
}
76+
matchingIterators.put(entry.getKey(), new ScorerAndIterator(scorer, iterator, twoPhase));
6777
}
6878
}
6979
}
@@ -73,13 +83,13 @@ public void setNextReader(LeafReaderContext readerContext) throws IOException {
7383
public void process(HitContext hitContext) throws IOException {
7484
Map<String, Float> matches = new LinkedHashMap<>();
7585
int doc = hitContext.docId();
76-
for (Map.Entry<String, Scorer> entry : matchingIterators.entrySet()) {
77-
Scorer scorer = entry.getValue();
78-
if (scorer.iterator().docID() < doc) {
79-
scorer.iterator().advance(doc);
86+
for (Map.Entry<String, ScorerAndIterator> entry : matchingIterators.entrySet()) {
87+
ScorerAndIterator query = entry.getValue();
88+
if (query.approximation.docID() < doc) {
89+
query.approximation.advance(doc);
8090
}
81-
if (scorer.iterator().docID() == doc) {
82-
matches.put(entry.getKey(), scorer.score());
91+
if (query.approximation.docID() == doc && (query.twoPhase == null || query.twoPhase.matches())) {
92+
matches.put(entry.getKey(), query.scorer.score());
8393
}
8494
}
8595
hitContext.hit().matchedQueries(matches);

0 commit comments

Comments
 (0)