mirror of https://github.com/apache/lucene.git
Rewrite queries with no SHOULD clauses and minimumShouldMatch > 0 to a MatchNoDocsQuery.
Closes #14026
This commit is contained in:
parent
c44b3f9338
commit
b732543b57
|
@ -604,23 +604,20 @@ public class BooleanQuery extends Query implements Iterable<BooleanClause> {
|
||||||
// Important(this can only be processed after nested clauses have been flattened)
|
// Important(this can only be processed after nested clauses have been flattened)
|
||||||
{
|
{
|
||||||
final Collection<Query> shoulds = clauseSets.get(Occur.SHOULD);
|
final Collection<Query> shoulds = clauseSets.get(Occur.SHOULD);
|
||||||
if (shoulds.size() > 0) {
|
if (shoulds.size() < minimumNumberShouldMatch) {
|
||||||
if (shoulds.size() < minimumNumberShouldMatch) {
|
return new MatchNoDocsQuery("SHOULD clause count less than minimumNumberShouldMatch");
|
||||||
return new MatchNoDocsQuery("SHOULD clause count less than minimumNumberShouldMatch");
|
}
|
||||||
}
|
if (shoulds.size() > 0 && shoulds.size() == minimumNumberShouldMatch) {
|
||||||
|
BooleanQuery.Builder builder = new BooleanQuery.Builder();
|
||||||
if (shoulds.size() == minimumNumberShouldMatch) {
|
for (BooleanClause clause : clauses) {
|
||||||
BooleanQuery.Builder builder = new BooleanQuery.Builder();
|
if (clause.occur() == Occur.SHOULD) {
|
||||||
for (BooleanClause clause : clauses) {
|
builder.add(clause.query(), Occur.MUST);
|
||||||
if (clause.occur() == Occur.SHOULD) {
|
} else {
|
||||||
builder.add(clause.query(), Occur.MUST);
|
builder.add(clause);
|
||||||
} else {
|
|
||||||
builder.add(clause);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return builder.build();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return builder.build();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -401,14 +401,12 @@ public class TestBooleanRewrites extends LuceneTestCase {
|
||||||
|
|
||||||
bq =
|
bq =
|
||||||
new BooleanQuery.Builder()
|
new BooleanQuery.Builder()
|
||||||
.setMinimumNumberShouldMatch(random().nextInt(5))
|
|
||||||
.add(new TermQuery(new Term("foo", "bar")), Occur.MUST)
|
.add(new TermQuery(new Term("foo", "bar")), Occur.MUST)
|
||||||
.add(new TermQuery(new Term("foo", "baz")), Occur.MUST)
|
.add(new TermQuery(new Term("foo", "baz")), Occur.MUST)
|
||||||
.add(new MatchAllDocsQuery(), Occur.FILTER)
|
.add(new MatchAllDocsQuery(), Occur.FILTER)
|
||||||
.build();
|
.build();
|
||||||
Query expected =
|
Query expected =
|
||||||
new BooleanQuery.Builder()
|
new BooleanQuery.Builder()
|
||||||
.setMinimumNumberShouldMatch(bq.getMinimumNumberShouldMatch())
|
|
||||||
.add(new TermQuery(new Term("foo", "bar")), Occur.MUST)
|
.add(new TermQuery(new Term("foo", "bar")), Occur.MUST)
|
||||||
.add(new TermQuery(new Term("foo", "baz")), Occur.MUST)
|
.add(new TermQuery(new Term("foo", "baz")), Occur.MUST)
|
||||||
.build();
|
.build();
|
||||||
|
@ -476,7 +474,22 @@ public class TestBooleanRewrites extends LuceneTestCase {
|
||||||
Query query = randomBooleanQuery(random());
|
Query query = randomBooleanQuery(random());
|
||||||
final TopDocs td1 = searcher1.search(query, 100);
|
final TopDocs td1 = searcher1.search(query, 100);
|
||||||
final TopDocs td2 = searcher2.search(query, 100);
|
final TopDocs td2 = searcher2.search(query, 100);
|
||||||
assertEquals(td1, td2);
|
try {
|
||||||
|
assertEquals(td1, td2);
|
||||||
|
} catch (AssertionError e) {
|
||||||
|
System.out.println(query);
|
||||||
|
Query rewritten = query;
|
||||||
|
do {
|
||||||
|
query = rewritten;
|
||||||
|
rewritten = query.rewrite(searcher1);
|
||||||
|
System.out.println(rewritten);
|
||||||
|
TopDocs tdx = searcher2.search(rewritten, 100);
|
||||||
|
if (td2.totalHits.value() != tdx.totalHits.value()) {
|
||||||
|
System.out.println("Bad");
|
||||||
|
}
|
||||||
|
} while (query != rewritten);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
searcher1.getIndexReader().close();
|
searcher1.getIndexReader().close();
|
||||||
|
|
Loading…
Reference in New Issue