/*
 * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 * or more contributor license agreements. Licensed under the "Elastic License
 * 2.0", the "GNU Affero General Public License v3.0 only", and the "Server Side
 * Public License v 1"; you may not use this file except in compliance with, at
 * your election, the "Elastic License 2.0", the "GNU Affero General Public
 * License v3.0 only", or the "Server Side Public License, v 1".
 */

package org.elasticsearch.script.mustache;

import org.elasticsearch.TransportVersions;
import org.elasticsearch.action.index.IndexRequestBuilder;
import org.elasticsearch.action.search.SearchRequest;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.index.IndexNotFoundException;
import org.elasticsearch.plugins.Plugin;
import org.elasticsearch.script.ScriptType;
import org.elasticsearch.script.mustache.MultiSearchTemplateResponse.Item;
import org.elasticsearch.search.DummyQueryParserPlugin;
import org.elasticsearch.search.FailBeforeCurrentVersionQueryBuilder;
import org.elasticsearch.search.SearchService;
import org.elasticsearch.test.ESIntegTestCase;
import org.elasticsearch.xcontent.XContentParseException;
import org.elasticsearch.xcontent.json.JsonXContent;

import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHitCount;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertResponse;
import static org.elasticsearch.xcontent.XContentFactory.jsonBuilder;
import static org.hamcrest.Matchers.arrayWithSize;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.greaterThan;
import static org.hamcrest.Matchers.instanceOf;
import static org.hamcrest.core.Is.is;

public class MultiSearchTemplateIT extends ESIntegTestCase {

    @Override
    protected Collection<Class<? extends Plugin>> nodePlugins() {
        return List.of(MustachePlugin.class, DummyQueryParserPlugin.class);
    }

    @Override
    protected Settings nodeSettings(int nodeOrdinal, Settings otherSettings) {
        return Settings.builder()
            .put(super.nodeSettings(nodeOrdinal, otherSettings))
            .put(SearchService.CCS_VERSION_CHECK_SETTING.getKey(), "true")
            .build();
    }

