Fixed score explain is for `custom_filters_score` query.
Only the explain of the filter was included. This fix adds an explain for the inner query and wraps it in a top-level explanation.
This commit is contained in:
parent
6e66f45f58
commit
81a6940ad3
|
@ -168,11 +168,17 @@ public class FiltersFunctionScoreQuery extends Query {
|
||||||
filterFunction.function.setNextReader(reader);
|
filterFunction.function.setNextReader(reader);
|
||||||
Explanation functionExplanation = filterFunction.function.explainFactor(doc);
|
Explanation functionExplanation = filterFunction.function.explainFactor(doc);
|
||||||
float sc = getValue() * subQueryExpl.getValue() * functionExplanation.getValue();
|
float sc = getValue() * subQueryExpl.getValue() * functionExplanation.getValue();
|
||||||
Explanation res = new ComplexExplanation(true, sc, "custom score, product of:");
|
Explanation filterExplanation = new ComplexExplanation(true, sc, "custom score, product of:");
|
||||||
res.addDetail(new Explanation(1.0f, "match filter: " + filterFunction.filter.toString()));
|
filterExplanation.addDetail(new Explanation(1.0f, "match filter: " + filterFunction.filter.toString()));
|
||||||
res.addDetail(functionExplanation);
|
filterExplanation.addDetail(functionExplanation);
|
||||||
res.addDetail(new Explanation(getValue(), "queryBoost"));
|
filterExplanation.addDetail(new Explanation(getValue(), "queryBoost"));
|
||||||
return res;
|
|
||||||
|
// top level score = subquery.score * filter.score (this already has the query boost)
|
||||||
|
float topLevelScore = subQueryExpl.getValue() * sc;
|
||||||
|
Explanation topLevel = new ComplexExplanation(true, topLevelScore, "custom score, score mode [" + scoreMode.toString().toLowerCase() + "]");
|
||||||
|
topLevel.addDetail(subQueryExpl);
|
||||||
|
topLevel.addDetail(filterExplanation);
|
||||||
|
return topLevel;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -19,6 +19,8 @@
|
||||||
|
|
||||||
package org.elasticsearch.test.integration.search.customscore;
|
package org.elasticsearch.test.integration.search.customscore;
|
||||||
|
|
||||||
|
import org.apache.lucene.search.Explanation;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
import org.elasticsearch.action.search.SearchResponse;
|
import org.elasticsearch.action.search.SearchResponse;
|
||||||
import org.elasticsearch.action.search.SearchType;
|
import org.elasticsearch.action.search.SearchType;
|
||||||
import org.elasticsearch.client.Client;
|
import org.elasticsearch.client.Client;
|
||||||
|
@ -38,6 +40,7 @@ import static org.elasticsearch.search.builder.SearchSourceBuilder.searchSource;
|
||||||
import static org.hamcrest.MatcherAssert.assertThat;
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
import static org.hamcrest.Matchers.anyOf;
|
import static org.hamcrest.Matchers.anyOf;
|
||||||
import static org.hamcrest.Matchers.equalTo;
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
import static org.testng.Assert.assertNotNull;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -63,6 +66,81 @@ public class CustomScoreSearchTests extends AbstractNodesTests {
|
||||||
return client("node1");
|
return client("node1");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testScoreExplainBug_2283() throws Exception {
|
||||||
|
client.admin().indices().prepareDelete().execute().actionGet();
|
||||||
|
client.admin().indices().prepareCreate("test").setSettings(settingsBuilder().put("index.number_of_shards", 1)).execute().actionGet();
|
||||||
|
ClusterHealthResponse healthResponse = client.admin().cluster().prepareHealth("test").setWaitForYellowStatus().execute().actionGet();
|
||||||
|
assertThat(healthResponse.timedOut(), equalTo(false));
|
||||||
|
|
||||||
|
client.prepareIndex("test", "type", "1").setSource("field", "value1", "color", "red").execute().actionGet();
|
||||||
|
client.prepareIndex("test", "type", "2").setSource("field", "value2", "color", "blue").execute().actionGet();
|
||||||
|
client.prepareIndex("test", "type", "3").setSource("field", "value3", "color", "red").execute().actionGet();
|
||||||
|
client.prepareIndex("test", "type", "4").setSource("field", "value4", "color", "blue").execute().actionGet();
|
||||||
|
|
||||||
|
client.admin().indices().prepareRefresh().execute().actionGet();
|
||||||
|
|
||||||
|
SearchResponse searchResponse = client.prepareSearch("test")
|
||||||
|
.setQuery(customFiltersScoreQuery(matchAllQuery())
|
||||||
|
.add(termFilter("field", "value4"), "2")
|
||||||
|
.add(termFilter("field", "value2"), "3")
|
||||||
|
.scoreMode("first"))
|
||||||
|
.setExplain(true)
|
||||||
|
.execute().actionGet();
|
||||||
|
|
||||||
|
assertThat(Arrays.toString(searchResponse.shardFailures()), searchResponse.failedShards(), equalTo(0));
|
||||||
|
|
||||||
|
assertThat(searchResponse.hits().totalHits(), equalTo(4l));
|
||||||
|
assertThat(searchResponse.hits().getAt(0).id(), equalTo("2"));
|
||||||
|
assertThat(searchResponse.hits().getAt(0).score(), equalTo(3.0f));
|
||||||
|
logger.info("--> Hit[0] {} Explanation:\n {}", searchResponse.hits().getAt(0).id(), searchResponse.hits().getAt(0).explanation());
|
||||||
|
Explanation explanation = searchResponse.hits().getAt(0).explanation();
|
||||||
|
assertNotNull(explanation);
|
||||||
|
assertThat(explanation.isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getValue(), equalTo(3f));
|
||||||
|
assertThat(explanation.getDescription(), equalTo("custom score, score mode [first]"));
|
||||||
|
|
||||||
|
assertThat(explanation.getDetails().length, equalTo(2));
|
||||||
|
assertThat(explanation.getDetails()[0].isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getDetails()[0].getValue(), equalTo(1f));
|
||||||
|
assertThat(explanation.getDetails()[0].getDetails().length, equalTo(2));
|
||||||
|
assertThat(explanation.getDetails()[1].isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getDetails()[1].getValue(), equalTo(3f));
|
||||||
|
assertThat(explanation.getDetails()[1].getDetails().length, equalTo(3));
|
||||||
|
|
||||||
|
// Same query but with boost
|
||||||
|
searchResponse = client.prepareSearch("test")
|
||||||
|
.setQuery(customFiltersScoreQuery(matchAllQuery())
|
||||||
|
.add(termFilter("field", "value4"), "2")
|
||||||
|
.add(termFilter("field", "value2"), "3")
|
||||||
|
.boost(2)
|
||||||
|
.scoreMode("first"))
|
||||||
|
.setExplain(true)
|
||||||
|
.execute().actionGet();
|
||||||
|
|
||||||
|
assertThat(Arrays.toString(searchResponse.shardFailures()), searchResponse.failedShards(), equalTo(0));
|
||||||
|
|
||||||
|
assertThat(searchResponse.hits().totalHits(), equalTo(4l));
|
||||||
|
assertThat(searchResponse.hits().getAt(0).id(), equalTo("2"));
|
||||||
|
assertThat(searchResponse.hits().getAt(0).score(), equalTo(6f));
|
||||||
|
logger.info("--> Hit[0] {} Explanation:\n {}", searchResponse.hits().getAt(0).id(), searchResponse.hits().getAt(0).explanation());
|
||||||
|
explanation = searchResponse.hits().getAt(0).explanation();
|
||||||
|
assertNotNull(explanation);
|
||||||
|
assertThat(explanation.isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getValue(), equalTo(6f));
|
||||||
|
assertThat(explanation.getDescription(), equalTo("custom score, score mode [first]"));
|
||||||
|
|
||||||
|
assertThat(explanation.getDetails().length, equalTo(2));
|
||||||
|
assertThat(explanation.getDetails()[0].isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getDetails()[0].getValue(), equalTo(1f));
|
||||||
|
assertThat(explanation.getDetails()[0].getDetails().length, equalTo(2));
|
||||||
|
assertThat(explanation.getDetails()[1].isMatch(), equalTo(true));
|
||||||
|
assertThat(explanation.getDetails()[1].getValue(), equalTo(6f));
|
||||||
|
assertThat(explanation.getDetails()[1].getDetails().length, equalTo(3));
|
||||||
|
assertThat(explanation.getDetails()[1].getDetails()[2].getDescription(), equalTo("queryBoost"));
|
||||||
|
assertThat(explanation.getDetails()[1].getDetails()[2].getValue(), equalTo(2f));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testCustomScriptBoost() throws Exception {
|
public void testCustomScriptBoost() throws Exception {
|
||||||
client.admin().indices().prepareDelete().execute().actionGet();
|
client.admin().indices().prepareDelete().execute().actionGet();
|
||||||
|
|
Loading…
Reference in New Issue