diff --git a/x-pack/plugin/esql-core/src/main/java/org/elasticsearch/xpack/esql/core/expression/FieldAttribute.java b/x-pack/plugin/esql-core/src/main/java/org/elasticsearch/xpack/esql/core/expression/FieldAttribute.java index 0c19c12df4b5b..5a31be1ef5b70 100644 --- a/x-pack/plugin/esql-core/src/main/java/org/elasticsearch/xpack/esql/core/expression/FieldAttribute.java +++ b/x-pack/plugin/esql-core/src/main/java/org/elasticsearch/xpack/esql/core/expression/FieldAttribute.java @@ -34,8 +34,8 @@ * * To adequately represent e.g. union types, the name of the attribute can be altered because we may have multiple synthetic field * attributes that really belong to the same underlying field. For instance, if a multi-typed field is used both as {@code field::string} - * and {@code field::ip}, we'll generate 2 field attributes called {@code $$field$converted_to$string} and {@code $$field$converted_to$ip} - * but still referring to the same underlying field. + * and {@code field::ip}, we'll generate 2 field attributes called {@code $$field$converted_to$keyword} and {@code $$field$converted_to$ip} + * which still refer to the same underlying index field. */ public class FieldAttribute extends TypedAttribute { @@ -163,6 +163,15 @@ public FieldName fieldName() { return lazyFieldName; } + /** + * The name of the attribute. Can deviate from the field name e.g. in case of union types. For the physical field name, use + * {@link FieldAttribute#fieldName()}. + */ + @Override + public String name() { + return super.name(); + } + public EsField.Exact getExactInfo() { return field.getExactInfo(); } @@ -176,7 +185,7 @@ public FieldAttribute exactAttribute() { } private FieldAttribute innerField(EsField type) { - return new FieldAttribute(source(), name(), name() + "." + type.getName(), type, nullable(), id(), synthetic()); + return new FieldAttribute(source(), fieldName().string, name() + "." + type.getName(), type, nullable(), id(), synthetic()); } @Override diff --git a/x-pack/plugin/esql/qa/testFixtures/src/main/resources/lookup-join.csv-spec b/x-pack/plugin/esql/qa/testFixtures/src/main/resources/lookup-join.csv-spec index c4bc19e274d64..88e5ca2f482ef 100644 --- a/x-pack/plugin/esql/qa/testFixtures/src/main/resources/lookup-join.csv-spec +++ b/x-pack/plugin/esql/qa/testFixtures/src/main/resources/lookup-join.csv-spec @@ -732,6 +732,28 @@ language.id:integer | language.name:text | language.name.keyword:keyword | langu 2 | French | French | FR ; +############################################### +# union type behavior +############################################### + +joinOnMultiTypedMatchFieldCastToInteger +required_capability: join_lookup_v12 + +FROM apps, apps_short METADATA _index +| EVAL language_code = id::integer +| KEEP _index, language_code +| WHERE language_code < 3 +| LOOKUP JOIN languages_lookup ON language_code +| SORT _index ASC, language_code ASC +; + +_index:keyword | language_code:integer | language_name:keyword +apps | 1 | English +apps | 2 | French +apps_short | 1 | English +apps_short | 2 | French +; + ############################################### # Tests with clientips_lookup index ############################################### diff --git a/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupFromIndexIT.java b/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupFromIndexIT.java index e6b96e2e1267d..c652cb09be6f9 100644 --- a/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupFromIndexIT.java +++ b/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupFromIndexIT.java @@ -54,6 +54,7 @@ import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.xpack.core.async.AsyncExecutionId; import org.elasticsearch.xpack.esql.core.expression.Alias; +import org.elasticsearch.xpack.esql.core.expression.FieldAttribute; import org.elasticsearch.xpack.esql.core.expression.ReferenceAttribute; import org.elasticsearch.xpack.esql.core.tree.Source; import org.elasticsearch.xpack.esql.core.type.DataType; @@ -229,7 +230,7 @@ private void runLookup(DataType keyType, PopulateIndices populateIndices) throws keyType, "lookup", "lookup", - "key", + new FieldAttribute.FieldName("key"), List.of(new Alias(Source.EMPTY, "l", new ReferenceAttribute(Source.EMPTY, "l", DataType.LONG))), Source.EMPTY ); diff --git a/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupJoinTypesIT.java b/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupJoinTypesIT.java index 264c90da63144..df021f27a31fe 100644 --- a/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupJoinTypesIT.java +++ b/x-pack/plugin/esql/src/internalClusterTest/java/org/elasticsearch/xpack/esql/action/LookupJoinTypesIT.java @@ -8,6 +8,7 @@ package org.elasticsearch.xpack.esql.action; import org.apache.lucene.tests.util.LuceneTestCase; +import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; import org.elasticsearch.action.index.IndexRequestBuilder; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.index.mapper.extras.MapperExtrasPlugin; @@ -61,7 +62,6 @@ import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.is; -import static org.hamcrest.Matchers.nullValue; /** * This test suite tests the lookup join functionality in ESQL with various data types. @@ -69,60 +69,26 @@ * types being tested on the left of the pair, and then creates that many other lookup indexes, each with a single document containing * exactly two fields: the field to join on, and a field to return. * The assertion is that for valid combinations, the return result should exist, and for invalid combinations an exception should be thrown. - * If no exception is thrown, and no result is returned, our validation rules are not aligned with the internal behaviour (ie. a bug). - * Since the `LOOKUP JOIN` command requires the match field name to be the same between the main index and the lookup index, - * we will have field names that correctly represent the type of the field in the main index, but not the type of the field - * in the lookup index. This can be confusing, but it is important to remember that the field names are not the same as the types. - * For example, if we are testing the pairs (double, double), (double, float), (float, double) and (float, float), - * we will create the following indexes: - *
- *
index_double_double
- *
Index containing a single document with a field of type 'double' like:
- *         {
- *             "field_double": 1.0,  // this is mapped as type 'double'
- *             "other": "value"
- *         }
- *     
- *
index_double_float
- *
Index containing a single document with a field of type 'float' like:
- *         {
- *             "field_double": 1.0,  // this is mapped as type 'float' (a float with the name of the main index field)
- *             "other": "value"
- *         }
- *     
- *
index_float_double
- *
Index containing a single document with a field of type 'double' like:
- *         {
- *             "field_float": 1.0,  // this is mapped as type 'double' (a double with the name of the main index field)
- *             "other": "value"
- *         }
- *     
- *
index_float_float
- *
Index containing single document with a field of type 'float' like:
- *         {
- *             "field_float": 1.0,  // this is mapped as type 'float'
- *             "other": "value"
- *         }
- *     
- *
index
- *
Index containing document like:
- *         {
- *             "field_double": 1.0,  // this is mapped as type 'double'
- *             "field_float": 1.0    // this is mapped as type 'float'
- *         }
- *     
- *
- * Note that the lookup indexes have fields with a name that matches the type in the main index, and not the type actually used in the - * lookup index. Instead, the mapped type should be the type of the right-hand side of the pair being tested. - * Then we can run queries like: - *
- *     FROM index | LOOKUP JOIN index_double_float ON field_double | KEEP other
- * 
- * And assert that the result exists and is equal to "value". + * If no exception is thrown, and no result is returned, our validation rules are not aligned with the internal behaviour (i.e. a bug). + * Let's assume we want to test a lookup using a byte field in the main index and integer in the lookup index, then we'll create 2 indices, + * named {@code main_index} and {@code lookup_byte_integer} resp. + * The main index contains a field called {@code main_byte} and the lookup index has {@code lookup_integer}. To test the pair, we run + * {@code FROM main_index | RENAME main_byte AS lookup_integer | LOOKUP JOIN lookup_index ON lookup_integer | KEEP other} + * and assert that the result exists and is equal to "value". + * For tests using union types, the same applies but there are additional main indices so that we have actual mapping conflicts. + * E.g. the field {@code main_byte} will occur another time in the index {@code main_byte_as_short} when we're testing a byte-short union + * type. */ +// TODO: This suite creates a lot of indices. It should be sufficient to just create 1 main index with 1 field per relevant type and 1 +// lookup index with 1 field per relevant type; only union types require additional main indices so we can have the same field mapped to +// different types. @ClusterScope(scope = SUITE, numClientNodes = 1, numDataNodes = 1) @LuceneTestCase.SuppressFileSystems(value = "HandleLimitFS") public class LookupJoinTypesIT extends ESIntegTestCase { + private static final String MAIN_INDEX_PREFIX = "main_"; + private static final String MAIN_INDEX = MAIN_INDEX_PREFIX + "index"; + private static final String LOOKUP_INDEX_PREFIX = "lookup_"; + protected Collection> nodePlugins() { return List.of( EsqlPlugin.class, @@ -189,6 +155,17 @@ protected Collection> nodePlugins() { } } + // Union types; non-exhaustive and can be extended + { + TestConfigs configs = testConfigurations.computeIfAbsent("union-types", TestConfigs::new); + configs.addUnionTypePasses(SHORT, INTEGER, INTEGER); + configs.addUnionTypePasses(BYTE, DOUBLE, LONG); + configs.addUnionTypePasses(DATETIME, DATE_NANOS, DATE_NANOS); + configs.addUnionTypePasses(DATE_NANOS, DATETIME, DATETIME); + configs.addUnionTypePasses(SCALED_FLOAT, HALF_FLOAT, DOUBLE); + configs.addUnionTypePasses(TEXT, KEYWORD, KEYWORD); + } + // Tests for all unsupported types DataType[] unsupported = Join.UNSUPPORTED_TYPES; { @@ -270,20 +247,21 @@ protected Collection> nodePlugins() { } } - // Make sure we have never added two configurations with the same index name + // Make sure we have never added two configurations with the same lookup index name. + // This prevents accidentally adding the same test config to two different groups. Set knownTypes = new HashSet<>(); for (TestConfigs configs : testConfigurations.values()) { for (TestConfig config : configs.configs.values()) { - if (knownTypes.contains(config.indexName())) { - throw new IllegalArgumentException("Duplicate index name: " + config.indexName()); + if (knownTypes.contains(config.lookupIndexName())) { + throw new IllegalArgumentException("Duplicate lookup index name: " + config.lookupIndexName()); } - knownTypes.add(config.indexName()); + knownTypes.add(config.lookupIndexName()); } } } private static boolean existingIndex(Collection existing, DataType mainType, DataType lookupType) { - String indexName = "index_" + mainType.esType() + "_" + lookupType.esType(); + String indexName = LOOKUP_INDEX_PREFIX + mainType.esType() + "_" + lookupType.esType(); return existing.stream().anyMatch(c -> c.exists(indexName)); } @@ -319,79 +297,50 @@ public void testLookupJoinOthers() { testLookupJoinTypes("others"); } + public void testLookupJoinUnionTypes() { + testLookupJoinTypes("union-types"); + } + private void testLookupJoinTypes(String group) { - initIndexes(group); - initData(group); - for (TestConfig config : testConfigurations.get(group).configs.values()) { + TestConfigs configs = testConfigurations.get(group); + initIndexes(configs); + initData(configs); + for (TestConfig config : configs.values()) { if ((isValidDataType(config.mainType()) && isValidDataType(config.lookupType())) == false) { continue; } - String query = String.format( - Locale.ROOT, - "FROM index | LOOKUP JOIN %s ON %s | KEEP other", - config.indexName(), - config.fieldName() - ); config.validateMainIndex(); config.validateLookupIndex(); - config.testQuery(query); + config.validateAdditionalMainIndex(); + + config.doTest(); } } - private void initIndexes(String group) { - Collection configs = testConfigurations.get(group).configs.values(); - String propertyPrefix = "{\n \"properties\" : {\n"; - String propertySuffix = " }\n}\n"; - // The main index will have many fields, one of each type to use in later type specific joins - String mainFields = propertyPrefix + configs.stream() - .map(TestConfig::mainPropertySpec) - .distinct() - .collect(Collectors.joining(",\n ")) + propertySuffix; - assertAcked(prepareCreate("index").setMapping(mainFields)); - - Settings.Builder settings = Settings.builder() - .put("index.number_of_shards", 1) - .put("index.number_of_replicas", 0) - .put("index.mode", "lookup"); - configs.forEach( - // Each lookup index will get a document with a field to join on, and a results field to get back - (c) -> assertAcked( - prepareCreate(c.indexName()).setSettings(settings.build()) - .setMapping(propertyPrefix + c.lookupPropertySpec() + propertySuffix) - ) - ); + private void initIndexes(TestConfigs configs) { + for (TestMapping mapping : configs.indices()) { + CreateIndexRequestBuilder builder = prepareCreate(mapping.indexName).setMapping(mapping.propertiesAsJson()); + if (mapping.settings != null) { + builder = builder.setSettings(mapping.settings); + } + + assertAcked(builder); + } } - private void initData(String group) { - Collection configs = testConfigurations.get(group).configs.values(); - int docId = 0; - List indexRequests = new ArrayList<>(configs.size()); - for (TestConfig config : configs) { - String doc = String.format(Locale.ROOT, """ - { - %s, - "other": "value" - } - """, lookupPropertyFor(config)); - var indexRequest = client().prepareIndex().setIndex(config.indexName()).setId("" + (++docId)).setSource(doc, XContentType.JSON); + private void initData(TestConfigs configs) { + List docs = configs.docs(); + List indexRequests = new ArrayList<>(docs.size()); + + for (TestDocument doc : docs) { + var indexRequest = client().prepareIndex().setIndex(doc.indexName()).setId(doc.id).setSource(doc.source, XContentType.JSON); indexRequests.add(indexRequest); } indexRandom(true, indexRequests); - List mainProperties = configs.stream().map(this::mainPropertyFor).distinct().collect(Collectors.toList()); - index("index", "1", String.format(Locale.ROOT, """ - { - %s - } - """, String.join(",\n ", mainProperties))); - refresh("index"); - } - - private String lookupPropertyFor(TestConfig config) { - return String.format(Locale.ROOT, "\"%s\": %s", config.fieldName(), sampleDataTextFor(config.lookupType())); } - private String mainPropertyFor(TestConfig config) { - return String.format(Locale.ROOT, "\"%s\": %s", config.fieldName(), sampleDataTextFor(config.mainType())); + private static String propertyFor(String fieldName, DataType type) { + return String.format(Locale.ROOT, "\"%s\": %s", fieldName, sampleDataTextFor(type)); } private static String sampleDataTextFor(DataType type) { @@ -426,6 +375,40 @@ private static Object sampleDataFor(DataType type) { }; } + public record TestMapping(String indexName, Collection properties, Settings settings) { + + private static final String PROPERTY_PREFIX = "{\n \"properties\" : {\n"; + private static final String PROPERTY_SUFFIX = " }\n}\n"; + + /** + * {@link TestMapping#indexName} and {@link TestMapping#settings} should be the same across the collection, otherwise they're + * obtained from an arbitrary element. + */ + public static TestMapping mergeProperties(Collection mappings) { + TestMapping lastMapping = null; + + Set properties = new HashSet<>(); + for (TestMapping mapping : mappings) { + properties.addAll(mapping.properties); + lastMapping = mapping; + } + String indexName = lastMapping == null ? null : lastMapping.indexName; + Settings settings = lastMapping == null ? null : lastMapping.settings; + + return new TestMapping(indexName, properties, settings); + } + + public static String propertiesAsJson(Collection properties) { + return PROPERTY_PREFIX + String.join(", ", properties) + PROPERTY_SUFFIX; + } + + public String propertiesAsJson() { + return propertiesAsJson(properties); + } + }; + + private record TestDocument(String indexName, String id, String source) {}; + private static class TestConfigs { final String group; final Map configs; @@ -435,23 +418,98 @@ private static class TestConfigs { this.configs = new LinkedHashMap<>(); } + public List indices() { + List results = new ArrayList<>(); + + // The main index will have many fields, one of each type to use in later type specific joins + List mainIndices = new ArrayList<>(); + for (TestConfig config : configs.values()) { + mainIndices.add(config.mainIndex()); + TestMapping otherIndex = config.additionalMainIndex(); + if (otherIndex != null) { + results.add(otherIndex); + } + } + TestMapping mainIndex = TestMapping.mergeProperties(mainIndices); + + results.add(mainIndex); + + configs.values() + .forEach( + // Each lookup index will get a document with a field to join on, and a results field to get back + (c) -> results.add(c.lookupIndex()) + ); + + return results; + } + + public List docs() { + List results = new ArrayList<>(); + + int docId = 0; + for (TestConfig config : configs.values()) { + String doc = String.format(Locale.ROOT, """ + { + %s, + "other": "value" + } + """, propertyFor(config.lookupFieldName(), config.lookupType())); + results.add(new TestDocument(config.lookupIndexName(), "" + (++docId), doc)); + } + + List mainProperties = configs.values() + .stream() + .map(c -> propertyFor(c.mainFieldName(), c.mainType())) + .distinct() + .collect(Collectors.toList()); + results.add(new TestDocument(MAIN_INDEX, "1", String.format(Locale.ROOT, """ + { + %s + } + """, String.join(",\n ", mainProperties)))); + + for (TestConfig config : configs.values()) { + TestMapping additionalIndex = config.additionalMainIndex(); + if (additionalIndex != null) { + String doc = String.format(Locale.ROOT, """ + { + %s + } + """, propertyFor(config.mainFieldName(), ((TestConfigPassesUnionType) config).otherMainType())); + // TODO: Casting to TestConfigPassesUnionType is an ugly hack; better to derive the test data from the TestMapping or + // from the TestConfig. + results.add(new TestDocument(additionalIndex.indexName, "1", doc)); + } + } + + return results; + } + + private Collection values() { + return configs.values(); + } + private boolean exists(String indexName) { return configs.containsKey(indexName); } private void add(TestConfig config) { - if (configs.containsKey(config.indexName())) { - throw new IllegalArgumentException("Duplicate index name: " + config.indexName()); + if (configs.containsKey(config.lookupIndexName())) { + throw new IllegalArgumentException("Duplicate lookup index name: " + config.lookupIndexName()); } - configs.put(config.indexName(), config); + configs.put(config.lookupIndexName(), config); } private void addPasses(DataType mainType, DataType lookupType) { - add(new TestConfigPasses(mainType, lookupType, true)); + add(new TestConfigPasses(mainType, lookupType)); + } + + private void addUnionTypePasses(DataType mainType, DataType otherMainType, DataType lookupType) { + add(new TestConfigPassesUnionType(mainType, otherMainType, lookupType)); } private void addFails(DataType mainType, DataType lookupType) { - String fieldName = "field_" + mainType.esType(); + String fieldName = LOOKUP_INDEX_PREFIX + lookupType.esType(); String errorMessage = String.format( Locale.ROOT, "JOIN left field [%s] of type [%s] is incompatible with right field [%s] of type [%s]", @@ -471,7 +529,7 @@ private void addFails(DataType mainType, DataType lookupType) { } private void addFailsUnsupported(DataType mainType, DataType lookupType) { - String fieldName = "field_" + mainType.esType(); + String fieldName = "lookup_" + lookupType.esType(); String errorMessage = String.format( Locale.ROOT, "JOIN with right field [%s] of type [%s] is not supported", @@ -494,36 +552,75 @@ interface TestConfig { DataType lookupType(); - default String indexName() { - return "index_" + mainType().esType() + "_" + lookupType().esType(); + default TestMapping mainIndex() { + return new TestMapping(MAIN_INDEX, List.of(propertySpecFor(mainFieldName(), mainType())), null); } - default String fieldName() { - return "field_" + mainType().esType(); + /** Make sure the left index has the expected fields and types */ + default void validateMainIndex() { + validateIndex(MAIN_INDEX, mainFieldName(), sampleDataFor(mainType())); } - default String mainPropertySpec() { - return propertySpecFor(fieldName(), mainType(), ""); + /** + * The same across main indices (necessary for union types). + */ + default String mainFieldName() { + return MAIN_INDEX_PREFIX + mainType().esType(); } - default String lookupPropertySpec() { - return propertySpecFor(fieldName(), lookupType(), ", \"other\": { \"type\" : \"keyword\" }"); + /** + * Used for union types. Will have the same main field name, but using a different type. + */ + default TestMapping additionalMainIndex() { + return null; } - /** Make sure the left index has the expected fields and types */ - default void validateMainIndex() { - validateIndex("index", fieldName(), sampleDataFor(mainType())); + /** Make sure the additional indexes have the expected fields and types */ + default void validateAdditionalMainIndex() { + return; + } + + default String lookupIndexName() { + return LOOKUP_INDEX_PREFIX + mainType().esType() + "_" + lookupType().esType(); + } + + default TestMapping lookupIndex() { + return new TestMapping( + lookupIndexName(), + List.of(propertySpecFor(lookupFieldName(), lookupType()), "\"other\": { \"type\" : \"keyword\" }"), + Settings.builder().put("index.number_of_shards", 1).put("index.number_of_replicas", 0).put("index.mode", "lookup").build() + ); } /** Make sure the lookup index has the expected fields and types */ default void validateLookupIndex() { - validateIndex(indexName(), fieldName(), sampleDataFor(lookupType())); + validateIndex(lookupIndexName(), lookupFieldName(), sampleDataFor(lookupType())); } - void testQuery(String query); + default String lookupFieldName() { + return LOOKUP_INDEX_PREFIX + lookupType().esType(); + } + + default String testQuery() { + String mainField = mainFieldName(); + String lookupField = lookupFieldName(); + String lookupIndex = lookupIndexName(); + + return String.format( + Locale.ROOT, + "FROM %s | RENAME %s AS %s | LOOKUP JOIN %s ON %s | KEEP other", + MAIN_INDEX, + mainField, + lookupField, + lookupIndex, + lookupField + ); + } + + void doTest(); } - private static String propertySpecFor(String fieldName, DataType type, String extra) { + private static String propertySpecFor(String fieldName, DataType type) { if (type == SCALED_FLOAT) { return String.format( Locale.ROOT, @@ -531,9 +628,9 @@ private static String propertySpecFor(String fieldName, DataType type, String ex fieldName, type.esType(), SCALING_FACTOR - ) + extra; + ); } - return String.format(Locale.ROOT, "\"%s\": { \"type\" : \"%s\" }", fieldName, type.esType().replaceAll("cartesian_", "")) + extra; + return String.format(Locale.ROOT, "\"%s\": { \"type\" : \"%s\" }", fieldName, type.esType().replaceAll("cartesian_", "")); } private static void validateIndex(String indexName, String fieldName, Object expectedValue) { @@ -548,27 +645,91 @@ private static void validateIndex(String indexName, String fieldName, Object exp } } - private record TestConfigPasses(DataType mainType, DataType lookupType, boolean hasResults) implements TestConfig { + /** + * Test case for a pair of types that can successfully be used in {@code LOOKUP JOIN}. + */ + private record TestConfigPasses(DataType mainType, DataType lookupType) implements TestConfig { @Override - public void testQuery(String query) { + public void doTest() { + String query = testQuery(); try (var response = EsqlQueryRequestBuilder.newRequestBuilder(client()).query(query).get()) { Iterator results = response.response().column(0).iterator(); assertTrue("Expected at least one result for query: " + query, results.hasNext()); Object indexedResult = response.response().column(0).iterator().next(); - if (hasResults) { - assertThat("Expected valid result: " + query, indexedResult, equalTo("value")); - } else { - assertThat("Expected empty results for query: " + query, indexedResult, is(nullValue())); - } + assertThat("Expected valid result: " + query, indexedResult, equalTo("value")); + } + } + } + + /** + * Test case for a {@code LOOKUP JOIN} where a field with a mapping conflict is cast to the type of the lookup field. + */ + private record TestConfigPassesUnionType(DataType mainType, DataType otherMainType, DataType lookupType) implements TestConfig { + @Override + public String lookupIndexName() { + // Override so it doesn't clash with other lookup indices from non-union type tests. + return LOOKUP_INDEX_PREFIX + mainType().esType() + "_union_" + otherMainType().esType() + "_" + lookupType().esType(); + } + + private String additionalIndexName() { + return mainFieldName() + "_as_" + otherMainType().typeName(); + } + + @Override + public TestMapping additionalMainIndex() { + return new TestMapping(additionalIndexName(), List.of(propertySpecFor(mainFieldName(), otherMainType)), null); + } + + @Override + public void validateAdditionalMainIndex() { + validateIndex(additionalIndexName(), mainFieldName(), sampleDataFor(otherMainType)); + } + + @Override + public String testQuery() { + String mainField = mainFieldName(); + String lookupField = lookupFieldName(); + String lookupIndex = lookupIndexName(); + + return String.format( + Locale.ROOT, + "FROM %s, %s | EVAL %s = %s::%s | LOOKUP JOIN %s ON %s | KEEP other", + MAIN_INDEX, + additionalIndexName(), + lookupField, + mainField, + lookupType.typeName(), + lookupIndex, + lookupField + ); + } + + @Override + public void doTest() { + String query = testQuery(); + try (var response = EsqlQueryRequestBuilder.newRequestBuilder(client()).query(query).get()) { + Iterator results = response.response().column(0).iterator(); + + assertTrue("Expected at least two results for query, but result was empty: " + query, results.hasNext()); + Object indexedResult = results.next(); + assertThat("Expected valid result: " + query, indexedResult, equalTo("value")); + + assertTrue("Expected at least two results for query: " + query, results.hasNext()); + indexedResult = results.next(); + assertThat("Expected valid result: " + query, indexedResult, equalTo("value")); } } } + /** + * Test case for a pair of types that generate an error message when used in {@code LOOKUP JOIN}. + */ private record TestConfigFails(DataType mainType, DataType lookupType, Class exception, Consumer assertion) implements TestConfig { @Override - public void testQuery(String query) { + public void doTest() { + String query = testQuery(); E e = expectThrows( exception(), "Expected exception " + exception().getSimpleName() + " but no exception was thrown: " + query, diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/AbstractLookupService.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/AbstractLookupService.java index a963d6d738918..ea78252197e35 100644 --- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/AbstractLookupService.java +++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/AbstractLookupService.java @@ -71,6 +71,7 @@ import org.elasticsearch.transport.TransportService; import org.elasticsearch.xpack.esql.EsqlIllegalArgumentException; import org.elasticsearch.xpack.esql.core.expression.Alias; +import org.elasticsearch.xpack.esql.core.expression.FieldAttribute; import org.elasticsearch.xpack.esql.core.expression.NamedExpression; import org.elasticsearch.xpack.esql.core.tree.Source; import org.elasticsearch.xpack.esql.core.type.DataType; @@ -415,8 +416,13 @@ private static Operator extractFieldsOperator( ) { List fields = new ArrayList<>(extractFields.size()); for (NamedExpression extractField : extractFields) { + String fieldName = extractField instanceof FieldAttribute fa ? fa.fieldName().string() + // Cases for Alias and ReferenceAttribute: only required for ENRICH (Alias in case of ENRICH ... WITH x = field) + // (LOOKUP JOIN uses FieldAttributes) + : extractField instanceof Alias a ? ((NamedExpression) a.child()).name() + : extractField.name(); BlockLoader loader = shardContext.blockLoader( - extractField instanceof Alias a ? ((NamedExpression) a.child()).name() : extractField.name(), + fieldName, extractField.dataType() == DataType.UNSUPPORTED, MappedFieldType.FieldExtractPreference.NONE ); diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperator.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperator.java index b3a98c37ca37c..2b41edcbeff0b 100644 --- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperator.java +++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperator.java @@ -23,6 +23,7 @@ import org.elasticsearch.core.Releasables; import org.elasticsearch.tasks.CancellableTask; import org.elasticsearch.xcontent.XContentBuilder; +import org.elasticsearch.xpack.esql.core.expression.FieldAttribute; import org.elasticsearch.xpack.esql.core.expression.NamedExpression; import org.elasticsearch.xpack.esql.core.tree.Source; import org.elasticsearch.xpack.esql.core.type.DataType; @@ -45,7 +46,7 @@ public record Factory( DataType inputDataType, String lookupIndexPattern, String lookupIndex, - String matchField, + FieldAttribute.FieldName matchField, List loadFields, Source source ) implements OperatorFactory { @@ -56,7 +57,7 @@ public String describe() { + " input_type=" + inputDataType + " match_field=" - + matchField + + matchField.string() + " load_fields=" + loadFields + " inputChannel=" @@ -76,7 +77,7 @@ public Operator get(DriverContext driverContext) { inputDataType, lookupIndexPattern, lookupIndex, - matchField, + matchField.string(), loadFields, source ); diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/fulltext/FullTextFunction.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/fulltext/FullTextFunction.java index b7bf29276511c..5510fbddd27c7 100644 --- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/fulltext/FullTextFunction.java +++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/fulltext/FullTextFunction.java @@ -396,6 +396,8 @@ protected Map resolvedOptions() throws InvalidArgumentException return Map.of(); } + // TODO: this should likely be replaced by calls to FieldAttribute#fieldName; the MultiTypeEsField case looks + // wrong if `fieldAttribute` is a subfield, e.g. `parent.child` - multiTypeEsField#getName will just return `child`. public static String getNameFromFieldAttribute(FieldAttribute fieldAttribute) { String fieldName = fieldAttribute.name(); if (fieldAttribute.field() instanceof MultiTypeEsField multiTypeEsField) { diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/plan/logical/join/Join.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/plan/logical/join/Join.java index f939dfff5be51..59fdc902273b9 100644 --- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/plan/logical/join/Join.java +++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/plan/logical/join/Join.java @@ -158,6 +158,9 @@ public AttributeSet rightReferences() { return Expressions.references(config().rightFields()); } + /** + * The output fields obtained from the right child. + */ public List rightOutputFields() { AttributeSet leftInputs = left().outputSet(); diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/planner/LocalExecutionPlanner.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/planner/LocalExecutionPlanner.java index db2862234c136..83d071361ecfe 100644 --- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/planner/LocalExecutionPlanner.java +++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/planner/LocalExecutionPlanner.java @@ -766,10 +766,11 @@ private PhysicalOperation planLookupJoin(LookupJoinExec join, LocalExecutionPlan ); } - private record MatchConfig(String fieldName, int channel, DataType type) { + private record MatchConfig(FieldAttribute.FieldName fieldName, int channel, DataType type) { private MatchConfig(FieldAttribute match, Layout.ChannelAndType input) { - // Note, this handles TEXT fields with KEYWORD subfields - this(match.exactAttribute().name(), input.channel(), input.type()); + // TODO: Using exactAttribute was supposed to handle TEXT fields with KEYWORD subfields - but we don't allow these in lookup + // indices, so the call to exactAttribute looks redundant now. + this(match.exactAttribute().fieldName(), input.channel(), input.type()); } } diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperatorTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperatorTests.java index badc164f24722..dabcc6cbce895 100644 --- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperatorTests.java +++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/enrich/LookupFromIndexOperatorTests.java @@ -56,6 +56,7 @@ import org.elasticsearch.threadpool.TestThreadPool; import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.transport.TransportService; +import org.elasticsearch.xpack.esql.core.expression.FieldAttribute; import org.elasticsearch.xpack.esql.core.expression.NamedExpression; import org.elasticsearch.xpack.esql.core.expression.ReferenceAttribute; import org.elasticsearch.xpack.esql.core.tree.Source; @@ -134,7 +135,7 @@ protected Operator.OperatorFactory simple(SimpleOptions options) { int inputChannel = 0; DataType inputDataType = DataType.LONG; String lookupIndex = "idx"; - String matchField = "match"; + FieldAttribute.FieldName matchField = new FieldAttribute.FieldName("match"); List loadFields = List.of( new ReferenceAttribute(Source.EMPTY, "lkwd", DataType.KEYWORD), new ReferenceAttribute(Source.EMPTY, "lint", DataType.INTEGER)