    public void testBasic() throws Exception {
        createIndex("msearch");
        final int numDocs = randomIntBetween(10, 100);
        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[numDocs];
        for (int i = 0; i < numDocs; i++) {
            indexRequestBuilders[i] = prepareIndex("msearch").setId(String.valueOf(i)).setSource("odd", (i % 2 == 0), "group", (i % 3));
        }
        indexRandom(true, indexRequestBuilders);

        final String template = Strings.toString(
            jsonBuilder().startObject()
                .startObject("query")
                .startObject("{{query_type}}")
                .field("{{field_name}}", "{{field_value}}")
                .endObject()
                .endObject()
                .endObject()
        );

        MultiSearchTemplateRequest multiRequest = new MultiSearchTemplateRequest();

        // Search #1
        SearchTemplateRequest search1 = new SearchTemplateRequest();
        search1.setRequest(new SearchRequest("msearch"));
        search1.setScriptType(ScriptType.INLINE);
        search1.setScript(template);

        Map<String, Object> params1 = new HashMap<>();
        params1.put("query_type", "match");
        params1.put("field_name", "odd");
        params1.put("field_value", true);
        search1.setScriptParams(params1);
        multiRequest.add(search1);

        // Search #2 (Simulate is true)
        SearchTemplateRequest search2 = new SearchTemplateRequest();
        search2.setRequest(new SearchRequest("msearch"));
        search2.setScriptType(ScriptType.INLINE);
        search2.setScript(template);
        search2.setSimulate(true);

        Map<String, Object> params2 = new HashMap<>();
        params2.put("query_type", "match_phrase_prefix");
        params2.put("field_name", "message");
        params2.put("field_value", "quick brown f");
        search2.setScriptParams(params2);
        multiRequest.add(search2);

        // Search #3
        SearchTemplateRequest search3 = new SearchTemplateRequest();
        search3.setRequest(new SearchRequest("msearch"));
        search3.setScriptType(ScriptType.INLINE);
        search3.setScript(template);
        search3.setSimulate(false);

        Map<String, Object> params3 = new HashMap<>();
        params3.put("query_type", "term");
        params3.put("field_name", "odd");
        params3.put("field_value", "false");
        search3.setScriptParams(params3);
        multiRequest.add(search3);

        // Search #4 (Fail because of unknown index)
        SearchTemplateRequest search4 = new SearchTemplateRequest();
        search4.setRequest(new SearchRequest("unknown"));
        search4.setScriptType(ScriptType.INLINE);
        search4.setScript(template);

        Map<String, Object> params4 = new HashMap<>();
        params4.put("query_type", "match");
        params4.put("field_name", "group");
        params4.put("field_value", "test");
        search4.setScriptParams(params4);
        multiRequest.add(search4);

        // Search #5 (Simulate is true)
        SearchTemplateRequest search5 = new SearchTemplateRequest();
        search5.setRequest(new SearchRequest("msearch"));
        search5.setScriptType(ScriptType.INLINE);
        search5.setScript("{{! ignore me }}{\"query\":{\"terms\":{\"group\":[{{#groups}}{{.}},{{/groups}}]}}}");
        search5.setSimulate(true);

        Map<String, Object> params5 = new HashMap<>();
        params5.put("groups", Arrays.asList(1, 2, 3));
        search5.setScriptParams(params5);
        multiRequest.add(search5);

        assertResponse(client().execute(MustachePlugin.MULTI_SEARCH_TEMPLATE_ACTION, multiRequest), response -> {
            assertThat(response.getResponses(), arrayWithSize(5));
            assertThat(response.getTook().millis(), greaterThan(0L));

            MultiSearchTemplateResponse.Item response1 = response.getResponses()[0];
            assertThat(response1.isFailure(), is(false));
            SearchTemplateResponse searchTemplateResponse1 = response1.getResponse();
            assertThat(searchTemplateResponse1.hasResponse(), is(true));
            assertHitCount(searchTemplateResponse1.getResponse(), (numDocs / 2) + (numDocs % 2));
            assertThat(searchTemplateResponse1.getSource().utf8ToString(), equalTo("""
                {"query":{"match":{"odd":"true"}}}"""));

            MultiSearchTemplateResponse.Item response2 = response.getResponses()[1];
            assertThat(response2.isFailure(), is(false));
            SearchTemplateResponse searchTemplateResponse2 = response2.getResponse();
            assertThat(searchTemplateResponse2.hasResponse(), is(false));
            assertThat(searchTemplateResponse2.getSource().utf8ToString(), equalTo("""
                {"query":{"match_phrase_prefix":{"message":"quick brown f"}}}"""));

            MultiSearchTemplateResponse.Item response3 = response.getResponses()[2];
            assertThat(response3.isFailure(), is(false));
            SearchTemplateResponse searchTemplateResponse3 = response3.getResponse();
            assertThat(searchTemplateResponse3.hasResponse(), is(true));
            assertHitCount(searchTemplateResponse3.getResponse(), (numDocs / 2));
            assertThat(searchTemplateResponse3.getSource().utf8ToString(), equalTo("""
                {"query":{"term":{"odd":"false"}}}"""));

            MultiSearchTemplateResponse.Item response4 = response.getResponses()[3];
            assertThat(response4.isFailure(), is(true));
            assertThat(response4.getFailure(), instanceOf(IndexNotFoundException.class));
            assertThat(response4.getFailure().getMessage(), equalTo("no such index [unknown]"));

            MultiSearchTemplateResponse.Item response5 = response.getResponses()[4];
            assertThat(response5.isFailure(), is(true));
            assertNull(response5.getResponse());
            assertThat(response5.getFailure(), instanceOf(XContentParseException.class));
        });
    }

    /**
    * Test that triggering the CCS compatibility check with a query that shouldn't go to the minor before TransportVersion.current() works
    */
    public void testCCSCheckCompatibility() throws Exception {
        String templateString = """
            {
            "source": "{ \\"query\\":{\\"fail_before_current_version\\":{}} }"
            }""";
        SearchTemplateRequest searchTemplateRequest = SearchTemplateRequest.fromXContent(
            createParser(JsonXContent.jsonXContent, templateString)
        );
        searchTemplateRequest.setRequest(new SearchRequest());
        MultiSearchTemplateRequest request = new MultiSearchTemplateRequest();
        request.add(searchTemplateRequest);
        assertResponse(client().execute(MustachePlugin.MULTI_SEARCH_TEMPLATE_ACTION, request), multiSearchTemplateResponse -> {
            Item response = multiSearchTemplateResponse.getResponses()[0];
            assertTrue(response.isFailure());
            Exception ex = response.getFailure();
            assertThat(
                ex.getMessage(),
                containsString("[class org.elasticsearch.action.search.SearchRequest] is not compatible with version")
            );
            assertThat(ex.getMessage(), containsString("'search.check_ccs_compatibility' setting is enabled."));

            String expectedCause = Strings.format(
                "[fail_before_current_version] was released first in version %s, failed compatibility "
                    + "check trying to send it to node with version %s",
                FailBeforeCurrentVersionQueryBuilder.FUTURE_VERSION.toReleaseVersion(),
                TransportVersions.MINIMUM_CCS_VERSION.toReleaseVersion()
            );
            String actualCause = ex.getCause().getMessage();
            assertEquals(expectedCause, actualCause);
        });
    }
}
