diff --git a/src/main/java/org/opensearch/plugin/insights/core/service/categorizer/QueryShapeGenerator.java b/src/main/java/org/opensearch/plugin/insights/core/service/categorizer/QueryShapeGenerator.java index b5eff30a..f3db022a 100644 --- a/src/main/java/org/opensearch/plugin/insights/core/service/categorizer/QueryShapeGenerator.java +++ b/src/main/java/org/opensearch/plugin/insights/core/service/categorizer/QueryShapeGenerator.java @@ -14,6 +14,8 @@ import java.util.List; import java.util.Map; import java.util.function.Function; +import org.apache.lucene.util.BytesRef; +import org.opensearch.common.hash.MurmurHash3; import org.opensearch.core.common.io.stream.NamedWriteable; import org.opensearch.index.query.AbstractGeometryQueryBuilder; import org.opensearch.index.query.CommonTermsQueryBuilder; @@ -76,6 +78,18 @@ public class QueryShapeGenerator { static final Map, List>> AGG_FIELD_DATA_MAP = FieldDataMapHelper.getAggFieldDataMap(); static final Map, List>> SORT_FIELD_DATA_MAP = FieldDataMapHelper.getSortFieldDataMap(); + /** + * Method to get query shape hash code given a source + * @param source search request source + * @param showFields whether to include field data in query shape + * @return Hash code of query shape as a MurmurHash3.Hash128 object (128-bit) + */ + public static MurmurHash3.Hash128 getShapeHashCode(SearchSourceBuilder source, Boolean showFields) { + final String shape = buildShape(source, showFields); + final BytesRef shapeBytes = new BytesRef(shape); + return MurmurHash3.hash128(shapeBytes.bytes, 0, shapeBytes.length, 0, new MurmurHash3.Hash128()); + } + /** * Method to build search query shape given a source * @param source search request source diff --git a/src/test/java/org/opensearch/plugin/insights/SearchSourceBuilderUtils.java b/src/test/java/org/opensearch/plugin/insights/SearchSourceBuilderUtils.java new file mode 100644 index 00000000..18b2f242 --- /dev/null +++ b/src/test/java/org/opensearch/plugin/insights/SearchSourceBuilderUtils.java @@ -0,0 +1,422 @@ +/* + * SPDX-License-Identifier: Apache-2.0 + * + * The OpenSearch Contributors require contributions made to + * this file be licensed under the Apache-2.0 license or a + * compatible open source license. + */ + +package org.opensearch.plugin.insights; + +import org.opensearch.index.query.BoolQueryBuilder; +import org.opensearch.index.query.MatchQueryBuilder; +import org.opensearch.index.query.QueryBuilders; +import org.opensearch.index.query.RangeQueryBuilder; +import org.opensearch.index.query.RegexpQueryBuilder; +import org.opensearch.index.query.TermQueryBuilder; +import org.opensearch.search.aggregations.bucket.terms.SignificantTextAggregationBuilder; +import org.opensearch.search.aggregations.bucket.terms.TermsAggregationBuilder; +import org.opensearch.search.aggregations.metrics.TopHitsAggregationBuilder; +import org.opensearch.search.aggregations.pipeline.AvgBucketPipelineAggregationBuilder; +import org.opensearch.search.aggregations.pipeline.DerivativePipelineAggregationBuilder; +import org.opensearch.search.aggregations.pipeline.MaxBucketPipelineAggregationBuilder; +import org.opensearch.search.aggregations.support.ValueType; +import org.opensearch.search.builder.SearchSourceBuilder; +import org.opensearch.search.sort.SortOrder; + +public class SearchSourceBuilderUtils { + + public static SearchSourceBuilder createDefaultSearchSourceBuilder() { + /* + * { + * "size": 0, + * "query": { + * "bool": { + * "must": [ + * { + * "term": { + * "field1": "value2" + * } + * } + * ], + * "filter": [ + * { + * "match": { + * "field2": "php" + * } + * }, + * { + * "range": { + * "field4": {} + * } + * } + * ], + * "should": [ + * { + * "regexp": { + * "field3": "text" + * } + * } + * ] + * } + * }, + * "aggs": { + * "agg1": { + * "terms": { + * "field": "type" + * }, + * "meta": { + * "user_value_type_hint": "string" + * }, + * "aggs": { + * "pipeline-agg1": { + * "derivative": { + * "buckets_path": "bucket1" + * } + * }, + * "child-agg3": { + * "terms": { + * "field": "key.sub3" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * } + * } + * }, + * "agg2": { + * "terms": { + * "field": "model" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * }, + * "agg3": { + * "terms": { + * "field": "key" + * }, + * "meta": { + * "user_value_type_hint": "string" + * }, + * "aggs": { + * "pipeline-agg2": { + * "max_bucket": { + * "buckets_path": "bucket2" + * } + * }, + * "child-agg1": { + * "terms": { + * "field": "key.sub1" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * }, + * "child-agg2": { + * "terms": { + * "field": "key.sub2" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * } + * } + * }, + * "top_hits": { + * "top_hits": { + * "stored_fields": ["_none_"] + * } + * }, + * "sig_text": { + * "significant_text": { + * "field": "agg4", + * "filter_duplicate_text": true + * } + * }, + * "pipeline-agg4": { + * "max_bucket": { + * "buckets_path": "bucket4" + * } + * }, + * "pipeline-agg3": { + * "derivative": { + * "buckets_path": "bucket3" + * } + * }, + * "pipeline-agg5": { + * "avg_bucket": { + * "buckets_path": "bucket5" + * } + * } + * }, + * "sort": [ + * { + * "color": { + * "order": "desc" + * } + * }, + * { + * "vendor": { + * "order": "desc" + * } + * }, + * { + * "price": { + * "order": "asc" + * } + * }, + * { + * "album": { + * "order": "asc" + * } + * } + * ] + * } + */ + SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); + sourceBuilder.size(0); + // build query + TermQueryBuilder termQueryBuilder = QueryBuilders.termQuery("field1", "value2"); + MatchQueryBuilder matchQueryBuilder = QueryBuilders.matchQuery("field2", "php"); + RegexpQueryBuilder regexpQueryBuilder = new RegexpQueryBuilder("field3", "text"); + RangeQueryBuilder rangeQueryBuilder = new RangeQueryBuilder("field4"); + sourceBuilder.query( + new BoolQueryBuilder().must(termQueryBuilder).filter(matchQueryBuilder).should(regexpQueryBuilder).filter(rangeQueryBuilder) + ); + // build aggregation + sourceBuilder.aggregation( + new TermsAggregationBuilder("agg1").userValueTypeHint(ValueType.STRING) + .field("type") + .subAggregation(new DerivativePipelineAggregationBuilder("pipeline-agg1", "bucket1")) + .subAggregation(new TermsAggregationBuilder("child-agg3").userValueTypeHint(ValueType.STRING).field("key.sub3")) + ); + sourceBuilder.aggregation(new TermsAggregationBuilder("agg2").userValueTypeHint(ValueType.STRING).field("model")); + sourceBuilder.aggregation( + new TermsAggregationBuilder("agg3").userValueTypeHint(ValueType.STRING) + .field("key") + .subAggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg2", "bucket2")) + .subAggregation(new TermsAggregationBuilder("child-agg1").userValueTypeHint(ValueType.STRING).field("key.sub1")) + .subAggregation(new TermsAggregationBuilder("child-agg2").userValueTypeHint(ValueType.STRING).field("key.sub2")) + ); + sourceBuilder.aggregation(new TopHitsAggregationBuilder("top_hits").storedField("_none_")); + sourceBuilder.aggregation(new SignificantTextAggregationBuilder("sig_text", "agg4").filterDuplicateText(true)); + sourceBuilder.aggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg4", "bucket4")); + sourceBuilder.aggregation(new DerivativePipelineAggregationBuilder("pipeline-agg3", "bucket3")); + sourceBuilder.aggregation(new AvgBucketPipelineAggregationBuilder("pipeline-agg5", "bucket5")); + // build sort + sourceBuilder.sort("color", SortOrder.DESC); + sourceBuilder.sort("vendor", SortOrder.DESC); + sourceBuilder.sort("price", SortOrder.ASC); + sourceBuilder.sort("album", SortOrder.ASC); + + return sourceBuilder; + } + + public static SearchSourceBuilder createQuerySearchSourceBuilder() { + /* + * { + * "size": 0, + * "query": { + * "bool": { + * "must": [ + * { + * "term": { + * "field1": "html" + * } + * } + * ], + * "filter": [ + * { + * "match": { + * "field2": "php" + * } + * }, + * { + * "range": { + * "field4": { + * "from": "2023-01-01T00:00:00Z", + * "to": "2023-12-31T23:59:59Z", + * "format": "strict_date_optional_time" + * } + * } + * } + * ], + * "should": [ + * { + * "regexp": { + * "field3": "text" + * } + * } + * ] + * } + * } + * } + */ + SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); + sourceBuilder.size(0); + TermQueryBuilder termQueryBuilder = QueryBuilders.termQuery("field1", "html"); + MatchQueryBuilder matchQueryBuilder = QueryBuilders.matchQuery("field2", "php"); + RegexpQueryBuilder regexpQueryBuilder = new RegexpQueryBuilder("field3", "text"); + RangeQueryBuilder rangeQueryBuilder = new RangeQueryBuilder("field4").from("2023-01-01T00:00:00Z") + .to("2023-12-31T23:59:59Z") + .format("strict_date_optional_time"); + sourceBuilder.query( + new BoolQueryBuilder().must(termQueryBuilder).filter(matchQueryBuilder).should(regexpQueryBuilder).filter(rangeQueryBuilder) + ); + return sourceBuilder; + } + + public static SearchSourceBuilder createAggregationSearchSourceBuilder() { + /* + * { + * "aggs": { + * "agg1": { + * "terms": { + * "field": "type" + * }, + * "meta": { + * "user_value_type_hint": "string" + * }, + * "aggs": { + * "pipeline-agg1": { + * "derivative": { + * "buckets_path": "bucket1" + * } + * }, + * "child-agg3": { + * "terms": { + * "field": "key.sub3" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * } + * } + * }, + * "agg2": { + * "terms": { + * "field": "model" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * }, + * "agg3": { + * "terms": { + * "field": "key" + * }, + * "meta": { + * "user_value_type_hint": "string" + * }, + * "aggs": { + * "pipeline-agg2": { + * "max_bucket": { + * "buckets_path": "bucket2" + * } + * }, + * "child-agg1": { + * "terms": { + * "field": "key.sub1" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * }, + * "child-agg2": { + * "terms": { + * "field": "key.sub2" + * }, + * "meta": { + * "user_value_type_hint": "string" + * } + * } + * } + * }, + * "top_hits": { + * "top_hits": { + * "stored_fields": ["_none_"] + * } + * }, + * "sig_text": { + * "significant_text": { + * "field": "agg4", + * "filter_duplicate_text": true + * } + * }, + * "pipeline-agg4": { + * "max_bucket": { + * "buckets_path": "bucket4" + * } + * }, + * "pipeline-agg3": { + * "derivative": { + * "buckets_path": "bucket3" + * } + * }, + * "pipeline-agg5": { + * "avg_bucket": { + * "buckets_path": "bucket5" + * } + * } + * } + * } + */ + SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); + sourceBuilder.aggregation( + new TermsAggregationBuilder("agg1").userValueTypeHint(ValueType.STRING) + .field("type") + .subAggregation(new DerivativePipelineAggregationBuilder("pipeline-agg1", "bucket1")) + .subAggregation(new TermsAggregationBuilder("child-agg3").userValueTypeHint(ValueType.STRING).field("key.sub3")) + ); + sourceBuilder.aggregation(new TermsAggregationBuilder("agg2").userValueTypeHint(ValueType.STRING).field("model")); + sourceBuilder.aggregation( + new TermsAggregationBuilder("agg3").userValueTypeHint(ValueType.STRING) + .field("key") + .subAggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg2", "bucket2")) + .subAggregation(new TermsAggregationBuilder("child-agg1").userValueTypeHint(ValueType.STRING).field("key.sub1")) + .subAggregation(new TermsAggregationBuilder("child-agg2").userValueTypeHint(ValueType.STRING).field("key.sub2")) + ); + sourceBuilder.aggregation(new TopHitsAggregationBuilder("top_hits").storedField("_none_")); + sourceBuilder.aggregation(new SignificantTextAggregationBuilder("sig_text", "agg4").filterDuplicateText(true)); + sourceBuilder.aggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg4", "bucket4")); + sourceBuilder.aggregation(new DerivativePipelineAggregationBuilder("pipeline-agg3", "bucket3")); + sourceBuilder.aggregation(new AvgBucketPipelineAggregationBuilder("pipeline-agg5", "bucket5")); + return sourceBuilder; + } + + public static SearchSourceBuilder createSortSearchSourceBuilder() { + /* + * { + * "sort": [ + * { + * "color": { + * "order": "desc" + * } + * }, + * { + * "vendor": { + * "order": "desc" + * } + * }, + * { + * "price": { + * "order": "asc" + * } + * }, + * { + * "album": { + * "order": "asc" + * } + * } + * ] + * } + */ + SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); + sourceBuilder.sort("color", SortOrder.DESC); + sourceBuilder.sort("vendor", SortOrder.DESC); + sourceBuilder.sort("price", SortOrder.ASC); + sourceBuilder.sort("album", SortOrder.ASC); + return sourceBuilder; + } +} diff --git a/src/test/java/org/opensearch/plugin/insights/core/service/categorizor/QueryShapeGeneratorTests.java b/src/test/java/org/opensearch/plugin/insights/core/service/categorizor/QueryShapeGeneratorTests.java index a0320a15..5e04fa9f 100644 --- a/src/test/java/org/opensearch/plugin/insights/core/service/categorizor/QueryShapeGeneratorTests.java +++ b/src/test/java/org/opensearch/plugin/insights/core/service/categorizor/QueryShapeGeneratorTests.java @@ -8,61 +8,16 @@ package org.opensearch.plugin.insights.core.service.categorizor; -import org.opensearch.index.query.BoolQueryBuilder; -import org.opensearch.index.query.MatchQueryBuilder; -import org.opensearch.index.query.QueryBuilders; -import org.opensearch.index.query.RangeQueryBuilder; -import org.opensearch.index.query.RegexpQueryBuilder; -import org.opensearch.index.query.TermQueryBuilder; +import org.opensearch.common.hash.MurmurHash3; +import org.opensearch.plugin.insights.SearchSourceBuilderUtils; import org.opensearch.plugin.insights.core.service.categorizer.QueryShapeGenerator; -import org.opensearch.search.aggregations.bucket.terms.SignificantTextAggregationBuilder; -import org.opensearch.search.aggregations.bucket.terms.TermsAggregationBuilder; -import org.opensearch.search.aggregations.metrics.TopHitsAggregationBuilder; -import org.opensearch.search.aggregations.pipeline.AvgBucketPipelineAggregationBuilder; -import org.opensearch.search.aggregations.pipeline.DerivativePipelineAggregationBuilder; -import org.opensearch.search.aggregations.pipeline.MaxBucketPipelineAggregationBuilder; -import org.opensearch.search.aggregations.support.ValueType; import org.opensearch.search.builder.SearchSourceBuilder; -import org.opensearch.search.sort.SortOrder; import org.opensearch.test.OpenSearchTestCase; public final class QueryShapeGeneratorTests extends OpenSearchTestCase { + public void testComplexSearch() { - SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); - sourceBuilder.size(0); - // build query - TermQueryBuilder termQueryBuilder = QueryBuilders.termQuery("field1", "value2"); - MatchQueryBuilder matchQueryBuilder = QueryBuilders.matchQuery("field2", "php"); - RegexpQueryBuilder regexpQueryBuilder = new RegexpQueryBuilder("field3", "text"); - RangeQueryBuilder rangeQueryBuilder = new RangeQueryBuilder("field4"); - sourceBuilder.query( - new BoolQueryBuilder().must(termQueryBuilder).filter(matchQueryBuilder).should(regexpQueryBuilder).filter(rangeQueryBuilder) - ); - // build agg - sourceBuilder.aggregation( - new TermsAggregationBuilder("agg1").userValueTypeHint(ValueType.STRING) - .field("type") - .subAggregation(new DerivativePipelineAggregationBuilder("pipeline-agg1", "bucket1")) - .subAggregation(new TermsAggregationBuilder("child-agg3").userValueTypeHint(ValueType.STRING).field("key.sub3")) - ); - sourceBuilder.aggregation(new TermsAggregationBuilder("agg2").userValueTypeHint(ValueType.STRING).field("model")); - sourceBuilder.aggregation( - new TermsAggregationBuilder("agg3").userValueTypeHint(ValueType.STRING) - .field("key") - .subAggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg2", "bucket2")) - .subAggregation(new TermsAggregationBuilder("child-agg1").userValueTypeHint(ValueType.STRING).field("key.sub1")) - .subAggregation(new TermsAggregationBuilder("child-agg2").userValueTypeHint(ValueType.STRING).field("key.sub2")) - ); - sourceBuilder.aggregation(new TopHitsAggregationBuilder("top_hits").storedField("_none_")); - sourceBuilder.aggregation(new SignificantTextAggregationBuilder("sig_text", "agg4").filterDuplicateText(true)); - sourceBuilder.aggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg4", "bucket4")); - sourceBuilder.aggregation(new DerivativePipelineAggregationBuilder("pipeline-agg3", "bucket3")); - sourceBuilder.aggregation(new AvgBucketPipelineAggregationBuilder("pipeline-agg5", "bucket5")); - // build sort - sourceBuilder.sort("color", SortOrder.DESC); - sourceBuilder.sort("vendor", SortOrder.DESC); - sourceBuilder.sort("price", SortOrder.ASC); - sourceBuilder.sort("album", SortOrder.ASC); + SearchSourceBuilder sourceBuilder = SearchSourceBuilderUtils.createDefaultSearchSourceBuilder(); String shapeShowFieldsTrue = QueryShapeGenerator.buildShape(sourceBuilder, true); String expectedShowFieldsTrue = "bool []\n" @@ -136,15 +91,7 @@ public void testComplexSearch() { } public void testQueryShape() { - SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); - sourceBuilder.size(0); - TermQueryBuilder termQueryBuilder = QueryBuilders.termQuery("field1", "value2"); - MatchQueryBuilder matchQueryBuilder = QueryBuilders.matchQuery("field2", "php"); - RegexpQueryBuilder regexpQueryBuilder = new RegexpQueryBuilder("field3", "text"); - RangeQueryBuilder rangeQueryBuilder = new RangeQueryBuilder("field4"); - sourceBuilder.query( - new BoolQueryBuilder().must(termQueryBuilder).filter(matchQueryBuilder).should(regexpQueryBuilder).filter(rangeQueryBuilder) - ); + SearchSourceBuilder sourceBuilder = SearchSourceBuilderUtils.createQuerySearchSourceBuilder(); String shapeShowFieldsTrue = QueryShapeGenerator.buildShape(sourceBuilder, true); String expectedShowFieldsTrue = "bool []\n" @@ -170,26 +117,7 @@ public void testQueryShape() { } public void testAggregationShape() { - SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); - sourceBuilder.aggregation( - new TermsAggregationBuilder("agg1").userValueTypeHint(ValueType.STRING) - .field("type") - .subAggregation(new DerivativePipelineAggregationBuilder("pipeline-agg1", "bucket1")) - .subAggregation(new TermsAggregationBuilder("child-agg3").userValueTypeHint(ValueType.STRING).field("key.sub3")) - ); - sourceBuilder.aggregation(new TermsAggregationBuilder("agg2").userValueTypeHint(ValueType.STRING).field("model")); - sourceBuilder.aggregation( - new TermsAggregationBuilder("agg3").userValueTypeHint(ValueType.STRING) - .field("key") - .subAggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg2", "bucket2")) - .subAggregation(new TermsAggregationBuilder("child-agg1").userValueTypeHint(ValueType.STRING).field("key.sub1")) - .subAggregation(new TermsAggregationBuilder("child-agg2").userValueTypeHint(ValueType.STRING).field("key.sub2")) - ); - sourceBuilder.aggregation(new TopHitsAggregationBuilder("top_hits").storedField("_none_")); - sourceBuilder.aggregation(new SignificantTextAggregationBuilder("sig_text", "agg4").filterDuplicateText(true)); - sourceBuilder.aggregation(new MaxBucketPipelineAggregationBuilder("pipeline-agg4", "bucket4")); - sourceBuilder.aggregation(new DerivativePipelineAggregationBuilder("pipeline-agg3", "bucket3")); - sourceBuilder.aggregation(new AvgBucketPipelineAggregationBuilder("pipeline-agg5", "bucket5")); + SearchSourceBuilder sourceBuilder = SearchSourceBuilderUtils.createAggregationSearchSourceBuilder(); String shapeShowFieldsTrue = QueryShapeGenerator.buildShape(sourceBuilder, true); String expectedShowFieldsTrue = "aggregation:\n" @@ -237,11 +165,7 @@ public void testAggregationShape() { } public void testSortShape() { - SearchSourceBuilder sourceBuilder = new SearchSourceBuilder(); - sourceBuilder.sort("color", SortOrder.DESC); - sourceBuilder.sort("vendor", SortOrder.DESC); - sourceBuilder.sort("price", SortOrder.ASC); - sourceBuilder.sort("album", SortOrder.ASC); + SearchSourceBuilder sourceBuilder = SearchSourceBuilderUtils.createSortSearchSourceBuilder(); String shapeShowFieldsTrue = QueryShapeGenerator.buildShape(sourceBuilder, true); String expectedShowFieldsTrue = "sort:\n" + " asc [album]\n" + " asc [price]\n" + " desc [color]\n" + " desc [vendor]\n"; @@ -251,4 +175,28 @@ public void testSortShape() { String expectedShowFieldsFalse = "sort:\n" + " asc\n" + " asc\n" + " desc\n" + " desc\n"; assertEquals(expectedShowFieldsFalse, shapeShowFieldsFalse); } + + public void testHashCode() { + // Create test source builders + SearchSourceBuilder defaultSourceBuilder = SearchSourceBuilderUtils.createDefaultSearchSourceBuilder(); + SearchSourceBuilder querySourceBuilder = SearchSourceBuilderUtils.createQuerySearchSourceBuilder(); + + // showFields true + MurmurHash3.Hash128 defaultHashTrue = QueryShapeGenerator.getShapeHashCode(defaultSourceBuilder, true); + MurmurHash3.Hash128 queryHashTrue = QueryShapeGenerator.getShapeHashCode(querySourceBuilder, true); + assertEquals(defaultHashTrue, QueryShapeGenerator.getShapeHashCode(defaultSourceBuilder, true)); + assertEquals(queryHashTrue, QueryShapeGenerator.getShapeHashCode(querySourceBuilder, true)); + assertNotEquals(defaultHashTrue, queryHashTrue); + + // showFields false + MurmurHash3.Hash128 defaultHashFalse = QueryShapeGenerator.getShapeHashCode(defaultSourceBuilder, false); + MurmurHash3.Hash128 queryHashFalse = QueryShapeGenerator.getShapeHashCode(querySourceBuilder, false); + assertEquals(defaultHashFalse, QueryShapeGenerator.getShapeHashCode(defaultSourceBuilder, false)); + assertEquals(queryHashFalse, QueryShapeGenerator.getShapeHashCode(querySourceBuilder, false)); + assertNotEquals(defaultHashFalse, queryHashFalse); + + // Compare field data on vs off + assertNotEquals(defaultHashTrue, defaultHashFalse); + assertNotEquals(queryHashTrue, queryHashFalse); + } }