/*
 * 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.index.mapper;

import org.apache.lucene.analysis.standard.StandardAnalyzer;
import org.elasticsearch.TransportVersion;
import org.elasticsearch.common.Strings;
import org.elasticsearch.common.compress.CompressedXContent;
import org.elasticsearch.common.lucene.Lucene;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.xcontent.XContentHelper;
import org.elasticsearch.index.IndexSettings;
import org.elasticsearch.index.IndexVersion;
import org.elasticsearch.index.analysis.AnalyzerScope;
import org.elasticsearch.index.analysis.IndexAnalyzers;
import org.elasticsearch.index.analysis.NamedAnalyzer;
import org.elasticsearch.index.mapper.FieldMapper.Parameter;
import org.elasticsearch.plugins.MapperPlugin;
import org.elasticsearch.plugins.Plugin;
import org.elasticsearch.script.ScriptCompiler;
import org.elasticsearch.xcontent.ToXContent;
import org.elasticsearch.xcontent.XContentBuilder;
import org.elasticsearch.xcontent.json.JsonXContent;

import java.io.IOException;
import java.util.Collection;
import java.util.EnumSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;

import static org.elasticsearch.index.mapper.MapperService.MergeReason.MAPPING_UPDATE;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.instanceOf;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

public class ParametrizedMapperTests extends MapperServiceTestCase {

    public enum DummyEnumType {
        NAME1,
        NAME2,
        NAME3;

        @Override
        public final String toString() {
            return name().toLowerCase(Locale.ROOT);
        }
    }

    public static class TestPlugin extends Plugin implements MapperPlugin {
        @Override
        public Map<String, Mapper.TypeParser> getMappers() {
            return Map.of("test_mapper", new TypeParser());
        }
    }

    @Override
    protected Collection<Plugin> getPlugins() {
        return List.of(new TestPlugin());
    }

    private static class StringWrapper {
        final String name;

        private StringWrapper(String name) {
            this.name = name;
        }

        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            StringWrapper that = (StringWrapper) o;
            return Objects.equals(name, that.name);
        }

        @Override
        public int hashCode() {
            return Objects.hash(name);
        }
    }

    private static TestMapper toType(Mapper in) {
        return (TestMapper) in;
    }

    public static class Builder extends FieldMapper.Builder {

        final Parameter<Boolean> fixed = Parameter.boolParam("fixed", false, m -> toType(m).fixed, true);
        final Parameter<Boolean> fixed2 = Parameter.boolParam("fixed2", false, m -> toType(m).fixed2, false)
            .addDeprecatedName("fixed2_old");
        final Parameter<String> variable = Parameter.stringParam("variable", true, m -> toType(m).variable, "default").acceptsNull();
        final Parameter<StringWrapper> wrapper = new Parameter<>("wrapper", false, () -> new StringWrapper("default"), (n, c, o) -> {
            if (o == null) return null;
            return new StringWrapper(o.toString());
        }, m -> toType(m).wrapper, (b, n, v) -> b.field(n, v.name), v -> "wrapper_" + v.name);
        final Parameter<Integer> intValue = Parameter.intParam("int_value", true, m -> toType(m).intValue, 5).addValidator(n -> {
            if (n > 50) {
                throw new IllegalArgumentException("Value of [n] cannot be greater than 50");
            }
        }).addValidator(n -> {
            if (n < 0) {
                throw new IllegalArgumentException("Value of [n] cannot be less than 0");
            }
        }).setMergeValidator((o, n, c) -> n >= o);
        final Parameter<NamedAnalyzer> analyzer = Parameter.analyzerParam(
            "analyzer",
            false,
            m -> toType(m).analyzer,
            () -> Lucene.KEYWORD_ANALYZER
        );
        final Parameter<NamedAnalyzer> searchAnalyzer = Parameter.analyzerParam(
            "search_analyzer",
            true,
            m -> toType(m).searchAnalyzer,
            analyzer::getValue
        );
        final Parameter<Boolean> index = Parameter.boolParam("index", false, m -> toType(m).index, true);
        final Parameter<String> required = Parameter.stringParam("required", true, m -> toType(m).required, null).addValidator(value -> {
            if (value == null) {
                throw new IllegalArgumentException("field [required] must be specified");
            }
        });
        final Parameter<DummyEnumType> enumField = Parameter.enumParam(
            "enum_field",
            true,
            m -> toType(m).enumField,
            DummyEnumType.NAME1,
            DummyEnumType.class
        );

        final Parameter<DummyEnumType> restrictedEnumField = Parameter.restrictedEnumParam(
            "restricted_enum_field",
            true,
            m -> toType(m).restrictedEnumField,
            DummyEnumType.NAME1,
            DummyEnumType.class,
            EnumSet.of(DummyEnumType.NAME1, DummyEnumType.NAME2)
        );

        protected Builder(String name) {
            super(name);
            // only output search analyzer if different to analyzer
            searchAnalyzer.setSerializerCheck(
                (id, ic, v) -> Objects.equals(analyzer.getValue().name(), searchAnalyzer.getValue().name()) == false
            );
        }

        @Override
        protected Parameter<?>[] getParameters() {
            return new Parameter<?>[] {
                fixed,
                fixed2,
                variable,
                index,
                wrapper,
                intValue,
                analyzer,
                searchAnalyzer,
                required,
                enumField,
                restrictedEnumField };
        }

        @Override
        public FieldMapper build(MapperBuilderContext context) {
            return new TestMapper(leafName(), context.buildFullName(leafName()), builderParams(this, context), this);
        }
    }

    public static class TypeParser implements Mapper.TypeParser {

        @Override
        public Mapper.Builder parse(String name, Map<String, Object> node, MappingParserContext parserContext)
            throws MapperParsingException {
            Builder builder = new Builder(name);
            builder.parse(name, parserContext, node);
            return builder;
        }
    }

    public static class TestMapper extends FieldMapper {

        private final boolean fixed;
        private final boolean fixed2;
        private final String variable;
        private final StringWrapper wrapper;
        private final int intValue;
        private final NamedAnalyzer analyzer;
        private final NamedAnalyzer searchAnalyzer;
        private final boolean index;
        private final String required;
        private final DummyEnumType enumField;
        private final DummyEnumType restrictedEnumField;

        protected TestMapper(String simpleName, String fullName, BuilderParams builderParams, ParametrizedMapperTests.Builder builder) {
            super(simpleName, new KeywordFieldMapper.KeywordFieldType(fullName), builderParams);
            this.fixed = builder.fixed.getValue();
            this.fixed2 = builder.fixed2.getValue();
            this.variable = builder.variable.getValue();
            this.wrapper = builder.wrapper.getValue();
            this.intValue = builder.intValue.getValue();
            this.analyzer = builder.analyzer.getValue();
            this.searchAnalyzer = builder.searchAnalyzer.getValue();
            this.index = builder.index.getValue();
            this.required = builder.required.getValue();
            this.enumField = builder.enumField.getValue();
            this.restrictedEnumField = builder.restrictedEnumField.getValue();
        }

        @Override
        public Builder getMergeBuilder() {
            return new ParametrizedMapperTests.Builder(leafName()).init(this);
        }

        @Override
        protected void parseCreateField(DocumentParserContext context) {

        }

        @Override
        protected String contentType() {
            return "test_mapper";
        }
    }

    private static TestMapper fromMapping(
        String mapping,
        IndexVersion version,
        TransportVersion transportVersion,
        boolean fromDynamicTemplate
    ) {
        MapperService mapperService = mock(MapperService.class);
        IndexAnalyzers indexAnalyzers = IndexAnalyzers.of(
            Map.of(
                "_standard",
                Lucene.STANDARD_ANALYZER,
                "_keyword",
                Lucene.KEYWORD_ANALYZER,
                "default",
                new NamedAnalyzer("default", AnalyzerScope.INDEX, new StandardAnalyzer())
            )
        );
        when(mapperService.getIndexAnalyzers()).thenReturn(indexAnalyzers);
        IndexSettings indexSettings = createIndexSettings(version, Settings.EMPTY);
        when(mapperService.getIndexSettings()).thenReturn(indexSettings);
        MappingParserContext pc = new MappingParserContext(s -> null, s -> {
            if (Objects.equals("keyword", s)) {
                return KeywordFieldMapper.PARSER;
            }
            if (Objects.equals("binary", s)) {
                return BinaryFieldMapper.PARSER;
            }
            return null;
        },
            name -> null,
            version,
            () -> transportVersion,
            () -> null,
            ScriptCompiler.NONE,
            mapperService.getIndexAnalyzers(),
            mapperService.getIndexSettings(),
            mapperService.getIndexSettings().getMode().idFieldMapperWithoutFieldData(),
            query -> {
                throw new UnsupportedOperationException();
            }
        );
        if (fromDynamicTemplate) {
            pc = pc.createDynamicTemplateContext(null);
        }
        return (TestMapper) new TypeParser().parse("field", XContentHelper.convertToMap(JsonXContent.jsonXContent, mapping, true), pc)
            .build(MapperBuilderContext.root(false, false));
    }

    private static TestMapper fromMapping(String mapping, IndexVersion version, TransportVersion transportVersion) {
        return fromMapping(mapping, version, transportVersion, false);
    }

    private static TestMapper fromMapping(String mapping) {
        return fromMapping(mapping, IndexVersion.current(), TransportVersion.current());
    }

    private String toStringWithDefaults(ToXContent value) throws IOException {
        XContentBuilder builder = JsonXContent.contentBuilder();
        ToXContent.Params params = new ToXContent.MapParams(Map.of("include_defaults", "true"));
        builder.startObject();
        value.toXContent(builder, params);
        builder.endObject();
        return Strings.toString(builder);

    }

    // defaults - create empty builder config, and serialize with and without defaults
    public void testDefaults() throws IOException {
        String mapping = """
            {"type":"test_mapper","required":"value"}""";

        TestMapper mapper = fromMapping(mapping);

        assertTrue(mapper.fixed);
        assertEquals("default", mapper.variable);

        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        assertEquals(XContentHelper.stripWhitespace("""
            {
              "field": {
                "type": "test_mapper",
                "fixed": true,
                "fixed2": false,
                "variable": "default",
                "index": true,
                "wrapper": "default",
                "int_value": 5,
                "analyzer": "_keyword",
                "required": "value",
                "enum_field": "name1",
                "restricted_enum_field": "name1"
              }
            }"""), toStringWithDefaults(mapper));
    }

    // merging - try updating 'fixed' and 'fixed2' should get an error, try updating 'variable' and verify update
    public void testMerging() {
        String mapping = """
            {"type":"test_mapper","fixed":false,"required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        TestMapper badMerge = fromMapping("""
            {"type":"test_mapper","fixed":true,"fixed2":true,"required":"value"}""");
        IllegalArgumentException e = expectThrows(
            IllegalArgumentException.class,
            () -> mapper.merge(badMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE))
        );
        String expectedError = """
            Mapper for [field] conflicts with existing mapper:
            \tCannot update parameter [fixed] from [false] to [true]
            \tCannot update parameter [fixed2] from [false] to [true]""";
        assertEquals(expectedError, e.getMessage());

        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));   // original mapping is unaffected

        // TODO: should we have to include 'fixed' here? Or should updates take as 'defaults' the existing values?
        TestMapper goodMerge = fromMapping("""
            {"type":"test_mapper","fixed":false,"variable":"updated","required":"value"}""");
        TestMapper merged = (TestMapper) mapper.merge(goodMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE));

        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper)); // original mapping is unaffected
        assertEquals("""
            {"field":{"type":"test_mapper","fixed":false,"variable":"updated","required":"value"}}\
            """, Strings.toString(merged));
    }

    // add multifield, verify, add second multifield, verify, overwrite second multifield
    public void testMultifields() throws IOException {
        String mapping = """
            {"type":"test_mapper","variable":"foo","required":"value","fields":{"sub":{"type":"keyword"}}}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        String addSubField = """
            {"type":"test_mapper","variable":"foo","required":"value","fields":{"sub2":{"type":"keyword"}}}""";
        TestMapper toMerge = fromMapping(addSubField);
        TestMapper merged = (TestMapper) mapper.merge(toMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE));
        assertEquals(XContentHelper.stripWhitespace("""
            {
              "field": {
                "type": "test_mapper",
                "variable": "foo",
                "required": "value",
                "fields": {
                  "sub": {
                    "type": "keyword"
                  },
                  "sub2": {
                    "type": "keyword"
                  }
                }
              }
            }"""), Strings.toString(merged));

        String badSubField = """
            {"type":"test_mapper","variable":"foo","required":"value","fields":{"sub2":{"type":"binary"}}}""";
        TestMapper badToMerge = fromMapping(badSubField);
        IllegalArgumentException e = expectThrows(
            IllegalArgumentException.class,
            () -> merged.merge(badToMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE))
        );
        assertEquals("mapper [field.sub2] cannot be changed from type [keyword] to [binary]", e.getMessage());
    }

    // add copy_to, verify
    public void testCopyTo() {
        String mapping = """
            {"type":"test_mapper","variable":"foo","required":"value","copy_to":["other"]}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        // On update, copy_to is completely replaced

        TestMapper toMerge = fromMapping("""
            {"type":"test_mapper","variable":"updated","required":"value","copy_to":["foo","bar"]}""");
        TestMapper merged = (TestMapper) mapper.merge(toMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE));
        assertEquals("""
            {"field":{"type":"test_mapper","variable":"updated","required":"value","copy_to":["foo","bar"]}}""", Strings.toString(merged));

        TestMapper removeCopyTo = fromMapping("""
            {"type":"test_mapper","variable":"updated","required":"value"}""");
        TestMapper noCopyTo = (TestMapper) merged.merge(
            removeCopyTo,
            MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE)
        );
        assertEquals("""
            {"field":{"type":"test_mapper","variable":"updated","required":"value"}}""", Strings.toString(noCopyTo));
    }

    public void testStoredSource() {
        String mapping = """
            {"type":"test_mapper","variable":"foo","required":"value","synthetic_source_keep":"none"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        mapping = """
            {"type":"test_mapper","variable":"foo","required":"value","synthetic_source_keep":"arrays"}""";
        mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        mapping = """
            {"type":"test_mapper","variable":"foo","required":"value","synthetic_source_keep":"all"}""";
        mapper = fromMapping(mapping);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        String mappingThrows = """
            {"type":"test_mapper","variable":"foo","required":"value","synthetic_source_keep":"no-such-value"}""";
        IllegalArgumentException e = expectThrows(IllegalArgumentException.class, () -> fromMapping(mappingThrows));
        assertEquals("Unknown synthetic_source_keep value [no-such-value], accepted values are [none,arrays,all]", e.getMessage());
    }

    public void testNullables() {
        String mapping = """
            {"type":"test_mapper","fixed":null,"required":"value"}""";
        MapperParsingException e = expectThrows(MapperParsingException.class, () -> fromMapping(mapping));
        assertEquals("[fixed] on mapper [field] of type [test_mapper] must not have a [null] value", e.getMessage());

        String fine = """
            {"type":"test_mapper","variable":null,"required":"value"}""";
        TestMapper mapper = fromMapping(fine);
        assertEquals("{\"field\":" + fine + "}", Strings.toString(mapper));
    }

    public void testObjectSerialization() throws IOException {
        String mapping = XContentHelper.stripWhitespace("""
            {
              "_doc": {
                "properties": {
                  "actual": {
                    "type": "double"
                  },
                  "bucket_count": {
                    "type": "long"
                  },
                  "bucket_influencers": {
                    "type": "nested",
                    "properties": {
                      "anomaly_score": {
                        "type": "double"
                      },
                      "bucket_span": {
                        "type": "long"
                      },
                      "is_interim": {
                        "type": "boolean"
                      }
                    }
                  }
                }
              }
            }""");

        MapperService mapperService = createMapperService(mapping);
        assertEquals(mapping, Strings.toString(mapperService.documentMapper().mapping()));

        mapperService.merge("_doc", new CompressedXContent(mapping), MAPPING_UPDATE);
        assertEquals(mapping, Strings.toString(mapperService.documentMapper().mapping()));
    }

    // test custom serializer
    public void testCustomSerialization() {
        String mapping = """
            {"type":"test_mapper","wrapper":"wrapped value","required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("wrapped value", mapper.wrapper.name);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        String conflict = """
            {"type":"test_mapper","wrapper":"new value","required":"value"}""";
        TestMapper toMerge = fromMapping(conflict);
        IllegalArgumentException e = expectThrows(
            IllegalArgumentException.class,
            () -> mapper.merge(toMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE))
        );
        assertEquals(
            "Mapper for [field] conflicts with existing mapper:\n"
                + "\tCannot update parameter [wrapper] from [wrapper_wrapped value] to [wrapper_new value]",
            e.getMessage()
        );
    }

    // test validator
    public void testParameterValidation() {
        String mapping = """
            {"type":"test_mapper","int_value":10,"required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals(10, mapper.intValue);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        IllegalArgumentException e = expectThrows(IllegalArgumentException.class, () -> fromMapping("""
            {"type":"test_mapper","int_value":60,"required":"value"}"""));
        assertEquals("Value of [n] cannot be greater than 50", e.getMessage());

        IllegalArgumentException e2 = expectThrows(IllegalArgumentException.class, () -> fromMapping("""
            {"type":"test_mapper","int_value":-60,"required":"value"}"""));
        assertEquals("Value of [n] cannot be less than 0", e2.getMessage());
    }

    // test deprecations
    public void testDeprecatedParameterName() {
        String mapping = """
            {"type":"test_mapper","fixed2_old":true,"required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertTrue(mapper.fixed2);
        assertWarnings("Parameter [fixed2_old] on mapper [field] is deprecated, use [fixed2]");
        assertEquals("""
            {"field":{"type":"test_mapper","fixed2":true,"required":"value"}}""", Strings.toString(mapper));
    }

    public void testAnalyzers() {
        String mapping = """
            {"type":"test_mapper","analyzer":"_standard","required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals(mapper.analyzer, Lucene.STANDARD_ANALYZER);
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        String withDef = """
            {"type":"test_mapper","analyzer":"default","required":"value"}""";
        mapper = fromMapping(withDef);
        assertEquals(mapper.analyzer.name(), "default");
        assertThat(mapper.analyzer.analyzer(), instanceOf(StandardAnalyzer.class));
        assertEquals("{\"field\":" + withDef + "}", Strings.toString(mapper));

        String badAnalyzer = """
            {"type":"test_mapper","analyzer":"wibble","required":"value"}""";
        IllegalArgumentException e = expectThrows(IllegalArgumentException.class, () -> fromMapping(badAnalyzer));
        assertEquals("analyzer [wibble] has not been configured in mappings", e.getMessage());

        TestMapper original = mapper;
        TestMapper toMerge = fromMapping(mapping);
        e = expectThrows(
            IllegalArgumentException.class,
            () -> original.merge(toMerge, MapperMergeContext.root(false, false, MAPPING_UPDATE, Long.MAX_VALUE))
        );
        assertEquals(
            "Mapper for [field] conflicts with existing mapper:\n" + "\tCannot update parameter [analyzer] from [default] to [_standard]",
            e.getMessage()
        );
    }

    public void testLinkedAnalyzers() throws IOException {
        String mapping = """
            {"type":"test_mapper","analyzer":"_standard","required":"value"}""";
        TestMapper mapper = fromMapping(mapping);
        assertEquals("_standard", mapper.analyzer.name());
        assertEquals("_standard", mapper.searchAnalyzer.name());
        assertEquals("{\"field\":" + mapping + "}", Strings.toString(mapper));

        String mappingWithSA = """
            {"type":"test_mapper","search_analyzer":"_standard","required":"value"}""";
        mapper = fromMapping(mappingWithSA);
        assertEquals("_keyword", mapper.analyzer.name());
        assertEquals("_standard", mapper.searchAnalyzer.name());
        assertEquals("{\"field\":" + mappingWithSA + "}", Strings.toString(mapper));

        String mappingWithBoth = """
            {"type":"test_mapper","analyzer":"default","search_analyzer":"_standard","required":"value"}""";
        mapper = fromMapping(mappingWithBoth);
        assertEquals("default", mapper.analyzer.name());
        assertEquals("_standard", mapper.searchAnalyzer.name());
        assertEquals("{\"field\":" + mappingWithBoth + "}", Strings.toString(mapper));

        // we've configured things so that search_analyzer is only output when different from
        // analyzer, no matter what the value of `include_defaults` is
        String mappingWithSame = """
            {"type":"test_mapper","analyzer":"default","search_analyzer":"default","required":"value"}""";
        mapper = fromMapping(mappingWithSame);
        assertEquals("default", mapper.analyzer.name());
        assertEquals("default", mapper.searchAnalyzer.name());
        assertEquals("""
            {"field":{"type":"test_mapper","analyzer":"default","required":"value"}}""", Strings.toString(mapper));

        assertEquals(XContentHelper.stripWhitespace("""
            {
              "field": {
                "type": "test_mapper",
                "fixed": true,
                "fixed2": false,
                "variable": "default",
                "index": true,
                "wrapper": "default",
                "int_value": 5,
                "analyzer": "default",
                "required": "value",
                "enum_field": "name1",
                "restricted_enum_field": "name1"
              }
            }"""), toStringWithDefaults(mapper));
    }

    public void testRequiredField() {
        {
            String mapping = "{\"type\":\"test_mapper\"}";
            IllegalArgumentException iae = expectThrows(IllegalArgumentException.class, () -> fromMapping(mapping));
            assertEquals("field [required] must be specified", iae.getMessage());
        }
        {
            String mapping = """
                {"type":"test_mapper","required":null}""";
            MapperParsingException exc = expectThrows(MapperParsingException.class, () -> fromMapping(mapping));
            assertEquals("[required] on mapper [field] of type [test_mapper] must not have a [null] value", exc.getMessage());
        }
    }

    public void testEnumField() {
        {
            String mapping = """
                {"type":"test_mapper","required":"a","enum_field":"baz"}""";
            MapperParsingException e = expectThrows(MapperParsingException.class, () -> fromMapping(mapping));
            assertEquals("Unknown value [baz] for field [enum_field] - accepted values are [name1, name2, name3]", e.getMessage());
        }
        {
            String mapping = """
                {"type":"test_mapper","required":"a","enum_field":"name3"}""";
            TestMapper mapper = fromMapping(mapping);
            assertEquals(DummyEnumType.NAME3, mapper.enumField);
        }
        {
            String mapping = """
                {"type":"test_mapper","required":"a"}""";
            TestMapper mapper = fromMapping(mapping);
            assertEquals(DummyEnumType.NAME1, mapper.enumField);
        }
    }

    public void testRestrictedEnumField() {
        {
            String mapping = """
                {"type":"test_mapper","required":"a","restricted_enum_field":"baz"}""";
            MapperParsingException e = expectThrows(MapperParsingException.class, () -> fromMapping(mapping));
            assertEquals("Unknown value [baz] for field [restricted_enum_field] - accepted values are [name1, name2]", e.getMessage());
        }
        {
            String mapping = """
                {"type":"test_mapper","required":"a","restricted_enum_field":"name3"}""";
            MapperParsingException e = expectThrows(MapperParsingException.class, () -> fromMapping(mapping));
            assertEquals("Unknown value [name3] for field [restricted_enum_field] - accepted values are [name1, name2]", e.getMessage());
        }
        {
            String mapping = """
                {"type":"test_mapper","required":"a","restricted_enum_field":"name2"}""";
            TestMapper mapper = fromMapping(mapping);
            assertEquals(DummyEnumType.NAME2, mapper.restrictedEnumField);
        }
        {
            String mapping = """
                {"type":"test_mapper","required":"a"}""";
            TestMapper mapper = fromMapping(mapping);
            assertEquals(DummyEnumType.NAME1, mapper.restrictedEnumField);
        }
    }

    public void testCustomMergeValidation() throws IOException {
        MapperService mapperService = createMapperService(fieldMapping(b -> {
            b.field("type", "test_mapper");
            b.field("required", "a");
            b.field("int_value", 10);
        }));

        IllegalArgumentException e = expectThrows(IllegalArgumentException.class, () -> merge(mapperService, fieldMapping(b -> {
            b.field("type", "test_mapper");
            b.field("required", "a");
            b.field("int_value", 5);    // custom merge validator says that int_value can only increase
        })));
        assertThat(e.getMessage(), containsString("int_value"));
    }
}
