From d9c5a6d2da6b34bec68a5b17daa69289434cf96f Mon Sep 17 00:00:00 2001 From: Robert Stupp Date: Sat, 18 Nov 2023 12:00:03 +0100 Subject: [PATCH] Bump cel-spec + googleapi submodules (#476) Build: maintenance + convenience changes: * add `compileAll` task * add IntelliJ-sync-task * bump jandex version to make it work with Java 21 * CI: Bump bazel version from 3.5.1 to 6.4.0 Bump cel-spec + googleapi submodules: Bump cel-spec submodule to [v0.13.0](https://github.com/google/cel-spec/releases/tag/v0.13.0), googleapi submodule to [f2d78630d2c1d5e20041dfff963e093de9298e4d](https://github.com/googleapis/googleapis/commit/f2d78630d2c1d5e20041dfff963e093de9298e4d). This was necessary due to recent build errors caused by missing (old) artifacts for the conformance bazel build. The submodule bumps bring a bunch of new conformance tests as well, which unveiled that some functionality that was not present in CEL-Go when CEL-Java was created, is required by the CEl-Spec. Summary of changes: * Bugfix in `o.p.c.checker.Standard` that fixes a type resolution error for equals and not-equals functions, when both parameters need to be up-casted. * Changes in many types (in `o.p.c.common.types.*T` classes) to automatically convert between numeric and null types. * equal and compare with a null and a non-null type no longer fail, but return `False` * equal and compare between different numeric types no longer fail, but return "the right" result * this includes that numeric CEL map keys can be heterogenous, e.g. an `int` can be retrieved using an `uint` or `double` key * Fix retrieval of milliseconds from `Duration` - must only return the milliseconds within the second * Un-ignore a bunch of conformance tests that pass fine now A few conformance tests have been added to `InterpreterTest`, but the majority of tests is left in the conformance tests. --- .github/workflows/main.yml | 2 +- build.gradle.kts | 18 ++ conformance/README.md | 7 +- conformance/build.gradle.kts | 9 +- conformance/run-conformance-tests.sh | 24 ++- .../cel/server/ConformanceServiceImpl.java | 18 +- .../main/proto/google/api/expr/conformance | 1 + .../cel/server/ConformanceServerTest.java | 16 +- core/build.gradle.kts | 2 + .../projectnessie/cel/checker/Standard.java | 4 +- .../projectnessie/cel/common/types/BoolT.java | 10 +- .../cel/common/types/BytesT.java | 11 +- .../cel/common/types/DoubleT.java | 88 ++++++--- .../cel/common/types/DurationT.java | 28 ++- .../projectnessie/cel/common/types/IntT.java | 118 ++++++++---- .../projectnessie/cel/common/types/ListT.java | 46 +++-- .../projectnessie/cel/common/types/MapT.java | 36 +++- .../projectnessie/cel/common/types/NullT.java | 15 +- .../cel/common/types/ObjectT.java | 9 +- .../cel/common/types/StringT.java | 38 ++-- .../cel/common/types/TimestampT.java | 11 +- .../projectnessie/cel/common/types/TypeT.java | 5 + .../projectnessie/cel/common/types/UintT.java | 146 ++++++++++----- .../cel/common/types/UnknownT.java | 4 +- .../cel/common/types/ref/BaseVal.java | 6 + .../cel/common/types/ref/Val.java | 2 + .../cel/interpreter/AttributeFactory.java | 171 +++++++++++++++++- .../cel/interpreter/Interpretable.java | 10 +- .../java/org/projectnessie/cel/CELTest.java | 5 +- .../cel/checker/CheckerTest.java | 7 +- .../cel/common/types/DoubleTest.java | 21 +++ .../cel/common/types/DurationTest.java | 4 +- .../cel/common/types/IntTest.java | 23 +++ .../cel/common/types/MapTest.java | 79 ++++++++ .../cel/common/types/StringTest.java | 5 +- .../cel/common/types/UintTest.java | 22 +++ .../cel/interpreter/InterpreterTest.java | 81 ++++++++- .../cel/interpreter/InterpreterTestCase.java | 25 ++- generated-pb/build.gradle.kts | 5 - gradle/libs.versions.toml | 4 +- submodules/cel-spec | 2 +- submodules/googleapis | 2 +- 42 files changed, 903 insertions(+), 237 deletions(-) create mode 120000 conformance/src/main/proto/google/api/expr/conformance diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index d460b13b..0e0c192e 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -98,7 +98,7 @@ jobs: if: ${{ matrix.java-version == '11' }} uses: jwlawson/actions-setup-bazel@v1.10.1 with: - bazel-version: '3.5.1' + bazel-version: '6.4.0' - name: Conformance tests if: ${{ matrix.java-version == '11' }} diff --git a/build.gradle.kts b/build.gradle.kts index 47a93fcd..424a9345 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -84,3 +84,21 @@ publishingHelper { nessieRepoName.set("cel-java") inceptionYear.set("2021") } + +idea.project.settings { + taskTriggers { + afterSync( + ":cel-generated-pb:jar", + ":cel-generated-pb:testJar", + ":cel-generated-antlr:shadowJar" + ) + } +} + +subprojects.forEach { + it.tasks.register("compileAll").configure { + group = "build" + description = "Runs all compilation and jar tasks" + dependsOn(tasks.withType(), tasks.withType()) + } +} diff --git a/conformance/README.md b/conformance/README.md index 2762beaa..ef4fab0b 100644 --- a/conformance/README.md +++ b/conformance/README.md @@ -11,8 +11,11 @@ The CEL-spec conformance test suite is written in Go and uses the bazel build to Required tools: * [Bazel build tool](https://bazel.build/) - - On Ubuntu run `apt-get install bazel-bootstrap` + + See [Bazel web site](https://bazel.build/install) for installation instructions. + Do **not** use the `bazel-bootstrap` package on Ubuntu, because it comes with + pre-compiled classes that are built with Java 17 or newer, preventing it from + running bazel using older Java versions. * gcc On Ubuntu run `apt-get install gcc` diff --git a/conformance/build.gradle.kts b/conformance/build.gradle.kts index 58da1452..b718358b 100644 --- a/conformance/build.gradle.kts +++ b/conformance/build.gradle.kts @@ -28,7 +28,10 @@ plugins { apply() -sourceSets.main { java.srcDir(layout.buildDirectory.dir("generated/source/proto/main/java")) } +sourceSets.main { + java.srcDir(layout.buildDirectory.dir("generated/source/proto/main/java")) + java.srcDir(layout.buildDirectory.dir("generated/source/proto/main/grpc")) +} dependencies { implementation(project(":cel-core")) @@ -58,6 +61,10 @@ configure { // Download from repositories artifact = "com.google.protobuf:protoc:${libs.versions.protobuf.get()}" } + plugins { + this.create("grpc") { artifact = "io.grpc:protoc-gen-grpc-java:${libs.versions.grpc.get()}" } + } + generateProtoTasks { all().configureEach { this.plugins.create("grpc") {} } } } // The protobuf-plugin should ideally do this diff --git a/conformance/run-conformance-tests.sh b/conformance/run-conformance-tests.sh index da46b510..33df1ad6 100755 --- a/conformance/run-conformance-tests.sh +++ b/conformance/run-conformance-tests.sh @@ -46,30 +46,33 @@ cel_java_skips=( # nested protobuf-object-structure, which gets rejected during gRPC/protobuf request # deserialization. Just skip those tests. "--skip_test=parse/nest/message_literal" - "--skip_test=parse/repeat/index" + # Proto equality specialties don't seem to be in effect for Java + "--skip_test=comparisons/eq_wrapper/eq_proto_nan_equal" + "--skip_test=comparisons/ne_literal/ne_proto_nan_not_equal" - # TODO Actual known issux to fix, a protobuf Any returned via this test is wrapped twice (Any in Any). + # TODO Actual known issue to fix, a protobuf Any returned via this test is wrapped twice (Any in Any). "--skip_test=dynamic/any/var" ) cel_go_skips=( - "--skip_test=comparisons/eq_literal/not_eq_list_false_vs_types,not_eq_map_false_vs_types" - "--skip_test=comparisons/in_map_literal/key_in_mixed_key_type_map_error" - "--skip_test=comparisons/eq_literal/not_eq_list_false_vs_types,not_eq_map_false_vs_types" - "--skip_test=comparisons/in_map_literal/key_in_mixed_key_type_map_error" "--skip_test=dynamic/int32/field_assign_proto2_range,field_assign_proto3_range" "--skip_test=dynamic/uint32/field_assign_proto2_range,field_assign_proto3_range" "--skip_test=dynamic/float/field_assign_proto2_range,field_assign_proto3_range" - "--skip_test=dynamic/value_null/literal_unset,field_read_proto2_unset,field_read_proto3_unset" "--skip_test=enums/legacy_proto2/assign_standalone_int_too_big,assign_standalone_int_too_neg" "--skip_test=enums/legacy_proto3/assign_standalone_int_too_big,assign_standalone_int_too_neg" "--skip_test=enums/strong_proto2" "--skip_test=enums/strong_proto3" - "--skip_test=fields/qualified_identifier_resolution/map_key_float,map_key_null,map_value_repeat_key" + # This conformance test is invalid nowadays + "--skip_test=fields/qualified_identifier_resolution/map_key_float" + # Unclear why the 'to_json_string' is expected to return a string, unlike the preceding to_json_number test. + "--skip_test=wrappers/uint64/to_json_string" + # TODO implement proper "toJson" at some point + "--skip_test=wrappers/field_mask/to_json" + "--skip_test=wrappers/timestamp/to_json" + "--skip_test=wrappers/empty/to_json" ) test_files=( - "tests/simple/testdata/plumbing.textproto" "tests/simple/testdata/basic.textproto" "tests/simple/testdata/comparisons.textproto" "tests/simple/testdata/conversions.textproto" @@ -83,11 +86,14 @@ test_files=( "tests/simple/testdata/macros.textproto" "tests/simple/testdata/namespace.textproto" "tests/simple/testdata/parse.textproto" + "tests/simple/testdata/plumbing.textproto" "tests/simple/testdata/proto2.textproto" "tests/simple/testdata/proto3.textproto" "tests/simple/testdata/string.textproto" + # TODO add when implemnting the string-extensions "tests/simple/testdata/string_ext.textproto" "tests/simple/testdata/timestamps.textproto" "tests/simple/testdata/unknowns.textproto" + "tests/simple/testdata/wrappers.textproto" ) bazel-bin/tests/simple/simple_test_/simple_test \ diff --git a/conformance/src/main/java/org/projectnessie/cel/server/ConformanceServiceImpl.java b/conformance/src/main/java/org/projectnessie/cel/server/ConformanceServiceImpl.java index af9968a3..a3d86aa6 100644 --- a/conformance/src/main/java/org/projectnessie/cel/server/ConformanceServiceImpl.java +++ b/conformance/src/main/java/org/projectnessie/cel/server/ConformanceServiceImpl.java @@ -39,20 +39,20 @@ import static org.projectnessie.cel.common.types.UnknownT.isUnknown; import static org.projectnessie.cel.common.types.UnknownT.unknownOf; -import com.google.api.expr.v1alpha1.CheckRequest; -import com.google.api.expr.v1alpha1.CheckResponse; -import com.google.api.expr.v1alpha1.ConformanceServiceGrpc.ConformanceServiceImplBase; +import com.google.api.expr.conformance.v1alpha1.CheckRequest; +import com.google.api.expr.conformance.v1alpha1.CheckResponse; +import com.google.api.expr.conformance.v1alpha1.ConformanceServiceGrpc.ConformanceServiceImplBase; +import com.google.api.expr.conformance.v1alpha1.EvalRequest; +import com.google.api.expr.conformance.v1alpha1.EvalResponse; +import com.google.api.expr.conformance.v1alpha1.IssueDetails; +import com.google.api.expr.conformance.v1alpha1.ParseRequest; +import com.google.api.expr.conformance.v1alpha1.ParseResponse; +import com.google.api.expr.conformance.v1alpha1.SourcePosition; import com.google.api.expr.v1alpha1.ErrorSet; -import com.google.api.expr.v1alpha1.EvalRequest; -import com.google.api.expr.v1alpha1.EvalResponse; import com.google.api.expr.v1alpha1.ExprValue; -import com.google.api.expr.v1alpha1.IssueDetails; import com.google.api.expr.v1alpha1.ListValue; import com.google.api.expr.v1alpha1.MapValue; import com.google.api.expr.v1alpha1.MapValue.Entry; -import com.google.api.expr.v1alpha1.ParseRequest; -import com.google.api.expr.v1alpha1.ParseResponse; -import com.google.api.expr.v1alpha1.SourcePosition; import com.google.api.expr.v1alpha1.UnknownSet; import com.google.api.expr.v1alpha1.Value; import com.google.protobuf.Any; diff --git a/conformance/src/main/proto/google/api/expr/conformance b/conformance/src/main/proto/google/api/expr/conformance new file mode 120000 index 00000000..04adab4d --- /dev/null +++ b/conformance/src/main/proto/google/api/expr/conformance @@ -0,0 +1 @@ +../../../../../../../submodules/googleapis/google/api/expr/conformance \ No newline at end of file diff --git a/conformance/src/test/java/org/projectnessie/cel/server/ConformanceServerTest.java b/conformance/src/test/java/org/projectnessie/cel/server/ConformanceServerTest.java index a75dda6c..ce79883b 100644 --- a/conformance/src/test/java/org/projectnessie/cel/server/ConformanceServerTest.java +++ b/conformance/src/test/java/org/projectnessie/cel/server/ConformanceServerTest.java @@ -20,22 +20,22 @@ import static org.projectnessie.cel.TestExpr.ExprLiteral; import static org.projectnessie.cel.Util.mapOf; -import com.google.api.expr.v1alpha1.CheckRequest; -import com.google.api.expr.v1alpha1.CheckResponse; +import com.google.api.expr.conformance.v1alpha1.CheckRequest; +import com.google.api.expr.conformance.v1alpha1.CheckResponse; +import com.google.api.expr.conformance.v1alpha1.ConformanceServiceGrpc; +import com.google.api.expr.conformance.v1alpha1.ConformanceServiceGrpc.ConformanceServiceBlockingStub; +import com.google.api.expr.conformance.v1alpha1.EvalRequest; +import com.google.api.expr.conformance.v1alpha1.EvalResponse; +import com.google.api.expr.conformance.v1alpha1.ParseRequest; +import com.google.api.expr.conformance.v1alpha1.ParseResponse; import com.google.api.expr.v1alpha1.CheckedExpr; -import com.google.api.expr.v1alpha1.ConformanceServiceGrpc; -import com.google.api.expr.v1alpha1.ConformanceServiceGrpc.ConformanceServiceBlockingStub; import com.google.api.expr.v1alpha1.Constant; import com.google.api.expr.v1alpha1.Constant.ConstantKindCase; -import com.google.api.expr.v1alpha1.EvalRequest; -import com.google.api.expr.v1alpha1.EvalResponse; import com.google.api.expr.v1alpha1.Expr; import com.google.api.expr.v1alpha1.Expr.Call; import com.google.api.expr.v1alpha1.Expr.ExprKindCase; import com.google.api.expr.v1alpha1.ExprValue; import com.google.api.expr.v1alpha1.ExprValue.KindCase; -import com.google.api.expr.v1alpha1.ParseRequest; -import com.google.api.expr.v1alpha1.ParseResponse; import com.google.api.expr.v1alpha1.ParsedExpr; import com.google.api.expr.v1alpha1.SourceInfo; import com.google.api.expr.v1alpha1.Type; diff --git a/core/build.gradle.kts b/core/build.gradle.kts index 7c6718c5..638a0927 100644 --- a/core/build.gradle.kts +++ b/core/build.gradle.kts @@ -37,6 +37,8 @@ dependencies { testImplementation(platform(libs.junit.bom)) testImplementation(libs.bundles.junit.testing) + testImplementation(libs.protobuf.java) + testImplementation(libs.guava) testRuntimeOnly(libs.junit.jupiter.engine) jmhImplementation(libs.jmh.core) diff --git a/core/src/main/java/org/projectnessie/cel/checker/Standard.java b/core/src/main/java/org/projectnessie/cel/checker/Standard.java index 8ee4c47c..e16ae573 100644 --- a/core/src/main/java/org/projectnessie/cel/checker/Standard.java +++ b/core/src/main/java/org/projectnessie/cel/checker/Standard.java @@ -162,13 +162,13 @@ Overloads.GreaterEqualsBytes, asList(Decls.Bytes, Decls.Bytes), Decls.Bool), Decls.newFunction( Operator.Equals.id, Decls.newParameterizedOverload( - Overloads.Equals, asList(paramA, paramA), Decls.Bool, typeParamAList))); + Overloads.Equals, asList(paramA, paramB), Decls.Bool, typeParamABList))); idents.add( Decls.newFunction( Operator.NotEquals.id, Decls.newParameterizedOverload( - Overloads.NotEquals, asList(paramA, paramA), Decls.Bool, typeParamAList))); + Overloads.NotEquals, asList(paramA, paramB), Decls.Bool, typeParamABList))); // Algebra. diff --git a/core/src/main/java/org/projectnessie/cel/common/types/BoolT.java b/core/src/main/java/org/projectnessie/cel/common/types/BoolT.java index 40b21937..5747c73d 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/BoolT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/BoolT.java @@ -107,10 +107,14 @@ public Val convertToType(Type typeVal) { /** Equal implements the ref.Val interface method. */ @Override public Val equal(Val other) { - if (!(other instanceof BoolT)) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Bool: + return Types.boolOf(b == ((BoolT) other).b); + case Null: + return False; + default: + return noSuchOverload(this, "equal", other); } - return Types.boolOf(b == ((BoolT) other).b); } /** Negate implements the traits.Negater interface method. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/BytesT.java b/core/src/main/java/org/projectnessie/cel/common/types/BytesT.java index 597ca249..0b1fa188 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/BytesT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/BytesT.java @@ -15,6 +15,7 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.Err.newErr; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; import static org.projectnessie.cel.common.types.Err.noSuchOverload; @@ -162,10 +163,14 @@ public Val convertToType(Type typeValue) { /** Equal implements the ref.Val interface method. */ @Override public Val equal(Val other) { - if (!(other instanceof BytesT)) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Bytes: + return boolOf(Arrays.equals(b, ((BytesT) other).b)); + case Null: + return False; + default: + return noSuchOverload(this, "equal", other); } - return boolOf(Arrays.equals(b, ((BytesT) other).b)); } /** Size implements the traits.Sizer interface method. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/DoubleT.java b/core/src/main/java/org/projectnessie/cel/common/types/DoubleT.java index 961e0051..5cfa40e0 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/DoubleT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/DoubleT.java @@ -15,6 +15,7 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; import static org.projectnessie.cel.common.types.Err.noSuchOverload; import static org.projectnessie.cel.common.types.Err.rangeError; @@ -31,7 +32,6 @@ import com.google.protobuf.Value; import java.math.BigDecimal; import java.math.BigInteger; -import java.util.Objects; import org.projectnessie.cel.common.types.ref.BaseVal; import org.projectnessie.cel.common.types.ref.Type; import org.projectnessie.cel.common.types.ref.TypeEnum; @@ -68,6 +68,11 @@ private DoubleT(double d) { this.d = d; } + @Override + public double doubleValue() { + return d; + } + /** Add implements traits.Adder.Add. */ @Override public Val add(Val other) { @@ -77,20 +82,6 @@ public Val add(Val other) { return doubleOf(d + ((DoubleT) other).d); } - /** Compare implements traits.Comparer.Compare. */ - @Override - public Val compare(Val other) { - if (!(other instanceof DoubleT)) { - return noSuchOverload(this, "compare", other); - } - double od = ((DoubleT) other).d; - if (d == od) { - // work around for special case of -0.0d == 0.0d (IEEE 754) - return IntZero; - } - return intOfCompare(Double.compare(d, od)); - } - /** ConvertToNative implements ref.Val.ConvertToNative. */ @SuppressWarnings("unchecked") @Override @@ -141,17 +132,23 @@ public Val convertToType(Type typeValue) { // (see https://github.com/google/cel-spec/blob/master/doc/langdef.md#numeric-values) switch (typeValue.typeEnum()) { case Int: + if (!Double.isFinite(d)) { + return rangeError(d, "int"); + } long r = (long) d; // ?? Math.round(d); if (r == Long.MIN_VALUE || r == Long.MAX_VALUE) { return rangeError(d, "int"); } return intOf(r); case Uint: + if (!Double.isFinite(d)) { + return rangeError(d, "uint"); + } // hack to support uint64 BigDecimal dec = new BigDecimal(d); BigInteger bi = dec.toBigInteger(); if (d < 0 || bi.compareTo(MAX_UINT64) > 0) { - return rangeError(d, "int"); + return rangeError(d, "uint"); } return uintOf(bi.longValue()); case Double: @@ -173,14 +170,51 @@ public Val divide(Val other) { return doubleOf(d / ((DoubleT) other).d); } + /** Compare implements traits.Comparer.Compare. */ + @Override + public Val compare(Val other) { + switch (other.type().typeEnum()) { + case Uint: + case Int: + case Double: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + double od = ((DoubleT) converted).d; + if (d == od) { + // work around for special case of -0.0d == 0.0d (IEEE 754) + return IntZero; + } + return intOfCompare(Double.compare(d, od)); + default: + return noSuchOverload(this, "compare", other); + } + } + /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - if (!(other instanceof DoubleT)) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Uint: + case Int: + case Double: + case String: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + double o = ((DoubleT) converted).d; + // TODO: Handle NaNs properly. + return boolOf(d == o); + case Null: + case Bytes: + case List: + case Map: + return False; + default: + return noSuchOverload(this, "equal", other); } - /** TODO: Handle NaNs properly. */ - return boolOf(d == ((DoubleT) other).d); } /** Multiply implements traits.Multiplier.Multiply. */ @@ -224,20 +258,16 @@ public boolean equals(Object o) { if (this == o) { return true; } - if (o == null || getClass() != o.getClass()) { + if (!(o instanceof Val)) { return false; } - DoubleT doubleT = (DoubleT) o; - double od = ((DoubleT) o).d; - if (d == od) { - // work around for special case of -0.0d == 0.0d (IEEE 754) - return true; - } - return Double.compare(doubleT.d, d) == 0; + // Defer to CEL's equal functionality to allow heterogeneous numeric map keys + return equal((Val) o).booleanValue(); } @Override public int hashCode() { - return Objects.hash(super.hashCode(), d); + // Used to allow heterogeneous numeric map keys + return (int) d; } } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/DurationT.java b/core/src/main/java/org/projectnessie/cel/common/types/DurationT.java index 4a683fa7..d3db8c5d 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/DurationT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/DurationT.java @@ -15,6 +15,7 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.Err.errDurationOutOfRange; import static org.projectnessie.cel.common.types.Err.errDurationOverflow; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; @@ -26,6 +27,7 @@ import com.google.protobuf.Any; import com.google.protobuf.Value; +import java.time.DateTimeException; import java.time.Duration; import java.time.ZonedDateTime; import java.time.format.DateTimeParseException; @@ -76,7 +78,17 @@ public static DurationT durationOf(String s) { try { dur = Duration.parse("PT" + s); } catch (DateTimeParseException e) { - dur = Duration.parse("P" + s); + try { + dur = Duration.parse("P" + s); + } catch (DateTimeParseException e2) { + // Not sure whether ns is the only case here, but this is at least the only case that + // triggered a conformance-test failure. + if (s.endsWith("ns")) { + dur = Duration.ofNanos(Long.parseLong(s.substring(0, s.length() - 2))); + } else { + throw new DateTimeException("Cannot parse duration '" + s + "'", e2); + } + } } return durationOf(dur); } @@ -216,10 +228,14 @@ public Val convertToType(Type typeValue) { /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - if (!(other instanceof DurationT)) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Duration: + return boolOf(d.equals(((DurationT) other).d)); + case Null: + return False; + default: + return noSuchOverload(this, "equal", other); } - return boolOf(d.equals(((DurationT) other).d)); } /** Negate implements traits.Negater.Negate. */ @@ -299,8 +315,6 @@ public static Val timeGetSeconds(Duration duration) { } public static Val timeGetMilliseconds(Duration duration) { - return IntT.intOf( - TimeUnit.SECONDS.toMillis(duration.getSeconds()) - + TimeUnit.NANOSECONDS.toMillis(duration.getNano())); + return IntT.intOf(TimeUnit.NANOSECONDS.toMillis(duration.getNano())); } } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/IntT.java b/core/src/main/java/org/projectnessie/cel/common/types/IntT.java index a2c11d7a..41cb5aa4 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/IntT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/IntT.java @@ -15,6 +15,8 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; +import static org.projectnessie.cel.common.types.DoubleT.DoubleType; import static org.projectnessie.cel.common.types.DoubleT.doubleOf; import static org.projectnessie.cel.common.types.Err.divideByZero; import static org.projectnessie.cel.common.types.Err.errIntOverflow; @@ -36,7 +38,6 @@ import com.google.protobuf.Int64Value; import com.google.protobuf.Value; import java.time.Instant; -import java.util.Objects; import org.projectnessie.cel.common.types.Overflow.OverflowException; import org.projectnessie.cel.common.types.ref.BaseVal; import org.projectnessie.cel.common.types.ref.Type; @@ -113,26 +114,9 @@ public long intValue() { return i; } - /** Add implements traits.Adder.Add. */ - @Override - public Val add(Val other) { - if (!(other instanceof IntT)) { - return noSuchOverload(this, "add", other); - } - try { - return IntT.intOf(Overflow.addInt64Checked(i, ((IntT) other).i)); - } catch (OverflowException e) { - return errIntOverflow; - } - } - - /** Compare implements traits.Comparer.Compare. */ @Override - public Val compare(Val other) { - if (!(other instanceof IntT)) { - return noSuchOverload(this, "compare", other); - } - return IntT.intOf(Long.compare(i, ((IntT) other).i)); + public double doubleValue() { + return (double) i; } /** ConvertToNative implements ref.Val.ConvertToNative. */ @@ -216,6 +200,62 @@ public Val convertToType(Type typeValue) { return newTypeConversionError(IntType, typeValue); } + /** Compare implements traits.Comparer.Compare. */ + @Override + public Val compare(Val other) { + switch (other.type().typeEnum()) { + case Double: + DoubleT cmp = (DoubleT) convertToType(DoubleType); + return cmp.compare(other); + case Uint: + if (i < 0L) { + // this int is negative, so it MUST be smaller than any uint + return IntNegOne; + } + if (other.intValue() < 0L) { + // the OTHER uint is > Integer.MAX_VALUE, so THIS int MUST be smaller + return IntNegOne; + } + return intOfCompare(Long.compareUnsigned(i, other.intValue())); + case Int: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + return intOfCompare(Long.compare(i, converted.intValue())); + default: + return noSuchOverload(this, "compare", other); + } + } + + /** Equal implements ref.Val.Equal. */ + @Override + public Val equal(Val other) { + switch (other.type().typeEnum()) { + case Double: + DoubleT cmp = (DoubleT) convertToType(DoubleType); + return cmp.equal(other); + case Uint: + if (other.intValue() < 0L) { + return False; + } + case Int: + case String: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + return boolOf(i == converted.intValue()); + case Null: + case Bytes: + case List: + case Map: + return False; + default: + return noSuchOverload(this, "equal", other); + } + } + /** Divide implements traits.Divider.Divide. */ @Override public Val divide(Val other) { @@ -233,15 +273,6 @@ public Val divide(Val other) { } } - /** Equal implements ref.Val.Equal. */ - @Override - public Val equal(Val other) { - if (!(other instanceof IntT)) { - return noSuchOverload(this, "equal", other); - } - return boolOf(i == ((IntT) other).i); - } - /** Modulo implements traits.Modder.Modulo. */ @Override public Val modulo(Val other) { @@ -282,6 +313,19 @@ public Val negate() { } } + /** Add implements traits.Adder.Add. */ + @Override + public Val add(Val other) { + if (!(other instanceof IntT)) { + return noSuchOverload(this, "add", other); + } + try { + return IntT.intOf(Overflow.addInt64Checked(i, ((IntT) other).i)); + } catch (OverflowException e) { + return errIntOverflow; + } + } + /** Subtract implements traits.Subtractor.Subtract. */ @Override public Val subtract(Val other) { @@ -312,22 +356,16 @@ public boolean equals(Object o) { if (this == o) { return true; } - if (o == null || getClass() != o.getClass()) { + if (!(o instanceof Val)) { return false; } - IntT intT = (IntT) o; - return i == intT.i; + // Defer to CEL's equal functionality to allow heterogeneous numeric map keys + return equal((Val) o).booleanValue(); } @Override public int hashCode() { - return Objects.hash(super.hashCode(), i); - } - - /** - * isJSONSafe indicates whether the int is safely representable as a floating point value in JSON. - */ - public boolean isJSONSafe() { - return i >= minIntJSON && i <= maxIntJSON; + // Used to allow heterogeneous numeric map keys + return (int) i; } } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/ListT.java b/core/src/main/java/org/projectnessie/cel/common/types/ListT.java index c4c48d88..6aa842e6 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/ListT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/ListT.java @@ -196,8 +196,11 @@ public Val equal(Val other) { if (isError(e2)) { return e2; } - if (e1.type() != e2.type()) { - return noSuchOverload(e1, Operator.Equals.id, e2); + if (!e1.type().equals(e2.type())) { + e2 = e2.convertToType(e2.type()); + if (e2.type().typeEnum() == TypeEnum.Err) { + return noSuchOverload(e1, Operator.Equals.id, e2); + } } if (e1.equal(e2) != True) { return False; @@ -208,23 +211,12 @@ public Val equal(Val other) { @Override public Val contains(Val value) { - Type firstType = null; - Type mixedType = null; for (long i = 0; i < size; i++) { Val elem = get(intOf(i)); - Type elemType = elem.type(); - if (firstType == null) { - firstType = elemType; - } else if (!firstType.equals(elemType)) { - mixedType = elemType; - } if (value.equal(elem) == True) { return True; } } - if (mixedType != null) { - return noSuchOverload(value, Operator.In.id, firstType, mixedType); - } return False; } @@ -303,8 +295,18 @@ public Val add(Val other) { @Override public Val get(Val index) { - if (!(index instanceof IntT)) { - return valOrErr(index, "unsupported index type '%s' in list", index.type()); + switch (index.type().typeEnum()) { + case Int: + case Uint: + break; + case Double: + double od = index.doubleValue(); + if (Math.rint(od) != od) { + return newErr("invalid_argument"); + } + break; + default: + return valOrErr(index, "unsupported index type '%s' in list", index.type()); } int sz = array.length; int i = (int) index.intValue(); @@ -369,8 +371,18 @@ public Val add(Val other) { @Override public Val get(Val index) { - if (!(index instanceof IntT)) { - return valOrErr(index, "unsupported index type '%s' in list", index.type()); + switch (index.type().typeEnum()) { + case Int: + case Uint: + break; + case Double: + double od = index.doubleValue(); + if (Math.rint(od) != od) { + return newErr("invalid_argument"); + } + break; + default: + return valOrErr(index, "unsupported index type '%s' in list", index.type()); } int sz = array.length; int i = (int) index.intValue(); diff --git a/core/src/main/java/org/projectnessie/cel/common/types/MapT.java b/core/src/main/java/org/projectnessie/cel/common/types/MapT.java index a1f08e6c..e9008eae 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/MapT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/MapT.java @@ -18,8 +18,8 @@ import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.BoolT.True; import static org.projectnessie.cel.common.types.Err.isError; +import static org.projectnessie.cel.common.types.Err.newErr; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; -import static org.projectnessie.cel.common.types.Err.noSuchOverload; import static org.projectnessie.cel.common.types.StringT.StringType; import static org.projectnessie.cel.common.types.TypeT.TypeType; import static org.projectnessie.cel.common.types.Types.boolOf; @@ -30,7 +30,6 @@ import java.util.HashMap; import java.util.Map; import java.util.Objects; -import org.projectnessie.cel.common.operators.Operator; import org.projectnessie.cel.common.types.ref.BaseVal; import org.projectnessie.cel.common.types.ref.Type; import org.projectnessie.cel.common.types.ref.TypeAdapter; @@ -58,7 +57,17 @@ public static Val newWrappedMap(TypeAdapter adapter, Map value) { public static Val newMaybeWrappedMap(TypeAdapter adapter, Map value) { Map newMap = new HashMap<>(value.size() * 4 / 3 + 1); - value.forEach((k, v) -> newMap.put(adapter.nativeToValue(k), adapter.nativeToValue(v))); + for (Map.Entry entry : value.entrySet()) { + Val k = adapter.nativeToValue(entry.getKey()); + Val v = adapter.nativeToValue(entry.getValue()); + if (k.type().typeEnum() == TypeEnum.Null) { + return newErr("unsupported key type"); + } + if (newMap.putIfAbsent(k, v) != null) { + // Prevent duplicate keys, error out. + return newErr("Failed with repeated key"); + } + } return newWrappedMap(adapter, newMap); } @@ -165,10 +174,19 @@ public Val equal(Val other) { if (isError(oVal)) { return val; } - if (val.type() != oVal.type()) { - return noSuchOverload(val, Operator.Equals.id, oVal); - } + Val eq = val.equal(oVal); + if (eq == True) { + continue; + } + if (eq == False) { + return False; + } + + if (!val.type().equals(oVal.type())) { + return False; + } + eq = val.equal(oVal); if (eq instanceof Err) { return eq; } @@ -188,12 +206,12 @@ public Object value() { @Override public Val contains(Val value) { - return boolOf(map.containsKey(value)); + return boolOf(find(value) != null); } @Override public Val get(Val index) { - return map.get(index); + return find(index); } @Override @@ -203,6 +221,8 @@ public Val size() { @Override public Val find(Val key) { + // Note: no special handling for heterogenous numeric map keys needed, the Val type + // implementations implement .hashCode() and .equals() do deal with heterogenous numeric keys. return map.get(key); } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/NullT.java b/core/src/main/java/org/projectnessie/cel/common/types/NullT.java index 80ea8301..948126d2 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/NullT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/NullT.java @@ -15,6 +15,7 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.BoolT.True; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; import static org.projectnessie.cel.common.types.Err.noSuchOverload; @@ -95,10 +96,18 @@ public Val convertToType(Type typeValue) { /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - if (NullType != other.type()) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Null: + return True; + case Int: + case Uint: + case Double: + case String: + case Bytes: + return False; + default: + return noSuchOverload(this, "equal", other); } - return True; } /** Type implements ref.Val.Type. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/ObjectT.java b/core/src/main/java/org/projectnessie/cel/common/types/ObjectT.java index 96871e31..e7f8c100 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/ObjectT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/ObjectT.java @@ -15,8 +15,8 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.Err.newTypeConversionError; -import static org.projectnessie.cel.common.types.Err.noSuchOverload; import static org.projectnessie.cel.common.types.Types.boolOf; import java.util.Objects; @@ -24,6 +24,7 @@ import org.projectnessie.cel.common.types.ref.Type; import org.projectnessie.cel.common.types.ref.TypeAdapter; import org.projectnessie.cel.common.types.ref.TypeDescription; +import org.projectnessie.cel.common.types.ref.TypeEnum; import org.projectnessie.cel.common.types.ref.Val; import org.projectnessie.cel.common.types.traits.FieldTester; import org.projectnessie.cel.common.types.traits.Indexer; @@ -57,9 +58,13 @@ public Val convertToType(Type typeVal) { @Override public Val equal(Val other) { + if (other.type().typeEnum() != TypeEnum.Object) { + return False; + } if (!typeDesc.name().equals(other.type().typeName())) { - return noSuchOverload(this, "equal", other); + return False; } + return boolOf(this.value.equals(other.value())); } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/StringT.java b/core/src/main/java/org/projectnessie/cel/common/types/StringT.java index 2d29ba1f..96275a2b 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/StringT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/StringT.java @@ -90,16 +90,6 @@ public Val add(Val other) { return new StringT(s + ((StringT) other).s); } - /** Compare implements traits.Comparer.Compare. */ - @Override - public Val compare(Val other) { - if (!(other instanceof StringT)) { - return noSuchOverload(this, "compare", other); - } - - return intOfCompare(s.compareTo(((StringT) other).s)); - } - /** ConvertToNative implements ref.Val.ConvertToNative. */ @SuppressWarnings("unchecked") @Override @@ -164,13 +154,35 @@ public Val convertToType(Type typeVal) { } } + /** Compare implements traits.Comparer.Compare. */ + @Override + public Val compare(Val other) { + switch (other.type().typeEnum()) { + case String: + return intOfCompare(s.compareTo(((StringT) other).s)); + case Null: + return False; + default: + return noSuchOverload(this, "compare", other); + } + } + /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - if (!(other instanceof StringT)) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case String: + return boolOf(s.equals(((StringT) other).s)); + case Int: + case Uint: + case Double: + case Bool: + return boolOf(s.equals(((StringT) other.convertToType(StringType)).s)); + case Null: + return False; + default: + return noSuchOverload(this, "equal", other); } - return boolOf(s.equals(((StringT) other).s)); } /** Match implements traits.Matcher.Match. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/TimestampT.java b/core/src/main/java/org/projectnessie/cel/common/types/TimestampT.java index 05ec1fa3..c16cf220 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/TimestampT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/TimestampT.java @@ -15,6 +15,7 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.DurationT.durationOf; import static org.projectnessie.cel.common.types.Err.errDurationOverflow; import static org.projectnessie.cel.common.types.Err.errTimestampOutOfRange; @@ -352,10 +353,14 @@ public Val convertToType(Type typeValue) { /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - if (TimestampType != other.type()) { - return noSuchOverload(this, "equal", other); + switch (other.type().typeEnum()) { + case Timestamp: + return boolOf(t.equals(((TimestampT) other).t)); + case Null: + return False; + default: + return noSuchOverload(this, "equal", other); } - return boolOf(t.equals(((TimestampT) other).t)); } /** Receive implements traits.Reciever.Receive. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/TypeT.java b/core/src/main/java/org/projectnessie/cel/common/types/TypeT.java index 3eced2fd..29305291 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/TypeT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/TypeT.java @@ -79,6 +79,11 @@ public long intValue() { throw new UnsupportedOperationException(); } + @Override + public double doubleValue() { + throw new UnsupportedOperationException(); + } + /** ConvertToNative implements ref.Val.ConvertToNative. */ @Override public T convertToNative(Class typeDesc) { diff --git a/core/src/main/java/org/projectnessie/cel/common/types/UintT.java b/core/src/main/java/org/projectnessie/cel/common/types/UintT.java index de2b9432..04d88528 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/UintT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/UintT.java @@ -15,6 +15,8 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.BoolT.False; +import static org.projectnessie.cel.common.types.DoubleT.DoubleType; import static org.projectnessie.cel.common.types.DoubleT.doubleOf; import static org.projectnessie.cel.common.types.Err.divideByZero; import static org.projectnessie.cel.common.types.Err.errUintOverflow; @@ -22,7 +24,9 @@ import static org.projectnessie.cel.common.types.Err.newTypeConversionError; import static org.projectnessie.cel.common.types.Err.noSuchOverload; import static org.projectnessie.cel.common.types.Err.rangeError; +import static org.projectnessie.cel.common.types.IntT.IntOne; import static org.projectnessie.cel.common.types.IntT.intOf; +import static org.projectnessie.cel.common.types.IntT.intOfCompare; import static org.projectnessie.cel.common.types.IntT.maxIntJSON; import static org.projectnessie.cel.common.types.StringT.stringOf; import static org.projectnessie.cel.common.types.Types.boolOf; @@ -32,7 +36,6 @@ import com.google.protobuf.UInt64Value; import com.google.protobuf.Value; import java.math.BigInteger; -import java.util.Objects; import org.projectnessie.cel.common.ULong; import org.projectnessie.cel.common.types.Overflow.OverflowException; import org.projectnessie.cel.common.types.ref.BaseVal; @@ -87,26 +90,9 @@ public long intValue() { return i; } - /** Add implements traits.Adder.Add. */ - @Override - public Val add(Val other) { - if (other.type() != UintType) { - return noSuchOverload(this, "add", other); - } - try { - return uintOf(Overflow.addUint64Checked(i, ((UintT) other).i)); - } catch (OverflowException e) { - return errUintOverflow; - } - } - - /** Compare implements traits.Comparer.Compare. */ @Override - public Val compare(Val other) { - if (other.type() != UintType) { - return noSuchOverload(this, "compare", other); - } - return intOf(Long.compareUnsigned(i, ((UintT) other).i)); + public double doubleValue() { + return (double) i; } /** ConvertToNative implements ref.Val.ConvertToNative. */ @@ -114,13 +100,12 @@ public Val compare(Val other) { @Override public T convertToNative(Class typeDesc) { if (typeDesc == Long.class || typeDesc == long.class || typeDesc == Object.class) { - if (i < 0) { - Err.throwErrorAsIllegalStateException(rangeError(i, "Java long")); - } + // no "is negative" check here, because there is no Java representation of an + // unsigned long, reusing Java's signed long return (T) Long.valueOf(i); } if (typeDesc == Integer.class || typeDesc == int.class) { - if (i < 0 || i > Integer.MAX_VALUE) { + if (i < Integer.MIN_VALUE || i > Integer.MAX_VALUE) { Err.throwErrorAsIllegalStateException(rangeError(i, "Java int")); } return (T) Integer.valueOf((int) i); @@ -180,6 +165,80 @@ public Val convertToType(Type typeValue) { return newTypeConversionError(UintType, typeValue); } + /** Compare implements traits.Comparer.Compare. */ + @Override + public Val compare(Val other) { + switch (other.type().typeEnum()) { + case Int: + if (other.type().typeEnum() == TypeEnum.Err) { + return other; + } + if (other.intValue() < 0L) { + // the other int is < 0, so any uint is greater + return IntOne; + } + if (i < 0L) { + // this uint is > Long.MAX_VALUE, so it MUST be greater than any signed int + return IntOne; + } + return intOfCompare(Long.compareUnsigned(i, other.intValue())); + case Double: + if (other.type().typeEnum() == TypeEnum.Err) { + return other; + } + if (other.doubleValue() < 0d) { + // the other int is < 0, so any uint is greater + return IntOne; + } + DoubleT cmp = (DoubleT) convertToType(DoubleType); + return cmp.compare(other); + case Uint: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + return intOfCompare(Long.compareUnsigned(i, ((UintT) converted).i)); + default: + return noSuchOverload(this, "compare", other); + } + } + + /** Equal implements ref.Val.Equal. */ + @Override + public Val equal(Val other) { + switch (other.type().typeEnum()) { + case Int: + if (other.type().typeEnum() == TypeEnum.Err) { + return other; + } + if (other.intValue() < 0L) { + // the other int is < 0, so no uint can be equal + return False; + } + if (i < 0L) { + // this uint is > Long.MAX_VALUE, so it CANNOT be equal + return False; + } + return boolOf(i == other.intValue()); + case Double: + return other.equal(this); + case Uint: + case String: + Val converted = other.convertToType(type()); + if (converted.type().typeEnum() == TypeEnum.Err) { + return converted; + } + return boolOf(i == converted.intValue()); + case Null: + case Bytes: + case List: + case Map: + return False; + default: + return noSuchOverload(this, "equal", other); + } + } + /** Divide implements traits.Divider.Divide. */ @Override public Val divide(Val other) { @@ -193,15 +252,6 @@ public Val divide(Val other) { return uintOf(i / otherInt); } - /** Equal implements ref.Val.Equal. */ - @Override - public Val equal(Val other) { - if (other.type() != UintType) { - return noSuchOverload(this, "equal", other); - } - return boolOf(i == ((UintT) other).i); - } - /** Modulo implements traits.Modder.Modulo. */ @Override public Val modulo(Val other) { @@ -228,6 +278,19 @@ public Val multiply(Val other) { } } + /** Add implements traits.Adder.Add. */ + @Override + public Val add(Val other) { + if (other.type() != UintType) { + return noSuchOverload(this, "add", other); + } + try { + return uintOf(Overflow.addUint64Checked(i, ((UintT) other).i)); + } catch (OverflowException e) { + return errUintOverflow; + } + } + /** Subtract implements traits.Subtractor.Subtract. */ @Override public Val subtract(Val other) { @@ -258,23 +321,20 @@ public boolean equals(Object o) { if (this == o) { return true; } - if (o == null || getClass() != o.getClass()) { + if (!(o instanceof Val)) { return false; } - UintT uintT = (UintT) o; - return i == uintT.i; + // Defer to CEL's equal functionality to allow heterogeneous numeric map keys + return equal((Val) o).booleanValue(); } @Override public int hashCode() { - return Objects.hash(super.hashCode(), i); + // Used to allow heterogeneous numeric map keys + return (int) i; } - /** - * isJSONSafe indicates whether the uint is safely representable as a floating point value in - * JSON. - */ - public boolean isJSONSafe() { - return i >= 0 && i <= IntT.maxIntJSON; + public String toString() { + return String.format("%s{%s}", type().typeName(), Long.toUnsignedString(i)); } } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/UnknownT.java b/core/src/main/java/org/projectnessie/cel/common/types/UnknownT.java index 42add63d..7630d266 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/UnknownT.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/UnknownT.java @@ -15,6 +15,8 @@ */ package org.projectnessie.cel.common.types; +import static org.projectnessie.cel.common.types.Types.boolOf; + import java.util.Objects; import org.projectnessie.cel.common.types.ref.BaseVal; import org.projectnessie.cel.common.types.ref.Type; @@ -70,7 +72,7 @@ public Val convertToType(Type typeVal) { /** Equal implements ref.Val.Equal. */ @Override public Val equal(Val other) { - return this; + return boolOf(other.type().typeEnum() == TypeEnum.Unknown); } /** Type implements ref.Val.Type. */ diff --git a/core/src/main/java/org/projectnessie/cel/common/types/ref/BaseVal.java b/core/src/main/java/org/projectnessie/cel/common/types/ref/BaseVal.java index 3dee153a..4d914a2a 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/ref/BaseVal.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/ref/BaseVal.java @@ -17,6 +17,7 @@ import static org.projectnessie.cel.common.types.BoolT.BoolType; import static org.projectnessie.cel.common.types.BoolT.True; +import static org.projectnessie.cel.common.types.DoubleT.DoubleType; import static org.projectnessie.cel.common.types.IntT.IntType; public abstract class BaseVal implements Val { @@ -48,4 +49,9 @@ public boolean booleanValue() { public long intValue() { return convertToType(IntType).intValue(); } + + @Override + public double doubleValue() { + return convertToType(DoubleType).doubleValue(); + } } diff --git a/core/src/main/java/org/projectnessie/cel/common/types/ref/Val.java b/core/src/main/java/org/projectnessie/cel/common/types/ref/Val.java index 34de489a..113285ba 100644 --- a/core/src/main/java/org/projectnessie/cel/common/types/ref/Val.java +++ b/core/src/main/java/org/projectnessie/cel/common/types/ref/Val.java @@ -50,4 +50,6 @@ public interface Val { boolean booleanValue(); long intValue(); + + double doubleValue(); } diff --git a/core/src/main/java/org/projectnessie/cel/interpreter/AttributeFactory.java b/core/src/main/java/org/projectnessie/cel/interpreter/AttributeFactory.java index ac5b5679..06008e16 100644 --- a/core/src/main/java/org/projectnessie/cel/interpreter/AttributeFactory.java +++ b/core/src/main/java/org/projectnessie/cel/interpreter/AttributeFactory.java @@ -17,6 +17,7 @@ import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.BoolT.True; +import static org.projectnessie.cel.common.types.DoubleT.doubleOf; import static org.projectnessie.cel.common.types.Err.ErrException; import static org.projectnessie.cel.common.types.Err.indexOutOfBoundsException; import static org.projectnessie.cel.common.types.Err.isError; @@ -27,6 +28,7 @@ import static org.projectnessie.cel.common.types.Err.noSuchOverload; import static org.projectnessie.cel.common.types.Err.throwErrorAsIllegalStateException; import static org.projectnessie.cel.common.types.IntT.intOf; +import static org.projectnessie.cel.common.types.NullT.NullValue; import static org.projectnessie.cel.common.types.StringT.stringOf; import static org.projectnessie.cel.common.types.Types.boolOf; import static org.projectnessie.cel.common.types.UintT.uintOf; @@ -43,7 +45,6 @@ import org.projectnessie.cel.common.ULong; import org.projectnessie.cel.common.containers.Container; import org.projectnessie.cel.common.types.Err; -import org.projectnessie.cel.common.types.NullT; import org.projectnessie.cel.common.types.ref.FieldType; import org.projectnessie.cel.common.types.ref.TypeAdapter; import org.projectnessie.cel.common.types.ref.TypeProvider; @@ -729,12 +730,18 @@ static Qualifier newQualifierStatic(TypeAdapter adapter, long id, Object v) { switch (val.type().typeEnum()) { case String: return new StringQualifier(id, (String) val.value(), val, adapter); + case Double: + return new DoubleQualifier(id, (double) val.value(), val, adapter); case Int: return new IntQualifier(id, val.intValue(), val, adapter); case Uint: return new UintQualifier(id, val.intValue(), val, adapter); case Bool: return new BoolQualifier(id, val.booleanValue(), val, adapter); + case Null: + // Not actually a qualifier, but conformance-tests require this, although it's actually an + // error condition. + return new NullQualifier(id, val, adapter); } } @@ -749,6 +756,10 @@ static Qualifier newQualifierStatic(TypeAdapter adapter, long id, Object v) { long i = ((Number) v).longValue(); return new IntQualifier(id, i, intOf(i), adapter); } + if (c == Double.class) { + double b = (Double) v; + return new DoubleQualifier(id, b, doubleOf(b), adapter); + } if (c == Boolean.class) { boolean b = (Boolean) v; return new BoolQualifier(id, b, boolOf(b), adapter); @@ -828,7 +839,7 @@ public Object qualify(org.projectnessie.cel.interpreter.Activation vars, Object obj = m.get(s); if (obj == null) { if (m.containsKey(s)) { - return NullT.NullValue; + return NullValue; } throw noSuchKeyException(s); } @@ -876,6 +887,105 @@ public String toString() { } } + final class DoubleQualifier implements Coster, ConstantQualifierEquator { + final long id; + final double value; + final Val celValue; + final TypeAdapter adapter; + + DoubleQualifier(long id, double value, Val celValue, TypeAdapter adapter) { + this.id = id; + this.value = value; + this.celValue = celValue; + this.adapter = adapter; + } + + /** ID is an implementation of the Qualifier interface method. */ + @Override + public long id() { + return id; + } + + /** Qualify implements the Qualifier interface method. */ + @SuppressWarnings("rawtypes") + @Override + public Object qualify(org.projectnessie.cel.interpreter.Activation vars, Object obj) { + double i = value; + if (obj instanceof Map) { + Map m = (Map) obj; + obj = m.get(i); + if (obj == null) { + obj = m.get((int) i); + } + if (obj == null) { + if (m.containsKey(i) || m.containsKey((int) i)) { + return null; + } + throw noSuchKeyException(i); + } + return obj; + } + if (obj.getClass().isArray()) { + int l = Array.getLength(obj); + if (i < 0 || i >= l) { + throw indexOutOfBoundsException(i); + } + obj = Array.get(obj, (int) i); + return obj; + } + if (obj instanceof List) { + List list = (List) obj; + int l = list.size(); + if (i < 0 || i >= l) { + throw indexOutOfBoundsException(i); + } + obj = list.get((int) i); + return obj; + } + if (isUnknown(obj)) { + return obj; + } + return refResolve(adapter, celValue, obj); + } + + /** Value implements the ConstantQualifier interface */ + @Override + public Val value() { + return celValue; + } + + /** Cost returns zero for constant field qualifiers */ + @Override + public Cost cost() { + return Cost.None; + } + + @Override + public boolean qualifierValueEquals(Object value) { + if (value instanceof ULong) { + return false; + } + if (value instanceof Number) { + return this.value == ((Number) value).doubleValue(); + } + return false; + } + + @Override + public String toString() { + return "DoubleQualifier{" + + "id=" + + id + + ", value=" + + value + + ", celValue=" + + celValue + + ", adapter=" + + adapter + + '}'; + } + } + final class IntQualifier implements Coster, ConstantQualifierEquator { final long id; final long value; @@ -1132,6 +1242,63 @@ public String toString() { } } + /** + * Not actually a qualifier, but conformance-tests require this, although it's actually an error + * condition. + */ + final class NullQualifier implements Coster, ConstantQualifierEquator { + final long id; + final Val celValue; + final TypeAdapter adapter; + + NullQualifier(long id, Val celValue, TypeAdapter adapter) { + this.id = id; + this.celValue = celValue; + this.adapter = adapter; + } + + /** ID is an implementation of the Qualifier interface method. */ + @Override + public long id() { + return id; + } + + /** Qualify implements the Qualifier interface method. */ + @Override + public Object qualify(org.projectnessie.cel.interpreter.Activation vars, Object obj) { + return null; + } + + /** Value implements the ConstantQualifier interface */ + @Override + public Val value() { + return NullValue; + } + + /** Cost returns zero for constant field qualifiers */ + @Override + public Cost cost() { + return Cost.None; + } + + @Override + public boolean qualifierValueEquals(Object value) { + return value == null || value == NullValue; + } + + @Override + public String toString() { + return "NullQualifier{" + + "id=" + + id + + ", celValue=" + + celValue + + ", adapter=" + + adapter + + '}'; + } + } + /** * fieldQualifier indicates that the qualification is a well-defined field with a known field * type. When the field type is known this can be used to improve the speed and efficiency of diff --git a/core/src/main/java/org/projectnessie/cel/interpreter/Interpretable.java b/core/src/main/java/org/projectnessie/cel/interpreter/Interpretable.java index 9480118d..7b367707 100644 --- a/core/src/main/java/org/projectnessie/cel/interpreter/Interpretable.java +++ b/core/src/main/java/org/projectnessie/cel/interpreter/Interpretable.java @@ -34,12 +34,14 @@ import java.util.Objects; import java.util.Set; import org.projectnessie.cel.common.operators.Operator; +import org.projectnessie.cel.common.types.Err; import org.projectnessie.cel.common.types.IterableT; import org.projectnessie.cel.common.types.IteratorT; import org.projectnessie.cel.common.types.Overloads; import org.projectnessie.cel.common.types.StringT; import org.projectnessie.cel.common.types.ref.FieldType; import org.projectnessie.cel.common.types.ref.TypeAdapter; +import org.projectnessie.cel.common.types.ref.TypeEnum; import org.projectnessie.cel.common.types.ref.TypeProvider; import org.projectnessie.cel.common.types.ref.Val; import org.projectnessie.cel.common.types.traits.Container; @@ -869,11 +871,17 @@ public Val eval(org.projectnessie.cel.interpreter.Activation ctx) { if (isUnknownOrError(keyVal)) { return keyVal; } + if (keyVal.type().typeEnum() == TypeEnum.Null) { + return newErr("unsupported key type"); + } Val valVal = vals[i].eval(ctx); if (isUnknownOrError(valVal)) { return valVal; } - entries.put(keyVal, valVal); + if (entries.putIfAbsent(keyVal, valVal) != null) { + // Prevent duplicate keys, error out. + return newErr("Failed with repeated key"); + } } return adapter.nativeToValue(entries); } diff --git a/core/src/test/java/org/projectnessie/cel/CELTest.java b/core/src/test/java/org/projectnessie/cel/CELTest.java index f0cd99ac..97cd7d22 100644 --- a/core/src/test/java/org/projectnessie/cel/CELTest.java +++ b/core/src/test/java/org/projectnessie/cel/CELTest.java @@ -49,6 +49,7 @@ import static org.projectnessie.cel.ProgramOption.functions; import static org.projectnessie.cel.ProgramOption.globals; import static org.projectnessie.cel.Util.mapOf; +import static org.projectnessie.cel.common.types.BoolT.False; import static org.projectnessie.cel.common.types.BoolT.True; import static org.projectnessie.cel.common.types.Err.isError; import static org.projectnessie.cel.common.types.Err.newErr; @@ -656,10 +657,10 @@ void ResidualAst_Complex() { assertThat(astIss.hasIssues()).isFalse(); Program prg = e.program(astIss.getAst(), evalOptions(OptTrackState, OptPartialEval)); EvalResult outDet = prg.eval(unkVars); - assertThat(outDet.getVal()).matches(UnknownT::isUnknown); + assertThat(outDet.getVal()).isSameAs(False); Ast residual = e.residualAst(astIss.getAst(), outDet.getEvalDetails()); String expr = astToString(residual); - assertThat(expr).isEqualTo("request.auth.claims.email == \"wiley@acme.co\""); + assertThat(expr).isEqualTo("false"); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/checker/CheckerTest.java b/core/src/test/java/org/projectnessie/cel/checker/CheckerTest.java index 38739f62..6bc4194c 100644 --- a/core/src/test/java/org/projectnessie/cel/checker/CheckerTest.java +++ b/core/src/test/java/org/projectnessie/cel/checker/CheckerTest.java @@ -751,10 +751,9 @@ static TestCase[] checkTestCases() { .idents( Decls.newVar( "x", Decls.newObjectType("google.api.expr.test.v1.proto3.TestAllTypes")))) - .error( - "ERROR: :1:16: found no matching overload for '_!=_' applied to '(int, null)'\n" - + " | x.single_int64 != null\n" - + " | ...............^"), + .r( + "_!=_(x~google.api.expr.test.v1.proto3.TestAllTypes^x.single_int64~int,null~null)~bool^not_equals") + .type(Decls.Bool), new TestCase() .i("x.single_int64_wrapper == null") .env( diff --git a/core/src/test/java/org/projectnessie/cel/common/types/DoubleTest.java b/core/src/test/java/org/projectnessie/cel/common/types/DoubleTest.java index 8bb14439..5dd224c2 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/DoubleTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/DoubleTest.java @@ -26,10 +26,12 @@ import static org.projectnessie.cel.common.types.IntT.IntType; import static org.projectnessie.cel.common.types.IntT.IntZero; import static org.projectnessie.cel.common.types.IntT.intOf; +import static org.projectnessie.cel.common.types.NullT.NullValue; import static org.projectnessie.cel.common.types.StringT.StringType; import static org.projectnessie.cel.common.types.StringT.stringOf; import static org.projectnessie.cel.common.types.TypeT.TypeType; import static org.projectnessie.cel.common.types.UintT.UintType; +import static org.projectnessie.cel.common.types.UintT.uintOf; import com.google.protobuf.Any; import com.google.protobuf.DoubleValue; @@ -53,6 +55,16 @@ void doubleCompare() { assertThat(gt.compare(lt).equal(IntOne)).isSameAs(True); assertThat(gt.compare(gt).equal(IntZero)).isSameAs(True); assertThat(gt.compare(TypeType)).matches(Err::isError); + + assertThat(doubleOf(1e70).compare(intOf(5))).isSameAs(IntOne); + assertThat(doubleOf(5).compare(intOf(5))).isSameAs(IntZero); + assertThat(doubleOf(5).compare(intOf(-5))).isSameAs(IntOne); + assertThat(doubleOf(5).compare(intOf(Integer.MAX_VALUE))).isSameAs(IntNegOne); + + assertThat(doubleOf(1e70).compare(doubleOf(5))).isSameAs(IntOne); + assertThat(doubleOf(5).compare(doubleOf(5))).isSameAs(IntZero); + assertThat(doubleOf(5).compare(doubleOf(1e70d))).isSameAs(IntNegOne); + assertThat(doubleOf(5).compare(doubleOf(1e-70d))).isSameAs(IntOne); } @Test @@ -161,6 +173,15 @@ void doubleDivide() { @Test void doubleEqual() { assertThat(doubleOf(0).equal(False)).matches(Err::isError); + assertThat(doubleOf(0).equal(NullValue)).isSameAs(False); + assertThat(doubleOf(0).equal(intOf(0))).isSameAs(True); + assertThat(doubleOf(0).equal(intOf(1))).isSameAs(False); + assertThat(doubleOf(0).equal(uintOf(0))).isSameAs(True); + assertThat(doubleOf(0).equal(uintOf(1))).isSameAs(False); + assertThat(doubleOf(0).equal(doubleOf(0))).isSameAs(True); + assertThat(doubleOf(0).equal(doubleOf(1))).isSameAs(False); + assertThat(doubleOf(0).equal(stringOf("0"))).isSameAs(True); + assertThat(doubleOf(0).equal(stringOf("1"))).isSameAs(False); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/common/types/DurationTest.java b/core/src/test/java/org/projectnessie/cel/common/types/DurationTest.java index 0b65be60..d6a70eed 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/DurationTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/DurationTest.java @@ -162,9 +162,9 @@ void durationGetSeconds() { @Test void durationGetMilliseconds() { - DurationT d = durationOf(ofSeconds(7506, 0)); + DurationT d = durationOf(ofSeconds(7506, 321456789)); Val min = d.receive(Overloads.TimeGetMilliseconds, Overloads.DurationToMilliseconds); - assertThat(min.equal(intOf(7506000))).isSameAs(True); + assertThat(min.equal(intOf(321))).isSameAs(True); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/common/types/IntTest.java b/core/src/test/java/org/projectnessie/cel/common/types/IntTest.java index 0c1cce09..17984d7f 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/IntTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/IntTest.java @@ -28,6 +28,7 @@ import static org.projectnessie.cel.common.types.IntT.IntZero; import static org.projectnessie.cel.common.types.IntT.intOf; import static org.projectnessie.cel.common.types.IntT.maxIntJSON; +import static org.projectnessie.cel.common.types.NullT.NullValue; import static org.projectnessie.cel.common.types.StringT.StringType; import static org.projectnessie.cel.common.types.StringT.stringOf; import static org.projectnessie.cel.common.types.TimestampT.TimestampType; @@ -73,6 +74,15 @@ void intCompare() { .isInstanceOf(Err.class) .extracting(Object::toString) .isEqualTo("no such overload: int.compare(type)"); + + assertThat(intOf(-5).compare(uintOf(5))).isSameAs(IntNegOne); + assertThat(intOf(5).compare(uintOf(5))).isSameAs(IntZero); + assertThat(intOf(5).compare(uintOf(0x8000000000000000L))).isSameAs(IntNegOne); + + assertThat(intOf(-5).compare(doubleOf(5))).isSameAs(IntNegOne); + assertThat(intOf(5).compare(doubleOf(5))).isSameAs(IntZero); + assertThat(intOf(5).compare(doubleOf(1e70d))).isSameAs(IntNegOne); + assertThat(intOf(5).compare(doubleOf(1e-70d))).isSameAs(IntOne); } @Test @@ -177,6 +187,19 @@ void intEqual() { .isInstanceOf(Err.class) .extracting(Object::toString) .isEqualTo("no such overload: int.equal(bool)"); + assertThat(intOf(0).equal(NullValue)).isSameAs(False); + assertThat(intOf(0).equal(stringOf("0"))).isSameAs(True); + assertThat(intOf(0).equal(stringOf("1"))).isSameAs(False); + assertThat(intOf(0).equal(intOf(0))).isSameAs(True); + assertThat(intOf(0).equal(intOf(1))).isSameAs(False); + assertThat(intOf(0).equal(uintOf(0))).isSameAs(True); + assertThat(intOf(0).equal(uintOf(1))).isSameAs(False); + assertThat(intOf(0x8000000000000000L).equal(uintOf(0x8000000000000000L))).isSameAs(False); + assertThat(intOf(0xffffffffffffffffL).equal(uintOf(0xffffffffffffffffL))).isSameAs(False); + assertThat(intOf(0).equal(doubleOf(0))).isSameAs(True); + assertThat(intOf(0).equal(doubleOf(1))).isSameAs(False); + assertThat(intOf(0).equal(stringOf("0"))).isSameAs(True); + assertThat(intOf(0).equal(stringOf("1"))).isSameAs(False); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/common/types/MapTest.java b/core/src/test/java/org/projectnessie/cel/common/types/MapTest.java index e5bbc419..f409c6f7 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/MapTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/MapTest.java @@ -15,11 +15,90 @@ */ package org.projectnessie.cel.common.types; +import static org.assertj.core.api.Assertions.assertThat; +import static org.projectnessie.cel.common.types.BoolT.True; +import static org.projectnessie.cel.common.types.DoubleT.doubleOf; +import static org.projectnessie.cel.common.types.IntT.intOf; +import static org.projectnessie.cel.common.types.MapT.newWrappedMap; +import static org.projectnessie.cel.common.types.StringT.stringOf; +import static org.projectnessie.cel.common.types.Types.boolOf; +import static org.projectnessie.cel.common.types.UintT.uintOf; +import static org.projectnessie.cel.common.types.pb.ProtoTypeRegistry.newRegistry; + +import com.google.common.collect.ImmutableMap; +import java.util.HashMap; +import java.util.Map; import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.Test; +import org.projectnessie.cel.common.types.ref.Val; public class MapTest { + @Test + void heterogenousKeys() { + Map javaMap = + ImmutableMap.of( + intOf(1), + stringOf("one"), + uintOf(2), + stringOf("two"), + doubleOf(3.1d), + stringOf("three"), + boolOf(true), + stringOf("true"), + stringOf("str"), + stringOf("string")); + MapT celMap = (MapT) newWrappedMap(newRegistry(), javaMap); + + assertThat(celMap.size()).isEqualTo(intOf(javaMap.size())); + + assertThat(celMap.find(intOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.find(uintOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.find(doubleOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.find(intOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.find(uintOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.find(doubleOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.find(intOf(3))).isNull(); + assertThat(celMap.find(uintOf(3))).isNull(); + assertThat(celMap.find(doubleOf(3.1d))).isEqualTo(stringOf("three")); + assertThat(celMap.find(boolOf(true))).isEqualTo(stringOf("true")); + assertThat(celMap.find(stringOf("str"))).isEqualTo(stringOf("string")); + + assertThat(celMap.get(intOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.get(uintOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.get(doubleOf(1))).isEqualTo(stringOf("one")); + assertThat(celMap.get(intOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.get(uintOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.get(doubleOf(2))).isEqualTo(stringOf("two")); + assertThat(celMap.get(intOf(3))).isNull(); + assertThat(celMap.get(uintOf(3))).isNull(); + assertThat(celMap.get(doubleOf(3.1d))).isEqualTo(stringOf("three")); + assertThat(celMap.get(boolOf(true))).isEqualTo(stringOf("true")); + assertThat(celMap.get(stringOf("str"))).isEqualTo(stringOf("string")); + + assertThat(celMap.contains(intOf(1))).isSameAs(True); + assertThat(celMap.contains(uintOf(1))).isSameAs(True); + assertThat(celMap.contains(doubleOf(1))).isSameAs(True); + + assertThat(celMap.contains(intOf(1))).isSameAs(True); + assertThat(celMap.contains(uintOf(1))).isSameAs(True); + assertThat(celMap.contains(doubleOf(1))).isSameAs(True); + + assertThat(celMap.find(intOf(42))).isNull(); + + IteratorT iter = celMap.iterator(); + Map mapFromIter = new HashMap<>(); + while (iter.hasNext() == True) { + Val key = iter.next(); + mapFromIter.put(key, celMap.find(key)); + } + assertThat(mapFromIter).hasSize(javaMap.size()).containsAllEntriesOf(javaMap); + + assertThat(celMap.convertToNative(Map.class)) + .isEqualTo( + ImmutableMap.of(1L, "one", 2L, "two", 3.1d, "three", true, "true", "str", "string")); + } + // type testStruct struct { // M string // Details []string diff --git a/core/src/test/java/org/projectnessie/cel/common/types/StringTest.java b/core/src/test/java/org/projectnessie/cel/common/types/StringTest.java index f9efb246..63e48256 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/StringTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/StringTest.java @@ -155,10 +155,7 @@ void stringConvertToType() { void stringEqual() { assertThat(stringOf("hello").equal(stringOf("hello"))).isSameAs(True); assertThat(stringOf("hello").equal(stringOf("hell"))).isSameAs(False); - assertThat(stringOf("c").equal(intOf(99))) - .isInstanceOf(Err.class) - .extracting(Object::toString) - .isEqualTo("no such overload: string.equal(int)"); + assertThat(stringOf("c").equal(intOf(99))).isSameAs(False); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/common/types/UintTest.java b/core/src/test/java/org/projectnessie/cel/common/types/UintTest.java index ba471ff0..4622b39d 100644 --- a/core/src/test/java/org/projectnessie/cel/common/types/UintTest.java +++ b/core/src/test/java/org/projectnessie/cel/common/types/UintTest.java @@ -29,6 +29,7 @@ import static org.projectnessie.cel.common.types.IntT.intOf; import static org.projectnessie.cel.common.types.IntT.maxIntJSON; import static org.projectnessie.cel.common.types.MapT.MapType; +import static org.projectnessie.cel.common.types.NullT.NullValue; import static org.projectnessie.cel.common.types.StringT.StringType; import static org.projectnessie.cel.common.types.StringT.stringOf; import static org.projectnessie.cel.common.types.TypeT.TypeType; @@ -70,6 +71,16 @@ void uintCompare() { .isInstanceOf(Err.class) .extracting(Object::toString) .isEqualTo("no such overload: uint.compare(type)"); + + assertThat(uintOf(0xffff800000000000L).compare(intOf(5))).isSameAs(IntOne); + assertThat(uintOf(5).compare(intOf(5))).isSameAs(IntZero); + assertThat(uintOf(5).compare(intOf(-5))).isSameAs(IntOne); + assertThat(uintOf(5).compare(intOf(Integer.MAX_VALUE))).isSameAs(IntNegOne); + + assertThat(uintOf(0xffff800000000000L).compare(doubleOf(5))).isSameAs(IntOne); + assertThat(uintOf(5).compare(doubleOf(5))).isSameAs(IntZero); + assertThat(uintOf(5).compare(doubleOf(1e70d))).isSameAs(IntNegOne); + assertThat(uintOf(5).compare(doubleOf(1e-70d))).isSameAs(IntOne); } @Test @@ -169,6 +180,17 @@ void uintEqual() { .isInstanceOf(Err.class) .extracting(Object::toString) .isEqualTo("no such overload: uint.equal(bool)"); + assertThat(uintOf(0).equal(NullValue)).isSameAs(False); + assertThat(uintOf(0).equal(intOf(0))).isSameAs(True); + assertThat(uintOf(0).equal(intOf(1))).isSameAs(False); + assertThat(uintOf(0x8000000000000000L).equal(intOf(0x8000000000000000L))).isSameAs(False); + assertThat(uintOf(0xffffffffffffffffL).equal(intOf(0xffffffffffffffffL))).isSameAs(False); + assertThat(uintOf(0).equal(uintOf(0))).isSameAs(True); + assertThat(uintOf(0).equal(uintOf(1))).isSameAs(False); + assertThat(uintOf(0).equal(doubleOf(0))).isSameAs(True); + assertThat(uintOf(0).equal(doubleOf(1))).isSameAs(False); + assertThat(uintOf(0).equal(stringOf("0"))).isSameAs(True); + assertThat(uintOf(0).equal(stringOf("1"))).isSameAs(False); } @Test diff --git a/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTest.java b/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTest.java index d0e8ec3c..ce58f0ad 100644 --- a/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTest.java +++ b/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTest.java @@ -239,6 +239,60 @@ TestCase err(String err) { @SuppressWarnings("unused") static TestCase[] testCases() { return new TestCase[] { + new TestCase(InterpreterTestCase.map_key_null) + .expr("{null:false}[null]") + .err("message: unsupported key type"), + new TestCase(InterpreterTestCase.map_value_repeat_key_heterogeneous) + .expr("{0: 1, 0u: 2}[0.0]") + .err("message: Failed with repeated key"), + new TestCase(InterpreterTestCase.map_key_mixed_numbers_lossy_double_key) + .expr("{1u: 1.0, 2: 2.0, 3u: 3.0}[3.1]") + .err("no such key: double{3.1}"), + new TestCase(InterpreterTestCase.zero_based_double_error) + .expr("[7, 8, 9][dyn(0.1)]") + .err("invalid_argument"), + new TestCase(InterpreterTestCase.zero_based_double).expr("[7, 8, 9][dyn(0.0)]").out(intOf(7)), + new TestCase(InterpreterTestCase.not_int32_eq_uint) + .expr("Int32Value{value: 34} == dyn(UInt64Value{value: 18446744073709551615u})") + .container("google.protobuf") + .out(False), + new TestCase(InterpreterTestCase.not_uint32_eq_double) + .expr("UInt32Value{value: 34u} == dyn(DoubleValue{value: 18446744073709551616.0})") + .container("google.protobuf") + .out(False), + new TestCase(InterpreterTestCase.eq_proto_different_types) + .expr("dyn(TestAllTypes{}) == dyn(NestedTestAllTypes{})") + .container("google.api.expr.test.v1.proto2") + .types( + com.google.api.expr.test.v1.proto2.TestAllTypesProto.TestAllTypes + .getDefaultInstance()) + .out(False), + new TestCase(InterpreterTestCase.not_lt_dyn_big_uint_int) + .expr("dyn(9223372036854775808u) < 1") + .out(False), + new TestCase(InterpreterTestCase.lt_dyn_int_big_uint) + .expr("dyn(1) < 9223372036854775808u") + .out(True), + new TestCase(InterpreterTestCase.lt_dyn_uint_big_double) + .expr("dyn(18446744073709551615u) < 18446744073709590000.0") + .out(True), + new TestCase(InterpreterTestCase.not_lt_dyn_uint_small_int) + .expr("dyn(1u) < -9223372036854775808") + .out(False), + new TestCase(InterpreterTestCase.lt_ne_dyn_int_double).expr("dyn(24) != 24.1").out(True), + new TestCase(InterpreterTestCase.eq_proto_nan_equal) + .expr( + "TestAllTypes{single_double: double('NaN')} == TestAllTypes{single_double: double('NaN')}") + .container("google.api.expr.test.v1.proto3") + .types( + com.google.api.expr.test.v1.proto3.TestAllTypesProto.TestAllTypes + .getDefaultInstance()) + // The outcome in the generated Java proto code is different than in the conformance-test, + // it is NOT: "For proto equality, fields with NaN value are treated as not equal." + .out(True), + new TestCase(InterpreterTestCase.eq_bool_not_null) + .expr("google.protobuf.BoolValue{} != null") + .out(True), new TestCase(InterpreterTestCase.literal_any) .expr( "google.protobuf.Any{type_url: 'type.googleapis.com/google.api.expr.test.v1.proto2.TestAllTypes', value: b'\\x08\\x96\\x01'}") @@ -272,9 +326,9 @@ static TestCase[] testCases() { com.google.api.expr.test.v1.proto3.TestAllTypesProto.TestAllTypes .getDefaultInstance()) .out(Struct.getDefaultInstance()), - new TestCase(InterpreterTestCase.elem_in_mixed_type_list_error) + new TestCase(InterpreterTestCase.elem_in_mixed_type_list2) .expr("'elem' in [1u, 'str', 2, b'bytes']") - .err("no such overload: string.@in(uint,bytes,...)"), + .out(False), new TestCase(InterpreterTestCase.elem_in_mixed_type_list) .expr("'elem' in [1, 'elem', 2]") .out(boolOf(true)), @@ -329,10 +383,22 @@ static TestCase[] testCases() { .setStandaloneEnumValue(-3) .build()) .out(intOf(-3)), - new TestCase(InterpreterTestCase.eq_list_elem_mixed_types_error) + new TestCase(InterpreterTestCase.eq_list_mixed_type_numbers) + .expr("[1.0, 2.0, 3] == [1u, 2, 3u]") + .out(True), + new TestCase(InterpreterTestCase.not_eq_list_mixed_type_numbers) + .expr("[1.0, 2.1] == [1u, 2]") + .out(False), + new TestCase(InterpreterTestCase.eq_list_elem_mixed_types_one_element) .expr("[1] == [1.0]") - .unchecked() - .err("no such overload: int._==_(double)"), + .out(True), + new TestCase(InterpreterTestCase.eq_list_elem_one_element) + .expr("['str'] == ['str']") + .out(True), + new TestCase(InterpreterTestCase.not_eq_list_one_element) + .expr("['str'] == ['blah']") + .out(False), + new TestCase(InterpreterTestCase.not_eq_list_one_element2).expr("[1] == [2]").out(False), new TestCase(InterpreterTestCase.parse_nest_message_literal) .container("google.api.expr.test.v1.proto3") .expr( @@ -1135,10 +1201,10 @@ static TestCase[] testCases() { .env(Decls.newVar("x", Decls.Duration)) .in( "x", - com.google.protobuf.Duration.newBuilder().setSeconds(123).setNanos(123456789).build()) + com.google.protobuf.Duration.newBuilder().setSeconds(123).setNanos(321456789).build()) .cost(costOf(2, 2)) .exhaustiveCost(costOf(2, 2)) - .out(123123), + .out(321), new TestCase(InterpreterTestCase.timestamp_get_hours_tz) .expr("timestamp('2009-02-13T23:31:30Z').getHours('2:00')") .out(intOf(1)) @@ -1563,6 +1629,7 @@ private void typeConversionOptCheck( if (tc.out != null) { assertThat(i).isInstanceOf(InterpretableConst.class); InterpretableConst ic = (InterpretableConst) i; + ic.value().equal(tc.out); assertThat(ic.value()).extracting(o -> o.equal(tc.out)).isSameAs(True); } } diff --git a/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTestCase.java b/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTestCase.java index 97045a41..2f0dbdf4 100644 --- a/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTestCase.java +++ b/core/src/test/java/org/projectnessie/cel/interpreter/InterpreterTestCase.java @@ -40,8 +40,14 @@ public enum InterpreterTestCase { cond_bad_type, duration_get_milliseconds, elem_in_mixed_type_list, - elem_in_mixed_type_list_error, + elem_in_mixed_type_list2, + eq_bool_not_null, + eq_list_elem_mixed_types_one_element, + eq_list_elem_one_element, eq_list_elem_mixed_types_error, + eq_list_mixed_type_numbers, + eq_proto_different_types, + eq_proto_nan_equal, in_list, in_map, index, @@ -63,6 +69,11 @@ public enum InterpreterTestCase { literal_pb_struct, literal_var, literal_any, + lt_dyn_int_big_uint, + lt_dyn_uint_big_double, + lt_ne_dyn_int_double, + map_key_mixed_numbers_lossy_double_key, + map_value_repeat_key_heterogeneous, timestamp_eq_timestamp, timestamp_ne_timestamp, timestamp_lt_timestamp, @@ -82,9 +93,17 @@ public enum InterpreterTestCase { macro_has_pb2_field, macro_has_pb3_field, macro_map, + map_key_null, matches, nested_proto_field, nested_proto_field_with_index, + not_eq_list_one_element, + not_eq_list_one_element2, + not_eq_list_mixed_type_numbers, + not_int32_eq_uint, + not_uint32_eq_double, + not_lt_dyn_big_uint_int, + not_lt_dyn_uint_small_int, or_true_1st, or_true_2nd, or_false, @@ -118,5 +137,7 @@ public enum InterpreterTestCase { select_subsumed_field, select_empty_repeated_nested, root_null_handling, - root_no_such_attribute + root_no_such_attribute, + zero_based_double, + zero_based_double_error } diff --git a/generated-pb/build.gradle.kts b/generated-pb/build.gradle.kts index 07d87732..97a8ffc5 100644 --- a/generated-pb/build.gradle.kts +++ b/generated-pb/build.gradle.kts @@ -29,7 +29,6 @@ apply() sourceSets.main { java.srcDir(layout.buildDirectory.dir("generated/source/proto/main/java")) - java.srcDir(layout.buildDirectory.dir("generated/source/proto/main/grpc")) java.destinationDirectory.set(layout.buildDirectory.dir("classes/java/generated")) } @@ -56,10 +55,6 @@ configure { // Download from repositories artifact = "com.google.protobuf:protoc:${libs.versions.protobuf.get()}" } - plugins { - this.create("grpc") { artifact = "io.grpc:protoc-gen-grpc-java:${libs.versions.grpc.get()}" } - } - generateProtoTasks { all().configureEach { this.plugins.create("grpc") {} } } } reflectionConfig { diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index cc87c0f3..6a8e739b 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -6,10 +6,9 @@ errorpronePlugin = "3.1.0" errorproneSlf4j = "0.1.20" googleJavaFormat = "1.18.1" grpc = "1.59.0" -guava = "31.1-jre" immutables = "2.10.0" jacoco = "0.8.11" -jandex = "2.4.3.Final" +jandex = "3.1.5" jandexPlugin = "1.90" jmh = "1.37" junit = "5.10.1" @@ -44,6 +43,7 @@ google-java-format = { module = "com.google.googlejavaformat:google-java-format" grpc-protobuf = { module = "io.grpc:grpc-protobuf", version.ref = "grpc" } grpc-stub = { module = "io.grpc:grpc-stub", version.ref = "grpc" } grpc-netty-shaded = { module = "io.grpc:grpc-netty-shaded", version.ref = "grpc" } +guava = { module = "com.google.guava:guava", version = "32.1.3-jre" } idea-ext = { module = "gradle.plugin.org.jetbrains.gradle.plugin.idea-ext:gradle-idea-ext", version = "1.1.7" } immutables-value-annotations = { module = "org.immutables:value-annotations", version.ref = "immutables" } immutables-value-processor = { module = "org.immutables:value-processor", version.ref = "immutables" } diff --git a/submodules/cel-spec b/submodules/cel-spec index 8ff24bab..7eb4db1a 160000 --- a/submodules/cel-spec +++ b/submodules/cel-spec @@ -1 +1 @@ -Subproject commit 8ff24babd31ef9caab6f24f303b7bd7528e2954b +Subproject commit 7eb4db1aa8cebecb71b2b33a0ced33b9ae5f4fdc diff --git a/submodules/googleapis b/submodules/googleapis index 0bcee5c6..f2d78630 160000 --- a/submodules/googleapis +++ b/submodules/googleapis @@ -1 +1 @@ -Subproject commit 0bcee5c673ecd59ffae84e8104aaa68c68e7e43a +Subproject commit f2d78630d2c1d5e20041dfff963e093de9298e4d