From c5da25754007af51029a676a8784b16be16d8995 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Iv=C3=A1n=20Cea=20Fontenla?=
Date: Thu, 1 Aug 2024 14:16:26 +0200
Subject: [PATCH] Add test cases for nulls and wrong types to aggregation tests
(#111482)
- Migrated the anyNullIsNull and wrong types cases to `AbstractFunctionTestCase`
- Minor fixes on anyNullIsNull to work with multi-row values. Just some conditions to return a List of null instead of a null. Everything else in these functions was mostly untouched
- Implemented it in some aggregations
- Fixed some errors around the aggregation tests code
Not all aggregations were migrated. Many of them have edge cases that don't work with some of those things.
For example, if `WEIGHTED_AVG(value, weight)` has a literal on the value, it ignores the weight, which makes anyNullIsNull fail as it expects a null return.
Such cases can be handled later.
Closes https://github.com/elastic/elasticsearch/issues/109917
---
.../expression/function/aggregate/Max.java | 2 +-
.../expression/function/aggregate/Min.java | 2 +-
.../expression/function/aggregate/Values.java | 11 +-
.../xpack/esql/analysis/AnalyzerTests.java | 8 +-
.../xpack/esql/analysis/VerifierTests.java | 4 +-
.../function/AbstractAggregationTestCase.java | 67 ++--
.../function/AbstractFunctionTestCase.java | 330 ++++++++++++++++++
.../AbstractScalarFunctionTestCase.java | 327 -----------------
.../expression/function/TestCaseSupplier.java | 4 +-
.../function/aggregate/AvgTests.java | 2 +-
.../function/aggregate/MaxTests.java | 73 +---
.../function/aggregate/MinTests.java | 73 +---
.../function/aggregate/ValuesTests.java | 6 +-
.../function/aggregate/WeightedAvgTests.java | 18 +-
14 files changed, 417 insertions(+), 510 deletions(-)
diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Max.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Max.java
index 4438ccec04c4c..22224628e23ad 100644
--- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Max.java
+++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Max.java
@@ -76,7 +76,7 @@ public Max replaceChildren(List newChildren) {
@Override
protected TypeResolution resolveType() {
return TypeResolutions.isType(
- this,
+ field(),
e -> e == DataType.BOOLEAN || e == DataType.DATETIME || e == DataType.IP || (e.isNumeric() && e != DataType.UNSIGNED_LONG),
sourceText(),
DEFAULT,
diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Min.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Min.java
index 490d227206e06..8e7bb6bc3e799 100644
--- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Min.java
+++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Min.java
@@ -76,7 +76,7 @@ public Min replaceChildren(List newChildren) {
@Override
protected TypeResolution resolveType() {
return TypeResolutions.isType(
- this,
+ field(),
e -> e == DataType.BOOLEAN || e == DataType.DATETIME || e == DataType.IP || (e.isNumeric() && e != DataType.UNSIGNED_LONG),
sourceText(),
DEFAULT,
diff --git a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Values.java b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Values.java
index 79276b26be6d5..136e1233601f9 100644
--- a/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Values.java
+++ b/x-pack/plugin/esql/src/main/java/org/elasticsearch/xpack/esql/expression/function/aggregate/Values.java
@@ -17,10 +17,10 @@
import org.elasticsearch.compute.aggregation.ValuesLongAggregatorFunctionSupplier;
import org.elasticsearch.xpack.esql.EsqlIllegalArgumentException;
import org.elasticsearch.xpack.esql.core.expression.Expression;
+import org.elasticsearch.xpack.esql.core.expression.TypeResolutions;
import org.elasticsearch.xpack.esql.core.tree.NodeInfo;
import org.elasticsearch.xpack.esql.core.tree.Source;
import org.elasticsearch.xpack.esql.core.type.DataType;
-import org.elasticsearch.xpack.esql.expression.EsqlTypeResolutions;
import org.elasticsearch.xpack.esql.expression.function.Example;
import org.elasticsearch.xpack.esql.expression.function.FunctionInfo;
import org.elasticsearch.xpack.esql.expression.function.Param;
@@ -30,6 +30,7 @@
import java.util.List;
import static org.elasticsearch.xpack.esql.core.expression.TypeResolutions.ParamOrdinal.DEFAULT;
+import static org.elasticsearch.xpack.esql.core.type.DataType.UNSIGNED_LONG;
public class Values extends AggregateFunction implements ToAggregator {
public static final NamedWriteableRegistry.Entry ENTRY = new NamedWriteableRegistry.Entry(Expression.class, "Values", Values::new);
@@ -84,7 +85,13 @@ public DataType dataType() {
@Override
protected TypeResolution resolveType() {
- return EsqlTypeResolutions.isNotSpatial(field(), sourceText(), DEFAULT);
+ return TypeResolutions.isType(
+ field(),
+ dt -> DataType.isSpatial(dt) == false && dt != UNSIGNED_LONG,
+ sourceText(),
+ DEFAULT,
+ "any type except unsigned_long and spatial types"
+ );
}
@Override
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/AnalyzerTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/AnalyzerTests.java
index 7333bd0e9f8a6..f0dd72e18ac2f 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/AnalyzerTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/AnalyzerTests.java
@@ -1834,13 +1834,13 @@ public void testUnsupportedTypesInStats() {
line 2:20: argument of [count_distinct(x)] must be [any exact type except unsigned_long, _source, or counter types],\
found value [x] type [unsigned_long]
line 2:39: argument of [max(x)] must be [boolean, datetime, ip or numeric except unsigned_long or counter types],\
- found value [max(x)] type [unsigned_long]
+ found value [x] type [unsigned_long]
line 2:47: argument of [median(x)] must be [numeric except unsigned_long or counter types],\
found value [x] type [unsigned_long]
line 2:58: argument of [median_absolute_deviation(x)] must be [numeric except unsigned_long or counter types],\
found value [x] type [unsigned_long]
line 2:88: argument of [min(x)] must be [boolean, datetime, ip or numeric except unsigned_long or counter types],\
- found value [min(x)] type [unsigned_long]
+ found value [x] type [unsigned_long]
line 2:96: first argument of [percentile(x, 10)] must be [numeric except unsigned_long],\
found value [x] type [unsigned_long]
line 2:115: argument of [sum(x)] must be [numeric except unsigned_long or counter types],\
@@ -1854,13 +1854,13 @@ public void testUnsupportedTypesInStats() {
line 2:10: argument of [avg(x)] must be [numeric except unsigned_long or counter types],\
found value [x] type [version]
line 2:18: argument of [max(x)] must be [boolean, datetime, ip or numeric except unsigned_long or counter types],\
- found value [max(x)] type [version]
+ found value [x] type [version]
line 2:26: argument of [median(x)] must be [numeric except unsigned_long or counter types],\
found value [x] type [version]
line 2:37: argument of [median_absolute_deviation(x)] must be [numeric except unsigned_long or counter types],\
found value [x] type [version]
line 2:67: argument of [min(x)] must be [boolean, datetime, ip or numeric except unsigned_long or counter types],\
- found value [min(x)] type [version]
+ found value [x] type [version]
line 2:75: first argument of [percentile(x, 10)] must be [numeric except unsigned_long], found value [x] type [version]
line 2:94: argument of [sum(x)] must be [numeric except unsigned_long or counter types], found value [x] type [version]""");
}
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/VerifierTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/VerifierTests.java
index 08b1ef9f6fef6..49372da04d8c3 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/VerifierTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/analysis/VerifierTests.java
@@ -494,7 +494,7 @@ public void testAggregateOnCounter() {
equalTo(
"1:20: argument of [min(network.bytes_in)] must be"
+ " [boolean, datetime, ip or numeric except unsigned_long or counter types],"
- + " found value [min(network.bytes_in)] type [counter_long]"
+ + " found value [network.bytes_in] type [counter_long]"
)
);
@@ -503,7 +503,7 @@ public void testAggregateOnCounter() {
equalTo(
"1:20: argument of [max(network.bytes_in)] must be"
+ " [boolean, datetime, ip or numeric except unsigned_long or counter types],"
- + " found value [max(network.bytes_in)] type [counter_long]"
+ + " found value [network.bytes_in] type [counter_long]"
)
);
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractAggregationTestCase.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractAggregationTestCase.java
index 25ff4f9c2122d..65425486ea4e0 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractAggregationTestCase.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractAggregationTestCase.java
@@ -57,6 +57,25 @@ public abstract class AbstractAggregationTestCase extends AbstractFunctionTestCa
* Use if possible, as this method may get updated with new checks in the future.
*
*/
+ protected static Iterable parameterSuppliersFromTypedDataWithDefaultChecks(
+ List suppliers,
+ boolean entirelyNullPreservesType,
+ PositionalErrorMessageSupplier positionalErrorMessageSupplier
+ ) {
+ return parameterSuppliersFromTypedData(
+ errorsForCasesWithoutExamples(
+ withNoRowsExpectingNull(anyNullIsNull(entirelyNullPreservesType, randomizeBytesRefsOffset(suppliers))),
+ positionalErrorMessageSupplier
+ )
+ );
+ }
+
+ // TODO: Remove and migrate everything to the method with all the parameters
+ /**
+ * @deprecated Use {@link #parameterSuppliersFromTypedDataWithDefaultChecks(List, boolean, PositionalErrorMessageSupplier)} instead.
+ * This method doesn't add all the default checks.
+ */
+ @Deprecated
protected static Iterable parameterSuppliersFromTypedDataWithDefaultChecks(List suppliers) {
return parameterSuppliersFromTypedData(withNoRowsExpectingNull(randomizeBytesRefsOffset(suppliers)));
}
@@ -119,24 +138,9 @@ public void testFold() {
Expression expression = buildLiteralExpression(testCase);
resolveExpression(expression, aggregatorFunctionSupplier -> {
- // An aggregation cannot be folded
- }, evaluableExpression -> {
- assertTrue(evaluableExpression.foldable());
- if (testCase.foldingExceptionClass() == null) {
- Object result = evaluableExpression.fold();
- // Decode unsigned longs into BigIntegers
- if (testCase.expectedType() == DataType.UNSIGNED_LONG && result != null) {
- result = NumericUtils.unsignedLongAsBigInteger((Long) result);
- }
- assertThat(result, testCase.getMatcher());
- if (testCase.getExpectedWarnings() != null) {
- assertWarnings(testCase.getExpectedWarnings());
- }
- } else {
- Throwable t = expectThrows(testCase.foldingExceptionClass(), evaluableExpression::fold);
- assertThat(t.getMessage(), equalTo(testCase.foldingExceptionMessage()));
- }
- });
+ // An aggregation cannot be folded.
+ // It's not an error either as not all aggregations are foldable.
+ }, this::evaluate);
}
private void aggregateSingleMode(Expression expression) {
@@ -263,13 +267,19 @@ private void aggregateWithIntermediates(Expression expression) {
}
private void evaluate(Expression evaluableExpression) {
- Object result;
- try (var evaluator = evaluator(evaluableExpression).get(driverContext())) {
- try (Block block = evaluator.eval(row(testCase.getDataValues()))) {
- result = toJavaObjectUnsignedLongAware(block, 0);
- }
+ assertTrue(evaluableExpression.foldable());
+
+ if (testCase.foldingExceptionClass() != null) {
+ Throwable t = expectThrows(testCase.foldingExceptionClass(), evaluableExpression::fold);
+ assertThat(t.getMessage(), equalTo(testCase.foldingExceptionMessage()));
+ return;
}
+ Object result = evaluableExpression.fold();
+ // Decode unsigned longs into BigIntegers
+ if (testCase.expectedType() == DataType.UNSIGNED_LONG && result != null) {
+ result = NumericUtils.unsignedLongAsBigInteger((Long) result);
+ }
assertThat(result, not(equalTo(Double.NaN)));
assert testCase.getMatcher().matches(Double.POSITIVE_INFINITY) == false;
assertThat(result, not(equalTo(Double.POSITIVE_INFINITY)));
@@ -435,16 +445,23 @@ private IntBlock makeGroupsVector(int groupStart, int groupEnd, int rowCount) {
*/
private void processPageGrouping(GroupingAggregator aggregator, Page inputPage, int groupCount) {
var groupSliceSize = 1;
+ var allValuesNull = IntStream.range(0, inputPage.getBlockCount())
+ .mapToObj(inputPage::getBlock)
+ .anyMatch(Block::areAllValuesNull);
// Add data to chunks of groups
for (int currentGroupOffset = 0; currentGroupOffset < groupCount;) {
- var seenGroupIds = new SeenGroupIds.Range(0, currentGroupOffset + groupSliceSize);
+ int groupSliceRemainingSize = Math.min(groupSliceSize, groupCount - currentGroupOffset);
+ var seenGroupIds = new SeenGroupIds.Range(0, allValuesNull ? 0 : currentGroupOffset + groupSliceRemainingSize);
var addInput = aggregator.prepareProcessPage(seenGroupIds, inputPage);
var positionCount = inputPage.getPositionCount();
var dataSliceSize = 1;
// Divide data in chunks
for (int currentDataOffset = 0; currentDataOffset < positionCount;) {
- try (var groups = makeGroupsVector(currentGroupOffset, currentGroupOffset + groupSliceSize, dataSliceSize)) {
+ int dataSliceRemainingSize = Math.min(dataSliceSize, positionCount - currentDataOffset);
+ try (
+ var groups = makeGroupsVector(currentGroupOffset, currentGroupOffset + groupSliceRemainingSize, dataSliceRemainingSize)
+ ) {
addInput.add(currentDataOffset, groups);
}
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractFunctionTestCase.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractFunctionTestCase.java
index 20c583d3ac898..0c4bd6fe38b6a 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractFunctionTestCase.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractFunctionTestCase.java
@@ -40,6 +40,7 @@
import org.elasticsearch.xpack.esql.core.expression.Expression;
import org.elasticsearch.xpack.esql.core.expression.FieldAttribute;
import org.elasticsearch.xpack.esql.core.expression.Literal;
+import org.elasticsearch.xpack.esql.core.expression.TypeResolutions;
import org.elasticsearch.xpack.esql.core.expression.predicate.nulls.IsNotNull;
import org.elasticsearch.xpack.esql.core.expression.predicate.nulls.IsNull;
import org.elasticsearch.xpack.esql.core.session.Configuration;
@@ -49,6 +50,8 @@
import org.elasticsearch.xpack.esql.core.util.NumericUtils;
import org.elasticsearch.xpack.esql.core.util.StringUtils;
import org.elasticsearch.xpack.esql.evaluator.EvalMapper;
+import org.elasticsearch.xpack.esql.expression.function.scalar.conditional.Greatest;
+import org.elasticsearch.xpack.esql.expression.function.scalar.nulls.Coalesce;
import org.elasticsearch.xpack.esql.expression.function.scalar.string.RLike;
import org.elasticsearch.xpack.esql.expression.function.scalar.string.WildcardLike;
import org.elasticsearch.xpack.esql.expression.predicate.operator.arithmetic.Add;
@@ -69,6 +72,7 @@
import org.elasticsearch.xpack.esql.planner.Layout;
import org.elasticsearch.xpack.esql.planner.PlannerUtils;
import org.elasticsearch.xpack.versionfield.Version;
+import org.hamcrest.Matcher;
import org.junit.After;
import org.junit.AfterClass;
@@ -95,6 +99,8 @@
import java.util.TreeSet;
import java.util.function.Function;
import java.util.stream.Collectors;
+import java.util.stream.IntStream;
+import java.util.stream.Stream;
import static java.util.Map.entry;
import static org.elasticsearch.compute.data.BlockUtils.toJavaObject;
@@ -106,6 +112,7 @@
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.hasSize;
import static org.hamcrest.Matchers.instanceOf;
+import static org.hamcrest.Matchers.nullValue;
/**
* Base class for function tests.
@@ -191,6 +198,318 @@ protected static Iterable parameterSuppliersFromTypedData(List
+ * Note: This won't add more than a single null to any existing test case,
+ * just to keep the number of test cases from exploding totally.
+ *
+ *
+ * @param entirelyNullPreservesType should a test case that only contains parameters
+ * with the {@code null} type keep it's expected type?
+ * This is mostly going to be {@code true}
+ * except for functions that base their type entirely
+ * on input types like {@link Greatest} or {@link Coalesce}.
+ */
+ protected static List anyNullIsNull(boolean entirelyNullPreservesType, List testCaseSuppliers) {
+ return anyNullIsNull(
+ testCaseSuppliers,
+ (nullPosition, nullValueDataType, original) -> entirelyNullPreservesType == false
+ && nullValueDataType == DataType.NULL
+ && original.getData().size() == 1 ? DataType.NULL : original.expectedType(),
+ (nullPosition, nullData, original) -> original
+ );
+ }
+
+ public interface ExpectedType {
+ DataType expectedType(int nullPosition, DataType nullValueDataType, TestCaseSupplier.TestCase original);
+ }
+
+ public interface ExpectedEvaluatorToString {
+ Matcher evaluatorToString(int nullPosition, TestCaseSupplier.TypedData nullData, Matcher original);
+ }
+
+ protected static List anyNullIsNull(
+ List testCaseSuppliers,
+ ExpectedType expectedType,
+ ExpectedEvaluatorToString evaluatorToString
+ ) {
+ typesRequired(testCaseSuppliers);
+ List suppliers = new ArrayList<>(testCaseSuppliers.size());
+ suppliers.addAll(testCaseSuppliers);
+
+ /*
+ * For each original test case, add as many copies as there were
+ * arguments, replacing one of the arguments with null and keeping
+ * the others.
+ *
+ * Also, if this was the first time we saw the signature we copy it
+ * *again*, replacing the argument with null, but annotating the
+ * argument's type as `null` explicitly.
+ */
+ Set> uniqueSignatures = new HashSet<>();
+ for (TestCaseSupplier original : testCaseSuppliers) {
+ boolean firstTimeSeenSignature = uniqueSignatures.add(original.types());
+ for (int nullPosition = 0; nullPosition < original.types().size(); nullPosition++) {
+ int finalNullPosition = nullPosition;
+ suppliers.add(new TestCaseSupplier(original.name() + " null in " + nullPosition, original.types(), () -> {
+ TestCaseSupplier.TestCase oc = original.get();
+ List data = IntStream.range(0, oc.getData().size()).mapToObj(i -> {
+ TestCaseSupplier.TypedData od = oc.getData().get(i);
+ if (i != finalNullPosition) {
+ return od;
+ }
+ return od.withData(od.isMultiRow() ? Collections.singletonList(null) : null);
+ }).toList();
+ TestCaseSupplier.TypedData nulledData = oc.getData().get(finalNullPosition);
+ return new TestCaseSupplier.TestCase(
+ data,
+ evaluatorToString.evaluatorToString(finalNullPosition, nulledData, oc.evaluatorToString()),
+ expectedType.expectedType(finalNullPosition, nulledData.type(), oc),
+ nullValue(),
+ null,
+ oc.getExpectedTypeError(),
+ null,
+ null
+ );
+ }));
+
+ if (firstTimeSeenSignature) {
+ List typesWithNull = IntStream.range(0, original.types().size())
+ .mapToObj(i -> i == finalNullPosition ? DataType.NULL : original.types().get(i))
+ .toList();
+ boolean newSignature = uniqueSignatures.add(typesWithNull);
+ if (newSignature) {
+ suppliers.add(new TestCaseSupplier(typesWithNull, () -> {
+ TestCaseSupplier.TestCase oc = original.get();
+ List data = IntStream.range(0, oc.getData().size())
+ .mapToObj(
+ i -> i == finalNullPosition
+ ? (oc.getData().get(i).isMultiRow()
+ ? TestCaseSupplier.TypedData.MULTI_ROW_NULL
+ : TestCaseSupplier.TypedData.NULL)
+ : oc.getData().get(i)
+ )
+ .toList();
+ return new TestCaseSupplier.TestCase(
+ data,
+ equalTo("LiteralsEvaluator[lit=null]"),
+ expectedType.expectedType(finalNullPosition, DataType.NULL, oc),
+ nullValue(),
+ null,
+ oc.getExpectedTypeError(),
+ null,
+ null
+ );
+ }));
+ }
+ }
+ }
+ }
+
+ return suppliers;
+ }
+
+ @FunctionalInterface
+ protected interface PositionalErrorMessageSupplier {
+ /**
+ * This interface defines functions to supply error messages for incorrect types in specific positions. Functions which have
+ * the same type requirements for all positions can simplify this with a lambda returning a string constant.
+ *
+ * @param validForPosition - the set of {@link DataType}s that the test infrastructure believes to be allowable in the
+ * given position.
+ * @param position - the zero-index position in the list of parameters the function has detected the bad argument to be.
+ * @return The string describing the acceptable parameters for that position. Note that this function should not return
+ * the full error string; that will be constructed by the test. Just return the type string for that position.
+ */
+ String apply(Set validForPosition, int position);
+ }
+
+ /**
+ * Adds test cases containing unsupported parameter types that assert
+ * that they throw type errors.
+ */
+ protected static List errorsForCasesWithoutExamples(
+ List testCaseSuppliers,
+ PositionalErrorMessageSupplier positionalErrorMessageSupplier
+ ) {
+ return errorsForCasesWithoutExamples(testCaseSuppliers, (i, v, t) -> typeErrorMessage(i, v, t, positionalErrorMessageSupplier));
+ }
+
+ /**
+ * Build the expected error message for an invalid type signature.
+ */
+ protected static String typeErrorMessage(
+ boolean includeOrdinal,
+ List> validPerPosition,
+ List types,
+ PositionalErrorMessageSupplier expectedTypeSupplier
+ ) {
+ int badArgPosition = -1;
+ for (int i = 0; i < types.size(); i++) {
+ if (validPerPosition.get(i).contains(types.get(i)) == false) {
+ badArgPosition = i;
+ break;
+ }
+ }
+ if (badArgPosition == -1) {
+ throw new IllegalStateException(
+ "Can't generate error message for these types, you probably need a custom error message function"
+ );
+ }
+ String ordinal = includeOrdinal ? TypeResolutions.ParamOrdinal.fromIndex(badArgPosition).name().toLowerCase(Locale.ROOT) + " " : "";
+ String expectedTypeString = expectedTypeSupplier.apply(validPerPosition.get(badArgPosition), badArgPosition);
+ String name = types.get(badArgPosition).typeName();
+ return ordinal + "argument of [] must be [" + expectedTypeString + "], found value [" + name + "] type [" + name + "]";
+ }
+
+ @FunctionalInterface
+ protected interface TypeErrorMessageSupplier {
+ String apply(boolean includeOrdinal, List> validPerPosition, List types);
+ }
+
+ protected static List errorsForCasesWithoutExamples(
+ List testCaseSuppliers,
+ TypeErrorMessageSupplier typeErrorMessageSupplier
+ ) {
+ typesRequired(testCaseSuppliers);
+ List suppliers = new ArrayList<>(testCaseSuppliers.size());
+ suppliers.addAll(testCaseSuppliers);
+
+ Set> valid = testCaseSuppliers.stream().map(TestCaseSupplier::types).collect(Collectors.toSet());
+ List> validPerPosition = validPerPosition(valid);
+
+ testCaseSuppliers.stream()
+ .map(s -> s.types().size())
+ .collect(Collectors.toSet())
+ .stream()
+ .flatMap(count -> allPermutations(count))
+ .filter(types -> valid.contains(types) == false)
+ /*
+ * Skip any cases with more than one null. Our tests don't generate
+ * the full combinatorial explosions of all nulls - just a single null.
+ * Hopefully , cases will function the same as ,
+ * cases.
+ */.filter(types -> types.stream().filter(t -> t == DataType.NULL).count() <= 1)
+ .map(types -> typeErrorSupplier(validPerPosition.size() != 1, validPerPosition, types, typeErrorMessageSupplier))
+ .forEach(suppliers::add);
+ return suppliers;
+ }
+
+ private static List append(List orig, DataType extra) {
+ List longer = new ArrayList<>(orig.size() + 1);
+ longer.addAll(orig);
+ longer.add(extra);
+ return longer;
+ }
+
+ protected static Stream representable() {
+ return DataType.types().stream().filter(DataType::isRepresentable);
+ }
+
+ protected static TestCaseSupplier typeErrorSupplier(
+ boolean includeOrdinal,
+ List> validPerPosition,
+ List types,
+ PositionalErrorMessageSupplier errorMessageSupplier
+ ) {
+ return typeErrorSupplier(includeOrdinal, validPerPosition, types, (o, v, t) -> typeErrorMessage(o, v, t, errorMessageSupplier));
+ }
+
+ /**
+ * Build a test case that asserts that the combination of parameter types is an error.
+ */
+ protected static TestCaseSupplier typeErrorSupplier(
+ boolean includeOrdinal,
+ List> validPerPosition,
+ List types,
+ TypeErrorMessageSupplier errorMessageSupplier
+ ) {
+ return new TestCaseSupplier(
+ "type error for " + TestCaseSupplier.nameFromTypes(types),
+ types,
+ () -> TestCaseSupplier.TestCase.typeError(
+ types.stream().map(type -> new TestCaseSupplier.TypedData(randomLiteral(type).value(), type, type.typeName())).toList(),
+ errorMessageSupplier.apply(includeOrdinal, validPerPosition, types)
+ )
+ );
+ }
+
+ private static List> validPerPosition(Set> valid) {
+ int max = valid.stream().mapToInt(List::size).max().getAsInt();
+ List> result = new ArrayList<>(max);
+ for (int i = 0; i < max; i++) {
+ result.add(new HashSet<>());
+ }
+ for (List signature : valid) {
+ for (int i = 0; i < signature.size(); i++) {
+ result.get(i).add(signature.get(i));
+ }
+ }
+ return result;
+ }
+
+ protected static Stream> allPermutations(int argumentCount) {
+ if (argumentCount == 0) {
+ return Stream.of(List.of());
+ }
+ if (argumentCount > 3) {
+ throw new IllegalArgumentException("would generate too many combinations");
+ }
+ Stream> stream = validFunctionParameters().map(List::of);
+ for (int i = 1; i < argumentCount; i++) {
+ stream = stream.flatMap(types -> validFunctionParameters().map(t -> append(types, t)));
+ }
+ return stream;
+ }
+
+ /**
+ * The types that are valid in function parameters. This is used by the
+ * function tests to enumerate all possible parameters to test error messages
+ * for invalid combinations.
+ */
+ public static Stream validFunctionParameters() {
+ return Arrays.stream(DataType.values()).filter(t -> {
+ if (t == DataType.UNSUPPORTED) {
+ // By definition, functions never support UNSUPPORTED
+ return false;
+ }
+ if (t == DataType.DOC_DATA_TYPE || t == DataType.PARTIAL_AGG) {
+ /*
+ * Doc and partial_agg are special and functions aren't
+ * defined to take these. They'll use them implicitly if needed.
+ */
+ return false;
+ }
+ if (t == DataType.OBJECT || t == DataType.NESTED) {
+ // Object and nested fields aren't supported by any functions yet
+ return false;
+ }
+ if (t == DataType.SOURCE || t == DataType.TSID_DATA_TYPE) {
+ // No functions take source or tsid fields yet. We'll make some eventually and remove this.
+ return false;
+ }
+ if (t == DataType.DATE_PERIOD || t == DataType.TIME_DURATION) {
+ // We don't test that functions don't take date_period or time_duration. We should.
+ return false;
+ }
+ if (t.isCounter()) {
+ /*
+ * For now, we're assuming no functions take counters
+ * as parameters. That's not true - some do. But we'll
+ * need to update the tests to handle that.
+ */
+ return false;
+ }
+ if (t.widenSmallNumeric() != t) {
+ // Small numeric types are widened long before they arrive at functions.
+ return false;
+ }
+
+ return true;
+ }).sorted();
+ }
+
/**
* Build an {@link Attribute} that loads a field.
*/
@@ -997,6 +1316,17 @@ protected static DataType[] strings() {
return DataType.types().stream().filter(DataType::isString).toArray(DataType[]::new);
}
+ /**
+ * Validate that we know the types for all the test cases already created
+ * @param suppliers - list of suppliers before adding in the illegal type combinations
+ */
+ protected static void typesRequired(List suppliers) {
+ String bad = suppliers.stream().filter(s -> s.types() == null).map(s -> s.name()).collect(Collectors.joining("\n"));
+ if (bad.equals("") == false) {
+ throw new IllegalArgumentException("types required but not found for these tests:\n" + bad);
+ }
+ }
+
/**
* Returns true if the current test case is for an aggregation function.
*
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractScalarFunctionTestCase.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractScalarFunctionTestCase.java
index 1caea78e79ad5..f4123af8abd0a 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractScalarFunctionTestCase.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/AbstractScalarFunctionTestCase.java
@@ -22,21 +22,15 @@
import org.elasticsearch.indices.CrankyCircuitBreakerService;
import org.elasticsearch.xpack.esql.TestBlockFactory;
import org.elasticsearch.xpack.esql.core.expression.Expression;
-import org.elasticsearch.xpack.esql.core.expression.TypeResolutions;
import org.elasticsearch.xpack.esql.core.type.DataType;
import org.elasticsearch.xpack.esql.core.util.NumericUtils;
-import org.elasticsearch.xpack.esql.expression.function.scalar.conditional.Greatest;
import org.elasticsearch.xpack.esql.expression.function.scalar.multivalue.AbstractMultivalueFunctionTestCase;
-import org.elasticsearch.xpack.esql.expression.function.scalar.nulls.Coalesce;
import org.elasticsearch.xpack.esql.optimizer.FoldNull;
import org.elasticsearch.xpack.esql.planner.PlannerUtils;
import org.hamcrest.Matcher;
import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.HashSet;
import java.util.List;
-import java.util.Locale;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
@@ -44,7 +38,6 @@
import java.util.concurrent.Future;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
-import java.util.stream.Stream;
import static org.elasticsearch.compute.data.BlockUtils.toJavaObject;
import static org.hamcrest.Matchers.either;
@@ -372,152 +365,6 @@ public final void testFold() {
}
}
- /**
- * Adds cases with {@code null} and asserts that the result is {@code null}.
- *
- * Note: This won't add more than a single null to any existing test case,
- * just to keep the number of test cases from exploding totally.
- *
- *
- * @param entirelyNullPreservesType should a test case that only contains parameters
- * with the {@code null} type keep it's expected type?
- * This is mostly going to be {@code true}
- * except for functions that base their type entirely
- * on input types like {@link Greatest} or {@link Coalesce}.
- */
- protected static List anyNullIsNull(boolean entirelyNullPreservesType, List testCaseSuppliers) {
- return anyNullIsNull(
- testCaseSuppliers,
- (nullPosition, nullValueDataType, original) -> entirelyNullPreservesType == false
- && nullValueDataType == DataType.NULL
- && original.getData().size() == 1 ? DataType.NULL : original.expectedType(),
- (nullPosition, nullData, original) -> original
- );
- }
-
- public interface ExpectedType {
- DataType expectedType(int nullPosition, DataType nullValueDataType, TestCaseSupplier.TestCase original);
- }
-
- public interface ExpectedEvaluatorToString {
- Matcher evaluatorToString(int nullPosition, TestCaseSupplier.TypedData nullData, Matcher original);
- }
-
- protected static List anyNullIsNull(
- List testCaseSuppliers,
- ExpectedType expectedType,
- ExpectedEvaluatorToString evaluatorToString
- ) {
- typesRequired(testCaseSuppliers);
- List suppliers = new ArrayList<>(testCaseSuppliers.size());
- suppliers.addAll(testCaseSuppliers);
-
- /*
- * For each original test case, add as many copies as there were
- * arguments, replacing one of the arguments with null and keeping
- * the others.
- *
- * Also, if this was the first time we saw the signature we copy it
- * *again*, replacing the argument with null, but annotating the
- * argument's type as `null` explicitly.
- */
- Set> uniqueSignatures = new HashSet<>();
- for (TestCaseSupplier original : testCaseSuppliers) {
- boolean firstTimeSeenSignature = uniqueSignatures.add(original.types());
- for (int nullPosition = 0; nullPosition < original.types().size(); nullPosition++) {
- int finalNullPosition = nullPosition;
- suppliers.add(new TestCaseSupplier(original.name() + " null in " + nullPosition, original.types(), () -> {
- TestCaseSupplier.TestCase oc = original.get();
- List data = IntStream.range(0, oc.getData().size()).mapToObj(i -> {
- TestCaseSupplier.TypedData od = oc.getData().get(i);
- return i == finalNullPosition ? od.withData(null) : od;
- }).toList();
- TestCaseSupplier.TypedData nulledData = oc.getData().get(finalNullPosition);
- return new TestCaseSupplier.TestCase(
- data,
- evaluatorToString.evaluatorToString(finalNullPosition, nulledData, oc.evaluatorToString()),
- expectedType.expectedType(finalNullPosition, nulledData.type(), oc),
- nullValue(),
- null,
- oc.getExpectedTypeError(),
- null,
- null
- );
- }));
-
- if (firstTimeSeenSignature) {
- List typesWithNull = IntStream.range(0, original.types().size())
- .mapToObj(i -> i == finalNullPosition ? DataType.NULL : original.types().get(i))
- .toList();
- boolean newSignature = uniqueSignatures.add(typesWithNull);
- if (newSignature) {
- suppliers.add(new TestCaseSupplier(typesWithNull, () -> {
- TestCaseSupplier.TestCase oc = original.get();
- List data = IntStream.range(0, oc.getData().size())
- .mapToObj(i -> i == finalNullPosition ? TestCaseSupplier.TypedData.NULL : oc.getData().get(i))
- .toList();
- return new TestCaseSupplier.TestCase(
- data,
- equalTo("LiteralsEvaluator[lit=null]"),
- expectedType.expectedType(finalNullPosition, DataType.NULL, oc),
- nullValue(),
- null,
- oc.getExpectedTypeError(),
- null,
- null
- );
- }));
- }
- }
- }
- }
-
- return suppliers;
-
- }
-
- /**
- * Adds test cases containing unsupported parameter types that assert
- * that they throw type errors.
- */
- protected static List errorsForCasesWithoutExamples(
- List testCaseSuppliers,
- PositionalErrorMessageSupplier positionalErrorMessageSupplier
- ) {
- return errorsForCasesWithoutExamples(
- testCaseSuppliers,
- (i, v, t) -> AbstractScalarFunctionTestCase.typeErrorMessage(i, v, t, positionalErrorMessageSupplier)
- );
- }
-
- protected static List errorsForCasesWithoutExamples(
- List testCaseSuppliers,
- TypeErrorMessageSupplier typeErrorMessageSupplier
- ) {
- typesRequired(testCaseSuppliers);
- List suppliers = new ArrayList<>(testCaseSuppliers.size());
- suppliers.addAll(testCaseSuppliers);
-
- Set> valid = testCaseSuppliers.stream().map(TestCaseSupplier::types).collect(Collectors.toSet());
- List> validPerPosition = validPerPosition(valid);
-
- testCaseSuppliers.stream()
- .map(s -> s.types().size())
- .collect(Collectors.toSet())
- .stream()
- .flatMap(count -> allPermutations(count))
- .filter(types -> valid.contains(types) == false)
- /*
- * Skip any cases with more than one null. Our tests don't generate
- * the full combinatorial explosions of all nulls - just a single null.
- * Hopefully , cases will function the same as ,
- * cases.
- */.filter(types -> types.stream().filter(t -> t == DataType.NULL).count() <= 1)
- .map(types -> typeErrorSupplier(validPerPosition.size() != 1, validPerPosition, types, typeErrorMessageSupplier))
- .forEach(suppliers::add);
- return suppliers;
- }
-
public static String errorMessageStringForBinaryOperators(
boolean includeOrdinal,
List> validPerPosition,
@@ -572,178 +419,4 @@ protected static List failureForCasesWithoutExamples(List suppliers) {
- String bad = suppliers.stream().filter(s -> s.types() == null).map(s -> s.name()).collect(Collectors.joining("\n"));
- if (bad.equals("") == false) {
- throw new IllegalArgumentException("types required but not found for these tests:\n" + bad);
- }
- }
-
- private static List> validPerPosition(Set> valid) {
- int max = valid.stream().mapToInt(List::size).max().getAsInt();
- List> result = new ArrayList<>(max);
- for (int i = 0; i < max; i++) {
- result.add(new HashSet<>());
- }
- for (List signature : valid) {
- for (int i = 0; i < signature.size(); i++) {
- result.get(i).add(signature.get(i));
- }
- }
- return result;
- }
-
- private static Stream> allPermutations(int argumentCount) {
- if (argumentCount == 0) {
- return Stream.of(List.of());
- }
- if (argumentCount > 3) {
- throw new IllegalArgumentException("would generate too many combinations");
- }
- Stream> stream = validFunctionParameters().map(List::of);
- for (int i = 1; i < argumentCount; i++) {
- stream = stream.flatMap(types -> validFunctionParameters().map(t -> append(types, t)));
- }
- return stream;
- }
-
- private static List append(List orig, DataType extra) {
- List longer = new ArrayList<>(orig.size() + 1);
- longer.addAll(orig);
- longer.add(extra);
- return longer;
- }
-
- @FunctionalInterface
- protected interface TypeErrorMessageSupplier {
- String apply(boolean includeOrdinal, List> validPerPosition, List types);
- }
-
- @FunctionalInterface
- protected interface PositionalErrorMessageSupplier {
- /**
- * This interface defines functions to supply error messages for incorrect types in specific positions. Functions which have
- * the same type requirements for all positions can simplify this with a lambda returning a string constant.
- *
- * @param validForPosition - the set of {@link DataType}s that the test infrastructure believes to be allowable in the
- * given position.
- * @param position - the zero-index position in the list of parameters the function has detected the bad argument to be.
- * @return The string describing the acceptable parameters for that position. Note that this function should not return
- * the full error string; that will be constructed by the test. Just return the type string for that position.
- */
- String apply(Set validForPosition, int position);
- }
-
- protected static TestCaseSupplier typeErrorSupplier(
- boolean includeOrdinal,
- List> validPerPosition,
- List types,
- PositionalErrorMessageSupplier errorMessageSupplier
- ) {
- return typeErrorSupplier(
- includeOrdinal,
- validPerPosition,
- types,
- (o, v, t) -> AbstractScalarFunctionTestCase.typeErrorMessage(o, v, t, errorMessageSupplier)
- );
- }
-
- /**
- * Build a test case that asserts that the combination of parameter types is an error.
- */
- protected static TestCaseSupplier typeErrorSupplier(
- boolean includeOrdinal,
- List> validPerPosition,
- List types,
- TypeErrorMessageSupplier errorMessageSupplier
- ) {
- return new TestCaseSupplier(
- "type error for " + TestCaseSupplier.nameFromTypes(types),
- types,
- () -> TestCaseSupplier.TestCase.typeError(
- types.stream().map(type -> new TestCaseSupplier.TypedData(randomLiteral(type).value(), type, type.typeName())).toList(),
- errorMessageSupplier.apply(includeOrdinal, validPerPosition, types)
- )
- );
- }
-
- /**
- * Build the expected error message for an invalid type signature.
- */
- protected static String typeErrorMessage(
- boolean includeOrdinal,
- List> validPerPosition,
- List types,
- PositionalErrorMessageSupplier expectedTypeSupplier
- ) {
- int badArgPosition = -1;
- for (int i = 0; i < types.size(); i++) {
- if (validPerPosition.get(i).contains(types.get(i)) == false) {
- badArgPosition = i;
- break;
- }
- }
- if (badArgPosition == -1) {
- throw new IllegalStateException(
- "Can't generate error message for these types, you probably need a custom error message function"
- );
- }
- String ordinal = includeOrdinal ? TypeResolutions.ParamOrdinal.fromIndex(badArgPosition).name().toLowerCase(Locale.ROOT) + " " : "";
- String expectedTypeString = expectedTypeSupplier.apply(validPerPosition.get(badArgPosition), badArgPosition);
- String name = types.get(badArgPosition).typeName();
- return ordinal + "argument of [] must be [" + expectedTypeString + "], found value [" + name + "] type [" + name + "]";
- }
-
- /**
- * The types that are valid in function parameters. This is used by the
- * function tests to enumerate all possible parameters to test error messages
- * for invalid combinations.
- */
- public static Stream validFunctionParameters() {
- return Arrays.stream(DataType.values()).filter(t -> {
- if (t == DataType.UNSUPPORTED) {
- // By definition, functions never support UNSUPPORTED
- return false;
- }
- if (t == DataType.DOC_DATA_TYPE || t == DataType.PARTIAL_AGG) {
- /*
- * Doc and partial_agg are special and functions aren't
- * defined to take these. They'll use them implicitly if needed.
- */
- return false;
- }
- if (t == DataType.OBJECT || t == DataType.NESTED) {
- // Object and nested fields aren't supported by any functions yet
- return false;
- }
- if (t == DataType.SOURCE || t == DataType.TSID_DATA_TYPE) {
- // No functions take source or tsid fields yet. We'll make some eventually and remove this.
- return false;
- }
- if (t == DataType.DATE_PERIOD || t == DataType.TIME_DURATION) {
- // We don't test that functions don't take date_period or time_duration. We should.
- return false;
- }
- if (t.isCounter()) {
- /*
- * For now, we're assuming no functions take counters
- * as parameters. That's not true - some do. But we'll
- * need to update the tests to handle that.
- */
- return false;
- }
- if (t.widenSmallNumeric() != t) {
- // Small numeric types are widened long before they arrive at functions.
- return false;
- }
-
- return true;
- }).sorted();
- }
-
}
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/TestCaseSupplier.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/TestCaseSupplier.java
index 3585e58bf97ab..6652cca0c4527 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/TestCaseSupplier.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/TestCaseSupplier.java
@@ -30,6 +30,7 @@
import java.time.Period;
import java.util.ArrayList;
import java.util.Arrays;
+import java.util.Collections;
import java.util.List;
import java.util.function.BiFunction;
import java.util.function.BinaryOperator;
@@ -1455,6 +1456,7 @@ public TypedData get() {
*/
public static class TypedData {
public static final TypedData NULL = new TypedData(null, DataType.NULL, "");
+ public static final TypedData MULTI_ROW_NULL = TypedData.multiRow(Collections.singletonList(null), DataType.NULL, "");
private final Object data;
private final DataType type;
@@ -1583,7 +1585,7 @@ public Literal asLiteral() {
throw new IllegalStateException("Multirow values require exactly 1 element to be a literal, got " + values.size());
}
- return new Literal(Source.synthetic(name), values, type);
+ return new Literal(Source.synthetic(name), values.get(0), type);
}
return new Literal(Source.synthetic(name), data, type);
}
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/AvgTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/AvgTests.java
index f456bd409059a..80737dac1aa58 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/AvgTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/AvgTests.java
@@ -53,7 +53,7 @@ public static Iterable parameters() {
)
);
- return parameterSuppliersFromTypedDataWithDefaultChecks(suppliers);
+ return parameterSuppliersFromTypedDataWithDefaultChecks(suppliers, true, (v, p) -> "numeric except unsigned_long or counter types");
}
@Override
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MaxTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MaxTests.java
index 1d489e0146ad3..52e908a51dd1e 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MaxTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MaxTests.java
@@ -49,73 +49,6 @@ public static Iterable parameters() {
suppliers.addAll(
List.of(
- // Surrogates
- new TestCaseSupplier(
- List.of(DataType.INTEGER),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5, 8, -2, 0, 200), DataType.INTEGER, "field")),
- "Max[field=Attribute[channel=0]]",
- DataType.INTEGER,
- equalTo(200)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.LONG),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5L, 8L, -2L, 0L, 200L), DataType.LONG, "field")),
- "Max[field=Attribute[channel=0]]",
- DataType.LONG,
- equalTo(200L)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.DOUBLE),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5., 8., -2., 0., 200.), DataType.DOUBLE, "field")),
- "Max[field=Attribute[channel=0]]",
- DataType.DOUBLE,
- equalTo(200.)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.DATETIME),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5L, 8L, 2L, 0L, 200L), DataType.DATETIME, "field")),
- "Max[field=Attribute[channel=0]]",
- DataType.DATETIME,
- equalTo(200L)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.BOOLEAN),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(true, false, false, true), DataType.BOOLEAN, "field")),
- "Max[field=Attribute[channel=0]]",
- DataType.BOOLEAN,
- equalTo(true)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.IP),
- () -> new TestCaseSupplier.TestCase(
- List.of(
- TestCaseSupplier.TypedData.multiRow(
- List.of(
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("127.0.0.1"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("::1"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("::"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("ffff::")))
- ),
- DataType.IP,
- "field"
- )
- ),
- "Max[field=Attribute[channel=0]]",
- DataType.IP,
- equalTo(new BytesRef(InetAddressPoint.encode(InetAddresses.forString("ffff::"))))
- )
- ),
-
// Folding
new TestCaseSupplier(
List.of(DataType.INTEGER),
@@ -180,7 +113,11 @@ public static Iterable parameters() {
)
);
- return parameterSuppliersFromTypedDataWithDefaultChecks(suppliers);
+ return parameterSuppliersFromTypedDataWithDefaultChecks(
+ suppliers,
+ false,
+ (v, p) -> "boolean, datetime, ip or numeric except unsigned_long or counter types"
+ );
}
@Override
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MinTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MinTests.java
index b5fb5b2c1c414..9514c817df497 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MinTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/MinTests.java
@@ -49,73 +49,6 @@ public static Iterable parameters() {
suppliers.addAll(
List.of(
- // Surrogates
- new TestCaseSupplier(
- List.of(DataType.INTEGER),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5, 8, -2, 0, 200), DataType.INTEGER, "field")),
- "Min[field=Attribute[channel=0]]",
- DataType.INTEGER,
- equalTo(-2)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.LONG),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5L, 8L, -2L, 0L, 200L), DataType.LONG, "field")),
- "Min[field=Attribute[channel=0]]",
- DataType.LONG,
- equalTo(-2L)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.DOUBLE),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5., 8., -2., 0., 200.), DataType.DOUBLE, "field")),
- "Min[field=Attribute[channel=0]]",
- DataType.DOUBLE,
- equalTo(-2.)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.DATETIME),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(5L, 8L, 2L, 0L, 200L), DataType.DATETIME, "field")),
- "Min[field=Attribute[channel=0]]",
- DataType.DATETIME,
- equalTo(0L)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.BOOLEAN),
- () -> new TestCaseSupplier.TestCase(
- List.of(TestCaseSupplier.TypedData.multiRow(List.of(true, false, false, true), DataType.BOOLEAN, "field")),
- "Min[field=Attribute[channel=0]]",
- DataType.BOOLEAN,
- equalTo(false)
- )
- ),
- new TestCaseSupplier(
- List.of(DataType.IP),
- () -> new TestCaseSupplier.TestCase(
- List.of(
- TestCaseSupplier.TypedData.multiRow(
- List.of(
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("127.0.0.1"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("::1"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("::"))),
- new BytesRef(InetAddressPoint.encode(InetAddresses.forString("ffff::")))
- ),
- DataType.IP,
- "field"
- )
- ),
- "Min[field=Attribute[channel=0]]",
- DataType.IP,
- equalTo(new BytesRef(InetAddressPoint.encode(InetAddresses.forString("::"))))
- )
- ),
-
// Folding
new TestCaseSupplier(
List.of(DataType.INTEGER),
@@ -180,7 +113,11 @@ public static Iterable parameters() {
)
);
- return parameterSuppliersFromTypedDataWithDefaultChecks(suppliers);
+ return parameterSuppliersFromTypedDataWithDefaultChecks(
+ suppliers,
+ false,
+ (v, p) -> "boolean, datetime, ip or numeric except unsigned_long or counter types"
+ );
}
@Override
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/ValuesTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/ValuesTests.java
index 23b70b94d0d7f..55320543d0ec3 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/ValuesTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/ValuesTests.java
@@ -53,7 +53,11 @@ public static Iterable parameters() {
MultiRowTestCaseSupplier.stringCases(1, 20, DataType.TEXT)
).flatMap(List::stream).map(ValuesTests::makeSupplier).collect(Collectors.toCollection(() -> suppliers));
- return parameterSuppliersFromTypedDataWithDefaultChecks(suppliers);
+ return parameterSuppliersFromTypedDataWithDefaultChecks(
+ suppliers,
+ false,
+ (v, p) -> "any type except unsigned_long and spatial types"
+ );
}
@Override
diff --git a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/WeightedAvgTests.java b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/WeightedAvgTests.java
index 2ba091437f237..2c2ffc97f268c 100644
--- a/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/WeightedAvgTests.java
+++ b/x-pack/plugin/esql/src/test/java/org/elasticsearch/xpack/esql/expression/function/aggregate/WeightedAvgTests.java
@@ -52,11 +52,11 @@ public static Iterable parameters() {
List.of(
// Folding
new TestCaseSupplier(
- List.of(DataType.INTEGER),
+ List.of(DataType.INTEGER, DataType.INTEGER),
() -> new TestCaseSupplier.TestCase(
List.of(
- TestCaseSupplier.TypedData.multiRow(List.of(5), DataType.INTEGER, "field"),
- TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "field")
+ TestCaseSupplier.TypedData.multiRow(List.of(5), DataType.INTEGER, "number"),
+ TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "weight")
),
"WeightedAvg[number=Attribute[channel=0],weight=Attribute[channel=1]]",
DataType.DOUBLE,
@@ -64,11 +64,11 @@ public static Iterable parameters() {
)
),
new TestCaseSupplier(
- List.of(DataType.LONG),
+ List.of(DataType.LONG, DataType.INTEGER),
() -> new TestCaseSupplier.TestCase(
List.of(
- TestCaseSupplier.TypedData.multiRow(List.of(5L), DataType.LONG, "field"),
- TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "field")
+ TestCaseSupplier.TypedData.multiRow(List.of(5L), DataType.LONG, "number"),
+ TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "weight")
),
"WeightedAvg[number=Attribute[channel=0],weight=Attribute[channel=1]]",
DataType.DOUBLE,
@@ -76,11 +76,11 @@ public static Iterable parameters() {
)
),
new TestCaseSupplier(
- List.of(DataType.DOUBLE),
+ List.of(DataType.DOUBLE, DataType.INTEGER),
() -> new TestCaseSupplier.TestCase(
List.of(
- TestCaseSupplier.TypedData.multiRow(List.of(5.), DataType.DOUBLE, "field"),
- TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "field")
+ TestCaseSupplier.TypedData.multiRow(List.of(5.), DataType.DOUBLE, "number"),
+ TestCaseSupplier.TypedData.multiRow(List.of(100), DataType.INTEGER, "weight")
),
"WeightedAvg[number=Attribute[channel=0],weight=Attribute[channel=1]]",
DataType.DOUBLE,