diff --git a/CHANGELOG.md b/CHANGELOG.md index d9e6627082..89d4d7814f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -32,6 +32,7 @@ Thank you to all who have contributed! ### Changed - Change `StaticType.AnyOfType`'s `.toString` to not perform `.flatten()` +- Change modeling of `COALESCE` and `NULLIF` to dedicated nodes in logical plan ### Deprecated - The current SqlBlock, SqlDialect, and SqlLayout are marked as deprecated and will be slightly changed in the next release. diff --git a/partiql-plan/src/main/resources/partiql_plan.ion b/partiql-plan/src/main/resources/partiql_plan.ion index 1e970807f3..78ab9f5ab2 100644 --- a/partiql-plan/src/main/resources/partiql_plan.ion +++ b/partiql-plan/src/main/resources/partiql_plan.ion @@ -135,6 +135,15 @@ rex::{ ], }, + nullif::{ + value: rex, + nullifier: rex + }, + + coalesce::{ + args: list::[rex] + }, + collection::{ values: list::[rex], }, diff --git a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/ir/Nodes.kt b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/ir/Nodes.kt index 0d444cb340..db65156fa8 100644 --- a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/ir/Nodes.kt +++ b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/ir/Nodes.kt @@ -51,10 +51,12 @@ import org.partiql.planner.internal.ir.builder.RexOpCallDynamicCandidateBuilder import org.partiql.planner.internal.ir.builder.RexOpCallStaticBuilder import org.partiql.planner.internal.ir.builder.RexOpCaseBranchBuilder import org.partiql.planner.internal.ir.builder.RexOpCaseBuilder +import org.partiql.planner.internal.ir.builder.RexOpCoalesceBuilder import org.partiql.planner.internal.ir.builder.RexOpCollectionBuilder import org.partiql.planner.internal.ir.builder.RexOpErrBuilder import org.partiql.planner.internal.ir.builder.RexOpGlobalBuilder import org.partiql.planner.internal.ir.builder.RexOpLitBuilder +import org.partiql.planner.internal.ir.builder.RexOpNullifBuilder import org.partiql.planner.internal.ir.builder.RexOpPathIndexBuilder import org.partiql.planner.internal.ir.builder.RexOpPathKeyBuilder import org.partiql.planner.internal.ir.builder.RexOpPathSymbolBuilder @@ -312,6 +314,8 @@ internal data class Rex( is Path -> visitor.visitRexOpPath(this, ctx) is Call -> visitor.visitRexOpCall(this, ctx) is Case -> visitor.visitRexOpCase(this, ctx) + is Nullif -> visitor.visitRexOpNullif(this, ctx) + is Coalesce -> visitor.visitRexOpCoalesce(this, ctx) is Collection -> visitor.visitRexOpCollection(this, ctx) is Struct -> visitor.visitRexOpStruct(this, ctx) is Pivot -> visitor.visitRexOpPivot(this, ctx) @@ -567,6 +571,47 @@ internal data class Rex( } } + internal data class Nullif( + @JvmField + internal val value: Rex, + @JvmField + internal val nullifier: Rex, + ) : Op() { + internal override val children: List by lazy { + val kids = mutableListOf() + kids.add(value) + kids.add(nullifier) + kids.filterNotNull() + } + + internal override fun accept(visitor: PlanVisitor, ctx: C): R = + visitor.visitRexOpNullif(this, ctx) + + internal companion object { + @JvmStatic + internal fun builder(): RexOpNullifBuilder = RexOpNullifBuilder() + } + } + + internal data class Coalesce( + @JvmField + internal val args: List, + ) : Op() { + override val children: List by lazy { + val kids = mutableListOf() + kids.addAll(args) + kids.filterNotNull() + } + + override fun accept(visitor: PlanVisitor, ctx: C): R = + visitor.visitRexOpCoalesce(this, ctx) + + internal companion object { + @JvmStatic + internal fun builder(): RexOpCoalesceBuilder = RexOpCoalesceBuilder() + } + } + internal data class Collection( @JvmField internal val values: List, ) : Op() { diff --git a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/PlanTransform.kt b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/PlanTransform.kt index 89741aadb2..531eb1c093 100644 --- a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/PlanTransform.kt +++ b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/PlanTransform.kt @@ -180,6 +180,17 @@ internal object PlanTransform : PlanBaseVisitor() { branches = node.branches.map { visitRexOpCaseBranch(it, ctx) }, default = visitRex(node.default, ctx) ) + override fun visitRexOpNullif(node: Rex.Op.Nullif, ctx: ProblemCallback) = + org.partiql.plan.Rex.Op.Nullif( + value = visitRex(node.value, ctx), + nullifier = visitRex(node.nullifier, ctx), + ) + + override fun visitRexOpCoalesce(node: Rex.Op.Coalesce, ctx: ProblemCallback) = + org.partiql.plan.Rex.Op.Coalesce( + args = node.args.map { visitRex(it, ctx) } + ) + override fun visitRexOpCaseBranch(node: Rex.Op.Case.Branch, ctx: ProblemCallback) = org.partiql.plan.Rex.Op.Case.Branch( condition = visitRex(node.condition, ctx), rex = visitRex(node.rex, ctx) diff --git a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/RexConverter.kt b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/RexConverter.kt index 498dea8a2c..8c30083c11 100644 --- a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/RexConverter.kt +++ b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/transforms/RexConverter.kt @@ -30,8 +30,10 @@ import org.partiql.planner.internal.ir.identifierQualified import org.partiql.planner.internal.ir.identifierSymbol import org.partiql.planner.internal.ir.rex import org.partiql.planner.internal.ir.rexOpCallStatic +import org.partiql.planner.internal.ir.rexOpCoalesce import org.partiql.planner.internal.ir.rexOpCollection import org.partiql.planner.internal.ir.rexOpLit +import org.partiql.planner.internal.ir.rexOpNullif import org.partiql.planner.internal.ir.rexOpPathIndex import org.partiql.planner.internal.ir.rexOpPathKey import org.partiql.planner.internal.ir.rexOpPathSymbol @@ -107,7 +109,7 @@ internal object RexConverter { private fun visitExprCoerce(node: Expr, ctx: Env, coercion: Rex.Op.Subquery.Coercion = Rex.Op.Subquery.Coercion.SCALAR): Rex { val rex = super.visitExpr(node, ctx) return when (rex.op is Rex.Op.Select) { - true -> rex(StaticType.ANY, rexOpSubquery(rex.op as Rex.Op.Select, coercion)) + true -> rex(StaticType.ANY, rexOpSubquery(rex.op, coercion)) else -> rex } } @@ -439,44 +441,21 @@ internal object RexConverter { return rex(type, call) } - // coalesce(expr1, expr2, ... exprN) -> - // CASE - // WHEN expr1 IS NOT NULL THEN EXPR1 - // ... - // WHEN exprn is NOT NULL THEN exprn - // ELSE NULL END - override fun visitExprCoalesce(node: Expr.Coalesce, ctx: Env): Rex = plan { + override fun visitExprCoalesce(node: Expr.Coalesce, ctx: Env): Rex { val type = StaticType.ANY - val createBranch: (Rex) -> Rex.Op.Case.Branch = { expr: Rex -> - val updatedCondition = rex(type, negate(call("is_null", expr))) - rexOpCaseBranch(updatedCondition, expr) + val args = node.args.map { arg -> + visitExprCoerce(arg, ctx) } - - val branches = node.args.map { - createBranch(visitExpr(it, ctx)) - }.toMutableList() - - val defaultRex = rex(type = StaticType.NULL, op = rexOpLit(value = nullValue())) - val op = rexOpCase(branches, defaultRex) - rex(type, op) + val op = rexOpCoalesce(args) + return rex(type, op) } - // nullIf(expr1, expr2) -> - // CASE - // WHEN expr1 = expr2 THEN NULL - // ELSE expr1 END - override fun visitExprNullIf(node: Expr.NullIf, ctx: Env): Rex = plan { + override fun visitExprNullIf(node: Expr.NullIf, ctx: Env): Rex { val type = StaticType.ANY - val expr1 = visitExpr(node.value, ctx) - val expr2 = visitExpr(node.nullifier, ctx) - val id = identifierSymbol(Expr.Binary.Op.EQ.name.lowercase(), Identifier.CaseSensitivity.SENSITIVE) - val fn = fnUnresolved(id, true) - val call = rexOpCallStatic(fn, listOf(expr1, expr2)) - val branches = listOf( - rexOpCaseBranch(rex(type, call), rex(type = StaticType.NULL, op = rexOpLit(value = nullValue()))), - ) - val op = rexOpCase(branches.toMutableList(), expr1) - rex(type, op) + val value = visitExprCoerce(node.value, ctx) + val nullifier = visitExprCoerce(node.nullifier, ctx) + val op = rexOpNullif(value, nullifier) + return rex(type, op) } /** diff --git a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/DynamicTyper.kt b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/DynamicTyper.kt index eb271817bd..9ae72cc6ad 100644 --- a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/DynamicTyper.kt +++ b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/DynamicTyper.kt @@ -79,7 +79,15 @@ internal class DynamicTyper { */ fun accumulate(type: StaticType) { val nonAbsentTypes = mutableSetOf() - for (t in type.flatten().allTypes) { + val flatType = type.flatten() + if (flatType == StaticType.ANY) { + // Use ANY runtime; do not expand ANY + types.add(flatType) + args.add(ANY) + calculate(ANY) + return + } + for (t in flatType.allTypes) { when (t) { is NullType -> nullable = true is MissingType -> missable = true @@ -121,7 +129,7 @@ internal class DynamicTyper { if (missable) modifiers.add(StaticType.MISSING) // If at top supertype, then return union of all accumulated types if (supertype == ANY) { - return StaticType.unionOf(types + modifiers) to null + return StaticType.unionOf(types + modifiers).flatten() to null } // If a collection, then return union of all accumulated types as these coercion rules are not defined by SQL. if (supertype == STRUCT || supertype == BAG || supertype == LIST || supertype == SEXP) { diff --git a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/PlanTyper.kt b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/PlanTyper.kt index 9ee9bfbbb6..bf7845fb3c 100644 --- a/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/PlanTyper.kt +++ b/partiql-planner/src/main/kotlin/org/partiql/planner/internal/typer/PlanTyper.kt @@ -55,9 +55,11 @@ import org.partiql.planner.internal.ir.rexOpCallDynamic import org.partiql.planner.internal.ir.rexOpCallDynamicCandidate import org.partiql.planner.internal.ir.rexOpCallStatic import org.partiql.planner.internal.ir.rexOpCaseBranch +import org.partiql.planner.internal.ir.rexOpCoalesce import org.partiql.planner.internal.ir.rexOpCollection import org.partiql.planner.internal.ir.rexOpErr import org.partiql.planner.internal.ir.rexOpLit +import org.partiql.planner.internal.ir.rexOpNullif import org.partiql.planner.internal.ir.rexOpPathIndex import org.partiql.planner.internal.ir.rexOpPathKey import org.partiql.planner.internal.ir.rexOpPathSymbol @@ -760,6 +762,54 @@ internal class PlanTyper( return rex(type, op) } + // COALESCE(v1, v2,..., vN) + // == + // CASE + // WHEN v1 IS NOT NULL THEN v1 -- WHEN branch always a boolean + // WHEN v2 IS NOT NULL THEN v2 -- WHEN branch always a boolean + // ... -- similarly for v3..vN-1 + // ELSE vN + // END + // --> minimal common supertype of(, , ..., ) + override fun visitRexOpCoalesce(node: Rex.Op.Coalesce, ctx: StaticType?): Rex { + val args = node.args.map { visitRex(it, it.type) }.toMutableList() + val typer = DynamicTyper() + args.forEach { v -> + typer.accumulate(v.type) + } + val (type, mapping) = typer.mapping() + if (mapping != null) { + assert(mapping.size == args.size) { "Coercion mappings `len ${mapping.size}` did not match the number of COALESCE arguments `len ${args.size}`" } + for (i in args.indices) { + val (operand, target) = mapping[i] + if (operand == target) continue // skip; no coercion needed + val cast = env.fnResolver.cast(operand, target) + val rex = rex(type, rexOpCallStatic(fnResolved(cast), listOf(args[i]))) + args[i] = rex + } + } + val op = rexOpCoalesce(args) + return rex(type, op) + } + + // NULLIF(v1, v2) + // == + // CASE + // WHEN v1 = v2 THEN NULL -- WHEN branch always a boolean + // ELSE v1 + // END + // --> minimal common supertype of (NULL, ) + override fun visitRexOpNullif(node: Rex.Op.Nullif, ctx: StaticType?): Rex { + val value = visitRex(node.value, node.value.type) + val nullifier = visitRex(node.nullifier, node.nullifier.type) + val typer = DynamicTyper() + typer.accumulate(NULL) + typer.accumulate(value.type) + val (type, _) = typer.mapping() + val op = rexOpNullif(value, nullifier) + return rex(type, op) + } + /** * In this context, Boolean means PartiQLValueType Bool, which can be nullable. * Hence, we permit Static Type BOOL, Static Type NULL, Static Type Missing here. diff --git a/partiql-planner/src/main/resources/partiql_plan_internal.ion b/partiql-planner/src/main/resources/partiql_plan_internal.ion index d2cf418e67..667922ed6a 100644 --- a/partiql-planner/src/main/resources/partiql_plan_internal.ion +++ b/partiql-planner/src/main/resources/partiql_plan_internal.ion @@ -158,6 +158,15 @@ rex::{ ], }, + nullif::{ + value: rex, + nullifier: rex + }, + + coalesce::{ + args: list::[rex] + }, + collection::{ values: list::[rex], }, diff --git a/partiql-planner/src/test/kotlin/org/partiql/planner/internal/typer/PlanTyperTestsPorted.kt b/partiql-planner/src/test/kotlin/org/partiql/planner/internal/typer/PlanTyperTestsPorted.kt index de6fa6cbc0..db22b36dc4 100644 --- a/partiql-planner/src/test/kotlin/org/partiql/planner/internal/typer/PlanTyperTestsPorted.kt +++ b/partiql-planner/src/test/kotlin/org/partiql/planner/internal/typer/PlanTyperTestsPorted.kt @@ -2461,7 +2461,7 @@ class PlanTyperTestsPorted { SuccessTestCase( key = PartiQLTest.Key("basics", "case-when-11"), catalog = "pql", - expected = unionOf(StaticType.INT, StaticType.NULL, StaticType.MISSING), + expected = unionOf(StaticType.INT, StaticType.MISSING), ), SuccessTestCase( key = PartiQLTest.Key("basics", "case-when-12"), @@ -2572,6 +2572,219 @@ class PlanTyperTestsPorted { catalog = "pql", expected = MISSING ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "case-when-31"), + catalog = "pql", + expected = StaticType.ANY + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "case-when-32"), + catalog = "pql", + expected = StaticType.ANY + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "case-when-33"), + catalog = "pql", + expected = StaticType.ANY + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "case-when-34"), + catalog = "pql", + expected = StaticType.ANY + ), + ) + + @JvmStatic + fun nullIf() = listOf( + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-00"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-01"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-02"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-03"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-04"), + catalog = "pql", + expected = StaticType.INT8.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-05"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-06"), + catalog = "pql", + expected = StaticType.NULL + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-07"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-08"), + catalog = "pql", + expected = StaticType.NULL_OR_MISSING + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-09"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-10"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-11"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-12"), + catalog = "pql", + expected = StaticType.INT8.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-13"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-14"), + catalog = "pql", + expected = StaticType.STRING.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-15"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-16"), + catalog = "pql", + expected = unionOf(StaticType.INT2, StaticType.INT4, StaticType.INT8, StaticType.INT, StaticType.DECIMAL, StaticType.NULL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-17"), + catalog = "pql", + expected = StaticType.INT4.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "nullif-18"), + catalog = "pql", + expected = StaticType.ANY + ), + ) + + @JvmStatic + fun coalesce() = listOf( + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-00"), + catalog = "pql", + expected = StaticType.INT4 + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-01"), + catalog = "pql", + expected = StaticType.INT4 + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-02"), + catalog = "pql", + expected = StaticType.DECIMAL + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-03"), + catalog = "pql", + expected = unionOf(StaticType.NULL, StaticType.DECIMAL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-04"), + catalog = "pql", + expected = unionOf(StaticType.NULL, StaticType.MISSING, StaticType.DECIMAL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-05"), + catalog = "pql", + expected = unionOf(StaticType.NULL, StaticType.MISSING, StaticType.DECIMAL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-06"), + catalog = "pql", + expected = StaticType.INT4 + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-07"), + catalog = "pql", + expected = StaticType.INT4 + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-08"), + catalog = "pql", + expected = StaticType.INT8 + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-09"), + catalog = "pql", + expected = StaticType.INT8.asNullable() + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-10"), + catalog = "pql", + expected = unionOf(StaticType.INT8, StaticType.NULL, StaticType.MISSING) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-11"), + catalog = "pql", + expected = unionOf(StaticType.INT8, StaticType.STRING) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-12"), + catalog = "pql", + expected = unionOf(StaticType.INT8, StaticType.NULL, StaticType.STRING) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-13"), + catalog = "pql", + expected = unionOf(StaticType.INT2, StaticType.INT4, StaticType.INT8, StaticType.INT, StaticType.DECIMAL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-14"), + catalog = "pql", + expected = unionOf(StaticType.INT2, StaticType.INT4, StaticType.INT8, StaticType.INT, StaticType.DECIMAL, StaticType.STRING) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-15"), + catalog = "pql", + expected = unionOf(StaticType.INT2, StaticType.INT4, StaticType.INT8, StaticType.INT, StaticType.DECIMAL, StaticType.STRING, StaticType.NULL) + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-16"), + catalog = "pql", + expected = StaticType.ANY + ), + SuccessTestCase( + key = PartiQLTest.Key("basics", "coalesce-17"), + catalog = "pql", + expected = StaticType.ANY + ), ) @JvmStatic @@ -3227,6 +3440,16 @@ class PlanTyperTestsPorted { @Execution(ExecutionMode.CONCURRENT) fun testCaseWhens(tc: TestCase) = runTest(tc) + @ParameterizedTest + @MethodSource("nullIf") + @Execution(ExecutionMode.CONCURRENT) + fun testNullIf(tc: TestCase) = runTest(tc) + + @ParameterizedTest + @MethodSource("coalesce") + @Execution(ExecutionMode.CONCURRENT) + fun testCoalesce(tc: TestCase) = runTest(tc) + @ParameterizedTest @MethodSource("subqueryCases") @Execution(ExecutionMode.CONCURRENT) diff --git a/partiql-planner/src/testFixtures/resources/inputs/basics/case.sql b/partiql-planner/src/testFixtures/resources/inputs/basics/case.sql index ba1589e803..590e088578 100644 --- a/partiql-planner/src/testFixtures/resources/inputs/basics/case.sql +++ b/partiql-planner/src/testFixtures/resources/inputs/basics/case.sql @@ -85,8 +85,7 @@ CASE t_item.t_string END; --#[case-when-11] --- type: (int|null|missing) --- TODO should really be (int|missing) but our translation of coalesce doesn't consider types. +-- type: (int|missing) COALESCE(CAST(t_item.t_string AS INT), 1); -- ----------------------------- @@ -257,6 +256,42 @@ CASE t_item.t_string ELSE MISSING END; +-- ----------------------------- +-- Any Branches +-- ----------------------------- + +--#[case-when-31] +-- type: (any) +CASE t_item.t_string + WHEN 'a' THEN t_item.t_any + WHEN 'b' THEN t_item.t_int32 + ELSE NULL +END; + +--#[case-when-32] +-- type: (any) +CASE t_item.t_string + WHEN 'a' THEN t_item.t_int32 + WHEN 'b' THEN t_item.t_any + ELSE NULL +END; + +--#[case-when-33] +-- type: (any) +CASE t_item.t_string + WHEN 'a' THEN t_item.t_int32 + WHEN 'b' THEN NULL + ELSE t_item.t_any +END; + +--#[case-when-34] +-- type: (any) +CASE t_item.t_string + WHEN 'a' THEN t_item.t_int32_null + WHEN 'b' THEN t_item.t_any + ELSE t_item.t_any +END; + -- ----------------------------- -- (Unused) old tests -- ----------------------------- diff --git a/partiql-planner/src/testFixtures/resources/inputs/basics/coalesce.sql b/partiql-planner/src/testFixtures/resources/inputs/basics/coalesce.sql new file mode 100644 index 0000000000..c8e10d18fa --- /dev/null +++ b/partiql-planner/src/testFixtures/resources/inputs/basics/coalesce.sql @@ -0,0 +1,71 @@ +--#[coalesce-00] +-- type: (int32) +COALESCE(1); + +--#[coalesce-01] +-- type: (int32) +COALESCE(1, 2); + +--#[coalesce-02] +-- type: (decimal) +COALESCE(1, 1.23); + +--#[coalesce-03] +-- type: (null | decimal) +COALESCE(NULL, 1, 1.23); + +--#[coalesce-04] +-- type: (null | missing | decimal) +COALESCE(NULL, MISSING, 1, 1.23); + +--#[coalesce-05] +-- type: (null | missing | decimal); same as above +COALESCE(1, 1.23, NULL, MISSING); + +--#[coalesce-06] +-- type: (int32) +COALESCE(t_item.t_int32); + +--#[coalesce-07] +-- type: (int32) +COALESCE(t_item.t_int32, t_item.t_int32); + +--#[coalesce-08] +-- type: (int64) +COALESCE(t_item.t_int64, t_item.t_int32); + +--#[coalesce-09] +-- type: (int64 | null) +COALESCE(t_item.t_int64_null, t_item.t_int32, t_item.t_int32_null); + +--#[coalesce-10] +-- type: (int64 | null | missing) +COALESCE(t_item.t_int64_null, t_item.t_int32, t_item.t_int32_null, MISSING); + +--#[coalesce-11] +-- type: (int64 | string) +COALESCE(t_item.t_int64, t_item.t_string); + +--#[coalesce-12] +-- type: (int64 | null | string) +COALESCE(t_item.t_int64_null, t_item.t_string); + +--#[coalesce-13] +-- type: (int16 | int32 | int64 | int | decimal) +COALESCE(t_item.t_num_exact, t_item.t_int32); + +--#[coalesce-14] +-- type: (int16 | int32 | int64 | int | decimal, string) +COALESCE(t_item.t_num_exact, t_item.t_string); + +--#[coalesce-15] +-- type: (int16 | int32 | int64 | int | decimal, string, null) +COALESCE(t_item.t_num_exact, t_item.t_string, NULL); + +--#[coalesce-16] +-- type: (any) +COALESCE(t_item.t_any, t_item.t_int32); + +--#[coalesce-17] +-- type: (any) +COALESCE(t_item.t_int32, t_item.t_any); \ No newline at end of file diff --git a/partiql-planner/src/testFixtures/resources/inputs/basics/nullif.sql b/partiql-planner/src/testFixtures/resources/inputs/basics/nullif.sql new file mode 100644 index 0000000000..d03c3d863a --- /dev/null +++ b/partiql-planner/src/testFixtures/resources/inputs/basics/nullif.sql @@ -0,0 +1,77 @@ +--#[nullif-00] +-- Currently, no constant-folding. If there was, return type could be int32. +-- type: (int32 | null) +NULLIF(1, 2); + +--#[nullif-01] +-- Currently, no constant-folding. If there was, return type could be null. +-- type: (int32 | null) +NULLIF(1, 1); + +--#[nullif-02] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_int32); + +--#[nullif-03] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_int64); + +--#[nullif-04] +-- type: (int64 | null) +NULLIF(t_item.t_int64, t_item.t_int32); + +--#[nullif-05] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_null); + +--#[nullif-06] +-- type: (null) +NULLIF(t_item.t_null, t_item.t_int32); + +--#[nullif-07] +-- type: (int32 | null) +NULLIF(t_item.t_int32, MISSING); + +--#[nullif-08] +-- type: (missing | null) +NULLIF(MISSING, t_item.t_int32); + +--#[nullif-09] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_int32_null); + +--#[nullif-10] +-- type: (int32 | null) +NULLIF(t_item.t_int32_null, t_item.t_int32); + +--#[nullif-11] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_int64_null); + +--#[nullif-12] +-- type: (int64 | null) +NULLIF(t_item.t_int64_null, t_item.t_int32); + +--#[nullif-13] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_string); + +--#[nullif-14] +-- type: (string | null) +NULLIF(t_item.t_string, t_item.t_int32); + +--#[nullif-15] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_num_exact); + +--#[nullif-16] +-- type: (int16 | int32 | int64 | int | decimal | null) +NULLIF(t_item.t_num_exact, t_item.t_int32); + +--#[nullif-17] +-- type: (int32 | null) +NULLIF(t_item.t_int32, t_item.t_any); + +--#[nullif-18] +-- type: (any) +NULLIF(t_item.t_any, t_item.t_int32);