Skip to content

Commit f74dd87

Browse files
authored
Move dynamic mapping parameter configuration to @document and @field.
Original Pull Request #1872 Closes #1871
1 parent 3b921b7 commit f74dd87

File tree

10 files changed

+233
-14
lines changed

10 files changed

+233
-14
lines changed

src/main/java/org/springframework/data/elasticsearch/annotations/Document.java

+9
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@
3333
* @author Ivan Greene
3434
* @author Mark Paluch
3535
* @author Peter-Josef Meisch
36+
* @author Sascha Woo
3637
*/
3738
@Persistent
3839
@Inherited
@@ -112,4 +113,12 @@
112113
* @since 4.3
113114
*/
114115
WriteTypeHint writeTypeHint() default WriteTypeHint.DEFAULT;
116+
117+
/**
118+
* Controls how Elasticsearch dynamically adds fields to the document.
119+
*
120+
* @since 4.3
121+
*/
122+
Dynamic dynamic() default Dynamic.INHERIT;
123+
115124
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
/*
2+
* Copyright 2021 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.elasticsearch.annotations;
17+
18+
/**
19+
* Values for the {@code dynamic} mapping parameter.
20+
*
21+
* @author Sascha Woo
22+
* @since 4.3
23+
*/
24+
public enum Dynamic {
25+
/**
26+
* New fields are added to the mapping.
27+
*/
28+
TRUE,
29+
/**
30+
* New fields are added to the mapping as
31+
* <a href="https://www.elastic.co/guide/en/elasticsearch/reference/current/runtime.html">runtime fields</a>. These
32+
* fields are not indexed, and are loaded from {@code _source} at query time.
33+
*/
34+
RUNTIME,
35+
/**
36+
* New fields are ignored. These fields will not be indexed or searchable, but will still appear in the
37+
* {@code _source} field of returned hits. These fields will not be added to the mapping, and new fields must be added
38+
* explicitly.
39+
*/
40+
FALSE,
41+
/**
42+
* If new fields are detected, an exception is thrown and the document is rejected. New fields must be explicitly
43+
* added to the mapping.
44+
*/
45+
STRICT,
46+
/**
47+
* Inherit the dynamic setting from their parent object or from the mapping type.
48+
*/
49+
INHERIT
50+
}

src/main/java/org/springframework/data/elasticsearch/annotations/DynamicMapping.java

