Refactors ScriptQueryBuilder and Parser

Relates to #10217
Closes #12115

This PR is against the query-refactoring branch.
This commit is contained in:
Alex Ksikes 2015-07-08 08:27:53 +02:00
parent 4d4dc5cceb
commit 7099e6ca93
3 changed files with 197 additions and 101 deletions

View File

@ -19,24 +19,44 @@
package org.elasticsearch.index.query; package org.elasticsearch.index.query;
import org.apache.lucene.index.LeafReaderContext;
import org.apache.lucene.search.IndexSearcher;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.RandomAccessWeight;
import org.apache.lucene.search.Weight;
import org.apache.lucene.util.Bits;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.io.stream.StreamInput;
import org.elasticsearch.common.io.stream.StreamOutput;
import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentBuilder;
import org.elasticsearch.script.Script; import org.elasticsearch.script.*;
import org.elasticsearch.script.Script.ScriptField; import org.elasticsearch.script.Script.ScriptField;
import org.elasticsearch.search.lookup.SearchLookup;
import java.io.IOException; import java.io.IOException;
import java.util.Objects;
public class ScriptQueryBuilder extends AbstractQueryBuilder<ScriptQueryBuilder> { public class ScriptQueryBuilder extends AbstractQueryBuilder<ScriptQueryBuilder> {
public static final String NAME = "script"; public static final String NAME = "script";
static final ScriptQueryBuilder PROTOTYPE = new ScriptQueryBuilder((Script) null); static final ScriptQueryBuilder PROTOTYPE = new ScriptQueryBuilder(null);
private Script script;
private final Script script;
public ScriptQueryBuilder(Script script) { public ScriptQueryBuilder(Script script) {
this.script = script; this.script = script;
} }
public Script script() {
return this.script;
}
@Override
public String getName() {
return NAME;
}
@Override @Override
protected void doXContent(XContentBuilder builder, Params builderParams) throws IOException { protected void doXContent(XContentBuilder builder, Params builderParams) throws IOException {
builder.startObject(NAME); builder.startObject(NAME);
@ -46,7 +66,109 @@ public class ScriptQueryBuilder extends AbstractQueryBuilder<ScriptQueryBuilder>
} }
@Override @Override
public String getName() { protected Query doToQuery(QueryParseContext parseContext) throws IOException {
return NAME; return new ScriptQuery(script, parseContext.scriptService(), parseContext.lookup());
}
@Override
public QueryValidationException validate() {
QueryValidationException validationException = null;
if (this.script == null) {
validationException = addValidationError("script cannot be null", validationException);
}
return validationException;
}
static class ScriptQuery extends Query {
private final Script script;
private final SearchScript searchScript;
public ScriptQuery(Script script, ScriptService scriptService, SearchLookup searchLookup) {
this.script = script;
this.searchScript = scriptService.search(searchLookup, script, ScriptContext.Standard.SEARCH);
}
@Override
public String toString(String field) {
StringBuilder buffer = new StringBuilder();
buffer.append("ScriptFilter(");
buffer.append(script);
buffer.append(")");
return buffer.toString();
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (!super.equals(obj))
return false;
ScriptQuery other = (ScriptQuery) obj;
return Objects.equals(script, other.script);
}
@Override
public int hashCode() {
final int prime = 31;
int result = super.hashCode();
result = prime * result + Objects.hashCode(script);
return result;
}
@Override
public Weight createWeight(IndexSearcher searcher, boolean needsScores) throws IOException {
return new RandomAccessWeight(this) {
@Override
protected Bits getMatchingDocs(final LeafReaderContext context) throws IOException {
final LeafSearchScript leafScript = searchScript.getLeafSearchScript(context);
return new Bits() {
@Override
public boolean get(int doc) {
leafScript.setDocument(doc);
Object val = leafScript.run();
if (val == null) {
return false;
}
if (val instanceof Boolean) {
return (Boolean) val;
}
if (val instanceof Number) {
return ((Number) val).longValue() != 0;
}
throw new IllegalArgumentException("Can't handle type [" + val + "] in script filter");
}
@Override
public int length() {
return context.reader().maxDoc();
}
};
}
};
}
}
@Override
protected ScriptQueryBuilder doReadFrom(StreamInput in) throws IOException {
return new ScriptQueryBuilder(Script.readScript(in));
}
@Override
protected void doWriteTo(StreamOutput out) throws IOException {
script.writeTo(out);
}
@Override
protected int doHashCode() {
return Objects.hash(script);
}
@Override
protected boolean doEquals(ScriptQueryBuilder other) {
return Objects.equals(script, other.script);
} }
} }

View File

