Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add runtime fields to index mapping. #1820

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion src/main/asciidoc/reference/elasticsearch-misc.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -51,10 +51,24 @@ class Entity {

When Spring Data Elasticsearch creates the index mapping with the `IndexOperations.createMapping()` methods, it uses the annotations described in <<elasticsearch.mapping.meta-model.annotations>>, especially the `@Field` annotation. In addition to that it is possible to add the `@Mapping` annotation to a class. This annotation has the following properties:

* `mappingPath` a classpath resource in JSON format which is used as the mapping, no other mapping processing is done.
* `mappingPath` a classpath resource in JSON format; if this is not empty it is used as the mapping, no other mapping processing is done.
* `enabled` when set to false, this flag is written to the mapping and no further processing is done.
* `dateDetection` and `numericDetection` set the corresponding properties in the mapping when not set to `DEFAULT`.
* `dynamicDateFormats` when this String array is not empty, it defines the date formats used for automatic date detection.
* `runtimeFieldsPath` a classpath resource in JSON format containing the definition of runtime fields which is written to the index mappings, for example:
====
[source,json]
----
{
"day_of_week": {
"type": "keyword",
"script": {
"source": "emit(doc['@timestamp'].value.dayOfWeekEnum.getDisplayName(TextStyle.FULL, Locale.ROOT))"
}
}
}
----
====

[[elasticsearch.misc.filter]]
== Filter Builder
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
* Elasticsearch Mapping
*
* @author Mohsin Husen
* @author Peter-Josef Meisch
*/
@Persistent
@Inherited
Expand All @@ -42,6 +43,7 @@
* @since 4.2
*/
boolean enabled() default true;

/**
* whether date_detection is enabled
*
Expand All @@ -58,10 +60,20 @@

/**
* custom dynamic date formats
*
* @since 4.3
*/
String[] dynamicDateFormats() default {};

/**
* classpath to a JSON file containing the values for a runtime mapping definition. The file must contain the JSON
* object that is written as the value of the runtime property. {@see <a href=
* "https://www.elastic.co/guide/en/elasticsearch/reference/7.12/runtime-mapping-fields.html">elasticsearch doc</a>}
*
* @since 4.3
*/
String runtimeFieldsPath() default "";

enum Detection {
DEFAULT, TRUE, FALSE;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -219,8 +219,6 @@ protected Document buildMapping(Class<?> clazz) {
if (hasText(mappings)) {
return Document.parse(mappings);
}
} else {
LOGGER.info("mappingPath in @Mapping has to be defined. Building mappings using @Field");
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,7 @@ public class MappingBuilder {
private static final String DATE_DETECTION = "date_detection";
private static final String NUMERIC_DETECTION = "numeric_detection";
private static final String DYNAMIC_DATE_FORMATS = "dynamic_date_formats";
private static final String RUNTIME = "runtime";

private final ElasticsearchConverter elasticsearchConverter;

Expand Down Expand Up @@ -168,6 +169,10 @@ private void mapEntity(XContentBuilder builder, @Nullable ElasticsearchPersisten
if (mappingAnnotation.dynamicDateFormats().length > 0) {
builder.field(DYNAMIC_DATE_FORMATS, mappingAnnotation.dynamicDateFormats());
}

if (StringUtils.hasText(mappingAnnotation.runtimeFieldsPath())) {
addRuntimeFields(builder, mappingAnnotation.runtimeFieldsPath());
}
}

boolean writeNestedProperties = !isRootObject && (isAnyPropertyAnnotatedWithField(entity) || nestedOrObjectField);
Expand Down Expand Up @@ -222,6 +227,15 @@ private void mapEntity(XContentBuilder builder, @Nullable ElasticsearchPersisten

}

private void addRuntimeFields(XContentBuilder builder, String runtimeFieldsPath) throws IOException {

ClassPathResource runtimeFields = new ClassPathResource(runtimeFieldsPath);

if (runtimeFields.exists()) {
builder.rawField(RUNTIME, runtimeFields.getInputStream(), XContentType.JSON);
}
}

private void buildPropertyMapping(XContentBuilder builder, boolean isRootObject,
ElasticsearchPersistentProperty property) throws IOException {

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
import java.lang.Integer;
import java.lang.Object;
import java.math.BigDecimal;
import java.time.Instant;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
Expand Down Expand Up @@ -316,6 +317,16 @@ void shouldWriteDynamicDetectionValues() {
indexOps.delete();
}

@Test // #1816
@DisplayName("should write runtime fields")
void shouldWriteRuntimeFields() {

IndexOperations indexOps = operations.indexOps(RuntimeFieldEntity.class);
indexOps.create();
indexOps.putMapping();
indexOps.delete();
}

// region entities
@Document(indexName = "ignore-above-index")
static class IgnoreAboveEntity {
Expand Down Expand Up @@ -1130,6 +1141,14 @@ public void setAuthor(Author author) {
private static class DynamicDetectionMapping {
@Id @Nullable private String id;
}

@Document(indexName = "runtime-fields")
@Mapping(runtimeFieldsPath = "/mappings/runtime-fields.json")
private static class RuntimeFieldEntity {
@Id @Nullable private String id;
@Field(type = Date, format = DateFormat.epoch_millis, name = "@timestamp") @Nullable private Instant timestamp;
}

// endregion

}
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
import java.lang.Integer;
import java.lang.Object;
import java.math.BigDecimal;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.Collection;
Expand Down Expand Up @@ -855,7 +856,38 @@ void shouldWriteDynamicDateFormats() throws JSONException {
assertEquals(expected, mapping, true);
}

@Test // #1816
@DisplayName("should write runtime fields")
void shouldWriteRuntimeFields() throws JSONException {

String expected = "{\n" + //
" \"runtime\": {\n" + //
" \"day_of_week\": {\n" + //
" \"type\": \"keyword\",\n" + //
" \"script\": {\n" + //
" \"source\": \"emit(doc['@timestamp'].value.dayOfWeekEnum.getDisplayName(TextStyle.FULL, Locale.ROOT))\"\n" + //
" }\n" + //
" }\n" + //
" },\n" + //
" \"properties\": {\n" + //
" \"_class\": {\n" + //
" \"type\": \"keyword\",\n" + //
" \"index\": false,\n" + //
" \"doc_values\": false\n" + //
" },\n" + //
" \"@timestamp\": {\n" + //
" \"type\": \"date\",\n" + //
" \"format\": \"epoch_millis\"\n" + //
" }\n" + //
" }\n" + //
"}\n"; //

String mapping = getMappingBuilder().buildPropertyMapping(RuntimeFieldEntity.class);

assertEquals(expected, mapping, true);
}
// region entities

@Document(indexName = "ignore-above-index")
static class IgnoreAboveEntity {
@Nullable @Id private String id;
Expand Down Expand Up @@ -1778,7 +1810,7 @@ private static class DynamicDetectionMappingDefault {
}

@Document(indexName = "dynamic-dateformats-mapping")
@Mapping(dynamicDateFormats = {"date1", "date2"})
@Mapping(dynamicDateFormats = { "date1", "date2" })
private static class DynamicDateFormatsMapping {
@Id @Nullable private String id;
}
Expand All @@ -1794,5 +1826,12 @@ private static class DynamicDetectionMappingTrue {
private static class DynamicDetectionMappingFalse {
@Id @Nullable private String id;
}

@Document(indexName = "runtime-fields")
@Mapping(runtimeFieldsPath = "/mappings/runtime-fields.json")
private static class RuntimeFieldEntity {
@Id @Nullable private String id;
@Field(type = Date, format = DateFormat.epoch_millis, name = "@timestamp") @Nullable private Instant timestamp;
}
// endregion
}
8 changes: 8 additions & 0 deletions src/test/resources/mappings/runtime-fields.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"day_of_week": {
"type": "keyword",
"script": {
"source": "emit(doc['@timestamp'].value.dayOfWeekEnum.getDisplayName(TextStyle.FULL, Locale.ROOT))"
}
}
}