SOLR-5618: Fix false cache hits in queryResultCache when hashCodes are equal and duplicate filter queries exist in one of the requests

git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1556988 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Chris M. Hostetter 2014-01-09 23:55:46 +00:00
parent f96e58f45d
commit 6b33a8c593
4 changed files with 234 additions and 36 deletions

View File

@ -353,6 +353,9 @@ Bug Fixes
* SOLR-5543: Core swaps resulted in duplicate core entries in solr.xml when * SOLR-5543: Core swaps resulted in duplicate core entries in solr.xml when
using solr.xml persistence. (Bill Bell, Alan Woodward) using solr.xml persistence. (Bill Bell, Alan Woodward)
* SOLR-5618: Fix false cache hits in queryResultCache when hashCodes are equal
and duplicate filter queries exist in one of the requests (hossman)
Optimizations Optimizations
---------------------- ----------------------

View File

@ -21,6 +21,7 @@ import org.apache.lucene.search.Query;
import org.apache.lucene.search.Sort; import org.apache.lucene.search.Sort;
import org.apache.lucene.search.SortField; import org.apache.lucene.search.SortField;
import java.util.List; import java.util.List;
import java.util.ArrayList;
/** A hash key encapsulating a query, a list of filters, and a sort /** A hash key encapsulating a query, a list of filters, and a sort
* *
@ -47,6 +48,8 @@ public final class QueryResultKey {
if (filters != null) { if (filters != null) {
for (Query filt : filters) for (Query filt : filters)
// NOTE: simple summation used here so keys with the same filters but in
// different orders get the same hashCode
h += filt.hashCode(); h += filt.hashCode();
} }
@ -78,7 +81,7 @@ public final class QueryResultKey {
// first. // first.
if (this.sfields.length != other.sfields.length) return false; if (this.sfields.length != other.sfields.length) return false;
if (!this.query.equals(other.query)) return false; if (!this.query.equals(other.query)) return false;
if (!isEqual(this.filters, other.filters)) return false; if (!unorderedCompare(this.filters, other.filters)) return false;
for (int i=0; i<sfields.length; i++) { for (int i=0; i<sfields.length; i++) {
SortField sf1 = this.sfields[i]; SortField sf1 = this.sfields[i];
@ -89,17 +92,27 @@ public final class QueryResultKey {
return true; return true;
} }
/**
// Do fast version, expecting that filters are ordered and only * compares the two lists of queries in an unordered manner such that this method
// fall back to unordered compare on the first non-equal elements. * returns true if the 2 lists are the same size, and contain the same elements.
// This will only be called if the hash code of the entire key already *
// matched, so the slower unorderedCompare should pretty much never * This method should only be used if the lists come from QueryResultKeys which have
// be called if filter lists are generally ordered. * already been found to have equal hashCodes, since the unordered comparison aspects
private static boolean isEqual(List<Query> fqList1, List<Query> fqList2) { * of the logic are not cheap.
*
* @return true if the lists of equivilent other then the ordering
*/
private static boolean unorderedCompare(List<Query> fqList1, List<Query> fqList2) {
// Do fast version first, expecting that filters are usually in the same order
//
// Fall back to unordered compare logic on the first non-equal elements.
// The slower unorderedCompare should pretty much never be called if filter
// lists are generally ordered consistently
if (fqList1 == fqList2) return true; // takes care of identity and null cases if (fqList1 == fqList2) return true; // takes care of identity and null cases
if (fqList1 == null || fqList2 == null) return false; if (fqList1 == null || fqList2 == null) return false;
int sz = fqList1.size(); int sz = fqList1.size();
if (sz != fqList2.size()) return false; if (sz != fqList2.size()) return false;
for (int i = 0; i < sz; i++) { for (int i = 0; i < sz; i++) {
if (!fqList1.get(i).equals(fqList2.get(i))) { if (!fqList1.get(i).equals(fqList2.get(i))) {
return unorderedCompare(fqList1, fqList2, i); return unorderedCompare(fqList1, fqList2, i);
@ -108,18 +121,37 @@ public final class QueryResultKey {
return true; return true;
} }
/**
* Does an unordered comparison of the elements of two lists of queries starting at
* the specified start index.
*
* This method should only be called on lists which are the same size, and where
* all items with an index less then the specified start index are the same.
*
* @return true if the list items after start are equivilent other then the ordering
*/
private static boolean unorderedCompare(List<Query> fqList1, List<Query> fqList2, int start) { private static boolean unorderedCompare(List<Query> fqList1, List<Query> fqList2, int start) {
int sz = fqList1.size(); assert null != fqList1;
outer: assert null != fqList2;
final int sz = fqList1.size();
assert fqList2.size() == sz;
// SOLR-5618: if we had a garuntee that the lists never contained any duplicates,
// this logic could be a lot simplier
//
// (And of course: if the SolrIndexSearcher / QueryCommmand was ever changed to
// sort the filter query list, then this whole method could be eliminated).
final ArrayList<Query> set2 = new ArrayList<Query>(fqList2.subList(start, sz));
for (int i = start; i < sz; i++) { for (int i = start; i < sz; i++) {
Query q1 = fqList1.get(i); Query q1 = fqList1.get(i);
for (int j = start; j < sz; j++) { if ( ! set2.remove(q1) ) {
if (q1.equals(fqList2.get(j))) return false;
continue outer;
} }
return false;
} }
return true; return set2.isEmpty();
} }
} }

View File

@ -17,6 +17,7 @@
package org.apache.solr.core; package org.apache.solr.core;
import java.util.Arrays;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
@ -27,53 +28,175 @@ import org.apache.lucene.search.Query;
import org.apache.lucene.search.Sort; import org.apache.lucene.search.Sort;
import org.apache.lucene.search.SortField; import org.apache.lucene.search.SortField;
import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TermQuery;
import org.apache.lucene.util._TestUtil;
import org.apache.solr.SolrTestCaseJ4; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.search.QueryResultKey; import org.apache.solr.search.QueryResultKey;
import org.junit.Test; import org.junit.Test;
public class QueryResultKeyTest extends SolrTestCaseJ4 { public class QueryResultKeyTest extends SolrTestCaseJ4 {
@Test public void testFiltersOutOfOrder1() {
public void testFiltersHashCode() {
// the hashcode should be the same even when the list // the hashcode should be the same even when the list
// of filters is in a different order // of filters is in a different order
Sort sort = new Sort(new SortField("test", SortField.Type.INT)); Sort sort = new Sort(new SortField("test", SortField.Type.INT));
List<Query> filters = new ArrayList<Query>();
filters.add(new TermQuery(new Term("test", "field")));
filters.add(new TermQuery(new Term("test2", "field2")));
BooleanQuery query = new BooleanQuery(); BooleanQuery query = new BooleanQuery();
query.add(new TermQuery(new Term("test", "field")), Occur.MUST); query.add(new TermQuery(new Term("test", "field")), Occur.MUST);
List<Query> filters = Arrays.<Query>asList(new TermQuery(new Term("test", "field")),
new TermQuery(new Term("test2", "field2")));
QueryResultKey qrk1 = new QueryResultKey(query , filters, sort, 1); QueryResultKey qrk1 = new QueryResultKey(query , filters, sort, 1);
List<Query> filters2 = new ArrayList<Query>(); List<Query> filters2 = Arrays.<Query>asList(new TermQuery(new Term("test2", "field2")),
filters2.add(new TermQuery(new Term("test2", "field2"))); new TermQuery(new Term("test", "field")));
filters2.add(new TermQuery(new Term("test", "field")));
QueryResultKey qrk2 = new QueryResultKey(query , filters2, sort, 1); QueryResultKey qrk2 = new QueryResultKey(query , filters2, sort, 1);
assertKeyEquals(qrk1, qrk2);
assertEquals(qrk1.hashCode(), qrk2.hashCode());
} }
@Test @Test
public void testQueryResultKeySortedFilters() { public void testFiltersOutOfOrder2() {
Query fq1 = new TermQuery(new Term("test1", "field1")); Query fq1 = new TermQuery(new Term("test1", "field1"));
Query fq2 = new TermQuery(new Term("test2", "field2")); Query fq2 = new TermQuery(new Term("test2", "field2"));
Query query = new TermQuery(new Term("test3", "field3")); Query query = new TermQuery(new Term("test3", "field3"));
List<Query> filters = new ArrayList<Query>(); List<Query> filters = Arrays.asList(fq1, fq2);
filters.add(fq1);
filters.add(fq2);
QueryResultKey key = new QueryResultKey(query, filters, null, 0); QueryResultKey key = new QueryResultKey(query, filters, null, 0);
List<Query> newFilters = new ArrayList<Query>(); List<Query> newFilters = Arrays.asList(fq2, fq1);
newFilters.add(fq2);
newFilters.add(fq1);
QueryResultKey newKey = new QueryResultKey(query, newFilters, null, 0); QueryResultKey newKey = new QueryResultKey(query, newFilters, null, 0);
assertEquals(key, newKey); assertKeyEquals(key, newKey);
} }
public void testQueryResultKeyUnSortedFiltersWithDups() {
Query query = new TermQuery(new Term("main", "val"));
// we need Query clauses that have identical hashCodes
// but are not equal unless the term is equals
Query fq_aa = new FlatHashTermQuery("fq_a");
Query fq_ab = new FlatHashTermQuery("fq_a");
Query fq_ac = new FlatHashTermQuery("fq_a");
Query fq_zz = new FlatHashTermQuery("fq_z");
assertEquals(fq_aa.hashCode(), fq_ab.hashCode());
assertEquals(fq_aa.hashCode(), fq_ac.hashCode());
assertEquals(fq_aa.hashCode(), fq_zz.hashCode());
assertEquals(fq_aa, fq_ab);
assertEquals(fq_aa, fq_ac);
assertEquals(fq_ab, fq_aa);
assertEquals(fq_ab, fq_ac);
assertEquals(fq_ac, fq_aa);
assertEquals(fq_ac, fq_ab);
assertTrue( ! fq_aa.equals(fq_zz) );
assertTrue( ! fq_ab.equals(fq_zz) );
assertTrue( ! fq_ac.equals(fq_zz) );
assertTrue( ! fq_zz.equals(fq_aa) );
assertTrue( ! fq_zz.equals(fq_ab) );
assertTrue( ! fq_zz.equals(fq_ac) );
List<Query> filters1 = Arrays.asList(fq_aa, fq_ab);
List<Query> filters2 = Arrays.asList(fq_zz, fq_ac);
QueryResultKey key1 = new QueryResultKey(query, filters1, null, 0);
QueryResultKey key2 = new QueryResultKey(query, filters2, null, 0);
assertEquals(key1.hashCode(), key2.hashCode());
assertKeyNotEquals(key1, key2);
}
public void testRandomQueryKeyEquality() {
final int minIters = atLeast(100 * 1000);
final Query base = new FlatHashTermQuery("base");
// ensure we cover both code paths at least once
boolean didEquals = false;
boolean didNotEquals = false;
int iter = 1;
while (iter <= minIters || (! didEquals ) || (! didNotEquals ) ) {
iter++;
int[] numsA = smallArrayOfRandomNumbers();
int[] numsB = smallArrayOfRandomNumbers();
QueryResultKey aa = new QueryResultKey(base, buildFiltersFromNumbers(numsA), null, 0);
QueryResultKey bb = new QueryResultKey(base, buildFiltersFromNumbers(numsB), null, 0);
// now that we have our keys, sort the numbers so we know what to expect
Arrays.sort(numsA);
Arrays.sort(numsB);
if (Arrays.equals(numsA, numsB)) {
didEquals = true;
assertKeyEquals(aa, bb);
} else {
didNotEquals = true;
assertKeyNotEquals(aa, bb);
}
}
assert minIters <= iter;
}
/**
* does bi-directional equality check as well as verifying hashCode
*/
public void assertKeyEquals(QueryResultKey key1, QueryResultKey key2) {
assertNotNull(key1);
assertNotNull(key2);
assertEquals(key1.hashCode(), key2.hashCode());
assertEquals(key1, key2);
assertEquals(key2, key1);
}
/**
* does bi-directional check that the keys are <em>not</em> equals
*/
public void assertKeyNotEquals(QueryResultKey key1, QueryResultKey key2) {
assertTrue( ! key1.equals(key2) );
assertTrue( ! key2.equals(key1) );
}
/**
* returns a "small" list of "small" random numbers. The idea behind this method is
* that multiple calls have a decent change of returning two arrays which are the
* same size and contain the same numbers but in a differnet order.
*
* the array is garunteed to always have at least 1 element
*/
private int[] smallArrayOfRandomNumbers() {
int size = _TestUtil.nextInt(random(), 1, 5);
int[] result = new int[size];
for (int i=0; i < size; i++) {
result[i] = _TestUtil.nextInt(random(), 1, 5);
}
return result;
}
/**
* Creates an array of Filter queries using {@link FlatHashTermQuery} based on the
* specified ints
*/
private List<Query> buildFiltersFromNumbers(int[] values) {
ArrayList<Query> filters = new ArrayList<Query>(values.length);
for (int val : values) {
filters.add(new FlatHashTermQuery(String.valueOf(val)));
}
return filters;
}
/**
* Quick and dirty subclass of TermQuery that uses fixed field name and a constant
* value hashCode, regardless of the Term value.
*/
private static class FlatHashTermQuery extends TermQuery {
public FlatHashTermQuery(String val) {
super(new Term("some_field", val));
}
@Override
public int hashCode() {
return 42;
}
}
} }

View File

@ -20,6 +20,7 @@ package org.apache.solr.search;
import org.apache.lucene.util.OpenBitSet; import org.apache.lucene.util.OpenBitSet;
import org.apache.solr.SolrTestCaseJ4; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.common.params.SolrParams;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
import org.junit.BeforeClass; import org.junit.BeforeClass;
@ -259,6 +260,8 @@ public class TestFiltering extends SolrTestCaseJ4 {
} }
} }
assertU(commit()); assertU(commit());
// sanity check
assertJQ(req("q", "*:*"), "/response/numFound==" + model.indexSize);
int totalMatches=0; int totalMatches=0;
int nonZeros=0; int nonZeros=0;
@ -322,11 +325,10 @@ public class TestFiltering extends SolrTestCaseJ4 {
} catch (Exception e) { } catch (Exception e) {
// show the indexIter and queryIter for easier debugging // show the indexIter and queryIter for easier debugging
SolrException.log(log, e); SolrException.log(log, e);
String s= "FAILURE: iiter=" + iiter + " qiter=" + qiter + " request="+params; String s= "FAILURE: indexSize=" + model.indexSize + " iiter=" + iiter + " qiter=" + qiter + " request="+params;
log.error(s); log.error(s);
fail(s); fail(s);
} }
} }
// After making substantial changes to this test, make sure that we still get a // After making substantial changes to this test, make sure that we still get a
@ -336,4 +338,42 @@ public class TestFiltering extends SolrTestCaseJ4 {
} }
} }
public void testHossssSanity() throws Exception {
SolrParams match_0
= params("q", "{!frange v=val_i l=0 u=1}",
"fq", "{!frange v=val_i l=1 u=1}",
"fq", "{!frange v=val_i l=0 u=1}",
"fq", "-_query_:\"{!frange v=val_i l=1 u=1}\"",
"fq", "-_query_:\"{!frange v=val_i l=0 u=1}\"");
SolrParams match_1
= params("q", "{!frange v=val_i l=0 u=1}",
"fq", "{!frange v=val_i l=0 u=1}",
"fq", "{!frange v=val_i l=0 u=1}",
"fq", "-_query_:\"{!frange v=val_i l=1 u=1}\"",
"fq", "-_query_:\"{!frange v=val_i l=1 u=1}\"");
final int numDocs = 10;
for (int i = 0; i < numDocs; i++) {
String val = Integer.toString(i);
assertU(adoc("id",val,f,val));
}
assertU(commit());
// sanity check
assertJQ(req("q", "*:*"), "/response/numFound==" + numDocs);
// 1 then 0
assertJQ(req(match_1), "/response/numFound==1");
assertJQ(req(match_0), "/response/numFound==0");
// clear caches
assertU(commit());
// 0 then 1
assertJQ(req(match_0), "/response/numFound==0");
assertJQ(req(match_1), "/response/numFound==1");
}
} }