@ -19,20 +19,12 @@
package org.elasticsearch.index.query; package org.elasticsearch.index.query;
import com.google.common.base.Objects;
import org.apache.lucene.index.LeafReaderContext;
import org.apache.lucene.search.IndexSearcher;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.RandomAccessWeight;
import org.apache.lucene.search.Weight;
import org.apache.lucene.util.Bits;
import org.elasticsearch.common.ParseField;
import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.inject.Inject;
import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.script.*; import org.elasticsearch.script.Script;
import org.elasticsearch.script.Script.ScriptField; import org.elasticsearch.script.Script.ScriptField;
import org.elasticsearch.script.ScriptParameterParser;
import org.elasticsearch.script.ScriptParameterParser.ScriptParameterValue; import org.elasticsearch.script.ScriptParameterParser.ScriptParameterValue;
import org.elasticsearch.search.lookup.SearchLookup;
import java.io.IOException; import java.io.IOException;
import java.util.Map; import java.util.Map;
@ -42,7 +34,7 @@ import static com.google.common.collect.Maps.newHashMap;
/** /**
* *
*/ */
public class ScriptQueryParser extends BaseQueryParserTemp { public class ScriptQueryParser extends BaseQueryParser {
@Inject @Inject
public ScriptQueryParser() { public ScriptQueryParser() {
@ -54,20 +46,19 @@ public class ScriptQueryParser extends BaseQueryParserTemp {
} }
@Override @Override
public Query parse(QueryParseContext parseContext) throws IOException, QueryParsingException { public QueryBuilder fromXContent(QueryParseContext parseContext) throws IOException, QueryParsingException {
XContentParser parser = parseContext.parser(); XContentParser parser = parseContext.parser();
ScriptParameterParser scriptParameterParser = new ScriptParameterParser(); ScriptParameterParser scriptParameterParser = new ScriptParameterParser();
XContentParser.Token token;
// also, when caching, since its isCacheable is false, will result in loading all bit set... // also, when caching, since its isCacheable is false, will result in loading all bit set...
Script script = null; Script script = null;
Map<String, Object> params = null; Map<String, Object> params = null;
float boost = AbstractQueryBuilder.DEFAULT_BOOST; float boost = AbstractQueryBuilder.DEFAULT_BOOST;
String queryName = null; String queryName = null;
String currentFieldName = null;
XContentParser.Token token;
String currentFieldName = null;
while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) { while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
if (token == XContentParser.Token.FIELD_NAME) { if (token == XContentParser.Token.FIELD_NAME) {
currentFieldName = parser.currentName(); currentFieldName = parser.currentName();
@ -108,85 +99,9 @@ public class ScriptQueryParser extends BaseQueryParserTemp {
throw new QueryParsingException(parseContext, "script must be provided with a [script] filter"); throw new QueryParsingException(parseContext, "script must be provided with a [script] filter");
} }
Query query = new ScriptQuery(script, parseContext.scriptService(), parseContext.lookup()); return new ScriptQueryBuilder(script)
if (queryName != null) { .boost(boost)
parseContext.addNamedQuery(queryName, query); .queryName(queryName);
}
query.setBoost(boost);
return query;
}
static class ScriptQuery extends Query {
private final Script script;
private final SearchScript searchScript;
public ScriptQuery(Script script, ScriptService scriptService, SearchLookup searchLookup) {
this.script = script;
this.searchScript = scriptService.search(searchLookup, script, ScriptContext.Standard.SEARCH);
}
@Override
public String toString(String field) {
StringBuilder buffer = new StringBuilder();
buffer.append("ScriptFilter(");
buffer.append(script);
buffer.append(")");
return buffer.toString();
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (!super.equals(obj))
return false;
ScriptQuery other = (ScriptQuery) obj;
return Objects.equal(script, other.script);
}
@Override
public int hashCode() {
final int prime = 31;
int result = super.hashCode();
result = prime * result + Objects.hashCode(script);
return result;
}
@Override
public Weight createWeight(IndexSearcher searcher, boolean needsScores) throws IOException {
return new RandomAccessWeight(this) {
@Override
protected Bits getMatchingDocs(final LeafReaderContext context) throws IOException {
final LeafSearchScript leafScript = searchScript.getLeafSearchScript(context);
return new Bits() {
@Override
public boolean get(int doc) {
leafScript.setDocument(doc);
Object val = leafScript.run();
if (val == null) {
return false;
}
if (val instanceof Boolean) {
return (Boolean) val;
}
if (val instanceof Number) {
return ((Number) val).longValue() != 0;
}
throw new IllegalArgumentException("Can't handle type [" + val + "] in script filter");
}
@Override
public int length() {
return context.reader().maxDoc();
}
};
}
};
}
} }
@Override @Override

View File

@ -0,0 +1,59 @@
/*
* Licensed to Elasticsearch under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch licenses this file to you under
* the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.elasticsearch.index.query;
import org.apache.lucene.search.Query;
import org.elasticsearch.script.Script;
import org.elasticsearch.script.ScriptService.ScriptType;
import org.junit.Test;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import static org.hamcrest.Matchers.is;
public class ScriptQueryBuilderTest extends BaseQueryTestCase<ScriptQueryBuilder> {
@Override
protected ScriptQueryBuilder doCreateTestQueryBuilder() {
String script;
Map<String, Object> params = null;
if (randomBoolean()) {
script = "5 * 2 > param";
params = new HashMap<>();
params.put("param", 1);
} else {
script = "5 * 2 > 2";
}
return new ScriptQueryBuilder(new Script(script, ScriptType.INLINE, "expression", params));
}
@Override
protected Query doCreateExpectedQuery(ScriptQueryBuilder queryBuilder, QueryParseContext context) throws IOException {
return new ScriptQueryBuilder.ScriptQuery(queryBuilder.script(), context.scriptService(), context.lookup());
}
@Test
public void testValidate() {
ScriptQueryBuilder scriptQueryBuilder = new ScriptQueryBuilder(null);
assertThat(scriptQueryBuilder.validate().validationErrors().size(), is(1));
}
}