/*
 * 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.search.fetch.subphase.highlight;

import com.carrotsearch.randomizedtesting.generators.RandomPicks;

import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.analysis.TokenFilter;
import org.apache.lucene.analysis.TokenStream;
import org.apache.lucene.analysis.tokenattributes.CharTermAttribute;
import org.apache.lucene.search.join.ScoreMode;
import org.apache.lucene.tests.analysis.MockAnalyzer;
import org.apache.lucene.tests.analysis.MockTokenizer;
import org.elasticsearch.action.index.IndexRequestBuilder;
import org.elasticsearch.action.search.SearchRequest;
import org.elasticsearch.action.search.SearchRequestBuilder;
import org.elasticsearch.action.support.WriteRequest;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.geo.GeoPoint;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.settings.Settings.Builder;
import org.elasticsearch.common.time.DateFormatter;
import org.elasticsearch.common.util.Maps;
import org.elasticsearch.index.analysis.AbstractIndexAnalyzerProvider;
import org.elasticsearch.index.analysis.AnalyzerProvider;
import org.elasticsearch.index.analysis.PreConfiguredTokenFilter;
import org.elasticsearch.index.query.AbstractQueryBuilder;
import org.elasticsearch.index.query.CombinedFieldsQueryBuilder;
import org.elasticsearch.index.query.IdsQueryBuilder;
import org.elasticsearch.index.query.MatchQueryBuilder;
import org.elasticsearch.index.query.MultiMatchQueryBuilder;
import org.elasticsearch.index.query.QueryBuilder;
import org.elasticsearch.index.query.QueryBuilders;
import org.elasticsearch.index.query.functionscore.FunctionScoreQueryBuilder;
import org.elasticsearch.index.query.functionscore.RandomScoreFunctionBuilder;
import org.elasticsearch.indices.analysis.AnalysisModule;
import org.elasticsearch.plugins.AnalysisPlugin;
import org.elasticsearch.plugins.Plugin;
import org.elasticsearch.rest.RestStatus;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.elasticsearch.search.fetch.subphase.highlight.HighlightBuilder.BoundaryScannerType;
import org.elasticsearch.search.fetch.subphase.highlight.HighlightBuilder.Field;
import org.elasticsearch.search.sort.SortBuilders;
import org.elasticsearch.search.sort.SortOrder;
import org.elasticsearch.test.ESIntegTestCase;
import org.elasticsearch.test.InternalSettingsPlugin;
import org.elasticsearch.test.MockKeywordPlugin;
import org.elasticsearch.xcontent.XContentBuilder;
import org.elasticsearch.xcontent.XContentFactory;
import org.hamcrest.Matcher;
import org.hamcrest.Matchers;

import java.io.IOException;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.ExecutionException;

import static java.util.Collections.singletonList;
import static java.util.Collections.singletonMap;
import static org.elasticsearch.index.query.QueryBuilders.boolQuery;
import static org.elasticsearch.index.query.QueryBuilders.boostingQuery;
import static org.elasticsearch.index.query.QueryBuilders.combinedFieldsQuery;
import static org.elasticsearch.index.query.QueryBuilders.constantScoreQuery;
import static org.elasticsearch.index.query.QueryBuilders.existsQuery;
import static org.elasticsearch.index.query.QueryBuilders.fuzzyQuery;
import static org.elasticsearch.index.query.QueryBuilders.matchPhrasePrefixQuery;
import static org.elasticsearch.index.query.QueryBuilders.matchPhraseQuery;
import static org.elasticsearch.index.query.QueryBuilders.matchQuery;
import static org.elasticsearch.index.query.QueryBuilders.multiMatchQuery;
import static org.elasticsearch.index.query.QueryBuilders.nestedQuery;
import static org.elasticsearch.index.query.QueryBuilders.prefixQuery;
import static org.elasticsearch.index.query.QueryBuilders.queryStringQuery;
import static org.elasticsearch.index.query.QueryBuilders.rangeQuery;
import static org.elasticsearch.index.query.QueryBuilders.regexpQuery;
import static org.elasticsearch.index.query.QueryBuilders.termQuery;
import static org.elasticsearch.index.query.QueryBuilders.wildcardQuery;
import static org.elasticsearch.search.builder.SearchSourceBuilder.highlight;
import static org.elasticsearch.search.builder.SearchSourceBuilder.searchSource;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertAcked;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertFailures;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHighlight;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHitCount;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHitCountAndNoFailures;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertNoFailures;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertNoFailuresAndResponse;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertNotHighlighted;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertResponse;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertResponses;
import static org.elasticsearch.xcontent.XContentFactory.jsonBuilder;
import static org.hamcrest.Matchers.anyOf;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.hasKey;
import static org.hamcrest.Matchers.lessThan;
import static org.hamcrest.Matchers.not;
import static org.hamcrest.Matchers.startsWith;

public class HighlighterSearchIT extends ESIntegTestCase {
    // TODO as we move analyzers out of the core we need to move some of these into HighlighterWithAnalyzersTests
    private static final String[] ALL_TYPES = new String[] { "plain", "fvh", "unified" };

    @Override
    protected Collection<Class<? extends Plugin>> nodePlugins() {
        return Arrays.asList(InternalSettingsPlugin.class, MockKeywordPlugin.class, MockAnalysisPlugin.class);
    }

    public void testHighlightingWithKeywordIgnoreBoundaryScanner() throws IOException {
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("tags")
            .field("type", "keyword")
            .endObject()
            .startObject("sort")
            .field("type", "long")
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));
        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().array("tags", "foo bar", "foo bar", "foo bar", "foo baz").field("sort", 1).endObject())
            .get();
        prepareIndex("test").setId("2")
            .setSource(jsonBuilder().startObject().array("tags", "foo baz", "foo baz", "foo baz", "foo bar").field("sort", 2).endObject())
            .get();
        refresh();

        for (BoundaryScannerType scanner : BoundaryScannerType.values()) {
            assertResponse(
                prepareSearch().addSort(SortBuilders.fieldSort("sort"))
                    .setQuery(matchQuery("tags", "foo bar"))
                    .highlighter(new HighlightBuilder().field(new Field("tags")).numOfFragments(2).boundaryScannerType(scanner)),
                response -> {
                    assertHighlight(response, 0, "tags", 0, 2, equalTo("<em>foo bar</em>"));
                    assertHighlight(response, 0, "tags", 1, 2, equalTo("<em>foo bar</em>"));
                    assertHighlight(response, 1, "tags", 0, 1, equalTo("<em>foo bar</em>"));
                }
            );
        }
    }

    public void testHighlightingWithStoredKeyword() throws IOException {
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("text")
            .field("type", "keyword")
            .field("store", true)
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));
        prepareIndex("test").setId("1").setSource(jsonBuilder().startObject().field("text", "foo").endObject()).get();
        refresh();
        assertResponse(
            prepareSearch().setQuery(matchQuery("text", "foo")).highlighter(new HighlightBuilder().field(new Field("text"))),
            response -> assertHighlight(response, 0, "text", 0, equalTo("<em>foo</em>"))
        );
    }

    public void testHighlightingWithWildcardName() throws IOException {
        // test the kibana case with * as fieldname that will try highlight all fields including meta fields
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("text")
            .field("type", "text")
            .field("analyzer", "keyword")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));
        prepareIndex("test").setId("1").setSource(jsonBuilder().startObject().field("text", "text").endObject()).get();
        refresh();
        for (String type : ALL_TYPES) {
            assertResponse(
                prepareSearch().setQuery(constantScoreQuery(matchQuery("text", "text")))
                    .highlighter(new HighlightBuilder().field(new Field("*").highlighterType(type))),
                response -> assertHighlight(response, 0, "text", 0, equalTo("<em>text</em>"))
            );
        }
    }

    public void testFieldAlias() throws IOException {
        XContentBuilder mappings = jsonBuilder().startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("text")
            .field("type", "text")
            .field("store", true)
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .startObject("alias")
            .field("type", "alias")
            .field("path", "text")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));

        prepareIndex("test").setId("1").setSource("text", "foo").get();
        refresh();

        for (String type : ALL_TYPES) {
            HighlightBuilder builder = new HighlightBuilder().field(new Field("alias").highlighterType(type))
                .requireFieldMatch(randomBoolean());
            assertResponse(
                prepareSearch().setQuery(matchQuery("alias", "foo")).highlighter(builder),
                response -> assertHighlight(response, 0, "alias", 0, equalTo("<em>foo</em>"))
            );
        }
    }

    public void testFieldAliasWithSourceLookup() throws IOException {
        XContentBuilder mappings = jsonBuilder().startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("text")
            .field("type", "text")
            .field("analyzer", "whitespace")
            .field("store", false)
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .startObject("alias")
            .field("type", "alias")
            .field("path", "text")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));

        prepareIndex("test").setId("1").setSource("text", "foo bar").get();
        refresh();

        for (String type : ALL_TYPES) {
            HighlightBuilder builder = new HighlightBuilder().field(new Field("alias").highlighterType(type))
                .requireFieldMatch(randomBoolean());
            assertResponse(
                prepareSearch().setQuery(matchQuery("alias", "bar")).highlighter(builder),
                response -> assertHighlight(response, 0, "alias", 0, equalTo("foo <em>bar</em>"))
            );
        }
    }

    public void testFieldAliasWithWildcardField() throws IOException {
        XContentBuilder mappings = jsonBuilder().startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("keyword")
            .field("type", "keyword")
            .endObject()
            .startObject("alias")
            .field("type", "alias")
            .field("path", "keyword")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));

        prepareIndex("test").setId("1").setSource("keyword", "foo").get();
        refresh();

        HighlightBuilder builder = new HighlightBuilder().field(new Field("al*")).requireFieldMatch(false);
        assertResponse(
            prepareSearch().setQuery(matchQuery("alias", "foo")).highlighter(builder),
            response -> assertHighlight(response, 0, "alias", 0, equalTo("<em>foo</em>"))
        );
    }

    public void testHighlightingWhenFieldsAreNotStoredThereIsNoSource() throws IOException {
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("_source")
            .field("enabled", false)
            .endObject()
            .startObject("properties")
            .startObject("unstored_field")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .field("type", "text")
            .field("store", false)
            .endObject()
            .startObject("text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .field("type", "text")
            .field("store", true)
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));
        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().field("unstored_text", "text").field("text", "text").endObject())
            .get();
        refresh();
        for (String type : ALL_TYPES) {
            assertResponse(
                prepareSearch().setQuery(constantScoreQuery(matchQuery("text", "text")))
                    .highlighter(new HighlightBuilder().field(new Field("*").highlighterType(type))),
                response -> assertHighlight(response, 0, "text", 0, equalTo("<em>text</em>"))
            );
            assertNoFailuresAndResponse(
                prepareSearch().setQuery(constantScoreQuery(matchQuery("text", "text")))
                    .highlighter(new HighlightBuilder().field(new Field("unstored_text"))),
                response -> assertThat(response.getHits().getAt(0).getHighlightFields().size(), equalTo(0))
            );
        }
    }

    // see #3486
    public void testHighTermFrequencyDoc() throws IOException {
        assertAcked(prepareCreate("test").setMapping("name", "type=text,term_vector=with_positions_offsets,store=" + randomBoolean()));
        StringBuilder builder = new StringBuilder();
        for (int i = 0; i < 6000; i++) {
            builder.append("abc").append(" ");
        }
        prepareIndex("test").setId("1").setSource("name", builder.toString()).get();
        refresh();
        assertResponse(
            prepareSearch().setQuery(constantScoreQuery(matchQuery("name", "abc"))).highlighter(new HighlightBuilder().field("name")),
            response -> {
                assertHighlight(response, 0, "name", 0, startsWith("<em>abc</em> <em>abc</em> <em>abc</em> <em>abc</em>"));
            }
        );
    }

    public void testEnsureNoNegativeOffsets() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                "no_long_term",
                "type=text,term_vector=with_positions_offsets",
                "long_term",
                "type=text,term_vector=with_positions_offsets"
            )
        );

        prepareIndex("test").setId("1")
            .setSource(
                "no_long_term",
                "This is a test where foo is highlighed and should be highlighted",
                "long_term",
                "This is a test thisisaverylongwordandmakessurethisfails where foo is highlighed and should be highlighted"
            )
            .get();
        refresh();
        assertHighlight(
            prepareSearch().setQuery(matchQuery("long_term", "thisisaverylongwordandmakessurethisfails foo highlighed"))
                .highlighter(new HighlightBuilder().field("long_term", 18, 1).highlighterType("fvh")),
            0,
            "long_term",
            0,
            1,
            equalTo("<em>thisisaverylongwordandmakessurethisfails</em>")
        );

        assertNotHighlighted(
            prepareSearch().setQuery(matchPhraseQuery("no_long_term", "test foo highlighed").slop(3))
                .highlighter(new HighlightBuilder().field("no_long_term", 18, 1).highlighterType("fvh").postTags("</b>").preTags("<b>")),
            0,
            "no_long_term"
        );

        assertHighlight(
            prepareSearch().setQuery(matchPhraseQuery("no_long_term", "test foo highlighed").slop(3))
                .highlighter(new HighlightBuilder().field("no_long_term", 30, 1).highlighterType("fvh").postTags("</b>").preTags("<b>")),
            0,
            "no_long_term",
            0,
            1,
            equalTo("a <b>test</b> where <b>foo</b> is <b>highlighed</b> and")
        );
    }

    public void testSourceLookupHighlightingUsingPlainHighlighter() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    // we don't store title and don't use term vector, now lets see if it works...
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "no")
                    .endObject()
                    .startObject("attachments")
                    .startObject("properties")
                    .startObject("body")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "no")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource(
                    XContentFactory.jsonBuilder()
                        .startObject()
                        .field("title", "This is a test on the highlighting bug present in elasticsearch")
                        .startArray("attachments")
                        .startObject()
                        .field("body", "attachment 1")
                        .endObject()
                        .startObject()
                        .field("body", "attachment 2")
                        .endObject()
                        .endArray()
                        .endObject()
                );
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "bug")).highlighter(new HighlightBuilder().field("title", -1, 0)),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(
                        response,
                        i,
                        "title",
                        0,
                        equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch")
                    );
                }
            }
        );

        assertResponse(
            prepareSearch().setQuery(matchQuery("attachments.body", "attachment"))
                .highlighter(new HighlightBuilder().field("attachments.body", -1, 0)),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(response, i, "attachments.body", 0, equalTo("<em>attachment</em> 1"));
                    assertHighlight(response, i, "attachments.body", 1, equalTo("<em>attachment</em> 2"));
                }
            }
        );

    }

    public void testSourceLookupHighlightingUsingFastVectorHighlighter() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    // we don't store title, now lets see if it works...
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "with_positions_offsets")
                    .endObject()
                    .startObject("attachments")
                    .startObject("properties")
                    .startObject("body")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "with_positions_offsets")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource(
                    XContentFactory.jsonBuilder()
                        .startObject()
                        .field("title", "This is a test on the highlighting bug present in elasticsearch")
                        .startArray("attachments")
                        .startObject()
                        .field("body", "attachment 1")
                        .endObject()
                        .startObject()
                        .field("body", "attachment 2")
                        .endObject()
                        .endArray()
                        .endObject()
                );
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "bug")).highlighter(new HighlightBuilder().field("title", -1, 0)),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(
                        response,
                        i,
                        "title",
                        0,
                        equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch")
                    );
                }
            }
        );

        assertResponse(
            prepareSearch().setQuery(matchQuery("attachments.body", "attachment"))
                .highlighter(new HighlightBuilder().field("attachments.body", -1, 2))
                .execute(),
            response -> {
                for (int i = 0; i < 5; i++) {
                    assertHighlight(response, i, "attachments.body", 0, equalTo("<em>attachment</em> 1"));
                    assertHighlight(response, i, "attachments.body", 1, equalTo("<em>attachment</em> 2"));
                }
            }
        );
    }

    public void testSourceLookupHighlightingUsingPostingsHighlighter() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    // we don't store title, now lets see if it works...
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("index_options", "offsets")
                    .endObject()
                    .startObject("attachments")
                    .startObject("properties")
                    .startObject("body")
                    .field("type", "text")
                    .field("store", false)
                    .field("index_options", "offsets")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource(
                    XContentFactory.jsonBuilder()
                        .startObject()
                        .array(
                            "title",
                            "This is a test on the highlighting bug present in elasticsearch. Hopefully it works.",
                            "This is the second bug to perform highlighting on."
                        )
                        .startArray("attachments")
                        .startObject()
                        .field("body", "attachment for this test")
                        .endObject()
                        .startObject()
                        .field("body", "attachment 2")
                        .endObject()
                        .endArray()
                        .endObject()
                );
        }
        indexRandom(true, indexRequestBuilders);

        assertResponses(response -> {
            for (int i = 0; i < indexRequestBuilders.length; i++) {
                assertHighlight(
                    response,
                    i,
                    "title",
                    0,
                    equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch. Hopefully it works.")
                );
                assertHighlight(response, i, "title", 1, 2, equalTo("This is the second <em>bug</em> to perform highlighting on."));
            }
        },
            prepareSearch().setQuery(matchQuery("title", "bug"))
                // asking for the whole field to be highlighted
                .highlighter(new HighlightBuilder().field("title", -1, 0)),
            prepareSearch().setQuery(matchQuery("title", "bug"))
                // sentences will be generated out of each value
                .highlighter(new HighlightBuilder().field("title"))
        );

        assertResponse(
            prepareSearch().setQuery(matchQuery("attachments.body", "attachment"))
                .highlighter(new HighlightBuilder().field("attachments.body", -1, 2)),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(response, i, "attachments.body", 0, equalTo("<em>attachment</em> for this test"));
                    assertHighlight(response, i, "attachments.body", 1, 2, equalTo("<em>attachment</em> 2"));
                }
            }
        );
    }

    public void testHighlightIssue1994() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                "title",
                "type=text,store=false",
                "titleTV",
                "type=text,store=false,term_vector=with_positions_offsets"
            )
        );

        String[] titles = new String[] { "This is a test on the highlighting bug present in elasticsearch", "The bug is bugging us" };
        indexRandom(false, prepareIndex("test").setId("1").setSource("title", titles, "titleTV", titles));

        indexRandom(
            true,
            prepareIndex("test").setId("2").setSource("titleTV", new String[] { "some text to highlight", "highlight other text" })
        );

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "bug"))
                .highlighter(new HighlightBuilder().field("title", -1, 2).field("titleTV", -1, 2).requireFieldMatch(false)),
            response -> {
                assertHighlight(
                    response,
                    0,
                    "title",
                    0,
                    equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch")
                );
                assertHighlight(response, 0, "title", 1, 2, equalTo("The <em>bug</em> is bugging us"));
                assertHighlight(
                    response,
                    0,
                    "titleTV",
                    0,
                    equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch")
                );
                assertHighlight(response, 0, "titleTV", 1, 2, equalTo("The <em>bug</em> is bugging us"));
            }
        );
        assertResponse(
            prepareSearch().setQuery(matchQuery("titleTV", "highlight")).highlighter(new HighlightBuilder().field("titleTV", -1, 2)),
            response -> {
                assertHighlight(response, 0, "titleTV", 0, equalTo("some text to <em>highlight</em>"));
                assertHighlight(response, 0, "titleTV", 1, 2, equalTo("<em>highlight</em> other text"));
            }
        );
    }

    public void testGlobalHighlightingSettingsOverriddenAtFieldLevel() {
        createIndex("test");
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            new String[] { "this is a test", "this is the second test" },
            "field2",
            new String[] { "this is another test", "yet another test" }
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field1 and field2 produces different tags");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "test"))
            .highlighter(
                highlight().order("score")
                    .preTags("<global>")
                    .postTags("</global>")
                    .fragmentSize(1)
                    .numOfFragments(1)
                    .field(new HighlightBuilder.Field("field1").numOfFragments(2))
                    .field(
                        new HighlightBuilder.Field("field2").preTags("<field2>")
                            .postTags("</field2>")
                            .fragmentSize(50)
                            .requireFieldMatch(false)
                    )
            );

        assertResponse(prepareSearch("test").setSource(source), response -> {
            assertHighlight(response, 0, "field1", 0, 2, equalTo("<global>test</global>"));
            assertHighlight(response, 0, "field1", 1, 2, equalTo("<global>test</global>"));
            assertHighlight(response, 0, "field2", 0, 1, equalTo("yet another <field2>test</field2>"));
        });
    }

    // Issue #5175
    public void testHighlightingOnWildcardFields() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                "field-postings",
                "type=text,index_options=offsets",
                "field-fvh",
                "type=text,term_vector=with_positions_offsets",
                "field-plain",
                "type=text"
            )
        );
        ensureGreen();

        prepareIndex("test").setSource(
            "field-postings",
            "This is the first test sentence. Here is the second one.",
            "field-fvh",
            "This is the test with term_vectors",
            "field-plain",
            "This is the test for the plain highlighter"
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field*");
        SearchSourceBuilder source = searchSource()
            // postings hl doesn't support require_field_match, its field needs to be queried directly
            .query(termQuery("field-postings", "test"))
            .highlighter(highlight().field("field*").preTags("<xxx>").postTags("</xxx>").requireFieldMatch(false));

        assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
            assertHighlight(
                response,
                0,
                "field-postings",
                0,
                1,
                equalTo("This is the first <xxx>test</xxx> sentence. Here is the second one.")
            );
            assertHighlight(response, 0, "field-fvh", 0, 1, equalTo("This is the <xxx>test</xxx> with term_vectors"));
            assertHighlight(response, 0, "field-plain", 0, 1, equalTo("This is the <xxx>test</xxx> for the plain highlighter"));
        });
    }

    public void testPlainHighlighter() throws Exception {
        ensureGreen();

        prepareIndex("test").setSource("field1", "this is a test", "field2", "The quick brown fox jumps over the lazy dog").get();
        refresh();

        SearchSourceBuilder source = searchSource().query(termQuery("field1", "test"))
            .highlighter(highlight().highlighterType("plain").field("field1").order("score").preTags("<xxx>").postTags("</xxx>"));
        assertHighlight(prepareSearch("test").setSource(source), 0, "field1", 0, 1, equalTo("this is a <xxx>test</xxx>"));
    }

    public void testPlainHighlighterOrder() throws Exception {
        ensureGreen();

        prepareIndex("test").setSource("field1", "The quick brown fox jumps over the lazy brown dog but to no suprise the dog doesn't care")
            .get();
        refresh();

        {
            assertResponses(response -> {
                assertHighlight(response, 0, "field1", 0, 3, equalTo("The quick <xxx>brown</xxx> fox"));
                assertHighlight(response, 0, "field1", 1, 3, equalTo(" jumps over the lazy <xxx>brown</xxx> <xxx>dog</xxx>"));
                assertHighlight(response, 0, "field1", 2, 3, equalTo(" <xxx>dog</xxx> doesn't care"));
            },
                // fragments should be in order of appearance by default
                prepareSearch("test").setSource(
                    searchSource().query(matchQuery("field1", "brown dog"))
                        .highlighter(
                            highlight().highlighterType("plain").field("field1").preTags("<xxx>").postTags("</xxx>").fragmentSize(25)
                        )
                ),
                // lets be explicit about the order
                prepareSearch("test").setSource(
                    searchSource().query(matchQuery("field1", "brown dog"))
                        .highlighter(
                            highlight().highlighterType("plain")
                                .field("field1")
                                .order("none")
                                .preTags("<xxx>")
                                .postTags("</xxx>")
                                .fragmentSize(25)
                        )
                )
            );
        }
        {
            // order by score
            SearchSourceBuilder source = searchSource().query(matchQuery("field1", "brown dog"))
                .highlighter(
                    highlight().highlighterType("plain").order("score").field("field1").preTags("<xxx>").postTags("</xxx>").fragmentSize(25)
                );

            assertResponse(prepareSearch("test").setSource(source), response -> {
                assertHighlight(response, 0, "field1", 0, 3, equalTo(" jumps over the lazy <xxx>brown</xxx> <xxx>dog</xxx>"));
                assertHighlight(response, 0, "field1", 1, 3, equalTo("The quick <xxx>brown</xxx> fox"));
                assertHighlight(response, 0, "field1", 2, 3, equalTo(" <xxx>dog</xxx> doesn't care"));
            });
        }
    }

    public void testFastVectorHighlighter() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        indexRandom(
            true,
            prepareIndex("test").setSource("field1", "this is a test", "field2", "The quick brown fox jumps over the lazy dog")
        );

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "test"))
            .highlighter(highlight().field("field1", 100, 0).order("score").preTags("<xxx>").postTags("</xxx>"));

        assertHighlight(prepareSearch("test").setSource(source), 0, "field1", 0, 1, equalTo("this is a <xxx>test</xxx>"));

        logger.info("--> searching with boundary characters");
        source = searchSource().query(matchQuery("field2", "quick"))
            .highlighter(highlight().field("field2", 30, 1).boundaryChars(new char[] { ' ' }));

        assertHighlight(prepareSearch("test").setSource(source), 0, "field2", 0, 1, equalTo("The <em>quick</em> brown fox jumps over"));

        logger.info("--> searching with boundary characters on the field");
        source = searchSource().query(matchQuery("field2", "quick"))
            .highlighter(highlight().field(new Field("field2").fragmentSize(30).numOfFragments(1).boundaryChars(new char[] { ' ' })));

        assertHighlight(prepareSearch("test").setSource(source), 0, "field2", 0, 1, equalTo("The <em>quick</em> brown fox jumps over"));
    }

    public void testHighlighterWithSentenceBoundaryScanner() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("field1", "A sentence with few words. Another sentence with even more words."));

        for (String type : new String[] { "unified", "fvh" }) {
            logger.info("--> highlighting and searching on 'field' with sentence boundary_scanner");
            SearchSourceBuilder source = searchSource().query(termQuery("field1", "sentence"))
                .highlighter(
                    highlight().field("field1", 21, 2)
                        .highlighterType(type)
                        .preTags("<xxx>")
                        .postTags("</xxx>")
                        .boundaryScannerType(BoundaryScannerType.SENTENCE)
                );
            assertResponse(prepareSearch("test").setSource(source), response -> {
                assertHighlight(
                    response,
                    0,
                    "field1",
                    0,
                    2,
                    anyOf(equalTo("A <xxx>sentence</xxx> with few words"), equalTo("A <xxx>sentence</xxx> with few words. "))
                );

                assertHighlight(
                    response,
                    0,
                    "field1",
                    1,
                    2,
                    anyOf(equalTo("Another <xxx>sentence</xxx> with"), equalTo("Another <xxx>sentence</xxx> with even more words. "))
                );
            });
        }
    }

    public void testHighlighterWithSentenceBoundaryScannerAndLocale() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("field1", "A sentence with few words. Another sentence with even more words."));

        for (String type : new String[] { "fvh", "unified" }) {
            logger.info("--> highlighting and searching on 'field' with sentence boundary_scanner");
            SearchSourceBuilder source = searchSource().query(termQuery("field1", "sentence"))
                .highlighter(
                    highlight().field("field1", 21, 2)
                        .highlighterType(type)
                        .preTags("<xxx>")
                        .postTags("</xxx>")
                        .boundaryScannerType(BoundaryScannerType.SENTENCE)
                        .boundaryScannerLocale(Locale.ENGLISH.toLanguageTag())
                );

            assertResponse(prepareSearch("test").setSource(source), response -> {
                assertHighlight(
                    response,
                    0,
                    "field1",
                    0,
                    2,
                    anyOf(equalTo("A <xxx>sentence</xxx> with few words"), equalTo("A <xxx>sentence</xxx> with few words. "))
                );

                assertHighlight(
                    response,
                    0,
                    "field1",
                    1,
                    2,
                    anyOf(equalTo("Another <xxx>sentence</xxx> with"), equalTo("Another <xxx>sentence</xxx> with even more words. "))
                );
            });
        }
    }

    public void testHighlighterWithWordBoundaryScanner() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("field1", "some quick and hairy brown:fox jumped over the lazy dog"));

        logger.info("--> highlighting and searching on 'field' with word boundary_scanner");
        for (String type : new String[] { "unified", "fvh" }) {
            SearchSourceBuilder source = searchSource().query(termQuery("field1", "some"))
                .highlighter(
                    highlight().field("field1", 23, 1)
                        .highlighterType(type)
                        .preTags("<xxx>")
                        .postTags("</xxx>")
                        .boundaryScannerType(BoundaryScannerType.WORD)
                );

            assertHighlight(
                prepareSearch("test").setSource(source),
                0,
                "field1",
                0,
                1,
                anyOf(equalTo("<xxx>some</xxx> quick and hairy brown"), equalTo("<xxx>some</xxx>"))
            );
        }
    }

    public void testHighlighterWithWordBoundaryScannerAndLocale() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("field1", "some quick and hairy brown:fox jumped over the lazy dog"));

        for (String type : new String[] { "unified", "fvh" }) {
            SearchSourceBuilder source = searchSource().query(termQuery("field1", "some"))
                .highlighter(
                    highlight().field("field1", 23, 1)
                        .highlighterType(type)
                        .preTags("<xxx>")
                        .postTags("</xxx>")
                        .boundaryScannerType(BoundaryScannerType.WORD)
                        .boundaryScannerLocale(Locale.ENGLISH.toLanguageTag())
                );

            assertHighlight(
                prepareSearch("test").setSource(source),
                0,
                "field1",
                0,
                1,
                anyOf(equalTo("<xxx>some</xxx> quick and hairy brown"), equalTo("<xxx>some</xxx>"))
            );
        }
    }

    /**
     * The FHV can spend a long time highlighting degenerate documents if
     * phraseLimit is not set. Its default is now reasonably low.
     */
    public void testFVHManyMatches() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        // Index one megabyte of "t " over and over and over again
        String pattern = "t   ";
        String value = new String(new char[1024 * 256 / pattern.length()]).replace("\0", pattern);
        prepareIndex("test").setSource("field1", value).get();
        refresh();

        final long[] tookDefaultPhrase = new long[1];
        final long[] tookLargePhrase = new long[1];

        logger.info("--> highlighting and searching on field1 with default phrase limit");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "t"))
            .highlighter(highlight().highlighterType("fvh").field("field1", 20, 1).order("score").preTags("<xxx>").postTags("</xxx>"));
        assertResponse(client().search(new SearchRequest("test").source(source)), defaultPhraseLimit -> {
            assertHighlight(defaultPhraseLimit, 0, "field1", 0, 1, containsString("<xxx>t</xxx>"));
            tookDefaultPhrase[0] = defaultPhraseLimit.getTook().getMillis();
        });
        logger.info("--> highlighting and searching on field1 with large phrase limit");
        source = searchSource().query(termQuery("field1", "t"))
            .highlighter(
                highlight().highlighterType("fvh")
                    .field("field1", 20, 1)
                    .order("score")
                    .preTags("<xxx>")
                    .postTags("</xxx>")
                    .phraseLimit(30000)
            );
        assertResponse(client().search(new SearchRequest("test").source(source)), largePhraseLimit -> {
            assertHighlight(largePhraseLimit, 0, "field1", 0, 1, containsString("<xxx>t</xxx>"));
            tookLargePhrase[0] = largePhraseLimit.getTook().getMillis();
        });
        /*
         * I hate comparing times because it can be inconsistent but default is
         * in the neighborhood of 300ms and the large phrase limit is in the
         * neighborhood of 8 seconds.
         */
        assertThat(tookDefaultPhrase[0], lessThan(tookLargePhrase[0]));
    }

    public void testMatchedFieldsFvhRequireFieldMatch() throws Exception {
        checkMatchedFieldsCase(true, "fvh");
    }

    public void testMatchedFieldsFvhNoRequireFieldMatch() throws Exception {
        checkMatchedFieldsCase(false, "fvh");
    }

    public void testMatchedFieldsUnified() throws Exception {
        // unified highlighter requires that "require_field_match" is true when matched fields are used
        checkMatchedFieldsCase(true, "unified");
    }

    private void checkMatchedFieldsCase(boolean requireFieldMatch, String type) throws Exception {
        Settings.Builder settings = Settings.builder();
        settings.put(indexSettings());
        settings.put("index.analysis.analyzer.mock_english.tokenizer", "standard");
        settings.put("index.analysis.analyzer.mock_english.filter", "mock_snowball");
        assertAcked(
            prepareCreate("test").setSettings(settings)
                .setMapping(
                    XContentFactory.jsonBuilder()
                        .startObject()
                        .startObject("_doc")
                        .startObject("properties")
                        .startObject("foo")
                        .field("type", "text")
                        .field("term_vector", "with_positions_offsets")
                        .field("store", true)
                        .field("analyzer", "mock_english")
                        .startObject("fields")
                        .startObject("plain")
                        .field("type", "text")
                        .field("term_vector", "with_positions_offsets")
                        .field("analyzer", "standard")
                        .endObject()
                        .endObject()
                        .endObject()
                        .startObject("bar")
                        .field("type", "text")
                        .field("term_vector", "with_positions_offsets")
                        .field("store", true)
                        .field("analyzer", "mock_english")
                        .startObject("fields")
                        .startObject("plain")
                        .field("type", "text")
                        .field("term_vector", "with_positions_offsets")
                        .field("analyzer", "standard")
                        .endObject()
                        .endObject()
                        .endObject()
                        .endObject()
                        .endObject()
                        .endObject()
                )
        );
        ensureGreen();

        indexDoc("test", "1", "foo", "running with scissors");
        indexDoc(
            "test",
            "2",
            "foo",
            "cat cat junk junk junk junk junk junk junk cats junk junk",
            "bar",
            "cat cat junk junk junk junk junk junk junk cats junk junk"
        );
        indexDoc("test", "3", "foo", "weird", "bar", "result");
        refresh();

        Field fooField = new Field("foo").numOfFragments(1)
            .order("score")
            .fragmentSize(25)
            .highlighterType(type)
            .requireFieldMatch(requireFieldMatch);
        SearchRequestBuilder req = prepareSearch("test").highlighter(new HighlightBuilder().field(fooField));

        // First check highlighting without any matched fields set
        assertResponse(
            req.setQuery(queryStringQuery("running scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // And that matching a subfield doesn't automatically highlight it
        assertResponse(
            req.setQuery(queryStringQuery("foo.plain:running scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("running with <em>scissors</em>"))
        );

        // Add the subfield to the list of matched fields but don't match it. Everything should still work
        // like before we added it.
        fooField = new Field("foo").numOfFragments(1)
            .order("score")
            .fragmentSize(25)
            .highlighterType(type)
            .requireFieldMatch(requireFieldMatch);
        fooField.matchedFields("foo", "foo.plain");
        req = prepareSearch("test").highlighter(new HighlightBuilder().field(fooField));
        assertResponse(
            req.setQuery(queryStringQuery("running scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // Now make half the matches come from the stored field and half from just a matched field.
        assertResponse(
            req.setQuery(queryStringQuery("foo.plain:running scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // Now remove the stored field from the matched field list. That should work too.
        fooField = new Field("foo").numOfFragments(1)
            .order("score")
            .fragmentSize(25)
            .highlighterType(type)
            .requireFieldMatch(requireFieldMatch);
        fooField.matchedFields("foo.plain");
        req = prepareSearch("test").highlighter(new HighlightBuilder().field(fooField));
        // unified highlighter always keeps the original field in the list of matched fields
        String expectedHighlight0 = type.equals("unified") ? "<em>running</em> with <em>scissors</em>" : "<em>running</em> with scissors";
        assertResponse(
            req.setQuery(queryStringQuery("foo.plain:running scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight0))
        );

        // Now make sure boosted fields don't blow up when matched fields is both the subfield and stored field.
        fooField = new Field("foo").numOfFragments(1)
            .order("score")
            .fragmentSize(25)
            .highlighterType(type)
            .requireFieldMatch(requireFieldMatch);
        fooField.matchedFields("foo", "foo.plain");
        req = prepareSearch("test").highlighter(new HighlightBuilder().field(fooField));
        assertResponse(
            req.setQuery(queryStringQuery("foo.plain:running^5 scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // Now just all matches are against the matched field. This still returns highlighting.
        assertResponse(
            req.setQuery(queryStringQuery("foo.plain:running foo.plain:scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // And all matched field via the queryString's field parameter, just in case
        assertResponse(
            req.setQuery(queryStringQuery("running scissors").field("foo.plain")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // Finding the same string two ways is ok too
        assertResponse(
            req.setQuery(queryStringQuery("run foo.plain:running^5 scissors").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"))
        );

        // Unified and FVH highlighters break text into fragments differently
        String expectedHighlight1 = type.equals("unified") ? "junk junk junk <em>cats</em> junk" : "junk junk <em>cats</em> junk junk";

        // But we use the best found score when sorting fragments
        assertResponse(
            req.setQuery(queryStringQuery("cats foo.plain:cats^5").field("foo")),
            response -> assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight1))
        );

        // which can also be written by searching on the subfield
        assertResponse(
            req.setQuery(queryStringQuery("cats").field("foo").field("foo.plain", 5)),
            response -> assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight1))
        );

        // Speaking of two fields, you can have two fields, only one of which has matchedFields enabled
        QueryBuilder twoFieldsQuery = queryStringQuery("cats").field("foo").field("foo.plain", 5).field("bar").field("bar.plain", 5);
        Field barField = new Field("bar").numOfFragments(1)
            .order("score")
            .fragmentSize(25)
            .highlighterType(type)
            .requireFieldMatch(requireFieldMatch);
        assertResponse(req.setQuery(twoFieldsQuery).highlighter(new HighlightBuilder().field(fooField).field(barField)), response -> {
            assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight1));
            assertHighlight(response, 0, "bar", 0, equalTo("<em>cat</em> <em>cat</em> junk junk junk junk"));
        });
        // And you can enable matchedField highlighting on both
        barField.matchedFields("bar", "bar.plain");
        assertResponse(req.setQuery(twoFieldsQuery).highlighter(new HighlightBuilder().field(fooField).field(barField)), response -> {
            assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight1));
            assertHighlight(response, 0, "bar", 0, equalTo(expectedHighlight1));
        });

        // Setting a matchedField that isn't searched/doesn't exist is simply ignored.
        barField.matchedFields("bar", "candy");
        assertResponse(req.setQuery(twoFieldsQuery).highlighter(new HighlightBuilder().field(fooField).field(barField)), response -> {
            assertHighlight(response, 0, "foo", 0, equalTo(expectedHighlight1));
            assertHighlight(response, 0, "bar", 0, equalTo("<em>cat</em> <em>cat</em> junk junk junk junk"));
        });

        // If the stored field doesn't have a value it doesn't matter what you match, you get nothing.
        barField.matchedFields("bar", "foo.plain");
        assertResponse(
            req.setQuery(queryStringQuery("running scissors").field("foo.plain").field("bar"))
                .highlighter(new HighlightBuilder().field(fooField).field(barField)),
            response -> {
                assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"));
                assertThat(response.getHits().getAt(0).getHighlightFields(), not(hasKey("bar")));
            }
        );

        // If the stored field is found but the matched field isn't then you don't get a result either.
        // only applicable to fvh highlighter, as unified highlighter always keeps the original field in the list of matched fields
        if (type.equals("fvh")) {
            fooField.matchedFields("bar.plain");
            assertResponse(
                req.setQuery(queryStringQuery("running scissors").field("foo").field("foo.plain").field("bar").field("bar.plain"))
                    .highlighter(new HighlightBuilder().field(fooField).field(barField)),
                response -> assertThat(response.getHits().getAt(0).getHighlightFields(), not(hasKey("foo")))
            );
        }

        // But if you add the stored field to the list of matched fields then you'll get a result again
        fooField.matchedFields("foo", "bar.plain");
        assertResponse(
            req.setQuery(queryStringQuery("running scissors").field("foo").field("foo.plain").field("bar").field("bar.plain"))
                .highlighter(new HighlightBuilder().field(fooField).field(barField)),
            response -> {
                assertHighlight(response, 0, "foo", 0, equalTo("<em>running</em> with <em>scissors</em>"));
                assertThat(response.getHits().getAt(0).getHighlightFields(), not(hasKey("bar")));
            }
        );

        // You _can_ highlight fields that aren't subfields of one another.
        assertResponse(
            req.setQuery(queryStringQuery("weird").field("foo").field("foo.plain").field("bar").field("bar.plain"))
                .highlighter(new HighlightBuilder().field(fooField).field(barField)),
            response -> {
                assertHighlight(response, 0, "foo", 0, equalTo("<em>weird</em>"));
                assertHighlight(response, 0, "bar", 0, equalTo("<em>resul</em>t"));
            }
        );

        if (type.equals("unified")) {
            assertResponse(
                req.setQuery(queryStringQuery("result").field("foo").field("foo.plain").field("bar").field("bar.plain"))
                    .highlighter(new HighlightBuilder().field(fooField).field(barField)),
                response -> {
                    assertHighlight(response, 0, "bar", 0, equalTo("<em>result</em>"));
                }
            );
        } else {
            assertFailures(
                req.setQuery(queryStringQuery("result").field("foo").field("foo.plain").field("bar").field("bar.plain"))
                    .highlighter(new HighlightBuilder().field(fooField).field(barField)),
                RestStatus.INTERNAL_SERVER_ERROR,
                containsString("IndexOutOfBoundsException")
            );
        }
    }

    public void testFastVectorHighlighterManyDocs() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();

        int COUNT = between(20, 100);
        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[COUNT];
        for (int i = 0; i < COUNT; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i)).setSource("field1", "test " + i);
        }
        logger.info("--> indexing docs");
        indexRandom(true, indexRequestBuilders);

        logger.info("--> searching explicitly on field1 and highlighting on it");
        assertResponse(
            prepareSearch().setSize(COUNT)
                .setQuery(termQuery("field1", "test"))
                .highlighter(new HighlightBuilder().field("field1", 100, 0)),
            response -> {
                for (int i = 0; i < COUNT; i++) {
                    SearchHit hit = response.getHits().getHits()[i];
                    // LUCENE 3.1 UPGRADE: Caused adding the space at the end...
                    assertHighlight(response, i, "field1", 0, 1, equalTo("<em>test</em> " + hit.getId()));
                }
            }
        );
    }

    public XContentBuilder type1TermVectorMapping() throws IOException {
        return XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("field1")
            .field("type", "text")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .startObject("field2")
            .field("type", "text")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
    }

    public void testSameContent() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true,term_vector=with_positions_offsets"));

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < 5; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a test on the highlighting bug present in elasticsearch");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "bug")).highlighter(new HighlightBuilder().field("title", -1, 0)),
            response -> {
                for (int i = 0; i < 5; i++) {
                    assertHighlight(
                        response,
                        i,
                        "title",
                        0,
                        1,
                        equalTo("This is a test on the highlighting <em>bug</em> present in elasticsearch")
                    );
                }
            }
        );
    }

    public void testFastVectorHighlighterOffsetParameter() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true,term_vector=with_positions_offsets").get());

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < 5; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a test on the highlighting bug present in elasticsearch");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "bug"))
                .highlighter(new HighlightBuilder().field("title", 30, 1, 10).highlighterType("fvh")),
            response -> {
                for (int i = 0; i < 5; i++) {
                    // LUCENE 3.1 UPGRADE: Caused adding the space at the end...
                    assertHighlight(response, i, "title", 0, 1, equalTo("highlighting <em>bug</em> present in elasticsearch"));
                }
            }
        );
    }

    public void testEscapeHtml() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true"));

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a html escaping highlighting test for *&? elasticsearch");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 50, 1, 10)),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(
                        response,
                        i,
                        "title",
                        0,
                        1,
                        startsWith("This is a html escaping highlighting <em>test</em> for *&amp;?")
                    );
                }
            }
        );
    }

    public void testEscapeHtmlVector() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true,term_vector=with_positions_offsets"));

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < 5; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a html escaping highlighting test for *&? elasticsearch");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 30, 1, 10).highlighterType("plain")),
            response -> {
                for (int i = 0; i < 5; i++) {
                    assertHighlight(response, i, "title", 0, 1, equalTo(" highlighting <em>test</em> for *&amp;? elasticsearch"));
                }
            }
        );
    }

    public void testMultiMapperVectorWithStore() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", true)
                    .field("term_vector", "with_positions_offsets")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", true)
                    .field("term_vector", "with_positions_offsets")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();
        prepareIndex("test").setId("1").setSource("title", "this is a test").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 50, 1)),
            0,
            "title",
            0,
            1,
            equalTo("this is a <em>test</em>")
        );

        // search on title.key and highlight on title
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title.key", 50, 1)),
            0,
            "title.key",
            0,
            1,
            equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em>")
        );
    }

    public void testMultiMapperVectorFromSource() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "with_positions_offsets")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "with_positions_offsets")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();

        prepareIndex("test").setId("1").setSource("title", "this is a test").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 50, 1)),
            0,
            "title",
            0,
            1,
            equalTo("this is a <em>test</em>")
        );

        // search on title.key and highlight on title.key
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title.key", 50, 1)),
            0,
            "title.key",
            0,
            1,
            equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em>")
        );
    }

    public void testMultiMapperNoVectorWithStore() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", true)
                    .field("term_vector", "no")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", true)
                    .field("term_vector", "no")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );

        ensureGreen();
        prepareIndex("test").setId("1").setSource("title", "this is a test").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 50, 1)),
            0,
            "title",
            0,
            1,
            equalTo("this is a <em>test</em>")
        );

        // search on title.key and highlight on title
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title.key", 50, 1)),
            0,
            "title.key",
            0,
            1,
            equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em>")
        );
    }

    public void testMultiMapperNoVectorFromSource() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "no")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", false)
                    .field("term_vector", "no")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();
        prepareIndex("test").setId("1").setSource("title", "this is a test").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title", 50, 1)),
            0,
            "title",
            0,
            1,
            equalTo("this is a <em>test</em>")
        );

        // search on title.key and highlight on title.key
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test"))
                .highlighter(new HighlightBuilder().encoder("html").field("title.key", 50, 1)),
            0,
            "title.key",
            0,
            1,
            equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em>")
        );
    }

    public void testFastVectorHighlighterShouldFailIfNoTermVectors() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true,term_vector=no"));
        ensureGreen();

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < 5; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a test for the enabling fast vector highlighter");
        }
        indexRandom(true, indexRequestBuilders);

        assertNoFailures(
            prepareSearch().setQuery(matchPhraseQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().field("title", 50, 1, 10))
        );

        assertFailures(
            prepareSearch().setQuery(matchPhraseQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().field("title", 50, 1, 10).highlighterType("fvh")),
            RestStatus.BAD_REQUEST,
            containsString(
                "the field [title] should be indexed with term vector with position offsets to be used with fast vector highlighter"
            )
        );

        // should not fail if there is a wildcard
        assertNoFailures(
            prepareSearch().setQuery(matchPhraseQuery("title", "this is a test"))
                .highlighter(new HighlightBuilder().field("tit*", 50, 1, 10).highlighterType("fvh"))
        );
    }

    public void testDisableFastVectorHighlighter() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text,store=true,term_vector=with_positions_offsets,analyzer=classic"));
        ensureGreen();

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a test for the workaround for the fast vector highlighting SOLR-3724");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchPhraseQuery("title", "test for the workaround"))
                .highlighter(new HighlightBuilder().field("title", 50, 1, 10).highlighterType("fvh")),
            response -> {

                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    // Because of SOLR-3724 nothing is highlighted when FVH is used
                    assertNotHighlighted(response, i, "title");
                }
            }
        );

        assertResponses(response -> {
            for (int i = 0; i < indexRequestBuilders.length; i++) {
                assertHighlight(
                    response,
                    i,
                    "title",
                    0,
                    1,
                    equalTo("This is a <em>test</em> for the <em>workaround</em> for the fast vector highlighting SOLR-3724")
                );
            }
        },
            // Using plain highlighter instead of FVH
            prepareSearch().setQuery(matchPhraseQuery("title", "test for the workaround"))
                .highlighter(new HighlightBuilder().field("title", 50, 1, 10).highlighterType("plain")),
            // Using plain highlighter instead of FVH on the field level
            prepareSearch().setQuery(matchPhraseQuery("title", "test for the workaround"))
                .highlighter(
                    new HighlightBuilder().field(new HighlightBuilder.Field("title").highlighterType("plain")).highlighterType("plain")
                )
        );
    }

    public void testFSHHighlightAllMvFragments() throws Exception {
        assertAcked(prepareCreate("test").setMapping("tags", "type=text,term_vector=with_positions_offsets"));
        ensureGreen();
        prepareIndex("test").setId("1")
            .setSource(
                "tags",
                new String[] {
                    "this is a really long tag i would like to highlight",
                    "here is another one that is very long and has the tag token near the end" }
            )
            .get();
        refresh();

        assertResponse(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("tags", "tag"))
                .highlighter(new HighlightBuilder().field("tags", -1, 0).highlighterType("fvh")),
            response -> {
                assertHighlight(response, 0, "tags", 0, equalTo("this is a really long <em>tag</em> i would like to highlight"));
                assertHighlight(
                    response,
                    0,
                    "tags",
                    1,
                    2,
                    equalTo("here is another one that is very long and has the <em>tag</em> token near the end")
                );
            }
        );
    }

    public void testBoostingQuery() {
        createIndex("test");
        ensureGreen();
        prepareIndex("test").setSource("field1", "this is a test", "field2", "The quick brown fox jumps over the lazy dog").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boostingQuery(termQuery("field2", "brown"), termQuery("field2", "foobar")).negativeBoost(0.5f)
        ).highlighter(highlight().field("field2").order("score").preTags("<x>").postTags("</x>"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The quick <x>brown</x> fox jumps over the lazy dog")
        );
    }

    public void testBoostingQueryTermVector() throws IOException {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        ensureGreen();
        prepareIndex("test").setSource("field1", "this is a test", "field2", "The quick brown fox jumps over the lazy dog").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boostingQuery(termQuery("field2", "brown"), termQuery("field2", "foobar")).negativeBoost(0.5f)
        ).highlighter(highlight().field("field2").order("score").preTags("<x>").postTags("</x>"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The quick <x>brown</x> fox jumps over the lazy dog")
        );
    }

    public void testPlainHighlightDifferentFragmenter() throws Exception {
        assertAcked(prepareCreate("test").setMapping("tags", "type=text"));
        ensureGreen();
        prepareIndex("test").setId("1")
            .setSource(
                jsonBuilder().startObject()
                    .array(
                        "tags",
                        "this is a really long tag i would like to highlight",
                        "here is another one that is very long tag and has the tag token near the end"
                    )
                    .endObject()
            )
            .get();
        refresh();

        assertResponses(response -> {
            assertHighlight(response, 0, "tags", 0, equalTo("this is a really <em>long</em> <em>tag</em> i would like to highlight"));
            assertHighlight(
                response,
                0,
                "tags",
                1,
                2,
                equalTo("here is another one that is very <em>long</em> <em>tag</em> and has the tag token near the end")
            );
        },
            prepareSearch("test").setQuery(QueryBuilders.matchPhraseQuery("tags", "long tag"))
                .highlighter(
                    new HighlightBuilder().field(
                        new HighlightBuilder.Field("tags").highlighterType("plain").fragmentSize(-1).numOfFragments(2).fragmenter("simple")
                    )
                ),
            prepareSearch("test").setQuery(QueryBuilders.matchPhraseQuery("tags", "long tag"))
                .highlighter(
                    new HighlightBuilder().field(
                        new Field("tags").highlighterType("plain").fragmentSize(-1).numOfFragments(2).fragmenter("span")
                    )
                )
        );

        assertFailures(
            prepareSearch("test").setQuery(QueryBuilders.matchPhraseQuery("tags", "long tag"))
                .highlighter(
                    new HighlightBuilder().field(
                        new Field("tags").highlighterType("plain").fragmentSize(-1).numOfFragments(2).fragmenter("invalid")
                    )
                ),
            RestStatus.BAD_REQUEST,
            containsString("unknown fragmenter option [invalid] for the field [tags]")
        );
    }

    public void testPlainHighlighterMultipleFields() {
        createIndex("test");
        ensureGreen();

        indexDoc("test", "1", "field1", "The <b>quick<b> brown fox", "field2", "The <b>slow<b> brown fox");
        refresh();

        assertResponse(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("field1", "fox"))
                .highlighter(
                    new HighlightBuilder().field(
                        new HighlightBuilder.Field("field1").preTags("<1>").postTags("</1>").requireFieldMatch(true)
                    ).field(new HighlightBuilder.Field("field2").preTags("<2>").postTags("</2>").requireFieldMatch(false))
                ),
            response -> {
                assertHighlight(response, 0, "field1", 0, 1, equalTo("The <b>quick<b> brown <1>fox</1>"));
                assertHighlight(response, 0, "field2", 0, 1, equalTo("The <b>slow<b> brown <2>fox</2>"));
            }
        );
    }

    public void testFastVectorHighlighterMultipleFields() {
        assertAcked(
            prepareCreate("test").setMapping(
                "field1",
                "type=text,term_vector=with_positions_offsets",
                "field2",
                "type=text,term_vector=with_positions_offsets"
            )
        );
        ensureGreen();

        indexDoc("test", "1", "field1", "The <b>quick<b> brown fox", "field2", "The <b>slow<b> brown fox");
        refresh();

        assertResponse(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("field1", "fox"))
                .highlighter(
                    new HighlightBuilder().field(
                        new HighlightBuilder.Field("field1").preTags("<1>").postTags("</1>").requireFieldMatch(true)
                    ).field(new HighlightBuilder.Field("field2").preTags("<2>").postTags("</2>").requireFieldMatch(false))
                ),
            response -> {
                assertHighlight(response, 0, "field1", 0, 1, equalTo("The <b>quick<b> brown <1>fox</1>"));
                assertHighlight(response, 0, "field2", 0, 1, equalTo("The <b>slow<b> brown <2>fox</2>"));
            }
        );
    }

    public void testMissingStoredField() throws Exception {
        assertAcked(prepareCreate("test").setMapping("highlight_field", "type=text,store=true"));
        ensureGreen();
        prepareIndex("test").setId("1").setSource(jsonBuilder().startObject().field("field", "highlight").endObject()).get();
        refresh();

        // This query used to fail when the field to highlight was absent
        assertResponse(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("field", "highlight"))
                .highlighter(
                    new HighlightBuilder().field(
                        new HighlightBuilder.Field("highlight_field").fragmentSize(-1).numOfFragments(1).fragmenter("simple")
                    )
                ),
            response -> assertThat(response.getHits().getHits()[0].getHighlightFields().isEmpty(), equalTo(true))
        );
    }

    // Issue #3211
    public void testNumericHighlighting() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                "text",
                "type=text",
                "byte",
                "type=byte",
                "short",
                "type=short",
                "int",
                "type=integer",
                "long",
                "type=long",
                "float",
                "type=float",
                "double",
                "type=double"
            )
        );
        ensureGreen();

        prepareIndex("test").setId("1")
            .setSource("text", "elasticsearch test", "byte", 25, "short", 42, "int", 100, "long", -1, "float", 3.2f, "double", 42.42)
            .get();
        refresh();

        // Highlighting of numeric fields is not supported, but it should not raise errors
        // (this behavior is consistent with version 0.20)
        assertHitCount(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("text", "test"))
                .highlighter(
                    new HighlightBuilder().field("text")
                        .field("byte")
                        .field("short")
                        .field("int")
                        .field("long")
                        .field("float")
                        .field("double")
                ),
            1L
        );
    }

    // Issue #3200
    public void testResetTwice() throws Exception {
        assertAcked(
            prepareCreate("test").setSettings(
                Settings.builder().put(indexSettings()).put("analysis.analyzer.my_analyzer.type", "mock_whitespace").build()
            ).setMapping("text", "type=text,analyzer=my_analyzer")
        );
        ensureGreen();
        prepareIndex("test").setId("1").setSource("text", "elasticsearch test").get();
        refresh();

        // Mock tokenizer will throw an exception if it is resetted twice
        assertHitCount(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("text", "test")).highlighter(new HighlightBuilder().field("text")),
            1L
        );
    }

    public void testHighlightUsesHighlightQuery() throws IOException {
        assertAcked(
            prepareCreate("test").setMapping(
                "text",
                "type=text," + randomStoreField() + "term_vector=with_positions_offsets,index_options=offsets"
            )
        );
        ensureGreen();

        indexDoc("test", "1", "text", "Testing the highlight query feature");
        refresh();

        for (String type : ALL_TYPES) {
            HighlightBuilder.Field field = new HighlightBuilder.Field("text");
            HighlightBuilder highlightBuilder = new HighlightBuilder().field(field).highlighterType(type);
            SearchRequestBuilder search = prepareSearch("test").setQuery(QueryBuilders.matchQuery("text", "testing"))
                .highlighter(highlightBuilder);
            Matcher<String> searchQueryMatcher = equalTo("<em>Testing</em> the highlight query feature");

            assertResponse(search, response -> assertHighlight(response, 0, "text", 0, searchQueryMatcher));
            field = new HighlightBuilder.Field("text");

            Matcher<String> hlQueryMatcher = equalTo("Testing the highlight <em>query</em> feature");
            field.highlightQuery(matchQuery("text", "query"));
            highlightBuilder = new HighlightBuilder().field(field);
            search = prepareSearch("test").setQuery(QueryBuilders.matchQuery("text", "testing")).highlighter(highlightBuilder);
            assertResponse(search, response -> assertHighlight(response, 0, "text", 0, hlQueryMatcher));

            // Make sure the highlightQuery is taken into account when it is set on the highlight context instead of the field
            highlightBuilder.highlightQuery(matchQuery("text", "query"));
            field.highlighterType(type).highlightQuery(null);
            assertResponse(search, response -> assertHighlight(response, 0, "text", 0, hlQueryMatcher));
        }
    }

    private static String randomStoreField() {
        if (randomBoolean()) {
            return "store=true,";
        }
        return "";
    }

    public void testHighlightNoMatchSize() throws IOException {
        assertAcked(
            prepareCreate("test").setMapping(
                "text",
                "type=text," + randomStoreField() + "term_vector=with_positions_offsets,index_options=offsets"
            )
        );
        ensureGreen();

        String text = "I am pretty long so some of me should get cut off. Second sentence";
        indexDoc("test", "1", "text", text);
        refresh();

        // When you don't set noMatchSize you don't get any results if there isn't anything to highlight.
        HighlightBuilder.Field field = new HighlightBuilder.Field("text").fragmentSize(21).numOfFragments(1).highlighterType("plain");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        // When noMatchSize is set to 0 you also shouldn't get any
        field.highlighterType("plain").noMatchSize(0);
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        // When noMatchSize is between 0 and the size of the string
        field.highlighterType("plain").noMatchSize(21);
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so")
        );

        // The FVH also works but the fragment is longer than the plain highlighter because of boundary_max_scan
        field.highlighterType("fvh");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        // Unified hl also works but the fragment is longer than the plain highlighter because of the boundary is the word
        field.highlighterType("unified");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        // We can also ask for a fragment longer than the input string and get the whole string
        for (String type : new String[] { "plain", "unified" }) {
            field.highlighterType(type).noMatchSize(text.length() * 2).numOfFragments(0);
            assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));
        }

        field.highlighterType("fvh");
        assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));

        field.highlighterType("unified");
        assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));

        // We can also ask for a fragment exactly the size of the input field and get the whole field
        field.highlighterType("plain").noMatchSize(text.length());
        assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));

        field.highlighterType("fvh");
        assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));

        // unified hl returns the first sentence as the noMatchSize does not cross sentence boundary.
        field.highlighterType("unified");
        assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(text));

        // You can set noMatchSize globally in the highlighter as well
        field.highlighterType("plain").noMatchSize(null);
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field).noMatchSize(21)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so")
        );

        field.highlighterType("fvh");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field).noMatchSize(21)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        field.highlighterType("unified");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field).noMatchSize(21)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        // We don't break if noMatchSize is less than zero though
        field.highlighterType("plain").noMatchSize(randomIntBetween(Integer.MIN_VALUE, -1));
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        // Check when the requested fragment size equals the size of the string
        var anotherText = "I am unusual and don't end with your regular )token)";
        indexDoc("test", "1", "text", anotherText);
        refresh();
        for (String type : new String[] { "plain", "unified", "fvh" }) {
            field.highlighterType(type).noMatchSize(anotherText.length()).numOfFragments(0);
            assertHighlight(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text", 0, 1, equalTo(anotherText));
        }
    }

    public void testHighlightNoMatchSizeWithMultivaluedFields() throws IOException {
        assertAcked(
            prepareCreate("test").setMapping(
                "text",
                "type=text," + randomStoreField() + "term_vector=with_positions_offsets,index_options=offsets"
            )
        );
        ensureGreen();

        String text1 = "I am pretty long so some of me should get cut off. We'll see how that goes.";
        String text2 = "I am short";
        indexDoc("test", "1", "text", new String[] { text1, text2 });
        refresh();

        // The no match fragment should come from the first value of a multi-valued field
        HighlightBuilder.Field field = new HighlightBuilder.Field("text").fragmentSize(21)
            .numOfFragments(1)
            .highlighterType("plain")
            .noMatchSize(21);
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so")
        );

        field.highlighterType("fvh");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        field.highlighterType("unified");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am pretty long so some")
        );

        // And noMatchSize returns nothing when the first entry is empty string!
        indexDoc("test", "2", "text", new String[] { "", text2 });
        refresh();

        IdsQueryBuilder idsQueryBuilder = QueryBuilders.idsQuery().addIds("2");
        field.highlighterType("plain");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        // except for the unified highlighter which starts from the first string with actual content
        field.highlighterType("unified");
        assertHighlight(
            prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("I am short")
        );

        // But if the field was actually empty then you should get no highlighting field
        indexDoc("test", "3", "text", new String[] {});
        refresh();
        idsQueryBuilder = QueryBuilders.idsQuery().addIds("3");
        field.highlighterType("plain");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        // Same for if the field doesn't even exist on the document
        indexDoc("test", "4");
        refresh();

        idsQueryBuilder = QueryBuilders.idsQuery().addIds("4");
        field.highlighterType("plain");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(
            prepareSearch("test").setQuery(idsQueryBuilder).highlighter(new HighlightBuilder().field(field)),
            0,
            "postings"
        );

        // Again same if the field isn't mapped
        field = new HighlightBuilder.Field("unmapped").highlighterType("plain").noMatchSize(21);
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("fvh");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");

        field.highlighterType("unified");
        assertNotHighlighted(prepareSearch("test").highlighter(new HighlightBuilder().field(field)), 0, "text");
    }

    public void testHighlightNoMatchSizeNumberOfFragments() {
        assertAcked(
            prepareCreate("test").setMapping(
                "text",
                "type=text," + randomStoreField() + "term_vector=with_positions_offsets,index_options=offsets"
            )
        );
        ensureGreen();

        String text1 = "This is the first sentence. This is the second sentence." + HighlightUtils.PARAGRAPH_SEPARATOR;
        String text2 = "This is the third sentence. This is the fourth sentence.";
        String text3 = "This is the fifth sentence";
        indexDoc("test", "1", "text", new String[] { text1, text2, text3 });
        refresh();

        // The no match fragment should come from the first value of a multi-valued field
        HighlightBuilder.Field field = new HighlightBuilder.Field("text").fragmentSize(1)
            .numOfFragments(0)
            .highlighterType("plain")
            .noMatchSize(20);
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("This is the first")
        );

        field.highlighterType("fvh");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("This is the first sentence")
        );

        field.highlighterType("unified");
        assertHighlight(
            prepareSearch("test").highlighter(new HighlightBuilder().field(field)),
            0,
            "text",
            0,
            1,
            equalTo("This is the first sentence")
        );

        // if there's a match we only return the values with matches (whole value as number_of_fragments == 0)
        MatchQueryBuilder queryBuilder = QueryBuilders.matchQuery("text", "third fifth");
        field.highlighterType("plain");
        assertResponse(prepareSearch("test").setQuery(queryBuilder).highlighter(new HighlightBuilder().field(field)), response -> {
            assertHighlight(response, 0, "text", 0, 2, equalTo("This is the <em>third</em> sentence. This is the fourth sentence."));
            assertHighlight(response, 0, "text", 1, 2, equalTo("This is the <em>fifth</em> sentence"));
        });

        field.highlighterType("fvh");
        assertResponse(prepareSearch("test").setQuery(queryBuilder).highlighter(new HighlightBuilder().field(field)), response -> {
            assertHighlight(response, 0, "text", 0, 2, equalTo("This is the <em>third</em> sentence. This is the fourth sentence."));
            assertHighlight(response, 0, "text", 1, 2, equalTo("This is the <em>fifth</em> sentence"));
        });
        field.highlighterType("unified");
        assertResponse(prepareSearch("test").setQuery(queryBuilder).highlighter(new HighlightBuilder().field(field)), response -> {
            assertHighlight(response, 0, "text", 0, 2, equalTo("This is the <em>third</em> sentence. This is the fourth sentence."));
            assertHighlight(response, 0, "text", 1, 2, equalTo("This is the <em>fifth</em> sentence"));
        });
    }

    public void testPostingsHighlighter() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "this is a test", "field2", "The quick brown fox jumps over the lazy quick dog").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "test"))
            .highlighter(highlight().field("field1").preTags("<xxx>").postTags("</xxx>"));
        assertHighlight(client().search(new SearchRequest("test").source(source)), 0, "field1", 0, 1, equalTo("this is a <xxx>test</xxx>"));

        logger.info("--> searching on field1, highlighting on field1");
        source = searchSource().query(termQuery("field1", "test"))
            .highlighter(highlight().field("field1").preTags("<xxx>").postTags("</xxx>"));
        assertHighlight(client().search(new SearchRequest("test").source(source)), 0, "field1", 0, 1, equalTo("this is a <xxx>test</xxx>"));

        logger.info("--> searching on field2, highlighting on field2");
        source = searchSource().query(termQuery("field2", "quick"))
            .highlighter(highlight().field("field2").order("score").preTags("<xxx>").postTags("</xxx>"));
        assertHighlight(
            client().search(new SearchRequest("test").source(source)),
            0,
            "field2",
            0,
            1,
            equalTo("The <xxx>quick</xxx> brown fox jumps over the lazy <xxx>quick</xxx> dog")
        );

        logger.info("--> searching on field2, highlighting on field2");
        source = searchSource().query(matchPhraseQuery("field2", "quick brown"))
            .highlighter(highlight().field("field2").preTags("<xxx>").postTags("</xxx>"));
        assertHighlight(
            client().search(new SearchRequest("test").source(source)),
            0,
            "field2",
            0,
            1,
            equalTo("The <xxx>quick brown</xxx> fox jumps over the lazy quick dog")
        );

        // lets fall back to the standard highlighter then, what people would do to highlight query matches
        logger.info("--> searching on field2, highlighting on field2, falling back to the plain highlighter");
        source = searchSource().query(matchPhraseQuery("field2", "quick brown"))
            .highlighter(highlight().field("field2").preTags("<xxx>").postTags("</xxx>").highlighterType("plain").requireFieldMatch(false));
        assertHighlight(
            client().search(new SearchRequest("test").source(source)),
            0,
            "field2",
            0,
            1,
            equalTo("The <xxx>quick</xxx> <xxx>brown</xxx> fox jumps over the lazy quick dog")
        );
    }

    public void testPostingsHighlighterMultipleFields() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()).get());
        ensureGreen();

        indexDoc(
            "test",
            "1",
            "field1",
            "The <b>quick<b> brown fox. Second sentence.",
            "field2",
            "The <b>slow<b> brown fox. Second sentence."
        );
        refresh();
        assertHighlight(
            prepareSearch("test").setQuery(QueryBuilders.matchQuery("field1", "fox"))
                .highlighter(new HighlightBuilder().field(new Field("field1").preTags("<1>").postTags("</1>").requireFieldMatch(true))),
            0,
            "field1",
            0,
            1,
            equalTo("The <b>quick<b> brown <1>fox</1>. Second sentence.")
        );
    }

    public void testPostingsHighlighterNumberOfFragments() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setId("1")
            .setSource(
                "field1",
                "The quick brown fox jumps over the lazy dog. The lazy red fox jumps over the quick dog. "
                    + "The quick brown dog jumps over the lazy fox.",
                "field2",
                "The quick brown fox jumps over the lazy dog. The lazy red fox jumps over the quick dog. "
                    + "The quick brown dog jumps over the lazy fox."
            )
            .get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "fox"))
            .highlighter(highlight().field(new Field("field1").numOfFragments(5).preTags("<field1>").postTags("</field1>")));

        assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
            assertThat(response.getHits().getHits().length, equalTo(1));
            assertHighlight(
                response,
                0,
                "field1",
                0,
                2,
                equalTo(
                    "The quick brown <field1>fox</field1> jumps over the lazy dog."
                        + " The lazy red <field1>fox</field1> jumps over the quick dog."
                )
            );
            assertHighlight(response, 0, "field1", 1, 2, equalTo("The quick brown dog jumps over the lazy <field1>fox</field1>."));
        });
        prepareIndex("test").setId("2")
            .setSource(
                "field1",
                new String[] {
                    "The quick brown fox jumps over the lazy dog. Second sentence not finished",
                    "The lazy red fox jumps over the quick dog.",
                    "The quick brown dog jumps over the lazy fox." }
            )
            .get();
        refresh();

        source = searchSource().query(termQuery("field1", "fox"))
            .highlighter(highlight().field(new Field("field1").numOfFragments(0).preTags("<field1>").postTags("</field1>")));

        assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
            assertHitCount(response, 2L);

            for (SearchHit searchHit : response.getHits()) {
                if ("1".equals(searchHit.getId())) {
                    assertHighlight(
                        searchHit,
                        "field1",
                        0,
                        1,
                        equalTo(
                            "The quick brown <field1>fox</field1> jumps over the lazy dog. "
                                + "The lazy red <field1>fox</field1> jumps over the quick dog. "
                                + "The quick brown dog jumps over the lazy <field1>fox</field1>."
                        )
                    );
                } else if ("2".equals(searchHit.getId())) {
                    assertHighlight(
                        searchHit,
                        "field1",
                        0,
                        3,
                        equalTo("The quick brown <field1>fox</field1> jumps over the lazy dog. Second sentence not finished")
                    );
                    assertHighlight(searchHit, "field1", 1, 3, equalTo("The lazy red <field1>fox</field1> jumps over the quick dog."));
                    assertHighlight(searchHit, "field1", 2, 3, equalTo("The quick brown dog jumps over the lazy <field1>fox</field1>."));
                } else {
                    fail("Only hits with id 1 and 2 are returned");
                }
            }
        });
    }

    public void testMultiMatchQueryHighlight() throws Exception {
        XContentBuilder mapping = XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("field1")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .startObject("field2")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mapping));
        ensureGreen();
        prepareIndex("test").setSource("field1", "The quick brown fox jumps over", "field2", "The quick brown fox jumps over").get();
        refresh();
        final int iters = scaledRandomIntBetween(20, 30);
        for (int i = 0; i < iters; i++) {
            String highlighterType = rarely() ? null : RandomPicks.randomFrom(random(), ALL_TYPES);
            MultiMatchQueryBuilder.Type matchQueryType = RandomPicks.randomFrom(random(), MultiMatchQueryBuilder.Type.values());
            MultiMatchQueryBuilder multiMatchQueryBuilder = multiMatchQuery("the quick brown fox", "field1", "field2").type(matchQueryType);

            SearchSourceBuilder source = searchSource().query(multiMatchQueryBuilder)
                .highlighter(
                    highlight().highlightQuery(randomBoolean() ? multiMatchQueryBuilder : null)
                        .highlighterType(highlighterType)
                        .field(new Field("field1").requireFieldMatch(true).preTags("<field1>").postTags("</field1>"))
                );
            logger.info("Running multi-match type: [{}] highlight with type: [{}]", matchQueryType, highlighterType);
            assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
                assertHitCount(response, 1L);
                assertHighlight(
                    response,
                    0,
                    "field1",
                    0,
                    anyOf(
                        equalTo("<field1>The quick brown fox</field1> jumps over"),
                        equalTo("<field1>The</field1> <field1>quick</field1> <field1>brown</field1> <field1>fox</field1> jumps over")
                    )
                );
            });
        }
    }

    public void testCombinedFieldsQueryHighlight() throws Exception {
        XContentBuilder mapping = XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("field1")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .startObject("field2")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mapping));
        ensureGreen();

        prepareIndex("test").setSource("field1", "The quick brown fox jumps over", "field2", "The quick brown fox jumps over").get();
        refresh();

        for (String highlighterType : ALL_TYPES) {
            CombinedFieldsQueryBuilder multiMatchQueryBuilder = combinedFieldsQuery("the quick brown fox", "field1", "field2");
            SearchSourceBuilder source = searchSource().query(multiMatchQueryBuilder)
                .highlighter(
                    highlight().highlighterType(highlighterType)
                        .field(new Field("field1").requireFieldMatch(true).preTags("<field1>").postTags("</field1>"))
                );

            assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
                assertHitCount(response, 1L);
                assertHighlight(
                    response,
                    0,
                    "field1",
                    0,
                    equalTo("<field1>The</field1> <field1>quick</field1> <field1>brown</field1> <field1>fox</field1> jumps over")
                );
            });
        }
    }

    public void testPostingsHighlighterOrderByScore() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            new String[] {
                "This sentence contains one match, not that short. This sentence contains two sentence matches. "
                    + "This one contains no matches.",
                "This is the second value's first sentence. This one contains no matches. "
                    + "This sentence contains three sentence occurrences (sentence).",
                "One sentence match here and scored lower since the text is quite long, not that appealing. "
                    + "This one contains no matches." }
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "sentence"))
            .highlighter(highlight().field("field1").order("score"));

        assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
            Map<String, HighlightField> highlightFieldMap = response.getHits().getAt(0).getHighlightFields();
            assertThat(highlightFieldMap.size(), equalTo(1));
            HighlightField field1 = highlightFieldMap.get("field1");
            assertThat(field1.fragments().length, equalTo(4));
            assertThat(
                field1.fragments()[0].string(),
                equalTo("This <em>sentence</em> contains three <em>sentence</em> occurrences (<em>sentence</em>).")
            );
            assertThat(
                field1.fragments()[1].string(),
                equalTo(
                    "This <em>sentence</em> contains one match, not that short. "
                        + "This <em>sentence</em> contains two <em>sentence</em> matches."
                )
            );
            assertThat(
                field1.fragments()[2].string(),
                equalTo("This is the second value's first <em>sentence</em>. This one contains no matches.")
            );
            assertThat(
                field1.fragments()[3].string(),
                equalTo("One <em>sentence</em> match here and scored lower since the text is quite long, not that appealing.")
            );
        });
    }

    public void testMaxQueryOffsetDefault() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(type1PostingsffsetsMapping())
                .setSettings(Settings.builder().put("index.highlight.max_analyzed_offset", "10").build())
        );
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            new String[] {
                "This sentence contains one match, not that short. This sentence contains zero sentence matches. "
                    + "This one contains no matches.",
                "This is the second value's first sentence. This one contains no matches. "
                    + "This sentence contains three sentence occurrences (sentence).",
                "One sentence match here and scored lower since the text is quite long, not that appealing. "
                    + "This one contains no matches." }
        ).get();
        refresh();

        // Specific for this test: by passing "-1" as "maxAnalyzedOffset", the index highlight setting above will be used.
        SearchSourceBuilder source = searchSource().query(termQuery("field1", "sentence"))
            .highlighter(highlight().field("field1").order("score").maxAnalyzedOffset(-1));

        assertResponse(client().search(new SearchRequest("test").source(source)), response -> {
            Map<String, HighlightField> highlightFieldMap = response.getHits().getAt(0).getHighlightFields();
            assertThat(highlightFieldMap.size(), equalTo(1));
            HighlightField field1 = highlightFieldMap.get("field1");
            assertThat(field1.fragments().length, equalTo(1));
            assertThat(
                field1.fragments()[0].string(),
                equalTo("This <em>sentence</em> contains one match, not that short. This sentence contains zero sentence matches.")
            );
        });
    }

    public void testPostingsHighlighterEscapeHtml() throws Exception {
        assertAcked(prepareCreate("test").setMapping("title", "type=text," + randomStoreField() + "index_options=offsets"));

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < 5; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a html escaping highlighting test for *&? elasticsearch");
        }
        indexRandom(true, indexRequestBuilders);

        assertResponse(
            prepareSearch().setQuery(matchQuery("title", "test")).highlighter(new HighlightBuilder().field("title").encoder("html")),
            response -> {
                for (int i = 0; i < indexRequestBuilders.length; i++) {
                    assertHighlight(
                        response,
                        i,
                        "title",
                        0,
                        1,
                        equalTo("This is a html escaping highlighting <em>test</em> for *&amp;? elasticsearch")
                    );
                }
            }
        );
    }

    public void testPostingsHighlighterMultiMapperWithStore() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", true)
                    .field("index_options", "offsets")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", true)
                    .field("index_options", "offsets")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();
        prepareIndex("test").setId("1").setSource("title", "this is a test . Second sentence.").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertResponse(
            prepareSearch()
                // lets make sure we analyze the query and we highlight the resulting terms
                .setQuery(matchQuery("title", "This is a Test"))
                .highlighter(new HighlightBuilder().field("title")),
            response -> {

                assertHitCount(response, 1L);
                SearchHit hit = response.getHits().getAt(0);
                // stopwords are not highlighted since not indexed
                assertHighlight(hit, "title", 0, 1, equalTo("this is a <em>test</em> . Second sentence."));
            }
        );
        // search on title.key and highlight on title
        assertResponse(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test")).highlighter(new HighlightBuilder().field("title.key")),
            response -> {
                assertHitCount(response, 1L);

                // stopwords are now highlighted since we used only whitespace analyzer here
                assertHighlight(
                    response,
                    0,
                    "title.key",
                    0,
                    1,
                    equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em> . Second sentence.")
                );
            }
        );
    }

    public void testPostingsHighlighterMultiMapperFromSource() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", false)
                    .field("index_options", "offsets")
                    .field("analyzer", "classic")
                    .startObject("fields")
                    .startObject("key")
                    .field("type", "text")
                    .field("store", false)
                    .field("index_options", "offsets")
                    .field("analyzer", "whitespace")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();

        prepareIndex("test").setId("1").setSource("title", "this is a test").get();
        refresh();

        // simple search on body with standard analyzer with a simple field query
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title", "this is a test")).highlighter(new HighlightBuilder().field("title")),
            0,
            "title",
            0,
            1,
            equalTo("this is a <em>test</em>")
        );

        // search on title.key and highlight on title.key
        assertHighlight(
            prepareSearch().setQuery(matchQuery("title.key", "this is a test")).highlighter(new HighlightBuilder().field("title.key")),
            0,
            "title.key",
            0,
            1,
            equalTo("<em>this</em> <em>is</em> <em>a</em> <em>test</em>")
        );
    }

    public void testPostingsHighlighterShouldFailIfNoOffsets() throws Exception {
        assertAcked(
            prepareCreate("test").setMapping(
                jsonBuilder().startObject()
                    .startObject("_doc")
                    .startObject("properties")
                    .startObject("title")
                    .field("type", "text")
                    .field("store", true)
                    .field("index_options", "docs")
                    .endObject()
                    .endObject()
                    .endObject()
                    .endObject()
            )
        );
        ensureGreen();

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[5];
        for (int i = 0; i < indexRequestBuilders.length; i++) {
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("title", "This is a test for the postings highlighter");
        }
        indexRandom(true, indexRequestBuilders);

        assertNoFailures(
            prepareSearch().setQuery(matchQuery("title", "this is a test")).highlighter(new HighlightBuilder().field("title"))
        );
    }

    public void testPostingsHighlighterBoostingQuery() throws IOException, ExecutionException, InterruptedException {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();
        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boostingQuery(termQuery("field2", "brown"), termQuery("field2", "foobar")).negativeBoost(0.5f)
        ).highlighter(highlight().field("field2").preTags("<x>").postTags("</x>"));
        assertHighlight(
            client().search(new SearchRequest("test").source(source)),
            0,
            "field2",
            0,
            1,
            equalTo("The quick <x>brown</x> fox jumps over the lazy dog! Second sentence.")
        );
    }

    private static XContentBuilder type1PostingsffsetsMapping() throws IOException {
        return XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("field1")
            .field("type", "text")
            .field("index_options", "offsets")
            .endObject()
            .startObject("field2")
            .field("type", "text")
            .field("index_options", "offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
    }

    public void testPostingsHighlighterPrefixQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();
        logger.info("--> highlighting and searching on field2");

        SearchSourceBuilder source = searchSource().query(prefixQuery("field2", "qui")).highlighter(highlight().field("field2"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence.")
        );
    }

    public void testPostingsHighlighterFuzzyQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field2");
        SearchSourceBuilder source = searchSource().query(fuzzyQuery("field2", "quck")).highlighter(highlight().field("field2"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence.")
        );
    }

    public void testPostingsHighlighterRegexpQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field2");
        SearchSourceBuilder source = searchSource().query(regexpQuery("field2", "qu[a-l]+k")).highlighter(highlight().field("field2"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence.")
        );
    }

    public void testPostingsHighlighterWildcardQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field2");
        SearchSourceBuilder source = searchSource().query(wildcardQuery("field2", "qui*")).highlighter(highlight().field("field2"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence.")
        );

        source = searchSource().query(wildcardQuery("field2", "qu*k")).highlighter(highlight().field("field2"));
        assertResponse(prepareSearch("test").setSource(source), response -> {
            assertHitCount(response, 1L);

            assertHighlight(response, 0, "field2", 0, 1, equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence."));
        });
    }

    public void testPostingsHighlighterTermRangeQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "this is a test", "field2", "aaab").get();
        refresh();

        logger.info("--> highlighting and searching on field2");
        SearchSourceBuilder source = searchSource().query(rangeQuery("field2").gte("aaaa").lt("zzzz"))
            .highlighter(highlight().field("field2"));
        assertHighlight(prepareSearch("test").setSource(source), 0, "field2", 0, 1, equalTo("<em>aaab</em>"));
    }

    public void testPostingsHighlighterQueryString() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource(
            "field1",
            "this is a test",
            "field2",
            "The quick brown fox jumps over the lazy dog! Second sentence."
        ).get();
        refresh();

        logger.info("--> highlighting and searching on field2");
        SearchSourceBuilder source = searchSource().query(queryStringQuery("qui*").defaultField("field2"))
            .highlighter(highlight().field("field2"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field2",
            0,
            1,
            equalTo("The <em>quick</em> brown fox jumps over the lazy dog! Second sentence.")
        );
    }

    public void testPostingsHighlighterRegexpQueryWithinConstantScoreQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "The photography word will get highlighted").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(constantScoreQuery(regexpQuery("field1", "pho[a-z]+")))
            .highlighter(highlight().field("field1"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field1",
            0,
            1,
            equalTo("The <em>photography</em> word will get highlighted")
        );
    }

    public void testPostingsHighlighterMultiTermQueryMultipleLevels() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "The photography word will get highlighted").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boolQuery().should(boolQuery().mustNot(QueryBuilders.existsQuery("field1")))
                .should(matchQuery("field1", "test"))
                .should(constantScoreQuery(queryStringQuery("field1:photo*")))
        ).highlighter(highlight().field("field1"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field1",
            0,
            1,
            equalTo("The <em>photography</em> word will get highlighted")
        );
    }

    public void testPostingsHighlighterPrefixQueryWithinBooleanQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "The photography word will get highlighted").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boolQuery().must(prefixQuery("field1", "photo")).should(matchQuery("field1", "test").minimumShouldMatch("0"))
        ).highlighter(highlight().field("field1"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field1",
            0,
            1,
            equalTo("The <em>photography</em> word will get highlighted")
        );
    }

    public void testPostingsHighlighterQueryStringWithinFilteredQuery() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        prepareIndex("test").setSource("field1", "The photography word will get highlighted").get();
        refresh();

        logger.info("--> highlighting and searching on field1");
        SearchSourceBuilder source = searchSource().query(
            boolQuery().must(queryStringQuery("field1:photo*")).mustNot(existsQuery("field_null"))
        ).highlighter(highlight().field("field1"));
        assertHighlight(
            prepareSearch("test").setSource(source),
            0,
            "field1",
            0,
            1,
            equalTo("The <em>photography</em> word will get highlighted")
        );
    }

    public void testPostingsHighlighterManyDocs() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1PostingsffsetsMapping()));
        ensureGreen();

        int COUNT = between(20, 100);
        Map<String, String> prefixes = Maps.newMapWithExpectedSize(COUNT);

        IndexRequestBuilder[] indexRequestBuilders = new IndexRequestBuilder[COUNT];
        for (int i = 0; i < COUNT; i++) {
            // generating text with word to highlight in a different position
            // (https://github.com/elastic/elasticsearch/issues/4103)
            String prefix = randomAlphaOfLengthBetween(5, 30);
            prefixes.put(String.valueOf(i), prefix);
            indexRequestBuilders[i] = prepareIndex("test").setId(Integer.toString(i))
                .setSource("field1", "Sentence " + prefix + " test. Sentence two.");
        }
        logger.info("--> indexing docs");
        indexRandom(true, indexRequestBuilders);

        logger.info("--> searching explicitly on field1 and highlighting on it");
        SearchRequestBuilder searchRequestBuilder = prepareSearch().setSize(COUNT)
            .setQuery(termQuery("field1", "test"))
            .highlighter(new HighlightBuilder().field("field1"));
        assertResponse(searchRequestBuilder, response -> {
            assertHitCount(response, COUNT);
            assertThat(response.getHits().getHits().length, equalTo(COUNT));
            for (SearchHit hit : response.getHits()) {
                String prefix = prefixes.get(hit.getId());
                assertHighlight(hit, "field1", 0, 1, equalTo("Sentence " + prefix + " <em>test</em>. Sentence two."));
            }
        });
    }

    public void testDoesNotHighlightTypeName() throws Exception {
        XContentBuilder mapping = XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("foo")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mapping));
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("foo", "test typename"));

        for (String highlighter : ALL_TYPES) {
            assertHighlight(
                prepareSearch("test").setQuery(matchQuery("foo", "test"))
                    .highlighter(new HighlightBuilder().field("foo").highlighterType(highlighter).requireFieldMatch(false)),
                0,
                "foo",
                0,
                1,
                equalTo("<em>test</em> typename")
            );
        }
    }

    public void testDoesNotHighlightAliasFilters() throws Exception {
        XContentBuilder mapping = XContentFactory.jsonBuilder()
            .startObject()
            .startObject("_doc")
            .startObject("properties")
            .startObject("foo")
            .field("type", "text")
            .field("index_options", "offsets")
            .field("term_vector", "with_positions_offsets")
            .endObject()
            .endObject()
            .endObject()
            .endObject();
        assertAcked(prepareCreate("test").setMapping(mapping));
        assertAcked(
            indicesAdmin().prepareAliases(TEST_REQUEST_TIMEOUT, TEST_REQUEST_TIMEOUT)
                .addAlias("test", "filtered_alias", matchQuery("foo", "japanese"))
        );
        ensureGreen();

        indexRandom(true, prepareIndex("test").setSource("foo", "test japanese"));

        for (String highlighter : ALL_TYPES) {
            assertHighlight(
                prepareSearch("filtered_alias").setQuery(matchQuery("foo", "test"))
                    .highlighter(new HighlightBuilder().field("foo").highlighterType(highlighter).requireFieldMatch(false)),
                0,
                "foo",
                0,
                1,
                equalTo("<em>test</em> japanese")
            );
        }
    }

    public void testFastVectorHighlighterPhraseBoost() throws Exception {
        assertAcked(prepareCreate("test").setMapping(type1TermVectorMapping()));
        phraseBoostTestCase("fvh");
    }

    /**
     * Test phrase boosting over normal term matches.  Note that this will never pass with the plain highlighter
     * because it doesn't support the concept of terms having a different weight based on position.
     * @param highlighterType highlighter to test
     */
    private void phraseBoostTestCase(String highlighterType) {
        ensureGreen();
        String text = """
            words words junk junk junk junk junk junk junk junk highlight junk junk junk junk together junk
            """ + """
            junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk
            """.repeat(10) + """
            highlight words together
            """ + """
            junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk junk
            """.repeat(10);
        indexDoc("test", "1", "field1", text);
        refresh();

        // Match queries
        phraseBoostTestCaseForClauses(
            highlighterType,
            100f,
            matchQuery("field1", "highlight words together"),
            matchPhraseQuery("field1", "highlight words together")
        );

        // Query string with a single field
        phraseBoostTestCaseForClauses(
            highlighterType,
            100f,
            queryStringQuery("highlight words together").field("field1"),
            queryStringQuery("\"highlight words together\"").field("field1")
        );

        // Query string with a single field without dismax
        phraseBoostTestCaseForClauses(
            highlighterType,
            100f,
            queryStringQuery("highlight words together").field("field1"),
            queryStringQuery("\"highlight words together\"").field("field1")
        );

        // Query string with more than one field
        phraseBoostTestCaseForClauses(
            highlighterType,
            100f,
            queryStringQuery("highlight words together").field("field1").field("field2"),
            queryStringQuery("\"highlight words together\"").field("field1").field("field2")
        );

        // Query string boosting the field
        phraseBoostTestCaseForClauses(
            highlighterType,
            1f,
            queryStringQuery("highlight words together").field("field1"),
            queryStringQuery("\"highlight words together\"").field("field1", 100)
        );
    }

    private <P extends AbstractQueryBuilder<P>> void phraseBoostTestCaseForClauses(
        String highlighterType,
        float boost,
        QueryBuilder terms,
        P phrase
    ) {
        Matcher<String> highlightedMatcher = Matchers.either(containsString("<em>highlight words together</em>"))
            .or(containsString("<em>highlight</em> <em>words</em> <em>together</em>"));
        SearchRequestBuilder search = prepareSearch("test").highlighter(
            new HighlightBuilder().field("field1", 100, 1).order("score").highlighterType(highlighterType).requireFieldMatch(true)
        );

        // Try with a bool query
        phrase.boost(boost);
        assertHighlight(search.setQuery(boolQuery().must(terms).should(phrase)), 0, "field1", 0, 1, highlightedMatcher);
        phrase.boost(1);
        // Try with a boosting query
        assertHighlight(search.setQuery(boostingQuery(phrase, terms).boost(boost).negativeBoost(1)), 0, "field1", 0, 1, highlightedMatcher);
        // Try with a boosting query using a negative boost
        assertHighlight(
            search.setQuery(boostingQuery(phrase, terms).boost(1).negativeBoost(1 / boost)),
            0,
            "field1",
            0,
            1,
            highlightedMatcher
        );
    }

    public void testGeoFieldHighlightingWithDifferentHighlighters() throws IOException {
        // check that we do not get an exception for geo_point fields in case someone tries to highlight
        // it accidentially with a wildcard
        // see https://github.com/elastic/elasticsearch/issues/17537
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("geo_point")
            .field("type", "geo_point")
            .endObject()
            .startObject("text")
            .field("type", "text")
            .field("term_vector", "with_positions_offsets_payloads")
            .field("index_options", "offsets")
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));

        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().field("text", "Arbitrary text field which will should not cause a failure").endObject())
            .get();
        refresh();
        String highlighterType = randomFrom(ALL_TYPES);
        QueryBuilder query = QueryBuilders.boolQuery()
            .should(
                QueryBuilders.geoBoundingBoxQuery("geo_point")
                    .setCorners(61.10078883158897, -170.15625, -64.92354174306496, 118.47656249999999)
            )
            .should(QueryBuilders.termQuery("text", "failure"));
        assertNoFailuresAndResponse(
            prepareSearch().setSource(
                new SearchSourceBuilder().query(query).highlighter(new HighlightBuilder().field("*").highlighterType(highlighterType))
            ),
            response -> {
                assertThat(response.getHits().getTotalHits().value(), equalTo(1L));
                assertThat(response.getHits().getAt(0).getHighlightFields().get("text").fragments().length, equalTo(1));
            }
        );
    }

    public void testGeoFieldHighlightingWhenQueryGetsRewritten() throws IOException {
        // same as above but in this example the query gets rewritten during highlighting
        // see https://github.com/elastic/elasticsearch/issues/17537#issuecomment-244939633
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("loc")
            .field("type", "geo_point")
            .endObject()
            .startObject("jd")
            .field("type", "text")
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));
        ensureYellow();

        prepareIndex("test").setId("1")
            .setSource(
                jsonBuilder().startObject()
                    .field("jd", "some आवश्यकता है- आर्य समाज अनाथालय, 68 सिविल लाइन्स, बरेली को एक पुरूष" + " रस text")
                    .field("loc", "12.934059,77.610741")
                    .endObject()
            )
            .get();
        refresh();

        QueryBuilder query = QueryBuilders.functionScoreQuery(
            QueryBuilders.boolQuery()
                .filter(
                    QueryBuilders.geoBoundingBoxQuery("loc")
                        .setCorners(new GeoPoint(48.934059, 41.610741), new GeoPoint(-23.065941, 113.610741))
                )
        );
        assertHitCountAndNoFailures(
            prepareSearch().setSource(
                new SearchSourceBuilder().query(query).highlighter(new HighlightBuilder().highlighterType("plain").field("jd"))
            ),
            1
        );
    }

    public void testKeywordFieldHighlighting() throws IOException {
        // check that keyword highlighting works
        XContentBuilder mappings = jsonBuilder();
        mappings.startObject();
        mappings.startObject("_doc")
            .startObject("properties")
            .startObject("keyword_field")
            .field("type", "keyword")
            .endObject()
            .endObject()
            .endObject();
        mappings.endObject();
        assertAcked(prepareCreate("test").setMapping(mappings));

        prepareIndex("test").setId("1").setSource(jsonBuilder().startObject().field("keyword_field", "some text").endObject()).get();
        refresh();
        assertNoFailuresAndResponse(
            prepareSearch().setSource(
                new SearchSourceBuilder().query(QueryBuilders.matchQuery("keyword_field", "some text"))
                    .highlighter(new HighlightBuilder().field("*"))
            ),
            response -> {
                assertThat(response.getHits().getTotalHits().value(), equalTo(1L));
                HighlightField highlightField = response.getHits().getAt(0).getHighlightFields().get("keyword_field");
                assertThat(highlightField.fragments()[0].string(), equalTo("<em>some text</em>"));
            }
        );
    }

    public void testCopyToFields() throws Exception {
        XContentBuilder b = jsonBuilder().startObject().startObject("properties");
        b.startObject("foo");
        {
            b.field("type", "text");
            b.field("copy_to", "foo_copy");
        }
        b.endObject();
        b.startObject("foo_copy").field("type", "text").endObject();
        b.endObject().endObject();
        prepareCreate("test").setMapping(b).get();

        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().field("foo", "how now brown cow").endObject())
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();

        assertResponse(
            prepareSearch().setQuery(matchQuery("foo_copy", "brown")).highlighter(new HighlightBuilder().field(new Field("foo_copy"))),
            response -> {
                assertHitCount(response, 1);
                HighlightField field = response.getHits().getAt(0).getHighlightFields().get("foo_copy");
                assertThat(field.fragments().length, equalTo(1));
                assertThat(field.fragments()[0].string(), equalTo("how now <em>brown</em> cow"));
            }
        );
    }

    public void testACopyFieldWithNestedQuery() throws Exception {
        String mapping = Strings.toString(
            jsonBuilder().startObject()
                .startObject("properties")
                .startObject("foo")
                .field("type", "nested")
                .startObject("properties")
                .startObject("text")
                .field("type", "text")
                .field("copy_to", "foo_text")
                .endObject()
                .endObject()
                .endObject()
                .startObject("foo_text")
                .field("type", "text")
                .field("term_vector", "with_positions_offsets")
                .field("store", true)
                .endObject()
                .endObject()
                .endObject()
        );
        prepareCreate("test").setMapping(mapping).get();

        prepareIndex("test").setId("1")
            .setSource(
                jsonBuilder().startObject()
                    .startArray("foo")
                    .startObject()
                    .field("text", "brown")
                    .endObject()
                    .startObject()
                    .field("text", "cow")
                    .endObject()
                    .endArray()
                    .endObject()
            )
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();

        assertResponse(
            prepareSearch().setQuery(nestedQuery("foo", matchQuery("foo.text", "brown cow"), ScoreMode.None))
                .highlighter(new HighlightBuilder().field(new Field("foo_text").highlighterType("fvh")).requireFieldMatch(false)),
            response -> {
                assertHitCount(response, 1);
                HighlightField field = response.getHits().getAt(0).getHighlightFields().get("foo_text");
                assertThat(field.fragments().length, equalTo(2));
                assertThat(field.fragments()[0].string(), equalTo("<em>brown</em>"));
                assertThat(field.fragments()[1].string(), equalTo("<em>cow</em>"));
            }
        );
    }

    public void testFunctionScoreQueryHighlight() throws Exception {
        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().field("text", "brown").endObject())
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();

        assertResponse(
            prepareSearch().setQuery(new FunctionScoreQueryBuilder(QueryBuilders.prefixQuery("text", "bro")))
                .highlighter(new HighlightBuilder().field(new Field("text"))),
            response -> {
                assertHitCount(response, 1);
                HighlightField field = response.getHits().getAt(0).getHighlightFields().get("text");
                assertThat(field.fragments().length, equalTo(1));
                assertThat(field.fragments()[0].string(), equalTo("<em>brown</em>"));
            }
        );
    }

    public void testFiltersFunctionScoreQueryHighlight() throws Exception {
        prepareIndex("test").setId("1")
            .setSource(jsonBuilder().startObject().field("text", "brown").field("enable", "yes").endObject())
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();
        FunctionScoreQueryBuilder.FilterFunctionBuilder filterBuilder = new FunctionScoreQueryBuilder.FilterFunctionBuilder(
            QueryBuilders.termQuery("enable", "yes"),
            new RandomScoreFunctionBuilder()
        );

        assertResponse(
            prepareSearch().setQuery(
                new FunctionScoreQueryBuilder(
                    QueryBuilders.prefixQuery("text", "bro"),
                    new FunctionScoreQueryBuilder.FilterFunctionBuilder[] { filterBuilder }
                )
            ).highlighter(new HighlightBuilder().field(new Field("text"))),
            response -> {
                assertHitCount(response, 1);
                HighlightField field = response.getHits().getAt(0).getHighlightFields().get("text");
                assertThat(field.fragments().length, equalTo(1));
                assertThat(field.fragments()[0].string(), equalTo("<em>brown</em>"));
            }
        );
    }

    public void testHighlightQueryRewriteDatesWithNow() throws Exception {
        assertAcked(
            indicesAdmin().prepareCreate("index-1")
                .setMapping("d", "type=date", "field", "type=text,store=true,term_vector=with_positions_offsets")
                .setSettings(indexSettings(2, 0))
        );
        ZonedDateTime now = ZonedDateTime.now(ZoneOffset.UTC);
        DateFormatter formatter = DateFormatter.forPattern("strict_date_optional_time");
        indexRandom(
            true,
            prepareIndex("index-1").setId("1").setSource("d", formatter.format(now), "field", "hello world"),
            prepareIndex("index-1").setId("2").setSource("d", formatter.format(now.minusDays(1)), "field", "hello"),
            prepareIndex("index-1").setId("3").setSource("d", formatter.format(now.minusDays(2)), "field", "world")
        );
        ensureSearchable("index-1");
        for (int i = 0; i < 5; i++) {
            assertNoFailuresAndResponse(
                prepareSearch("index-1").addSort("d", SortOrder.DESC)
                    .setTrackScores(true)
                    .highlighter(highlight().field("field").preTags("<x>").postTags("</x>"))
                    .setQuery(
                        QueryBuilders.boolQuery()
                            .must(QueryBuilders.rangeQuery("d").gte("now-12h").lte("now").includeLower(true).includeUpper(true).boost(1.0f))
                            .should(QueryBuilders.termQuery("field", "hello"))
                    ),
                response -> {
                    assertThat(response.getHits().getTotalHits().value(), equalTo(1L));
                    assertHighlight(response, 0, "field", 0, 1, equalTo("<x>hello</x> world"));
                }
            );
        }
    }

    public void testWithNestedQuery() throws Exception {
        String mapping = Strings.toString(
            jsonBuilder().startObject()
                .startObject("properties")
                .startObject("text")
                .field("type", "text")
                .field("index_options", "offsets")
                .field("term_vector", "with_positions_offsets")
                .endObject()
                .startObject("foo")
                .field("type", "nested")
                .startObject("properties")
                .startObject("text")
                .field("type", "text")
                .endObject()
                .endObject()
                .endObject()
                .endObject()
                .endObject()
        );
        prepareCreate("test").setMapping(mapping).get();

        prepareIndex("test").setId("1")
            .setSource(
                jsonBuilder().startObject()
                    .startArray("foo")
                    .startObject()
                    .field("text", "brown shoes")
                    .endObject()
                    .startObject()
                    .field("text", "cow")
                    .endObject()
                    .endArray()
                    .field("text", "brown")
                    .endObject()
            )
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();

        for (String type : new String[] { "unified", "plain" }) {
            assertResponse(
                prepareSearch().setQuery(nestedQuery("foo", matchQuery("foo.text", "brown cow"), ScoreMode.None))
                    .highlighter(new HighlightBuilder().field(new Field("foo.text").highlighterType(type))),
                response -> {
                    assertHitCount(response, 1);
                    HighlightField field = response.getHits().getAt(0).getHighlightFields().get("foo.text");
                    assertThat(field.fragments().length, equalTo(2));
                    assertThat(field.fragments()[0].string(), equalTo("<em>brown</em> shoes"));
                    assertThat(field.fragments()[1].string(), equalTo("<em>cow</em>"));
                }
            );
            assertResponses(response -> {
                assertHitCount(response, 1);
                HighlightField field = response.getHits().getAt(0).getHighlightFields().get("foo.text");
                assertThat(field.fragments().length, equalTo(1));
                assertThat(field.fragments()[0].string(), equalTo("<em>brown</em> shoes"));
            },
                prepareSearch().setQuery(nestedQuery("foo", prefixQuery("foo.text", "bro"), ScoreMode.None))
                    .highlighter(new HighlightBuilder().field(new Field("foo.text").highlighterType(type))),
                prepareSearch().setQuery(nestedQuery("foo", matchPhrasePrefixQuery("foo.text", "bro"), ScoreMode.None))
                    .highlighter(new HighlightBuilder().field(new Field("foo.text").highlighterType(type)))
            );
            assertResponse(
                prepareSearch().setQuery(nestedQuery("foo", matchPhraseQuery("foo.text", "brown shoes"), ScoreMode.None))
                    .highlighter(new HighlightBuilder().field(new Field("foo.text").highlighterType(type))),
                response -> {
                    assertHitCount(response, 1);
                    HighlightField field = response.getHits().getAt(0).getHighlightFields().get("foo.text");
                    assertThat(field.fragments().length, equalTo(1));
                    assertThat(field.fragments()[0].string(), equalTo("<em>brown</em> <em>shoes</em>"));
                }
            );
        }

        // For unified and fvh highlighters we just check that the nested query is correctly extracted
        // but we highlight the root text field since nested documents cannot be highlighted with postings nor term vectors
        // directly.
        for (String type : ALL_TYPES) {
            assertResponse(
                prepareSearch().setQuery(nestedQuery("foo", prefixQuery("foo.text", "bro"), ScoreMode.None))
                    .highlighter(new HighlightBuilder().field(new Field("text").highlighterType(type).requireFieldMatch(false))),
                response -> {
                    assertHitCount(response, 1);
                    HighlightField field = response.getHits().getAt(0).getHighlightFields().get("text");
                    assertThat(field.fragments().length, equalTo(1));
                    assertThat(field.fragments()[0].string(), equalTo("<em>brown</em>"));
                }
            );
        }
    }

    public void testWithNormalizer() throws Exception {
        Builder builder = Settings.builder().put(indexSettings()).putList("index.analysis.normalizer.my_normalizer.filter", "lowercase");

        assertAcked(prepareCreate("test").setSettings(builder.build()).setMapping("keyword", "type=keyword,normalizer=my_normalizer"));
        ensureGreen();

        prepareIndex("test").setId("0").setSource("keyword", "Hello World").setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE).get();

        for (String highlighterType : new String[] { "unified", "plain" }) {
            assertResponse(
                prepareSearch().setQuery(matchQuery("keyword", "hello world"))
                    .highlighter(new HighlightBuilder().field(new Field("keyword").highlighterType(highlighterType))),
                response -> {
                    assertHitCount(response, 1);
                    HighlightField field = response.getHits().getAt(0).getHighlightFields().get("keyword");
                    assertThat(field.fragments().length, equalTo(1));
                    assertThat(field.fragments()[0].string(), equalTo("<em>hello world</em>"));
                }
            );
        }
    }

    public void testDisableHighlightIdField() throws Exception {
        assertAcked(prepareCreate("test").setMapping("keyword", "type=keyword"));
        ensureGreen();

        prepareIndex("test").setId("d33f85bf1e51e84d9ab38948db9f3a068e1fe5294f1d8603914ac8c7bcc39ca1")
            .setSource("keyword", "Hello World")
            .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE)
            .get();

        for (String highlighterType : new String[] { "plain", "unified" }) {
            assertResponse(
                prepareSearch().setQuery(matchQuery("_id", "d33f85bf1e51e84d9ab38948db9f3a068e1fe5294f1d8603914ac8c7bcc39ca1"))
                    .highlighter(new HighlightBuilder().field(new Field("*").highlighterType(highlighterType).requireFieldMatch(false))),
                response -> {
                    assertHitCount(response, 1);
                    assertNull(response.getHits().getAt(0).getHighlightFields().get("_id"));
                }
            );
        }
    }

    public static class MockAnalysisPlugin extends Plugin implements AnalysisPlugin {

        public final class MockSnowBall extends TokenFilter {
            private final CharTermAttribute termAtt = addAttribute(CharTermAttribute.class);

            /** Sole constructor. */
            MockSnowBall(TokenStream in) {
                super(in);
            }

            @Override
            public boolean incrementToken() throws IOException {
                if (input.incrementToken()) {
                    final char[] buffer = termAtt.buffer();
                    final int length = termAtt.length();
                    if (buffer[length - 1] == 's') {
                        termAtt.setLength(length - 1);
                    }
                    if (length > 3) {
                        if (buffer[length - 1] == 'g' && buffer[length - 2] == 'n' && buffer[length - 3] == 'i') {
                            termAtt.setLength(length - 3);
                        }
                    }
                    return true;
                } else return false;
            }
        }

        @Override
        public List<PreConfiguredTokenFilter> getPreConfiguredTokenFilters() {
            return singletonList(PreConfiguredTokenFilter.singleton("mock_snowball", false, MockSnowBall::new));
        }

        @Override
        public Map<String, AnalysisModule.AnalysisProvider<AnalyzerProvider<? extends Analyzer>>> getAnalyzers() {
            return singletonMap(
                "mock_whitespace",
                (indexSettings, environment, name, settings) -> new AbstractIndexAnalyzerProvider<Analyzer>(name) {

                    MockAnalyzer instance = new MockAnalyzer(random(), MockTokenizer.WHITESPACE, false);

                    @Override
                    public Analyzer get() {
                        return instance;
                    }
                }
            );
        }
    }
}
