From 8ccecc3a4cacb17f491f49c2156d238acc37c7a3 Mon Sep 17 00:00:00 2001 From: David Kyle Date: Mon, 28 Sep 2020 18:05:19 +0100 Subject: [PATCH] [ML][Transform] Filter null objects from field caps request (#62945) If the transform grouping is a script then exclude the field from the source index mappings fields caps request. A null object caused an NPE in the serialisation of FieldCapabilitiesIndexRequest. --- .../transform/transforms/pivot/SchemaUtil.java | 14 +++++++++++--- .../pivot/AggregationSchemaAndResultTests.java | 8 ++++++-- 2 files changed, 17 insertions(+), 5 deletions(-) diff --git a/x-pack/plugin/transform/src/main/java/org/elasticsearch/xpack/transform/transforms/pivot/SchemaUtil.java b/x-pack/plugin/transform/src/main/java/org/elasticsearch/xpack/transform/transforms/pivot/SchemaUtil.java index 32ad8fd704237..7bbf1d2faddff 100644 --- a/x-pack/plugin/transform/src/main/java/org/elasticsearch/xpack/transform/transforms/pivot/SchemaUtil.java +++ b/x-pack/plugin/transform/src/main/java/org/elasticsearch/xpack/transform/transforms/pivot/SchemaUtil.java @@ -25,6 +25,7 @@ import java.util.HashMap; import java.util.Map; +import java.util.Objects; import java.util.Set; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -77,6 +78,11 @@ public static void deduceMappings( .getGroups() .forEach( (destinationFieldName, group) -> { + // skip any fields that use scripts as there will be no source mapping + if (group.getScriptConfig() != null) { + return; + } + // We will always need the field name for the grouping to create the mapping fieldNamesForGrouping.put(destinationFieldName, group.getField()); // Sometimes the group config will supply a desired mapping as well @@ -107,7 +113,7 @@ public static void deduceMappings( getSourceFieldMappings( client, source, - allFieldNames.values().toArray(new String[0]), + allFieldNames.values().stream().filter(Objects::nonNull).toArray(String[]::new), ActionListener.wrap( sourceMappings -> listener.onResponse( resolveMappings( @@ -182,7 +188,8 @@ private static Map resolveMappings( } else if (destinationMapping != null) { targetMapping.put(targetFieldName, destinationMapping); } else { - logger.warn("Failed to deduce mapping for [{}], fall back to dynamic mapping.", targetFieldName); + logger.warn("Failed to deduce mapping for [{}], fall back to dynamic mapping. " + + "Create the destination index with complete mappings first to avoid deducing the mappings", targetFieldName); } }); @@ -192,7 +199,8 @@ private static Map resolveMappings( if (destinationMapping != null) { targetMapping.put(targetFieldName, destinationMapping); } else { - logger.warn("Failed to deduce mapping for [{}], fall back to keyword.", targetFieldName); + logger.warn("Failed to deduce mapping for [{}], fall back to keyword. " + + "Create the destination index with complete mappings first to avoid deducing the mappings", targetFieldName); targetMapping.put(targetFieldName, KeywordFieldMapper.CONTENT_TYPE); } }); diff --git a/x-pack/plugin/transform/src/test/java/org/elasticsearch/xpack/transform/transforms/pivot/AggregationSchemaAndResultTests.java b/x-pack/plugin/transform/src/test/java/org/elasticsearch/xpack/transform/transforms/pivot/AggregationSchemaAndResultTests.java index 1b5faa897fa04..a2e811993e522 100644 --- a/x-pack/plugin/transform/src/test/java/org/elasticsearch/xpack/transform/transforms/pivot/AggregationSchemaAndResultTests.java +++ b/x-pack/plugin/transform/src/test/java/org/elasticsearch/xpack/transform/transforms/pivot/AggregationSchemaAndResultTests.java @@ -127,11 +127,13 @@ public void testBasic() throws InterruptedException { AggregationConfig aggregationConfig = new AggregationConfig(Collections.emptyMap(), aggs); GroupConfig groupConfig = GroupConfigTests.randomGroupConfig(); PivotConfig pivotConfig = new PivotConfig(groupConfig, aggregationConfig, null); + long numGroupsWithoutScripts = groupConfig.getGroups().values().stream() + .filter(singleGroupSource -> singleGroupSource.getScriptConfig() == null).count(); this.>assertAsync( listener -> SchemaUtil.deduceMappings(client, pivotConfig, new String[] { "source-index" }, listener), mappings -> { - assertEquals(groupConfig.getGroups().size() + 10, mappings.size()); + assertEquals(numGroupsWithoutScripts + 10, mappings.size()); assertEquals("long", mappings.get("max_rating")); assertEquals("double", mappings.get("avg_rating")); assertEquals("long", mappings.get("count_rating")); @@ -189,11 +191,13 @@ public void testNested() throws InterruptedException { AggregationConfig aggregationConfig = new AggregationConfig(Collections.emptyMap(), aggs); GroupConfig groupConfig = GroupConfigTests.randomGroupConfig(); PivotConfig pivotConfig = new PivotConfig(groupConfig, aggregationConfig, null); + long numGroupsWithoutScripts = groupConfig.getGroups().values().stream() + .filter(singleGroupSource -> singleGroupSource.getScriptConfig() == null).count(); this.>assertAsync( listener -> SchemaUtil.deduceMappings(client, pivotConfig, new String[] { "source-index" }, listener), mappings -> { - assertEquals(groupConfig.getGroups().size() + 12, mappings.size()); + assertEquals(numGroupsWithoutScripts + 12, mappings.size()); assertEquals("long", mappings.get("filter_1")); assertEquals("object", mappings.get("filter_2")); assertEquals("long", mappings.get("filter_2.max_drinks_2"));