+3
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,14 @@
2626
* {@see <a href="https://www.elastic.co/guide/en/elasticsearch/reference/current/dynamic.html">elasticsearch doc</a>}
2727
*
2828
* @author Peter-Josef Meisch
29+
* @author Sascha Woo
2930
* @since 4.0
31+
* @deprecated since 4.3, use {@link Document#dynamic()} or {@link Field#dynamic()} instead.
3032
*/
3133
@Retention(RetentionPolicy.RUNTIME)
3234
@Target({ ElementType.TYPE, ElementType.FIELD })
3335
@Documented
36+
@Deprecated
3437
public @interface DynamicMapping {
3538

3639
DynamicMappingValue value() default DynamicMappingValue.True;

src/main/java/org/springframework/data/elasticsearch/annotations/DynamicMappingValue.java

+3
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,11 @@
1919
* values for the {@link DynamicMapping annotation}
2020
*
2121
* @author Peter-Josef Meisch
22+
* @author Sascha Woo
2223
* @since 4.0
24+
* @deprecated since 4.3, use {@link Document#dynamic()} or {@link Field#dynamic()} instead.
2325
*/
26+
@Deprecated
2427
public enum DynamicMappingValue {
2528
True, False, Strict
2629
}

src/main/java/org/springframework/data/elasticsearch/annotations/Field.java

+8
Original file line numberDiff line numberDiff line change
@@ -195,4 +195,12 @@
195195
* @since 4.2
196196
*/
197197
int dims() default -1;
198+
199+
/**
200+
* Controls how Elasticsearch dynamically adds fields to the inner object within the document.<br>
201+
* To be used in combination with {@link FieldType#Object} or {@link FieldType#Nested}
202+
*
203+
* @since 4.3
204+
*/
205+
Dynamic dynamic() default Dynamic.INHERIT;
198206
}

src/main/java/org/springframework/data/elasticsearch/core/index/MappingBuilder.java

+15-5
Original file line numberDiff line numberDiff line change
@@ -197,7 +197,9 @@ private void mapEntity(XContentBuilder builder, @Nullable ElasticsearchPersisten
197197
}
198198
}
199199

200-
if (dynamicMapping != null) {
200+
if (entity != null && entity.dynamic() != Dynamic.INHERIT) {
201+
builder.field(TYPE_DYNAMIC, entity.dynamic().name().toLowerCase());
202+
} else if (dynamicMapping != null) {
201203
builder.field(TYPE_DYNAMIC, dynamicMapping.value().name().toLowerCase());
202204
}
203205

@@ -440,8 +442,12 @@ private void addSingleFieldMapping(XContentBuilder builder, ElasticsearchPersist
440442

441443
builder.startObject(property.getFieldName());
442444

443-
if (nestedOrObjectField && dynamicMapping != null) {
444-
builder.field(TYPE_DYNAMIC, dynamicMapping.value().name().toLowerCase());
445+
if (nestedOrObjectField) {
446+
if (annotation.dynamic() != Dynamic.INHERIT) {
447+
builder.field(TYPE_DYNAMIC, annotation.dynamic().name().toLowerCase());
448+
} else if (dynamicMapping != null) {
449+
builder.field(TYPE_DYNAMIC, dynamicMapping.value().name().toLowerCase());
450+
}
445451
}
446452

447453
addFieldMappingParameters(builder, annotation, nestedOrObjectField);
@@ -489,8 +495,12 @@ private void addMultiFieldMapping(XContentBuilder builder, ElasticsearchPersiste
489495
// main field
490496
builder.startObject(property.getFieldName());
491497

492-
if (nestedOrObjectField && dynamicMapping != null) {
493-
builder.field(TYPE_DYNAMIC, dynamicMapping.value().name().toLowerCase());
498+
if (nestedOrObjectField) {
499+
if (annotation.mainField().dynamic() != Dynamic.INHERIT) {
500+
builder.field(TYPE_DYNAMIC, annotation.mainField().dynamic().name().toLowerCase());
501+
} else if (dynamicMapping != null) {
502+
builder.field(TYPE_DYNAMIC, dynamicMapping.value().name().toLowerCase());
503+
}
494504
}
495505

496506
addFieldMappingParameters(builder, annotation.mainField(), nestedOrObjectField);

src/main/java/org/springframework/data/elasticsearch/core/mapping/ElasticsearchPersistentEntity.java

+7
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
package org.springframework.data.elasticsearch.core.mapping;
1717

1818
import org.elasticsearch.index.VersionType;
19+
import org.springframework.data.elasticsearch.annotations.Dynamic;
1920
import org.springframework.data.elasticsearch.annotations.Field;
2021
import org.springframework.data.elasticsearch.core.index.Settings;
2122
import org.springframework.data.elasticsearch.core.join.JoinField;
@@ -160,4 +161,10 @@ default ElasticsearchPersistentProperty getRequiredSeqNoPrimaryTermProperty() {
160161
* @since 4.3
161162
*/
162163
boolean writeTypeHints();
164+
165+
/**
166+
* @return the {@code dynamic} mapping parameter value.
167+
* @since 4.3
168+
*/
169+
Dynamic dynamic();
163170
}

src/main/java/org/springframework/data/elasticsearch/core/mapping/SimpleElasticsearchPersistentEntity.java

+10-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
import org.springframework.core.annotation.AnnotatedElementUtils;
2626
import org.springframework.dao.InvalidDataAccessApiUsageException;
2727
import org.springframework.data.elasticsearch.annotations.Document;
28+
import org.springframework.data.elasticsearch.annotations.Dynamic;
2829
import org.springframework.data.elasticsearch.annotations.Field;
2930
import org.springframework.data.elasticsearch.annotations.FieldType;
3031
import org.springframework.data.elasticsearch.annotations.Routing;
@@ -73,6 +74,7 @@ public class SimpleElasticsearchPersistentEntity<T> extends BasicPersistentEntit
7374
private @Nullable ElasticsearchPersistentProperty joinFieldProperty;
7475
private @Nullable VersionType versionType;
7576
private boolean createIndexAndMapping;
77+
private final Dynamic dynamic;
7678
private final Map<String, ElasticsearchPersistentProperty> fieldNamePropertyCache = new ConcurrentHashMap<>();
7779
private final ConcurrentHashMap<String, Expression> routingExpressions = new ConcurrentHashMap<>();
7880
private @Nullable String routing;
@@ -102,8 +104,10 @@ public SimpleElasticsearchPersistentEntity(TypeInformation<T> typeInformation,
102104
this.indexName = document.indexName();
103105
this.versionType = document.versionType();
104106
this.createIndexAndMapping = document.createIndex();
107+
this.dynamic = document.dynamic();
108+
} else {
109+
this.dynamic = Dynamic.INHERIT;
105110
}
106-
107111
Routing routingAnnotation = AnnotatedElementUtils.findMergedAnnotation(clazz, Routing.class);
108112

109113
if (routingAnnotation != null) {
@@ -559,4 +563,9 @@ public FieldNamingStrategy getFieldNamingStrategy() {
559563
return fieldNamingStrategy;
560564
}
561565
}
566+
567+
@Override
568+
public Dynamic dynamic() {
569+
return dynamic;
570+
}
562571
}

src/test/java/org/springframework/data/elasticsearch/core/index/MappingBuilderIntegrationTests.java

+39-4
Original file line numberDiff line numberDiff line change
@@ -287,8 +287,18 @@ void shouldWriteMappingForDisabledProperty() {
287287
}
288288

289289
@Test // #1767
290-
@DisplayName("should write dynamic mapping entries")
291-
void shouldWriteDynamicMappingEntries() {
290+
@DisplayName("should write dynamic mapping annotations")
291+
void shouldWriteDynamicMappingAnnotations() {
292+
293+
IndexOperations indexOps = operations.indexOps(DynamicMappingAnnotationEntity.class);
294+
indexOps.create();
295+
indexOps.putMapping();
296+
297+
}
298+
299+
@Test // #1871
300+
@DisplayName("should write dynamic mapping")
301+
void shouldWriteDynamicMapping() {
292302

293303
IndexOperations indexOps = operations.indexOps(DynamicMappingEntity.class);
294304
indexOps.create();
@@ -1104,9 +1114,9 @@ public void setDense_vector(@Nullable float[] dense_vector) {
11041114
}
11051115
}
11061116

1107-
@Document(indexName = "dynamic-mapping")
1117+
@Document(indexName = "dynamic-mapping-annotation")
11081118
@DynamicMapping(DynamicMappingValue.False)
1109-
static class DynamicMappingEntity {
1119+
static class DynamicMappingAnnotationEntity {
11101120

11111121
@Nullable @DynamicMapping(DynamicMappingValue.Strict) @Field(type = FieldType.Object) private Author author;
11121122
@Nullable @DynamicMapping(DynamicMappingValue.False) @Field(
@@ -1124,6 +1134,31 @@ public void setAuthor(Author author) {
11241134
}
11251135
}
11261136

1137+
@Document(indexName = "dynamic-mapping", dynamic = Dynamic.FALSE)
1138+
static class DynamicMappingEntity {
1139+
1140+
@Nullable @Field(type = FieldType.Object) //
1141+
private Map<String, Object> objectInherit;
1142+
@Nullable @Field(type = FieldType.Object, dynamic = Dynamic.FALSE) //
1143+
private Map<String, Object> objectFalse;
1144+
@Nullable @Field(type = FieldType.Object, dynamic = Dynamic.TRUE) //
1145+
private Map<String, Object> objectTrue;
1146+
@Nullable @Field(type = FieldType.Object, dynamic = Dynamic.STRICT) //
1147+
private Map<String, Object> objectStrict;
1148+
@Nullable @Field(type = FieldType.Object, dynamic = Dynamic.RUNTIME) //
1149+
private Map<String, Object> objectRuntime;
1150+
@Nullable @Field(type = FieldType.Nested) //
1151+
private List<Map<String, Object>> nestedObjectInherit;
1152+
@Nullable @Field(type = FieldType.Nested, dynamic = Dynamic.FALSE) //
1153+
private List<Map<String, Object>> nestedObjectFalse;
1154+
@Nullable @Field(type = FieldType.Nested, dynamic = Dynamic.TRUE) //
1155+
private List<Map<String, Object>> nestedObjectTrue;
1156+
@Nullable @Field(type = FieldType.Nested, dynamic = Dynamic.STRICT) //
1157+
private List<Map<String, Object>> nestedObjectStrict;
1158+
@Nullable @Field(type = FieldType.Nested, dynamic = Dynamic.RUNTIME) //
1159+
private List<Map<String, Object>> nestedObjectRuntime;
1160+
}
1161+
11271162
@Document(indexName = "dynamic-detection-mapping-true")
11281163
@Mapping(dateDetection = Mapping.Detection.TRUE, numericDetection = Mapping.Detection.TRUE,
11291164
dynamicDateFormats = { "MM/dd/yyyy" })

0 commit comments

Comments
 (0)