diff --git a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeatureFieldMapper.java b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeatureFieldMapper.java index 9d34562f35733..75a97d45f5d2a 100644 --- a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeatureFieldMapper.java +++ b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeatureFieldMapper.java @@ -152,11 +152,7 @@ protected void parseCreateField(ParseContext context) throws IOException { float value; if (context.externalValueSet()) { Object v = context.externalValue(); - if (v instanceof Number) { - value = ((Number) v).floatValue(); - } else { - value = Float.parseFloat(v.toString()); - } + value = objectToFloat(v); } else if (context.parser().currentToken() == Token.VALUE_NULL) { // skip return; @@ -176,6 +172,19 @@ protected void parseCreateField(ParseContext context) throws IOException { context.doc().addWithKey(name(), new FeatureField("_feature", name(), value)); } + private Float objectToFloat(Object value) { + if (value instanceof Number) { + return ((Number) value).floatValue(); + } else { + return Float.parseFloat(value.toString()); + } + } + + @Override + protected Float parseSourceValue(Object value) { + return objectToFloat(value); + } + @Override protected String contentType() { return CONTENT_TYPE; diff --git a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeaturesFieldMapper.java b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeaturesFieldMapper.java index 4968bf5ec6b0e..760d6dbc6dfcf 100644 --- a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeaturesFieldMapper.java +++ b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/RankFeaturesFieldMapper.java @@ -159,6 +159,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new AssertionError("parse is implemented directly"); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override protected boolean indexedByDefault() { return false; diff --git a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapper.java b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapper.java index cc11b3e736add..b9b8e868dd01b 100644 --- a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapper.java +++ b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapper.java @@ -473,6 +473,13 @@ private static double objectToDouble(Object value) { return doubleValue; } + @Override + protected Double parseSourceValue(Object value) { + double doubleValue = objectToDouble(value); + double scalingFactor = fieldType().getScalingFactor(); + return Math.round(doubleValue * scalingFactor) / scalingFactor; + } + private static class ScaledFloatIndexFieldData extends IndexNumericFieldData { private final IndexNumericFieldData scaledFieldData; diff --git a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/SearchAsYouTypeFieldMapper.java b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/SearchAsYouTypeFieldMapper.java index 4ef205a921ecf..7b26e229db2cc 100644 --- a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/SearchAsYouTypeFieldMapper.java +++ b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/SearchAsYouTypeFieldMapper.java @@ -418,6 +418,11 @@ protected void parseCreateField(ParseContext context) { throw new UnsupportedOperationException(); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { @@ -459,6 +464,11 @@ protected void mergeOptions(FieldMapper other, List conflicts) { } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected String contentType() { return "shingle"; @@ -577,6 +587,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + @Override protected String contentType() { return CONTENT_TYPE; diff --git a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/TokenCountFieldMapper.java b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/TokenCountFieldMapper.java index fb6a41a0f7608..51da52e37e22f 100644 --- a/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/TokenCountFieldMapper.java +++ b/modules/mapper-extras/src/main/java/org/elasticsearch/index/mapper/TokenCountFieldMapper.java @@ -158,6 +158,11 @@ protected void parseCreateField(ParseContext context) throws IOException { context.doc().addAll(NumberFieldMapper.NumberType.INTEGER.createFields(fieldType().name(), tokenCount, indexed, docValued, stored)); } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + /** * Count position increments in a token stream. Package private for testing. * @param analyzer analyzer to create token stream diff --git a/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/RankFeatureFieldMapperTests.java b/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/RankFeatureFieldMapperTests.java index 5fa95e6e15f2f..e867370acc8ed 100644 --- a/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/RankFeatureFieldMapperTests.java +++ b/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/RankFeatureFieldMapperTests.java @@ -23,9 +23,12 @@ import org.apache.lucene.analysis.tokenattributes.TermFrequencyAttribute; import org.apache.lucene.document.FeatureField; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.index.IndexService; @@ -186,4 +189,12 @@ public void testRejectMultiValuedFields() throws MapperParsingException, IOExcep e.getCause().getMessage()); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + RankFeatureFieldMapper mapper = new RankFeatureFieldMapper.Builder("field").build(context); + + assertEquals(3.14f, mapper.parseSourceValue(3.14), 0.0001); + assertEquals(42.9f, mapper.parseSourceValue("42.9"), 0.0001); + } } diff --git a/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapperTests.java b/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapperTests.java index 7fa4a81ec5c13..f573c25ba176d 100644 --- a/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapperTests.java +++ b/modules/mapper-extras/src/test/java/org/elasticsearch/index/mapper/ScaledFloatFieldMapperTests.java @@ -21,9 +21,12 @@ import org.apache.lucene.index.DocValuesType; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.index.IndexService; @@ -398,4 +401,15 @@ public void testMeta() throws Exception { new CompressedXContent(mapping3), MergeReason.MAPPING_UPDATE); assertEquals(mapping3, mapper.mappingSource().toString()); } + + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + ScaledFloatFieldMapper mapper = new ScaledFloatFieldMapper.Builder("field") + .scalingFactor(100) + .build(context); + + assertEquals(3.14, mapper.parseSourceValue(3.1415926), 0.00001); + assertEquals(3.14, mapper.parseSourceValue("3.1415"), 0.00001); + } } diff --git a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/MetaJoinFieldMapper.java b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/MetaJoinFieldMapper.java index ed9fe5b76689c..eb479a3d56705 100644 --- a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/MetaJoinFieldMapper.java +++ b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/MetaJoinFieldMapper.java @@ -135,6 +135,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new IllegalStateException("Should never be called"); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException("The " + typeName() + " field is not stored in _source."); + } + @Override protected String contentType() { return CONTENT_TYPE; diff --git a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentIdFieldMapper.java b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentIdFieldMapper.java index 88a7c455c7bc6..5d1914e5e1771 100644 --- a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentIdFieldMapper.java +++ b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentIdFieldMapper.java @@ -185,6 +185,11 @@ protected void parseCreateField(ParseContext context) throws IOException { context.doc().add(new SortedDocValuesField(fieldType().name(), binaryValue)); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException("The " + typeName() + " field is not stored in _source."); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { ParentIdFieldMapper parentMergeWith = (ParentIdFieldMapper) other; diff --git a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentJoinFieldMapper.java b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentJoinFieldMapper.java index 7fbdee9d23ba2..b4ba5f00c2e85 100644 --- a/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentJoinFieldMapper.java +++ b/modules/parent-join/src/main/java/org/elasticsearch/join/mapper/ParentJoinFieldMapper.java @@ -347,6 +347,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new UnsupportedOperationException("parsing is implemented in parse(), this method should NEVER be called"); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override public void parse(ParseContext context) throws IOException { context.path().add(simpleName()); diff --git a/modules/percolator/src/main/java/org/elasticsearch/percolator/PercolatorFieldMapper.java b/modules/percolator/src/main/java/org/elasticsearch/percolator/PercolatorFieldMapper.java index 0df62e260c1f1..e9ed2bbbebb58 100644 --- a/modules/percolator/src/main/java/org/elasticsearch/percolator/PercolatorFieldMapper.java +++ b/modules/percolator/src/main/java/org/elasticsearch/percolator/PercolatorFieldMapper.java @@ -367,6 +367,11 @@ public void parse(ParseContext context) throws IOException { processQuery(query, context); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + static void createQueryBuilderField(Version indexVersion, BinaryFieldMapper qbField, QueryBuilder queryBuilder, ParseContext context) throws IOException { try (ByteArrayOutputStream stream = new ByteArrayOutputStream()) { diff --git a/plugins/analysis-icu/src/main/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapper.java b/plugins/analysis-icu/src/main/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapper.java index 108cbc719e08a..fd7c23f81d001 100644 --- a/plugins/analysis-icu/src/main/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapper.java +++ b/plugins/analysis-icu/src/main/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapper.java @@ -731,4 +731,9 @@ protected void parseCreateField(ParseContext context) throws IOException { createFieldNamesField(context); } } + + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } } diff --git a/plugins/analysis-icu/src/test/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapperTests.java b/plugins/analysis-icu/src/test/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapperTests.java index 21a9cce88fe54..28cd339e36bcd 100644 --- a/plugins/analysis-icu/src/test/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapperTests.java +++ b/plugins/analysis-icu/src/test/java/org/elasticsearch/index/mapper/ICUCollationKeywordFieldMapperTests.java @@ -26,9 +26,12 @@ import org.apache.lucene.index.IndexableField; import org.apache.lucene.index.IndexableFieldType; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.index.IndexService; @@ -483,4 +486,13 @@ public void testUpdateIgnoreAbove() throws IOException { indexService.mapperService().merge("type", new CompressedXContent(mapping), MergeReason.MAPPING_UPDATE); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + ICUCollationKeywordFieldMapper mapper = new ICUCollationKeywordFieldMapper.Builder("field").build(context); + + assertEquals("value", mapper.parseSourceValue("value")); + assertEquals("42", mapper.parseSourceValue(42L)); + assertEquals("true", mapper.parseSourceValue(true)); + } } diff --git a/plugins/mapper-annotated-text/src/main/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapper.java b/plugins/mapper-annotated-text/src/main/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapper.java index 94fd537f44846..5cde4e8ab264e 100644 --- a/plugins/mapper-annotated-text/src/main/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapper.java +++ b/plugins/mapper-annotated-text/src/main/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapper.java @@ -583,6 +583,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + @Override protected String contentType() { return CONTENT_TYPE; diff --git a/plugins/mapper-annotated-text/src/test/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapperTests.java b/plugins/mapper-annotated-text/src/test/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapperTests.java index 5acc8c9a82280..a365aa24cf336 100644 --- a/plugins/mapper-annotated-text/src/test/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapperTests.java +++ b/plugins/mapper-annotated-text/src/test/java/org/elasticsearch/index/mapper/annotatedtext/AnnotatedTextFieldMapperTests.java @@ -28,10 +28,12 @@ import org.apache.lucene.index.Terms; import org.apache.lucene.index.TermsEnum; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; import org.elasticsearch.action.bulk.BulkRequestBuilder; import org.elasticsearch.action.index.IndexRequest; import org.elasticsearch.action.termvectors.TermVectorsRequest; import org.elasticsearch.action.termvectors.TermVectorsResponse; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; @@ -44,8 +46,11 @@ import org.elasticsearch.index.IndexService; import org.elasticsearch.index.VersionType; import org.elasticsearch.index.engine.Engine; +import org.elasticsearch.index.mapper.ContentPath; import org.elasticsearch.index.mapper.DocumentMapper; import org.elasticsearch.index.mapper.DocumentMapperParser; +import org.elasticsearch.index.mapper.FieldMapper; +import org.elasticsearch.index.mapper.Mapper; import org.elasticsearch.index.mapper.MapperParsingException; import org.elasticsearch.index.mapper.MapperService.MergeReason; import org.elasticsearch.index.mapper.ParsedDocument; @@ -672,4 +677,19 @@ public void testEmptyName() throws IOException { assertThat(e.getMessage(), containsString("name cannot be empty string")); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + FieldMapper fieldMapper = new AnnotatedTextFieldMapper.Builder("field") + .indexAnalyzer(indexService.getIndexAnalyzers().getDefaultIndexAnalyzer()) + .searchAnalyzer(indexService.getIndexAnalyzers().getDefaultSearchAnalyzer()) + .searchQuoteAnalyzer(indexService.getIndexAnalyzers().getDefaultSearchQuoteAnalyzer()) + .build(context); + AnnotatedTextFieldMapper mapper = (AnnotatedTextFieldMapper) fieldMapper; + + assertEquals("value", mapper.parseSourceValue("value")); + assertEquals("42", mapper.parseSourceValue(42L)); + assertEquals("true", mapper.parseSourceValue(true)); + } } diff --git a/plugins/mapper-murmur3/src/main/java/org/elasticsearch/index/mapper/murmur3/Murmur3FieldMapper.java b/plugins/mapper-murmur3/src/main/java/org/elasticsearch/index/mapper/murmur3/Murmur3FieldMapper.java index 4be96ad4be03a..65a7c5051f01a 100644 --- a/plugins/mapper-murmur3/src/main/java/org/elasticsearch/index/mapper/murmur3/Murmur3FieldMapper.java +++ b/plugins/mapper-murmur3/src/main/java/org/elasticsearch/index/mapper/murmur3/Murmur3FieldMapper.java @@ -147,6 +147,11 @@ protected void parseCreateField(ParseContext context) } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + @Override protected boolean indexedByDefault() { return false; @@ -156,5 +161,4 @@ protected boolean indexedByDefault() { protected void mergeOptions(FieldMapper other, List conflicts) { } - } diff --git a/rest-api-spec/src/main/resources/rest-api-spec/test/search/330_fetch_fields.yml b/rest-api-spec/src/main/resources/rest-api-spec/test/search/330_fetch_fields.yml index f716e42f9e7eb..5653ae61a6c7d 100644 --- a/rest-api-spec/src/main/resources/rest-api-spec/test/search/330_fetch_fields.yml +++ b/rest-api-spec/src/main/resources/rest-api-spec/test/search/330_fetch_fields.yml @@ -21,7 +21,7 @@ setup: index: test id: 1 body: - keyword: [ "first", "second" ] + keyword: [ "x", "y" ] integer_range: gte: 0 lte: 42 @@ -39,8 +39,8 @@ setup: - is_true: hits.hits.0._id - is_true: hits.hits.0._source - - match: { hits.hits.0.fields.keyword.0: first } - - match: { hits.hits.0.fields.keyword.1: second } + - match: { hits.hits.0.fields.keyword.0: x } + - match: { hits.hits.0.fields.keyword.1: y } - match: { hits.hits.0.fields.integer_range.0.gte: 0 } - match: { hits.hits.0.fields.integer_range.0.lte: 42 } @@ -65,7 +65,7 @@ setup: index: test id: 1 body: - keyword: [ "value" ] + keyword: [ "x" ] - do: catch: bad_request @@ -76,3 +76,49 @@ setup: - match: { error.root_cause.0.type: "illegal_argument_exception" } - match: { error.root_cause.0.reason: "Unable to retrieve the requested [fields] since _source is disabled in the mappings for index [test]" } + +--- +"Test ignore malformed": + - do: + indices.create: + index: test + body: + settings: + number_of_shards: 1 + mappings: + properties: + keyword: + type: keyword + integer: + type: integer + ignore_malformed: true + + - do: + index: + index: test + id: 1 + body: + keyword: "x" + integer: 42 + + - do: + index: + index: test + id: 2 + body: + keyword: "y" + integer: "not an integer" + + - do: + indices.refresh: + index: [ test ] + + - do: + search: + index: test + body: + sort: [ keyword ] + fields: [ integer ] + + - match: { hits.hits.0.fields.integer.0: 42 } + - is_false: hits.hits.1.fields.integer diff --git a/server/src/main/java/org/elasticsearch/common/network/InetAddresses.java b/server/src/main/java/org/elasticsearch/common/network/InetAddresses.java index 21c5b4e04b25b..444c291f36a83 100644 --- a/server/src/main/java/org/elasticsearch/common/network/InetAddresses.java +++ b/server/src/main/java/org/elasticsearch/common/network/InetAddresses.java @@ -384,4 +384,17 @@ public static Tuple parseCidr(String maskedAddress) { throw new IllegalArgumentException("Expected [ip/prefix] but was [" + maskedAddress + "]"); } } + + /** + * Given an address and prefix length, returns the string representation of the range in CIDR notation. + * + * See {@link #toAddrString} for details on how the address is represented. + */ + public static String toCidrString(InetAddress address, int prefixLength) { + return new StringBuilder() + .append(toAddrString(address)) + .append("/") + .append(prefixLength) + .toString(); + } } diff --git a/server/src/main/java/org/elasticsearch/index/mapper/AbstractGeometryFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/AbstractGeometryFieldMapper.java index 0dbfc5070706d..0f23118bcdfea 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/AbstractGeometryFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/AbstractGeometryFieldMapper.java @@ -141,6 +141,11 @@ public Builder ignoreZValue(final boolean ignoreZValue) { } } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + public abstract static class TypeParser implements Mapper.TypeParser { protected abstract T newBuilder(String name, Map params); diff --git a/server/src/main/java/org/elasticsearch/index/mapper/BinaryFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/BinaryFieldMapper.java index 0a61d12cc24dd..bee261181b93f 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/BinaryFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/BinaryFieldMapper.java @@ -195,7 +195,11 @@ protected void parseCreateField(ParseContext context) throws IOException { // no doc values createFieldNamesField(context); } + } + @Override + protected Object parseSourceValue(Object value) { + return value; } @Override diff --git a/server/src/main/java/org/elasticsearch/index/mapper/BooleanFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/BooleanFieldMapper.java index f5d33845ef05c..cd6e1c85089b4 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/BooleanFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/BooleanFieldMapper.java @@ -30,6 +30,7 @@ import org.apache.lucene.search.TermQuery; import org.apache.lucene.search.TermRangeQuery; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.common.Booleans; import org.elasticsearch.common.Nullable; import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.support.XContentMapValues; @@ -256,6 +257,16 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + public Boolean parseSourceValue(Object value) { + if (value instanceof Boolean) { + return (Boolean) value; + } else { + String textValue = value.toString(); + return Booleans.parseBoolean(textValue.toCharArray(), 0, textValue.length(), false); + } + } + @Override public ParametrizedFieldMapper.Builder getMergeBuilder() { return new Builder(simpleName()).init(this); diff --git a/server/src/main/java/org/elasticsearch/index/mapper/CompletionFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/CompletionFieldMapper.java index 047fab17fc1fa..c23bc0a30bba4 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/CompletionFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/CompletionFieldMapper.java @@ -522,6 +522,15 @@ private void parse(ParseContext parseContext, Token token, } } + @Override + protected List parseSourceValue(Object value) { + if (value instanceof List) { + return (List) value; + } else { + return List.of(value); + } + } + static class CompletionInputMetadata { public final String input; public final Map> contexts; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/DateFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/DateFieldMapper.java index 5d80d18d7df61..2bd85094fd80d 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/DateFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/DateFieldMapper.java @@ -58,6 +58,7 @@ import java.time.Instant; import java.time.ZoneId; import java.time.ZoneOffset; +import java.time.ZonedDateTime; import java.util.Collections; import java.util.Iterator; import java.util.List; @@ -329,6 +330,7 @@ protected DateMathParser dateMathParser() { return dateMathParser; } + // Visible for testing. public long parse(String value) { return resolution.convert(DateFormatters.from(dateTimeFormatter().parse(value), dateTimeFormatter().locale()).toInstant()); } @@ -582,6 +584,15 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + public String parseSourceValue(Object value) { + String date = value.toString(); + long timestamp = fieldType().parse(date); + + ZonedDateTime dateTime = fieldType().resolution().toInstant(timestamp).atZone(ZoneOffset.UTC); + return fieldType().dateTimeFormatter().format(dateTime); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { final DateFieldMapper d = (DateFieldMapper) other; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/FieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/FieldMapper.java index 30f38ce16d8a3..80d76e954e9be 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/FieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/FieldMapper.java @@ -33,6 +33,7 @@ import org.elasticsearch.common.xcontent.support.AbstractXContentParser; import org.elasticsearch.index.analysis.NamedAnalyzer; import org.elasticsearch.index.mapper.FieldNamesFieldMapper.FieldNamesFieldType; +import org.elasticsearch.search.lookup.SourceLookup; import java.io.IOException; import java.util.ArrayList; @@ -269,6 +270,47 @@ public void parse(ParseContext context) throws IOException { */ protected abstract void parseCreateField(ParseContext context) throws IOException; + /** + * Given access to a document's _source, return this field's values. + * + * In addition to pulling out the values, mappers can parse them into a standard form. This + * method delegates parsing to {@link #parseSourceValue} for parsing. Most mappers will choose + * to override {@link #parseSourceValue} -- for example numeric field mappers make sure to + * parse the source value into a number of the right type. + * + * Some mappers may need more flexibility and can override this entire method instead. + * + * @param lookup a lookup structure over the document's source. + * @return a list a standardized field values. + */ + public List lookupValues(SourceLookup lookup) { + Object sourceValue = lookup.extractValue(name()); + if (sourceValue == null) { + return List.of(); + } + + List values = new ArrayList<>(); + if (parsesArrayValue()) { + return (List) parseSourceValue(sourceValue); + } else { + List sourceValues = sourceValue instanceof List ? (List) sourceValue : List.of(sourceValue); + for (Object value : sourceValues) { + Object parsedValue = parseSourceValue(value); + values.add(parsedValue); + } + } + return values; + } + + /** + * Given a value that has been extracted from a document's source, parse it into a standard + * format. This parsing logic should closely mirror the value parsing in + * {@link #parseCreateField} or {@link #parse}. + * + * Note that when overriding this method, {@link #lookupValues} should *not* be overridden. + */ + protected abstract Object parseSourceValue(Object value); + protected void createFieldNamesField(ParseContext context) { FieldNamesFieldType fieldNamesFieldType = context.docMapper().metadataMapper(FieldNamesFieldMapper.class).fieldType(); if (fieldNamesFieldType != null && fieldNamesFieldType.isEnabled()) { diff --git a/server/src/main/java/org/elasticsearch/index/mapper/IpFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/IpFieldMapper.java index ad18a1fa94b5e..b6709a9c88eff 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/IpFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/IpFieldMapper.java @@ -400,6 +400,12 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected String parseSourceValue(Object value) { + InetAddress address = InetAddresses.forString(value.toString()); + return InetAddresses.toAddrString(address); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { IpFieldMapper mergeWith = (IpFieldMapper) other; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/KeywordFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/KeywordFieldMapper.java index ee476631b2b86..8529e9b5b0cbf 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/KeywordFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/KeywordFieldMapper.java @@ -227,6 +227,11 @@ public Mapper.Builder parse(String name, Map node, ParserCont } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + public static final class KeywordFieldType extends StringFieldType { boolean hasNorms; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/MetadataFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/MetadataFieldMapper.java index 05b18099e7b1e..56c00ed8693f7 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/MetadataFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/MetadataFieldMapper.java @@ -86,6 +86,11 @@ public void postParse(ParseContext context) throws IOException { // do nothing } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException("The " + typeName() + " field is not stored in _source."); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { } diff --git a/server/src/main/java/org/elasticsearch/index/mapper/NumberFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/NumberFieldMapper.java index 106609f97cdeb..9ed666b0265a1 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/NumberFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/NumberFieldMapper.java @@ -1085,6 +1085,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected Number parseSourceValue(Object value) { + return fieldType().type.parse(value, coerce.value()); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { NumberFieldMapper m = (NumberFieldMapper) other; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/RangeFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/RangeFieldMapper.java index 28874363b456f..462ef975cc097 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/RangeFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/RangeFieldMapper.java @@ -52,6 +52,7 @@ import java.time.ZoneId; import java.time.ZoneOffset; import java.util.Collections; +import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.List; @@ -373,6 +374,26 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + @SuppressWarnings("unchecked") + protected Object parseSourceValue(Object value) { + RangeType rangeType = fieldType().rangeType(); + if (!(value instanceof Map)) { + assert rangeType == RangeType.IP; + Tuple ipRange = InetAddresses.parseCidr(value.toString()); + return InetAddresses.toCidrString(ipRange.v1(), ipRange.v2()); + } + + Map range = (Map) value; + Map parsedRange = new HashMap<>(); + for (Map.Entry entry : range.entrySet()) { + Object parsedValue = rangeType.parseValue(entry.getValue(), coerce.value(), fieldType().dateMathParser); + Object formattedValue = rangeType.formatValue(parsedValue, fieldType().dateTimeFormatter); + parsedRange.put(entry.getKey(), formattedValue); + } + return parsedRange; + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { RangeFieldMapper mergeWith = (RangeFieldMapper) other; diff --git a/server/src/main/java/org/elasticsearch/index/mapper/RangeType.java b/server/src/main/java/org/elasticsearch/index/mapper/RangeType.java index 86d79e3171573..f293d4ded6c61 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/RangeType.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/RangeType.java @@ -36,14 +36,17 @@ import org.elasticsearch.common.Nullable; import org.elasticsearch.common.geo.ShapeRelation; import org.elasticsearch.common.network.InetAddresses; +import org.elasticsearch.common.time.DateFormatter; import org.elasticsearch.common.time.DateMathParser; import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.index.query.QueryShardContext; import java.io.IOException; import java.net.InetAddress; +import java.time.Instant; import java.time.ZoneId; import java.time.ZoneOffset; +import java.time.ZonedDateTime; import java.util.ArrayList; import java.util.Arrays; import java.util.List; @@ -69,8 +72,9 @@ public InetAddress parseTo(RangeFieldMapper.RangeFieldType fieldType, XContentPa InetAddress address = InetAddresses.forString(parser.text()); return included ? address : nextDown(address); } + @Override - public InetAddress parse(Object value, boolean coerce) { + public InetAddress parseValue(Object value, boolean coerce, @Nullable DateMathParser dateMathParser) { if (value instanceof InetAddress) { return (InetAddress) value; } else { @@ -80,6 +84,12 @@ public InetAddress parse(Object value, boolean coerce) { return InetAddresses.forString(value.toString()); } } + + @Override + public Object formatValue(Object value, DateFormatter dateFormatter) { + return InetAddresses.toAddrString((InetAddress) value); + } + @Override public InetAddress minValue() { return InetAddressPoint.MIN_VALUE; @@ -170,22 +180,34 @@ private Query createQuery(String field, Object lower, Object upper, boolean incl public Field getRangeField(String name, RangeFieldMapper.Range r) { return new LongRange(name, new long[] {((Number)r.from).longValue()}, new long[] {((Number)r.to).longValue()}); } - private Number parse(DateMathParser dateMathParser, String dateStr) { - return dateMathParser.parse(dateStr, () -> {throw new IllegalArgumentException("now is not used at indexing time");}) - .toEpochMilli(); - } @Override public Number parseFrom(RangeFieldMapper.RangeFieldType fieldType, XContentParser parser, boolean coerce, boolean included) throws IOException { - Number value = parse(fieldType.dateMathParser, parser.text()); + Number value = parseValue(parser.text(), coerce, fieldType.dateMathParser); return included ? value : nextUp(value); } @Override public Number parseTo(RangeFieldMapper.RangeFieldType fieldType, XContentParser parser, boolean coerce, boolean included) throws IOException{ - Number value = parse(fieldType.dateMathParser, parser.text()); + Number value = parseValue(parser.text(), coerce, fieldType.dateMathParser); return included ? value : nextDown(value); } + + @Override + public Long parseValue(Object dateStr, boolean coerce, @Nullable DateMathParser dateMathParser) { + assert dateMathParser != null; + return dateMathParser.parse(dateStr.toString(), () -> { + throw new IllegalArgumentException("now is not used at indexing time"); + }).toEpochMilli(); + } + + @Override + public Object formatValue(Object value, DateFormatter dateFormatter) { + long timestamp = (long) value; + ZonedDateTime dateTime = Instant.ofEpochMilli(timestamp).atZone(ZoneOffset.UTC); + return dateFormatter.format(dateTime); + } + @Override public Long minValue() { return Long.MIN_VALUE; @@ -243,6 +265,7 @@ public Query rangeQuery(String field, boolean hasDocValues, Object lowerTerm, Ob return createRangeQuery(field, hasDocValues, low, high, includeLower, includeUpper, relation); } + @Override public Query withinQuery(String field, Object from, Object to, boolean includeLower, boolean includeUpper) { return LONG.withinQuery(field, from, to, includeLower, includeUpper); @@ -598,6 +621,15 @@ public List createFields(ParseContext context, String name, Rang } return fields; } + + public Object parseValue(Object value, boolean coerce, @Nullable DateMathParser dateMathParser) { + return numberType.parse(value, coerce); + } + + public Object formatValue(Object value, DateFormatter formatter) { + return value; + } + /** parses from value. rounds according to included flag */ public Object parseFrom(RangeFieldMapper.RangeFieldType fieldType, XContentParser parser, boolean coerce, boolean included) throws IOException { @@ -618,15 +650,12 @@ public Object parseTo(RangeFieldMapper.RangeFieldType fieldType, XContentParser public abstract Query withinQuery(String field, Object from, Object to, boolean includeFrom, boolean includeTo); public abstract Query containsQuery(String field, Object from, Object to, boolean includeFrom, boolean includeTo); public abstract Query intersectsQuery(String field, Object from, Object to, boolean includeFrom, boolean includeTo); - public Object parse(Object value, boolean coerce) { - return numberType.parse(value, coerce); - } public Query rangeQuery(String field, boolean hasDocValues, Object from, Object to, boolean includeFrom, boolean includeTo, ShapeRelation relation, @Nullable ZoneId timeZone, @Nullable DateMathParser dateMathParser, QueryShardContext context) { - Object lower = from == null ? minValue() : parse(from, false); - Object upper = to == null ? maxValue() : parse(to, false); + Object lower = from == null ? minValue() : parseValue(from, false, dateMathParser); + Object upper = to == null ? maxValue() : parseValue(to, false, dateMathParser); return createRangeQuery(field, hasDocValues, lower, upper, includeFrom, includeTo, relation); } diff --git a/server/src/main/java/org/elasticsearch/index/mapper/TextFieldMapper.java b/server/src/main/java/org/elasticsearch/index/mapper/TextFieldMapper.java index 503b2f98865a2..2c82aa720f96d 100644 --- a/server/src/main/java/org/elasticsearch/index/mapper/TextFieldMapper.java +++ b/server/src/main/java/org/elasticsearch/index/mapper/TextFieldMapper.java @@ -496,6 +496,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new UnsupportedOperationException(); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { @@ -522,6 +527,11 @@ protected void parseCreateField(ParseContext context) { throw new UnsupportedOperationException(); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { @@ -823,6 +833,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + @Override public Iterator iterator() { List subIterators = new ArrayList<>(); diff --git a/server/src/main/java/org/elasticsearch/search/fetch/subphase/FetchFieldsPhase.java b/server/src/main/java/org/elasticsearch/search/fetch/subphase/FetchFieldsPhase.java index e6d5558b36199..50513dee9c898 100644 --- a/server/src/main/java/org/elasticsearch/search/fetch/subphase/FetchFieldsPhase.java +++ b/server/src/main/java/org/elasticsearch/search/fetch/subphase/FetchFieldsPhase.java @@ -23,12 +23,15 @@ import org.apache.lucene.index.ReaderUtil; import org.elasticsearch.common.document.DocumentField; import org.elasticsearch.index.mapper.DocumentMapper; +import org.elasticsearch.index.mapper.IgnoredFieldMapper; import org.elasticsearch.search.SearchHit; import org.elasticsearch.search.fetch.FetchSubPhase; import org.elasticsearch.search.internal.SearchContext; import org.elasticsearch.search.lookup.SourceLookup; +import java.util.HashSet; import java.util.Map; +import java.util.Set; /** * A fetch sub-phase for high-level field retrieval. Given a list of fields, it @@ -59,10 +62,24 @@ public void hitsExecute(SearchContext context, SearchHit[] hits) { LeafReaderContext readerContext = context.searcher().getIndexReader().leaves().get(readerIndex); sourceLookup.setSegmentAndDocument(readerContext, hit.docId()); - Map documentFields = fieldValueRetriever.retrieve(sourceLookup); + Set ignoredFields = getIgnoredFields(hit); + Map documentFields = fieldValueRetriever.retrieve(sourceLookup, ignoredFields); for (Map.Entry entry : documentFields.entrySet()) { hit.setDocumentField(entry.getKey(), entry.getValue()); } } } + + private Set getIgnoredFields(SearchHit hit) { + DocumentField field = hit.field(IgnoredFieldMapper.NAME); + if (field == null) { + return Set.of(); + } + + Set ignoredFields = new HashSet<>(); + for (Object value : field.getValues()) { + ignoredFields.add((String) value); + } + return ignoredFields; + } } diff --git a/server/src/main/java/org/elasticsearch/search/fetch/subphase/FieldValueRetriever.java b/server/src/main/java/org/elasticsearch/search/fetch/subphase/FieldValueRetriever.java index e9d1b88a9199f..bf879c85f043a 100644 --- a/server/src/main/java/org/elasticsearch/search/fetch/subphase/FieldValueRetriever.java +++ b/server/src/main/java/org/elasticsearch/search/fetch/subphase/FieldValueRetriever.java @@ -20,15 +20,14 @@ package org.elasticsearch.search.fetch.subphase; import org.elasticsearch.common.document.DocumentField; -import org.elasticsearch.common.xcontent.support.XContentMapValues; -import org.elasticsearch.index.mapper.MappedFieldType; +import org.elasticsearch.index.mapper.DocumentFieldMappers; +import org.elasticsearch.index.mapper.FieldMapper; import org.elasticsearch.index.mapper.MapperService; import org.elasticsearch.search.lookup.SourceLookup; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; -import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; @@ -38,80 +37,60 @@ * Then given a specific document, it can retrieve the corresponding fields from the document's source. */ public class FieldValueRetriever { + private final DocumentFieldMappers fieldMappers; private final List fieldContexts; - private final Set sourcePaths; public static FieldValueRetriever create(MapperService mapperService, Collection fieldPatterns) { + DocumentFieldMappers fieldMappers = mapperService.documentMapper().mappers(); List fields = new ArrayList<>(); - Set sourcePaths = new HashSet<>(); for (String fieldPattern : fieldPatterns) { Collection concreteFields = mapperService.simpleMatchToFullName(fieldPattern); for (String field : concreteFields) { - MappedFieldType fieldType = mapperService.fieldType(field); - - if (fieldType != null) { + if (fieldMappers.getMapper(field) != null) { Set sourcePath = mapperService.sourcePath(field); fields.add(new FieldContext(field, sourcePath)); - sourcePaths.addAll(sourcePath); } } } - return new FieldValueRetriever(fields, sourcePaths); + return new FieldValueRetriever(fieldMappers, fields); } - private FieldValueRetriever(List fieldContexts, Set sourcePaths) { + private FieldValueRetriever(DocumentFieldMappers fieldMappers, + List fieldContexts) { + this.fieldMappers = fieldMappers; this.fieldContexts = fieldContexts; - this.sourcePaths = sourcePaths; } - @SuppressWarnings("unchecked") - public Map retrieve(SourceLookup sourceLookup) { - Map result = new HashMap<>(); - Map sourceValues = extractValues(sourceLookup, sourcePaths); - + public Map retrieve(SourceLookup sourceLookup, Set ignoredFields) { + Map documentFields = new HashMap<>(); for (FieldContext fieldContext : fieldContexts) { String field = fieldContext.fieldName; Set sourcePath = fieldContext.sourcePath; - List values = new ArrayList<>(); - for (String path : sourcePath) { - Object value = sourceValues.get(path); - if (value != null) { - if (value instanceof List) { - values.addAll((List) value); - } else { - values.add(value); - } - } + if (ignoredFields.contains(field)) { + continue; } - result.put(field, new DocumentField(field, values)); - } - return result; - } - /** - * For each of the provided paths, return its value in the source. Note that in contrast with - * {@link SourceLookup#extractRawValues}, array and object values can be returned. - */ - private static Map extractValues(SourceLookup sourceLookup, Set paths) { - Map result = new HashMap<>(paths.size()); - for (String path : paths) { - Object value = XContentMapValues.extractValue(path, sourceLookup); - if (value != null) { - result.put(path, value); + List parsedValues = new ArrayList<>(); + for (String path : sourcePath) { + FieldMapper fieldMapper = (FieldMapper) fieldMappers.getMapper(path); + List values = fieldMapper.lookupValues(sourceLookup); + parsedValues.addAll(values); } + documentFields.put(field, new DocumentField(field, parsedValues)); } - return result; + return documentFields; } private static class FieldContext { final String fieldName; final Set sourcePath; - FieldContext(String fieldName, Set sourcePath) { + FieldContext(String fieldName, + Set sourcePath) { this.fieldName = fieldName; this.sourcePath = sourcePath; } diff --git a/server/src/main/java/org/elasticsearch/search/lookup/SourceLookup.java b/server/src/main/java/org/elasticsearch/search/lookup/SourceLookup.java index e547f02ef0347..6393d4bf50ff0 100644 --- a/server/src/main/java/org/elasticsearch/search/lookup/SourceLookup.java +++ b/server/src/main/java/org/elasticsearch/search/lookup/SourceLookup.java @@ -132,6 +132,14 @@ public List extractRawValues(String path) { return XContentMapValues.extractRawValues(path, loadSourceIfNeeded()); } + /** + * For the provided path, return its value in the source. Note that in contrast with + * {@link SourceLookup#extractRawValues}, array and object values can be returned. + */ + public Object extractValue(String path) { + return XContentMapValues.extractValue(path, loadSourceIfNeeded()); + } + public Object filter(FetchSourceContext context) { return context.getFilter().apply(loadSourceIfNeeded()); } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/BooleanFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/BooleanFieldMapperTests.java index 0d5ed6d24803a..8666ea0073442 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/BooleanFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/BooleanFieldMapperTests.java @@ -33,9 +33,12 @@ import org.apache.lucene.store.ByteBuffersDirectory; import org.apache.lucene.store.Directory; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; @@ -293,4 +296,13 @@ public void testBoosts() throws Exception { assertEquals(new BoostQuery(new TermQuery(new Term("field", "T")), 2.0f), ft.termQuery("true", null)); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + BooleanFieldMapper mapper = new BooleanFieldMapper.Builder("field").build(context); + + assertTrue(mapper.parseSourceValue(true)); + assertFalse(mapper.parseSourceValue("false")); + assertFalse(mapper.parseSourceValue("")); + } } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/CompletionFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/CompletionFieldMapperTests.java index 274c8efd0e722..259f6d8c2a1ae 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/CompletionFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/CompletionFieldMapperTests.java @@ -18,6 +18,7 @@ */ package org.elasticsearch.index.mapper; +import org.apache.lucene.analysis.standard.StandardAnalyzer; import org.apache.lucene.document.SortedSetDocValuesField; import org.apache.lucene.index.IndexableField; import org.apache.lucene.search.Query; @@ -31,6 +32,8 @@ import org.apache.lucene.util.CharsRefBuilder; import org.apache.lucene.util.automaton.Operations; import org.apache.lucene.util.automaton.RegExp; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; @@ -42,6 +45,7 @@ import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.common.xcontent.json.JsonXContent; import org.elasticsearch.index.IndexService; +import org.elasticsearch.index.analysis.AnalyzerScope; import org.elasticsearch.index.analysis.NamedAnalyzer; import org.elasticsearch.test.ESSingleNodeTestCase; import org.hamcrest.FeatureMatcher; @@ -50,6 +54,7 @@ import org.hamcrest.core.CombinableMatcher; import java.io.IOException; +import java.util.List; import java.util.Map; import java.util.function.Function; @@ -931,6 +936,21 @@ public void testLimitOfContextMappings() throws Throwable { CompletionFieldMapper.COMPLETION_CONTEXTS_LIMIT + "] has been exceeded")); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + NamedAnalyzer defaultAnalyzer = new NamedAnalyzer("standard", AnalyzerScope.INDEX, new StandardAnalyzer()); + CompletionFieldMapper mapper = new CompletionFieldMapper.Builder("completion", defaultAnalyzer).build(context); + + assertEquals(List.of("value"), mapper.parseSourceValue("value")); + + List list = List.of("first", "second"); + assertEquals(list, mapper.parseSourceValue(list)); + + Map object = Map.of("input", List.of("first", "second"), "weight", "2.718"); + assertEquals(List.of(object), mapper.parseSourceValue(object)); + } + private Matcher suggestField(String value) { return Matchers.allOf(hasProperty(IndexableField::stringValue, equalTo(value)), Matchers.instanceOf(SuggestField.class)); diff --git a/server/src/test/java/org/elasticsearch/index/mapper/DateFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/DateFieldMapperTests.java index 7677e55449afa..5b77e50459fa0 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/DateFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/DateFieldMapperTests.java @@ -21,10 +21,13 @@ import org.apache.lucene.index.DocValuesType; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; import org.elasticsearch.bootstrap.JavaVersion; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.time.DateFormatter; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentType; @@ -464,4 +467,40 @@ public void testMeta() throws Exception { assertEquals(mapping3, mapper.mappingSource().toString()); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + DateFieldMapper mapper = new DateFieldMapper.Builder("field").build(context); + String date = "2020-05-15T21:33:02.000Z"; + assertEquals(date, mapper.parseSourceValue(date)); + assertEquals(date, mapper.parseSourceValue(1589578382000L)); + + DateFieldMapper mapperWithFormat = new DateFieldMapper.Builder("field") + .format("yyyy/MM/dd||epoch_millis") + .build(context); + String dateInFormat = "1990/12/29"; + assertEquals(dateInFormat, mapperWithFormat.parseSourceValue(dateInFormat)); + assertEquals(dateInFormat, mapperWithFormat.parseSourceValue(662428800000L)); + + DateFieldMapper mapperWithMillis = new DateFieldMapper.Builder("field") + .format("epoch_millis") + .build(context); + String dateInMillis = "662428800000"; + assertEquals(dateInMillis, mapperWithMillis.parseSourceValue(dateInMillis)); + assertEquals(dateInMillis, mapperWithMillis.parseSourceValue(662428800000L)); + } + + public void testParseSourceValueNanos() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + DateFieldMapper mapper = new DateFieldMapper.Builder("field") + .format("strict_date_time||epoch_millis") + .withResolution(DateFieldMapper.Resolution.NANOSECONDS) + .build(context); + String date = "2020-05-15T21:33:02.123456789Z"; + assertEquals("2020-05-15T21:33:02.123456789Z", mapper.parseSourceValue(date)); + assertEquals("2020-05-15T21:33:02.123Z", mapper.parseSourceValue(1589578382123L)); + } } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/DocumentFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/DocumentFieldMapperTests.java index 38db532d9fb0b..5cfa4609d9931 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/DocumentFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/DocumentFieldMapperTests.java @@ -102,6 +102,11 @@ static class FakeFieldMapper extends FieldMapper { protected void parseCreateField(ParseContext context) throws IOException { } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { diff --git a/server/src/test/java/org/elasticsearch/index/mapper/ExternalMapper.java b/server/src/test/java/org/elasticsearch/index/mapper/ExternalMapper.java index d8b8b0d213a3b..23ae5a9b6ec9c 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/ExternalMapper.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/ExternalMapper.java @@ -200,6 +200,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new UnsupportedOperationException(); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override public Iterator iterator() { return Iterators.concat(super.iterator(), Arrays.asList(binMapper, boolMapper, pointMapper, shapeMapper, stringMapper).iterator()); diff --git a/server/src/test/java/org/elasticsearch/index/mapper/FakeStringFieldMapper.java b/server/src/test/java/org/elasticsearch/index/mapper/FakeStringFieldMapper.java index a86a4a6d63f77..09c2a899f13fc 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/FakeStringFieldMapper.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/FakeStringFieldMapper.java @@ -133,6 +133,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { diff --git a/server/src/test/java/org/elasticsearch/index/mapper/IpFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/IpFieldMapperTests.java index 07ed508a624a1..0706e382fe830 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/IpFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/IpFieldMapperTests.java @@ -26,10 +26,13 @@ import org.apache.lucene.search.Query; import org.apache.lucene.search.TermQuery; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; import org.elasticsearch.common.network.InetAddresses; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; @@ -297,6 +300,16 @@ public void testEmptyName() throws IOException { assertThat(e.getMessage(), containsString("name cannot be empty string")); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + IpFieldMapper mapper = new IpFieldMapper.Builder("field").build(context); + + assertEquals("2001:db8::2:1", mapper.parseSourceValue("2001:db8::2:1")); + assertEquals("2001:db8::2:1", mapper.parseSourceValue("2001:db8:0:0:0:0:2:1")); + assertEquals("::1", mapper.parseSourceValue("0:0:0:0:0:0:0:1")); + } + @Override protected IpFieldMapper.Builder newBuilder() { return new IpFieldMapper.Builder("ip"); diff --git a/server/src/test/java/org/elasticsearch/index/mapper/IpRangeFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/IpRangeFieldMapperTests.java index 2f861ca6fc79c..8e06c938c8b7f 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/IpRangeFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/IpRangeFieldMapperTests.java @@ -20,10 +20,13 @@ import org.apache.lucene.index.DocValuesType; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; import org.elasticsearch.common.network.InetAddresses; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentType; @@ -80,4 +83,14 @@ public void testStoreCidr() throws Exception { assertThat(storedField.stringValue(), containsString(strVal)); } } + + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + RangeFieldMapper mapper = new RangeFieldMapper.Builder("field", RangeType.IP).build(context); + Map range = Map.of("gte", "2001:db8:0:0:0:0:2:1"); + assertEquals(Map.of("gte", "2001:db8::2:1"), mapper.parseSourceValue(range)); + assertEquals("2001:db8::2:1/32", mapper.parseSourceValue("2001:db8:0:0:0:0:2:1/32")); + } } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/KeywordFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/KeywordFieldMapperTests.java index 53fe049b8d964..128a44953f7cd 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/KeywordFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/KeywordFieldMapperTests.java @@ -29,6 +29,8 @@ import org.apache.lucene.search.similarities.BM25Similarity; import org.apache.lucene.search.similarities.BooleanSimilarity; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; @@ -166,6 +168,9 @@ public void testDefaults() throws Exception { // used by TermVectorsService assertArrayEquals(new String[] { "1234" }, TermVectorsService.getValues(doc.rootDoc().getFields("field"))); + + FieldMapper fieldMapper = (FieldMapper) mapper.mappers().getMapper("field"); + assertEquals("1234", fieldMapper.parseSourceValue("1234")); } public void testIgnoreAbove() throws IOException { @@ -624,4 +629,14 @@ public void testMeta() throws Exception { new CompressedXContent(mapping3), MergeReason.MAPPING_UPDATE); assertEquals(mapping3, mapper.mappingSource().toString()); } + + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + KeywordFieldMapper mapper = new KeywordFieldMapper.Builder("field").build(context); + + assertEquals("value", mapper.parseSourceValue("value")); + assertEquals("42", mapper.parseSourceValue(42L)); + assertEquals("true", mapper.parseSourceValue(true)); + } } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/NumberFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/NumberFieldMapperTests.java index 2514f93131398..7eefee2142419 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/NumberFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/NumberFieldMapperTests.java @@ -22,11 +22,14 @@ import com.carrotsearch.randomizedtesting.annotations.Timeout; import org.apache.lucene.index.DocValuesType; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; import org.elasticsearch.action.index.IndexRequest; import org.elasticsearch.action.index.IndexResponse; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ToXContentObject; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; @@ -401,6 +404,15 @@ public void testEmptyName() throws IOException { } } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + NumberFieldMapper mapper = new NumberFieldMapper.Builder("field", NumberType.INTEGER).build(context); + + assertEquals(3, mapper.parseSourceValue(3.14)); + assertEquals(42, mapper.parseSourceValue("42.9")); + } + @Timeout(millis = 30000) public void testOutOfRangeValues() throws IOException { final List> inputs = Arrays.asList( diff --git a/server/src/test/java/org/elasticsearch/index/mapper/ParametrizedMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/ParametrizedMapperTests.java index f330474f2dd79..5bd73a89099e0 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/ParametrizedMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/ParametrizedMapperTests.java @@ -172,6 +172,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override protected String contentType() { return "test_mapper"; diff --git a/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldMapperTests.java index 88d2db04f2b73..0659983e1ed62 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldMapperTests.java @@ -22,10 +22,13 @@ import org.apache.lucene.document.InetAddressPoint; import org.apache.lucene.index.DocValuesType; import org.apache.lucene.index.IndexableField; +import org.elasticsearch.Version; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; import org.elasticsearch.common.network.InetAddresses; +import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; @@ -40,6 +43,7 @@ import java.util.Collection; import java.util.HashSet; import java.util.Locale; +import java.util.Map; import java.util.Set; import static org.elasticsearch.index.query.RangeQueryBuilder.GTE_FIELD; @@ -49,6 +53,7 @@ import static org.hamcrest.Matchers.anyOf; import static org.hamcrest.Matchers.containsString; + public class RangeFieldMapperTests extends AbstractNumericFieldMapperTestCase { @Override @@ -486,4 +491,19 @@ public void testIllegalFormatField() throws Exception { assertEquals("Invalid format: [[test_format]]: Unknown pattern letter: t", e.getMessage()); } + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + RangeFieldMapper longMapper = new RangeFieldMapper.Builder("field", RangeType.LONG).build(context); + Map longRange = Map.of("gte", 3.14, "lt", "42.9"); + assertEquals(Map.of("gte", 3L, "lt", 42L), longMapper.parseSourceValue(longRange)); + + RangeFieldMapper dateMapper = new RangeFieldMapper.Builder("field", RangeType.DATE) + .format("yyyy/MM/dd||epoch_millis") + .build(context); + Map dateRange = Map.of("lt", "1990/12/29", "gte", 597429487111L); + assertEquals(Map.of("lt", "1990/12/29", "gte", "1988/12/06"), + dateMapper.parseSourceValue(dateRange)); + } } diff --git a/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldTypeTests.java b/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldTypeTests.java index f9feb59700a42..04ac4276c7a89 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldTypeTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/RangeFieldTypeTests.java @@ -237,13 +237,13 @@ public void testDateRangeQueryUsingMappingFormat() { assertEquals(1466062190000L, formatter.parseMillis(to)); RangeFieldType fieldType = new RangeFieldType(FIELDNAME, true, true, formatter, Collections.emptyMap()); - final Query query = fieldType.rangeQuery(from, to, true, true, relation, null, null, context); + final Query query = fieldType.rangeQuery(from, to, true, true, relation, null, fieldType.dateMathParser(), context); assertEquals("field:", query.toString()); // compare lower and upper bounds with what we would get on a `date` field DateFieldType dateFieldType = new DateFieldType(FIELDNAME, true, true, formatter, DateFieldMapper.Resolution.MILLISECONDS, Collections.emptyMap()); - final Query queryOnDateField = dateFieldType.rangeQuery(from, to, true, true, relation, null, null, context); + final Query queryOnDateField = dateFieldType.rangeQuery(from, to, true, true, relation, null, fieldType.dateMathParser(), context); assertEquals("field:[1465975790000 TO 1466062190999]", queryOnDateField.toString()); } @@ -464,9 +464,9 @@ private Object nextTo(Object from) throws Exception { } public void testParseIp() { - assertEquals(InetAddresses.forString("::1"), RangeType.IP.parse(InetAddresses.forString("::1"), randomBoolean())); - assertEquals(InetAddresses.forString("::1"), RangeType.IP.parse("::1", randomBoolean())); - assertEquals(InetAddresses.forString("::1"), RangeType.IP.parse(new BytesRef("::1"), randomBoolean())); + assertEquals(InetAddresses.forString("::1"), RangeType.IP.parseValue(InetAddresses.forString("::1"), randomBoolean(), null)); + assertEquals(InetAddresses.forString("::1"), RangeType.IP.parseValue("::1", randomBoolean(), null)); + assertEquals(InetAddresses.forString("::1"), RangeType.IP.parseValue(new BytesRef("::1"), randomBoolean(), null)); } public void testTermQuery() throws Exception { diff --git a/server/src/test/java/org/elasticsearch/index/mapper/TextFieldMapperTests.java b/server/src/test/java/org/elasticsearch/index/mapper/TextFieldMapperTests.java index 8a4eebf3ffe76..3071849c99cdf 100644 --- a/server/src/test/java/org/elasticsearch/index/mapper/TextFieldMapperTests.java +++ b/server/src/test/java/org/elasticsearch/index/mapper/TextFieldMapperTests.java @@ -49,7 +49,9 @@ import org.apache.lucene.search.spans.SpanOrQuery; import org.apache.lucene.search.spans.SpanTermQuery; import org.apache.lucene.util.BytesRef; +import org.elasticsearch.Version; import org.elasticsearch.action.index.IndexRequest; +import org.elasticsearch.cluster.metadata.IndexMetadata; import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.compress.CompressedXContent; @@ -1331,4 +1333,16 @@ public void testMeta() throws Exception { new CompressedXContent(mapping3), MergeReason.MAPPING_UPDATE); assertEquals(mapping3, mapper.mappingSource().toString()); } + + public void testParseSourceValue() { + Settings settings = Settings.builder().put(IndexMetadata.SETTING_VERSION_CREATED, Version.CURRENT.id).build(); + Mapper.BuilderContext context = new Mapper.BuilderContext(settings, new ContentPath()); + + FieldMapper fieldMapper = newBuilder().build(context); + TextFieldMapper mapper = (TextFieldMapper) fieldMapper; + + assertEquals("value", mapper.parseSourceValue("value")); + assertEquals("42", mapper.parseSourceValue(42L)); + assertEquals("true", mapper.parseSourceValue(true)); + } } diff --git a/server/src/test/java/org/elasticsearch/search/fetch/subphase/FieldValueRetrieverTests.java b/server/src/test/java/org/elasticsearch/search/fetch/subphase/FieldValueRetrieverTests.java index 92aa7eeacde4f..2b61fa2f8620e 100644 --- a/server/src/test/java/org/elasticsearch/search/fetch/subphase/FieldValueRetrieverTests.java +++ b/server/src/test/java/org/elasticsearch/search/fetch/subphase/FieldValueRetrieverTests.java @@ -32,6 +32,7 @@ import java.io.IOException; import java.util.List; import java.util.Map; +import java.util.Set; import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.hasItems; @@ -65,8 +66,8 @@ public void testObjectValues() throws IOException { MapperService mapperService = createMapperService(); XContentBuilder source = XContentFactory.jsonBuilder().startObject() .startObject("float_range") - .field("gte", 0.0) - .field("lte", 2.718) + .field("gte", 0.0f) + .field("lte", 2.718f) .endObject() .endObject(); @@ -76,14 +77,59 @@ public void testObjectValues() throws IOException { DocumentField rangeField = fields.get("float_range"); assertNotNull(rangeField); assertThat(rangeField.getValues().size(), equalTo(1)); - assertThat(rangeField.getValue(), equalTo(Map.of("gte", 0.0, "lte", 2.718))); + assertThat(rangeField.getValue(), equalTo(Map.of("gte", 0.0f, "lte", 2.718f))); + } + + public void testNonExistentField() throws IOException { + MapperService mapperService = createMapperService(); + XContentBuilder source = XContentFactory.jsonBuilder().startObject() + .field("field", "value") + .endObject(); + + Map fields = retrieveFields(mapperService, source, "non-existent"); + assertThat(fields.size(), equalTo(0)); + } + + public void testArrayValueMappers() throws IOException { + MapperService mapperService = createMapperService(); + + XContentBuilder source = XContentFactory.jsonBuilder().startObject() + .array("completion", "first", "second") + .endObject(); + + Map fields = retrieveFields(mapperService, source, "completion"); + assertThat(fields.size(), equalTo(1)); + + DocumentField field = fields.get("completion"); + assertNotNull(field); + assertThat(field.getValues().size(), equalTo(2)); + assertThat(field.getValues(), hasItems("first", "second")); + + // Test a field with multiple geo-points. + source = XContentFactory.jsonBuilder().startObject() + .startObject("completion") + .array("input", "first", "second") + .field("weight", "2.718") + .endObject() + .endObject(); + + fields = retrieveFields(mapperService, source, "completion"); + assertThat(fields.size(), equalTo(1)); + + field = fields.get("completion"); + assertNotNull(field); + assertThat(field.getValues().size(), equalTo(1)); + + Map expected = Map.of("input", List.of("first", "second"), + "weight", "2.718"); + assertThat(field.getValues().get(0), equalTo(expected)); } public void testFieldNamesWithWildcard() throws IOException { MapperService mapperService = createMapperService();; XContentBuilder source = XContentFactory.jsonBuilder().startObject() .array("field", "first", "second") - .field("integer_field", "third") + .field("integer_field", 333) .startObject("object") .field("field", "fourth") .endObject() @@ -100,7 +146,7 @@ public void testFieldNamesWithWildcard() throws IOException { DocumentField otherField = fields.get("integer_field"); assertNotNull(otherField); assertThat(otherField.getValues().size(), equalTo(1)); - assertThat(otherField.getValues(), hasItems("third")); + assertThat(otherField.getValues(), hasItems(333)); DocumentField objectField = fields.get("object.field"); assertNotNull(objectField); @@ -108,7 +154,6 @@ public void testFieldNamesWithWildcard() throws IOException { assertThat(objectField.getValues(), hasItems("fourth")); } - public void testFieldAliases() throws IOException { XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() .startObject("properties") @@ -226,15 +271,15 @@ private Map retrieveFields(MapperService mapperService, X sourceLookup.setSource(BytesReference.bytes(source)); FieldValueRetriever fetchFieldsLookup = FieldValueRetriever.create(mapperService, fieldPatterns); - return fetchFieldsLookup.retrieve(sourceLookup); + return fetchFieldsLookup.retrieve(sourceLookup, Set.of()); } - public MapperService createMapperService() throws IOException { XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() .startObject("properties") .startObject("field").field("type", "keyword").endObject() .startObject("integer_field").field("type", "integer").endObject() + .startObject("completion").field("type", "completion").endObject() .startObject("float_range").field("type", "float_range").endObject() .startObject("object") .startObject("properties") diff --git a/test/framework/src/main/java/org/elasticsearch/index/mapper/MockFieldMapper.java b/test/framework/src/main/java/org/elasticsearch/index/mapper/MockFieldMapper.java index 77e7f44880ab2..149ce7573c19a 100644 --- a/test/framework/src/main/java/org/elasticsearch/index/mapper/MockFieldMapper.java +++ b/test/framework/src/main/java/org/elasticsearch/index/mapper/MockFieldMapper.java @@ -89,6 +89,11 @@ protected String contentType() { protected void parseCreateField(ParseContext context) throws IOException { } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { diff --git a/x-pack/plugin/analytics/src/main/java/org/elasticsearch/xpack/analytics/mapper/HistogramFieldMapper.java b/x-pack/plugin/analytics/src/main/java/org/elasticsearch/xpack/analytics/mapper/HistogramFieldMapper.java index 685eda193d0b9..674c2e04eaead 100644 --- a/x-pack/plugin/analytics/src/main/java/org/elasticsearch/xpack/analytics/mapper/HistogramFieldMapper.java +++ b/x-pack/plugin/analytics/src/main/java/org/elasticsearch/xpack/analytics/mapper/HistogramFieldMapper.java @@ -164,6 +164,11 @@ protected void parseCreateField(ParseContext context) throws IOException { throw new UnsupportedOperationException("Parsing is implemented in parse(), this method should NEVER be called"); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + public static class HistogramFieldType extends MappedFieldType { public HistogramFieldType(String name, boolean hasDocValues, Map meta) { diff --git a/x-pack/plugin/mapper-constant-keyword/src/main/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapper.java b/x-pack/plugin/mapper-constant-keyword/src/main/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapper.java index e034e2b4ae268..3062d47e7dd8b 100644 --- a/x-pack/plugin/mapper-constant-keyword/src/main/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapper.java +++ b/x-pack/plugin/mapper-constant-keyword/src/main/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapper.java @@ -38,6 +38,7 @@ import org.elasticsearch.index.mapper.TypeParsers; import org.elasticsearch.index.query.QueryShardContext; import org.elasticsearch.search.aggregations.support.CoreValuesSourceType; +import org.elasticsearch.search.lookup.SourceLookup; import java.io.IOException; import java.time.ZoneId; @@ -262,6 +263,18 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + public List lookupValues(SourceLookup lookup) { + return fieldType().value == null + ? List.of() + : List.of(fieldType().value); + } + + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException("This should never be called, since lookupValues is implemented directly."); + } + @Override protected void mergeOptions(FieldMapper other, List conflicts) { ConstantKeywordFieldType newConstantKeywordFT = (ConstantKeywordFieldType) other.fieldType(); diff --git a/x-pack/plugin/mapper-constant-keyword/src/test/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapperTests.java b/x-pack/plugin/mapper-constant-keyword/src/test/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapperTests.java index c1e183d7c804c..acfd5689d6a2e 100644 --- a/x-pack/plugin/mapper-constant-keyword/src/test/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapperTests.java +++ b/x-pack/plugin/mapper-constant-keyword/src/test/java/org/elasticsearch/xpack/constantkeyword/mapper/ConstantKeywordFieldMapperTests.java @@ -13,18 +13,21 @@ import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.index.IndexService; import org.elasticsearch.index.mapper.DocumentMapper; +import org.elasticsearch.index.mapper.FieldMapper; import org.elasticsearch.index.mapper.FieldMapperTestCase; import org.elasticsearch.index.mapper.MapperParsingException; import org.elasticsearch.index.mapper.MapperService.MergeReason; import org.elasticsearch.index.mapper.ParsedDocument; import org.elasticsearch.index.mapper.SourceToParse; import org.elasticsearch.plugins.Plugin; +import org.elasticsearch.search.lookup.SourceLookup; import org.elasticsearch.xpack.constantkeyword.ConstantKeywordMapperPlugin; import org.elasticsearch.xpack.core.LocalStateCompositeXPackPlugin; import org.junit.Before; import java.util.Collection; import java.util.Collections; +import java.util.List; import java.util.Set; public class ConstantKeywordFieldMapperTests extends FieldMapperTestCase { @@ -133,4 +136,27 @@ public void testMeta() throws Exception { new CompressedXContent(mapping3), MergeReason.MAPPING_UPDATE); assertEquals(mapping3, mapper.mappingSource().toString()); } + + public void testLookupValues() throws Exception { + IndexService indexService = createIndex("test"); + String mapping = Strings.toString(XContentFactory.jsonBuilder().startObject().startObject("_doc") + .startObject("properties").startObject("field").field("type", "constant_keyword") + .endObject().endObject().endObject().endObject()); + DocumentMapper mapper = indexService.mapperService().merge("_doc", new CompressedXContent(mapping), MergeReason.MAPPING_UPDATE); + assertEquals(mapping, mapper.mappingSource().toString()); + + FieldMapper fieldMapper = (FieldMapper) mapper.mappers().getMapper("field"); + List values = fieldMapper.lookupValues(new SourceLookup()); + assertTrue(values.isEmpty()); + + String mapping2 = Strings.toString(XContentFactory.jsonBuilder().startObject().startObject("_doc") + .startObject("properties").startObject("field").field("type", "constant_keyword") + .field("value", "foo").endObject().endObject().endObject().endObject()); + mapper = indexService.mapperService().merge("_doc", new CompressedXContent(mapping2), MergeReason.MAPPING_UPDATE); + + fieldMapper = (FieldMapper) mapper.mappers().getMapper("field"); + values = fieldMapper.lookupValues(new SourceLookup()); + assertEquals(1, values.size()); + assertEquals("foo", values.get(0)); + } } diff --git a/x-pack/plugin/mapper-flattened/src/main/java/org/elasticsearch/xpack/flattened/mapper/FlatObjectFieldMapper.java b/x-pack/plugin/mapper-flattened/src/main/java/org/elasticsearch/xpack/flattened/mapper/FlatObjectFieldMapper.java index 6fc8183bca421..dabaf13e88f3f 100644 --- a/x-pack/plugin/mapper-flattened/src/main/java/org/elasticsearch/xpack/flattened/mapper/FlatObjectFieldMapper.java +++ b/x-pack/plugin/mapper-flattened/src/main/java/org/elasticsearch/xpack/flattened/mapper/FlatObjectFieldMapper.java @@ -574,6 +574,11 @@ protected void parseCreateField(ParseContext context) throws IOException { } } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override protected void doXContentBody(XContentBuilder builder, boolean includeDefaults, Params params) throws IOException { super.doXContentBody(builder, includeDefaults, params); diff --git a/x-pack/plugin/src/test/resources/rest-api-spec/test/constant_keyword/10_basic.yml b/x-pack/plugin/src/test/resources/rest-api-spec/test/constant_keyword/10_basic.yml index ad2e2677bf610..7a6c2a0d53b0d 100644 --- a/x-pack/plugin/src/test/resources/rest-api-spec/test/constant_keyword/10_basic.yml +++ b/x-pack/plugin/src/test/resources/rest-api-spec/test/constant_keyword/10_basic.yml @@ -184,3 +184,18 @@ setup: - match: {hits.hits.0._index: test1 } - match: {hits.hits.1._index: test1 } - match: {hits.hits.2._index: test2 } + +--- +"Field retrieval": + + - do: + search: + index: test* + body: + fields: [ foo ] + sort: [ { _index: asc } ] + + - match: { "hits.total.value": 3 } + - match: {hits.hits.0.fields.foo.0: bar } + - match: {hits.hits.1.fields.foo.0: bar } + - match: {hits.hits.2.fields.foo.0: baz } diff --git a/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/DenseVectorFieldMapper.java b/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/DenseVectorFieldMapper.java index 7e45286e28737..8e78a670c839c 100644 --- a/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/DenseVectorFieldMapper.java +++ b/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/DenseVectorFieldMapper.java @@ -205,6 +205,11 @@ public void parse(ParseContext context) throws IOException { context.doc().addWithKey(fieldType().name(), field); } + @Override + protected Object parseSourceValue(Object value) { + return value; + } + @Override protected boolean indexedByDefault() { return false; diff --git a/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/SparseVectorFieldMapper.java b/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/SparseVectorFieldMapper.java index 51cafa01ecdcc..280bb98083988 100644 --- a/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/SparseVectorFieldMapper.java +++ b/x-pack/plugin/vectors/src/main/java/org/elasticsearch/xpack/vectors/mapper/SparseVectorFieldMapper.java @@ -137,12 +137,16 @@ public void parse(ParseContext context) { throw new UnsupportedOperationException(ERROR_MESSAGE_7X); } - @Override protected void parseCreateField(ParseContext context) { throw new IllegalStateException("parse is implemented directly"); } + @Override + protected Object parseSourceValue(Object value) { + throw new UnsupportedOperationException(ERROR_MESSAGE_7X); + } + @Override protected String contentType() { return CONTENT_TYPE; diff --git a/x-pack/plugin/wildcard/src/main/java/org/elasticsearch/xpack/wildcard/mapper/WildcardFieldMapper.java b/x-pack/plugin/wildcard/src/main/java/org/elasticsearch/xpack/wildcard/mapper/WildcardFieldMapper.java index 8f0ef4b13f705..ba44ee2b2f5fc 100644 --- a/x-pack/plugin/wildcard/src/main/java/org/elasticsearch/xpack/wildcard/mapper/WildcardFieldMapper.java +++ b/x-pack/plugin/wildcard/src/main/java/org/elasticsearch/xpack/wildcard/mapper/WildcardFieldMapper.java @@ -948,6 +948,11 @@ protected void parseCreateField(ParseContext context) throws IOException { parseDoc.addAll(fields); } + @Override + protected String parseSourceValue(Object value) { + return value.toString(); + } + void createFields(String value, Document parseDoc, Listfields) throws IOException { if (value == null || value.length() > ignoreAbove) { return;