Added filter support in the `has_child` and `has_parent` filters.

Example:
```
curl -XPOST 'localhost:9200/_search' -d '{
  "query": {
    "filtered_query": {
      "query": {
        "match": {
          "title": "distributed systems"
        }
      },
      "filter": {
        "has_child": {
          "type": "tag",
          "filter": {
            "term": {
              "name": "book"
            }
          }
        }
      }
    }
  }
}'
```

Closes #2585
This commit is contained in:
Martijn van Groningen 2013-01-24 21:32:38 +01:00
parent a39469a252
commit 9013eeae8a
6 changed files with 136 additions and 15 deletions

View File

@ -352,7 +352,7 @@ public abstract class FilterBuilders {
/** /**
* A filter to filter based on the relationship between a shape and indexed shapes * A filter to filter based on the relationship between a shape and indexed shapes
* *
* @param name The shape field name * @param name The shape field name
* @param shape Shape to use in the filter * @param shape Shape to use in the filter
*/ */
public static GeoShapeFilterBuilder geoShapeFilter(String name, Shape shape) { public static GeoShapeFilterBuilder geoShapeFilter(String name, Shape shape) {
@ -392,10 +392,39 @@ public abstract class FilterBuilders {
return new HasChildFilterBuilder(type, query); return new HasChildFilterBuilder(type, query);
} }
/**
* Constructs a child filter, with the child type and the filter to run against child documents, with
* the result of the filter being the *parent* documents.
*
* @param type The child type
* @param filter The query to run against the child type
*/
public static HasChildFilterBuilder hasChildFilter(String type, FilterBuilder filter) {
return new HasChildFilterBuilder(type, filter);
}
/**
* Constructs a parent filter, with the parent type and the query to run against parent documents, with
* the result of the filter being the *child* documents.
*
* @param parentType The parent type
* @param query The query to run against the parent type
*/
public static HasParentFilterBuilder hasParentFilter(String parentType, QueryBuilder query) { public static HasParentFilterBuilder hasParentFilter(String parentType, QueryBuilder query) {
return new HasParentFilterBuilder(parentType, query); return new HasParentFilterBuilder(parentType, query);
} }
/**
* Constructs a parent filter, with the parent type and the filter to run against parent documents, with
* the result of the filter being the *child* documents.
*
* @param parentType The parent type
* @param filter The filter to run against the parent type
*/
public static HasParentFilterBuilder hasParentFilter(String parentType, FilterBuilder filter) {
return new HasParentFilterBuilder(parentType, filter);
}
public static BoolFilterBuilder boolFilter() { public static BoolFilterBuilder boolFilter() {
return new BoolFilterBuilder(); return new BoolFilterBuilder();
} }

View File

@ -28,19 +28,23 @@ import java.io.IOException;
*/ */
public class HasChildFilterBuilder extends BaseFilterBuilder { public class HasChildFilterBuilder extends BaseFilterBuilder {
private final FilterBuilder filterBuilder;
private final QueryBuilder queryBuilder; private final QueryBuilder queryBuilder;
private String childType; private String childType;
private String scope; private String scope;
private String filterName; private String filterName;
private String executionType; private String executionType;
public HasChildFilterBuilder(String type, QueryBuilder queryBuilder) { public HasChildFilterBuilder(String type, QueryBuilder queryBuilder) {
this.childType = type; this.childType = type;
this.queryBuilder = queryBuilder; this.queryBuilder = queryBuilder;
this.filterBuilder = null;
}
public HasChildFilterBuilder(String type, FilterBuilder filterBuilder) {
this.childType = type;
this.queryBuilder = null;
this.filterBuilder = filterBuilder;
} }
public HasChildFilterBuilder scope(String scope) { public HasChildFilterBuilder scope(String scope) {
@ -58,7 +62,7 @@ public class HasChildFilterBuilder extends BaseFilterBuilder {
/** /**
* Expert: Sets the low level child to parent filtering implementation. Can be: 'bitset' or 'uid' * Expert: Sets the low level child to parent filtering implementation. Can be: 'bitset' or 'uid'
* * <p/>
* This option is experimental and will be removed. * This option is experimental and will be removed.
*/ */
public HasChildFilterBuilder executionType(String executionType) { public HasChildFilterBuilder executionType(String executionType) {
@ -69,8 +73,13 @@ public class HasChildFilterBuilder extends BaseFilterBuilder {
@Override @Override
protected void doXContent(XContentBuilder builder, Params params) throws IOException { protected void doXContent(XContentBuilder builder, Params params) throws IOException {
builder.startObject(HasChildFilterParser.NAME); builder.startObject(HasChildFilterParser.NAME);
builder.field("query"); if (queryBuilder != null) {
queryBuilder.toXContent(builder, params); builder.field("query");
queryBuilder.toXContent(builder, params);
} else if (filterBuilder != null) {
builder.field("filter");
filterBuilder.toXContent(builder, params);
}
builder.field("child_type", childType); builder.field("child_type", childType);
if (scope != null) { if (scope != null) {
builder.field("_scope", scope); builder.field("_scope", scope);

View File

@ -23,6 +23,7 @@ import org.apache.lucene.search.Filter;
import org.apache.lucene.search.Query; import org.apache.lucene.search.Query;
import org.elasticsearch.common.Strings; import org.elasticsearch.common.Strings;
import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.inject.Inject;
import org.elasticsearch.common.lucene.search.XConstantScoreQuery;
import org.elasticsearch.common.lucene.search.XFilteredQuery; import org.elasticsearch.common.lucene.search.XFilteredQuery;
import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.index.mapper.DocumentMapper; import org.elasticsearch.index.mapper.DocumentMapper;
@ -74,6 +75,16 @@ public class HasChildFilterParser implements FilterParser {
} finally { } finally {
QueryParseContext.setTypes(origTypes); QueryParseContext.setTypes(origTypes);
} }
} else if ("filter".equals(currentFieldName)) {
// TODO handle `filter` element before `type` element...
String[] origTypes = QueryParseContext.setTypesWithPrevious(childType == null ? null : new String[]{childType});
try {
Filter innerFilter = parseContext.parseInnerFilter();
query = new XConstantScoreQuery(innerFilter);
queryFound = true;
} finally {
QueryParseContext.setTypes(origTypes);
}
} else { } else {
throw new QueryParsingException(parseContext.index(), "[has_child] filter does not support [" + currentFieldName + "]"); throw new QueryParsingException(parseContext.index(), "[has_child] filter does not support [" + currentFieldName + "]");
} }

View File

@ -29,18 +29,30 @@ import java.io.IOException;
public class HasParentFilterBuilder extends BaseFilterBuilder { public class HasParentFilterBuilder extends BaseFilterBuilder {
private final QueryBuilder queryBuilder; private final QueryBuilder queryBuilder;
private final FilterBuilder filterBuilder;
private final String parentType; private final String parentType;
private String scope; private String scope;
private String filterName; private String filterName;
private String executionType; private String executionType;
/** /**
* @param parentType The parent type * @param parentType The parent type
* @param parentQuery The query that will be matched with parent documents * @param parentQuery The query that will be matched with parent documents
*/ */
public HasParentFilterBuilder(String parentType, QueryBuilder parentQuery) { public HasParentFilterBuilder(String parentType, QueryBuilder parentQuery) {
this.parentType = parentType; this.parentType = parentType;
this.queryBuilder = parentQuery; this.queryBuilder = parentQuery;
this.filterBuilder = null;
}
/**
* @param parentType The parent type
* @param parentFilter The filter that will be matched with parent documents
*/
public HasParentFilterBuilder(String parentType, FilterBuilder parentFilter) {
this.parentType = parentType;
this.queryBuilder = null;
this.filterBuilder = parentFilter;
} }
public HasParentFilterBuilder scope(String scope) { public HasParentFilterBuilder scope(String scope) {
@ -55,7 +67,7 @@ public class HasParentFilterBuilder extends BaseFilterBuilder {
/** /**
* Expert: Sets the low level parent to child filtering implementation. Can be: 'bitset' or 'uid' * Expert: Sets the low level parent to child filtering implementation. Can be: 'bitset' or 'uid'
* * <p/>
* This option is experimental and will be removed. * This option is experimental and will be removed.
*/ */
public HasParentFilterBuilder executionType(String executionType) { public HasParentFilterBuilder executionType(String executionType) {
@ -66,8 +78,13 @@ public class HasParentFilterBuilder extends BaseFilterBuilder {
@Override @Override
protected void doXContent(XContentBuilder builder, Params params) throws IOException { protected void doXContent(XContentBuilder builder, Params params) throws IOException {
builder.startObject(HasParentFilterParser.NAME); builder.startObject(HasParentFilterParser.NAME);
builder.field("query"); if (queryBuilder != null) {
queryBuilder.toXContent(builder, params); builder.field("query");
queryBuilder.toXContent(builder, params);
} else if (filterBuilder != null) {
builder.field("filter");
filterBuilder.toXContent(builder, params);
}
builder.field("parent_type", parentType); builder.field("parent_type", parentType);
if (scope != null) { if (scope != null) {
builder.field("_scope", scope); builder.field("_scope", scope);

View File

@ -23,6 +23,7 @@ import org.apache.lucene.search.Filter;
import org.apache.lucene.search.Query; import org.apache.lucene.search.Query;
import org.elasticsearch.common.Strings; import org.elasticsearch.common.Strings;
import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.inject.Inject;
import org.elasticsearch.common.lucene.search.XConstantScoreQuery;
import org.elasticsearch.common.lucene.search.XFilteredQuery; import org.elasticsearch.common.lucene.search.XFilteredQuery;
import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.index.mapper.DocumentMapper; import org.elasticsearch.index.mapper.DocumentMapper;
@ -73,6 +74,16 @@ public class HasParentFilterParser implements FilterParser {
} finally { } finally {
QueryParseContext.setTypes(origTypes); QueryParseContext.setTypes(origTypes);
} }
} else if ("filter".equals(currentFieldName)) {
// TODO handle `filter` element before `type` element...
String[] origTypes = QueryParseContext.setTypesWithPrevious(parentType == null ? null : new String[]{parentType});
try {
Filter innerFilter = parseContext.parseInnerFilter();
query = new XConstantScoreQuery(innerFilter);
queryFound = true;
} finally {
QueryParseContext.setTypes(origTypes);
}
} else { } else {
throw new QueryParsingException(parseContext.index(), "[has_parent] filter does not support [" + currentFieldName + "]"); throw new QueryParsingException(parseContext.index(), "[has_parent] filter does not support [" + currentFieldName + "]");
} }

View File

@ -40,8 +40,7 @@ import java.util.Map;
import static com.google.common.collect.Maps.newHashMap; import static com.google.common.collect.Maps.newHashMap;
import static org.elasticsearch.common.xcontent.XContentFactory.jsonBuilder; import static org.elasticsearch.common.xcontent.XContentFactory.jsonBuilder;
import static org.elasticsearch.index.query.FilterBuilders.hasChildFilter; import static org.elasticsearch.index.query.FilterBuilders.*;
import static org.elasticsearch.index.query.FilterBuilders.hasParentFilter;
import static org.elasticsearch.index.query.QueryBuilders.*; import static org.elasticsearch.index.query.QueryBuilders.*;
import static org.elasticsearch.search.facet.FacetBuilders.termsFacet; import static org.elasticsearch.search.facet.FacetBuilders.termsFacet;
import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.MatcherAssert.assertThat;
@ -824,7 +823,7 @@ public class SimpleChildQuerySearchTests extends AbstractNodesTests {
assertThat(searchResponse.failedShards(), equalTo(0)); assertThat(searchResponse.failedShards(), equalTo(0));
assertThat(searchResponse.hits().totalHits(), equalTo(1l)); assertThat(searchResponse.hits().totalHits(), equalTo(1l));
client.prepareSearch("test") searchResponse = client.prepareSearch("test")
.setQuery(filteredQuery(matchAllQuery(), hasParentFilter("parent", matchAllQuery()))) .setQuery(filteredQuery(matchAllQuery(), hasParentFilter("parent", matchAllQuery())))
.execute().actionGet(); .execute().actionGet();
assertThat("Failures " + Arrays.toString(searchResponse.shardFailures()), searchResponse.shardFailures().length, equalTo(0)); assertThat("Failures " + Arrays.toString(searchResponse.shardFailures()), searchResponse.shardFailures().length, equalTo(0));
@ -1138,4 +1137,49 @@ public class SimpleChildQuerySearchTests extends AbstractNodesTests {
assertThat(response.hits().totalHits(), equalTo(0l)); assertThat(response.hits().totalHits(), equalTo(0l));
} }
@Test
public void testHasChildAndHasParentFilter_withFilter() throws Exception {
client.admin().indices().prepareDelete().execute().actionGet();
client.admin().indices().prepareCreate("test").setSettings(
ImmutableSettings.settingsBuilder()
.put("index.number_of_shards", 1)
.put("index.number_of_replicas", 0)
).execute().actionGet();
client.admin().cluster().prepareHealth().setWaitForGreenStatus().execute().actionGet();
client.admin().indices().preparePutMapping("test").setType("child").setSource(
jsonBuilder()
.startObject()
.startObject("type")
.startObject("_parent")
.field("type", "parent")
.endObject()
.endObject()
.endObject()
).execute().actionGet();
client.prepareIndex("test", "parent", "1").setSource("p_field", 1).execute().actionGet();
client.prepareIndex("test", "child", "2").setParent("1").setSource("c_field", 1).execute().actionGet();
client.admin().indices().prepareFlush("test").execute().actionGet();
client.prepareIndex("test", "type1", "3").setSource("p_field", "p_value1").execute().actionGet();
client.admin().indices().prepareFlush("test").execute().actionGet();
SearchResponse searchResponse = client.prepareSearch("test")
.setQuery(filteredQuery(matchAllQuery(), hasChildFilter("child", termFilter("c_field", 1))))
.execute().actionGet();
assertThat("Failures " + Arrays.toString(searchResponse.shardFailures()), searchResponse.shardFailures().length, equalTo(0));
assertThat(searchResponse.failedShards(), equalTo(0));
assertThat(searchResponse.hits().totalHits(), equalTo(1l));
assertThat(searchResponse.hits().hits()[0].id(), equalTo("1"));
searchResponse = client.prepareSearch("test")
.setQuery(filteredQuery(matchAllQuery(), hasParentFilter("parent", termFilter("p_field", 1))))
.execute().actionGet();
assertThat("Failures " + Arrays.toString(searchResponse.shardFailures()), searchResponse.shardFailures().length, equalTo(0));
assertThat(searchResponse.failedShards(), equalTo(0));
assertThat(searchResponse.hits().totalHits(), equalTo(1l));
assertThat(searchResponse.hits().hits()[0].id(), equalTo("2"));
}
} }