|
@@ -1,229 +0,0 @@
|
|
|
-/*
|
|
|
- * 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.script.mustache;
|
|
|
-
|
|
|
-import org.apache.lucene.index.memory.MemoryIndex;
|
|
|
-import org.apache.lucene.search.MatchAllDocsQuery;
|
|
|
-import org.apache.lucene.search.Query;
|
|
|
-import org.elasticsearch.common.bytes.BytesArray;
|
|
|
-import org.elasticsearch.common.xcontent.XContentBuilder;
|
|
|
-import org.elasticsearch.common.xcontent.XContentFactory;
|
|
|
-import org.elasticsearch.common.xcontent.XContentType;
|
|
|
-import org.elasticsearch.index.query.BoolQueryBuilder;
|
|
|
-import org.elasticsearch.index.query.MatchAllQueryBuilder;
|
|
|
-import org.elasticsearch.index.query.MatchQueryBuilder;
|
|
|
-import org.elasticsearch.index.query.QueryBuilder;
|
|
|
-import org.elasticsearch.index.query.QueryShardContext;
|
|
|
-import org.elasticsearch.index.query.TermQueryBuilder;
|
|
|
-import org.elasticsearch.plugins.Plugin;
|
|
|
-import org.elasticsearch.script.MockScriptPlugin;
|
|
|
-import org.elasticsearch.script.Script;
|
|
|
-import org.elasticsearch.script.ScriptType;
|
|
|
-import org.elasticsearch.search.internal.SearchContext;
|
|
|
-import org.elasticsearch.test.AbstractQueryTestCase;
|
|
|
-import org.junit.After;
|
|
|
-import org.junit.Before;
|
|
|
-
|
|
|
-import java.io.IOException;
|
|
|
-import java.util.Arrays;
|
|
|
-import java.util.Collection;
|
|
|
-import java.util.Collections;
|
|
|
-import java.util.HashMap;
|
|
|
-import java.util.Map;
|
|
|
-import java.util.function.Function;
|
|
|
-
|
|
|
-import static org.hamcrest.Matchers.containsString;
|
|
|
-
|
|
|
-public class TemplateQueryBuilderTests extends AbstractQueryTestCase<TemplateQueryBuilder> {
|
|
|
-
|
|
|
- /**
|
|
|
- * The query type all template tests will be based on.
|
|
|
- */
|
|
|
- private QueryBuilder templateBase;
|
|
|
-
|
|
|
- /**
|
|
|
- * All tests in this class cause deprecation warnings when a new {@link TemplateQueryBuilder} is created.
|
|
|
- * Instead of having to check them in every single test, we do it after each test is run
|
|
|
- */
|
|
|
- @After
|
|
|
- public void checkWarning() {
|
|
|
- assertWarnings("[template] query is deprecated, use search template api instead");
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- protected Collection<Class<? extends Plugin>> getPlugins() {
|
|
|
- return Arrays.asList(MustachePlugin.class, CustomScriptPlugin.class);
|
|
|
- }
|
|
|
-
|
|
|
- public static class CustomScriptPlugin extends MockScriptPlugin {
|
|
|
-
|
|
|
- @Override
|
|
|
- @SuppressWarnings("unchecked")
|
|
|
- protected Map<String, Function<Map<String, Object>, Object>> pluginScripts() {
|
|
|
- Map<String, Function<Map<String, Object>, Object>> scripts = new HashMap<>();
|
|
|
-
|
|
|
- scripts.put("{ \"match_all\" : {}}",
|
|
|
- s -> new BytesArray("{ \"match_all\" : {}}"));
|
|
|
-
|
|
|
- scripts.put("{ \"match_all\" : {\"_name\" : \"foobar\"}}",
|
|
|
- s -> new BytesArray("{ \"match_all\" : {\"_name\" : \"foobar\"}}"));
|
|
|
-
|
|
|
- scripts.put("{\n" +
|
|
|
- " \"term\" : {\n" +
|
|
|
- " \"foo\" : {\n" +
|
|
|
- " \"value\" : \"bar\",\n" +
|
|
|
- " \"boost\" : 2.0\n" +
|
|
|
- " }\n" +
|
|
|
- " }\n" +
|
|
|
- "}", s -> new BytesArray("{\n" +
|
|
|
- " \"term\" : {\n" +
|
|
|
- " \"foo\" : {\n" +
|
|
|
- " \"value\" : \"bar\",\n" +
|
|
|
- " \"boost\" : 2.0\n" +
|
|
|
- " }\n" +
|
|
|
- " }\n" +
|
|
|
- "}"));
|
|
|
- return scripts;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- @Before
|
|
|
- public void setup() {
|
|
|
- templateBase = new MatchQueryBuilder("field", "some values");
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- protected boolean supportsBoostAndQueryName() {
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- protected TemplateQueryBuilder doCreateTestQueryBuilder() {
|
|
|
- return new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mustache", templateBase.toString(), Collections.emptyMap()));
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- protected void doAssertLuceneQuery(TemplateQueryBuilder queryBuilder, Query query, SearchContext context) throws IOException {
|
|
|
- QueryShardContext queryShardContext = context.getQueryShardContext();
|
|
|
- assertEquals(rewrite(QueryBuilder.rewriteQuery(templateBase, queryShardContext).toQuery(queryShardContext)), rewrite(query));
|
|
|
- }
|
|
|
-
|
|
|
- public void testIllegalArgument() {
|
|
|
- expectThrows(IllegalArgumentException.class, () -> new TemplateQueryBuilder((Script) null));
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Override superclass test since template query doesn't support boost and queryName, so
|
|
|
- * we need to mutate other existing field in the test query.
|
|
|
- */
|
|
|
- @Override
|
|
|
- public void testUnknownField() throws IOException {
|
|
|
- TemplateQueryBuilder testQuery = createTestQueryBuilder();
|
|
|
- XContentType xContentType = randomFrom(XContentType.JSON, XContentType.YAML);
|
|
|
- String testQueryAsString = toXContent(testQuery, xContentType).string();
|
|
|
- String queryAsString = testQueryAsString.replace("inline", "bogusField");
|
|
|
- try {
|
|
|
- parseQuery(createParser(xContentType.xContent(), queryAsString));
|
|
|
- fail("IllegalArgumentException expected");
|
|
|
- } catch (IllegalArgumentException e) {
|
|
|
- assertThat(e.getMessage(), containsString("[script] unknown field [bogusField], parser not found"));
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- public void testJSONGeneration() throws IOException {
|
|
|
- Map<String, Object> vars = new HashMap<>();
|
|
|
- vars.put("template", "filled");
|
|
|
- TemplateQueryBuilder builder = new TemplateQueryBuilder("I am a $template string", ScriptType.INLINE, vars);
|
|
|
- XContentBuilder content = XContentFactory.jsonBuilder();
|
|
|
- content.startObject();
|
|
|
- builder.doXContent(content, null);
|
|
|
- content.endObject();
|
|
|
- content.close();
|
|
|
- assertEquals("{\"template\":{\"inline\":\"I am a $template string\",\"lang\":\"mustache\",\"params\":{\"template\":\"filled\"}}}",
|
|
|
- content.string());
|
|
|
- }
|
|
|
-
|
|
|
- public void testRawEscapedTemplate() throws IOException {
|
|
|
- String expectedTemplateString = "{\"match_{{template}}\": {}}\"";
|
|
|
- String query = "{\"template\": {\"inline\": \"{\\\"match_{{template}}\\\": {}}\\\"\",\"params\" : {\"template\" : \"all\"}}}";
|
|
|
- Map<String, Object> params = new HashMap<>();
|
|
|
- params.put("template", "all");
|
|
|
- QueryBuilder expectedBuilder = new TemplateQueryBuilder(expectedTemplateString, ScriptType.INLINE, params);
|
|
|
- assertParsedQuery(query, expectedBuilder);
|
|
|
- }
|
|
|
-
|
|
|
- public void testRawTemplate() throws IOException {
|
|
|
- String expectedTemplateString = "{\"match_{{template}}\":{}}";
|
|
|
- String query = "{\"template\": {\"inline\": {\"match_{{template}}\": {}},\"params\" : {\"template\" : \"all\"}}}";
|
|
|
- Map<String, Object> params = new HashMap<>();
|
|
|
- params.put("template", "all");
|
|
|
- QueryBuilder expectedBuilder = new TemplateQueryBuilder(expectedTemplateString, ScriptType.INLINE, params, XContentType.JSON);
|
|
|
- assertParsedQuery(query, expectedBuilder);
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void testMustRewrite() throws IOException {
|
|
|
- String query = "{ \"match_all\" : {}}";
|
|
|
- QueryBuilder builder = new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mockscript", query,
|
|
|
- Collections.singletonMap(Script.CONTENT_TYPE_OPTION, XContentType.JSON.mediaType()), Collections.emptyMap()));
|
|
|
- try {
|
|
|
- builder.toQuery(createShardContext());
|
|
|
- fail();
|
|
|
- } catch (UnsupportedOperationException ex) {
|
|
|
- assertEquals("this query must be rewritten first", ex.getMessage());
|
|
|
- }
|
|
|
- assertEquals(new MatchAllQueryBuilder(), builder.rewrite(createShardContext()));
|
|
|
- }
|
|
|
-
|
|
|
- public void testRewriteWithInnerName() throws IOException {
|
|
|
- final String query = "{ \"match_all\" : {\"_name\" : \"foobar\"}}";
|
|
|
- QueryBuilder builder = new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mockscript", query,
|
|
|
- Collections.singletonMap(Script.CONTENT_TYPE_OPTION, XContentType.JSON.mediaType()), Collections.emptyMap()));
|
|
|
- assertEquals(new MatchAllQueryBuilder().queryName("foobar"), builder.rewrite(createShardContext()));
|
|
|
-
|
|
|
- builder = new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mockscript", query, Collections.singletonMap(
|
|
|
- Script.CONTENT_TYPE_OPTION, XContentType.JSON.mediaType()), Collections.emptyMap())).queryName("outer");
|
|
|
- assertEquals(new BoolQueryBuilder().must(new MatchAllQueryBuilder().queryName("foobar")).queryName("outer"),
|
|
|
- builder.rewrite(createShardContext()));
|
|
|
- }
|
|
|
-
|
|
|
- public void testRewriteWithInnerBoost() throws IOException {
|
|
|
- final TermQueryBuilder query = new TermQueryBuilder("foo", "bar").boost(2);
|
|
|
- QueryBuilder builder = new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mockscript", query.toString(),
|
|
|
- Collections.singletonMap(Script.CONTENT_TYPE_OPTION, XContentType.JSON.mediaType()), Collections.emptyMap()));
|
|
|
- assertEquals(query, builder.rewrite(createShardContext()));
|
|
|
-
|
|
|
- builder = new TemplateQueryBuilder(new Script(ScriptType.INLINE, "mockscript", query.toString(),
|
|
|
- Collections.singletonMap(Script.CONTENT_TYPE_OPTION, XContentType.JSON.mediaType()), Collections.emptyMap())).boost(3);
|
|
|
- assertEquals(new BoolQueryBuilder().must(query).boost(3), builder.rewrite(createShardContext()));
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- protected Query rewrite(Query query) throws IOException {
|
|
|
- // TemplateQueryBuilder adds some optimization if the template and query builder have boosts / query names that wraps
|
|
|
- // the actual QueryBuilder that comes from the template into a BooleanQueryBuilder to give it an outer boost / name
|
|
|
- // this causes some queries to be not exactly equal but equivalent such that we need to rewrite them before comparing.
|
|
|
- if (query != null) {
|
|
|
- MemoryIndex idx = new MemoryIndex();
|
|
|
- return idx.createSearcher().rewrite(query);
|
|
|
- }
|
|
|
- return new MatchAllDocsQuery(); // null == *:*
|
|
|
- }
|
|
|
-}
|