org.apache.flink.table.expressions.comparison.scala Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.flink.table.expressions
import org.apache.calcite.rex.RexNode
import org.apache.calcite.sql.SqlOperator
import org.apache.calcite.sql.fun.SqlStdOperatorTable
import org.apache.calcite.tools.RelBuilder
import org.apache.flink.table.plan.logical.LogicalExprVisitor
import org.apache.flink.table.api.types.{DataTypes, InternalType, TimestampType}
import org.apache.flink.table.typeutils.TypeCheckUtils.{isArray, isComparable, isNumeric}
import org.apache.flink.table.typeutils.TypeUtils
import org.apache.flink.table.validate._
import scala.collection.JavaConversions._
abstract class BinaryComparison extends BinaryExpression {
private[flink] def sqlOperator: SqlOperator
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.call(sqlOperator, children.map(_.toRexNode))
}
override private[flink] def resultType = DataTypes.BOOLEAN
override private[flink] def validateInput(): ValidationResult =
(left.resultType, right.resultType) match {
case (lType, rType)
if isNumeric(lType) && isNumeric(rType) =>
ValidationSuccess
case (lType, rType)
if isComparable(lType) && lType == rType => ValidationSuccess
case (_: TimestampType, _: TimestampType) => ValidationSuccess
case (lType, rType) =>
ValidationFailure(
s"Comparison is only supported for numeric types and " +
s"comparable types of same type, got $lType and $rType")
}
}
case class EqualTo(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left === $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.EQUALS
override private[flink] def validateInput(): ValidationResult =
(left.resultType, right.resultType) match {
case (lType, rType)
if isNumeric(lType) && isNumeric(rType) =>
ValidationSuccess
case (lType, rType) if lType == rType => ValidationSuccess
case (lType, rType)
if isArray(lType) &&
TypeUtils.getExternalClassForType(lType) == TypeUtils.getExternalClassForType(rType) =>
ValidationSuccess
case (lType, rType) =>
ValidationFailure(s"Equality predicate on incompatible types: $lType and $rType")
}
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class NotEqualTo(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left !== $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.NOT_EQUALS
override private[flink] def validateInput(): ValidationResult =
(left.resultType, right.resultType) match {
case (lType, rType)
if isNumeric(lType) && isNumeric(rType) =>
ValidationSuccess
case (lType, rType) if lType == rType => ValidationSuccess
case (lType, rType)
if isArray(lType) &&
TypeUtils.getExternalClassForType(lType) == TypeUtils.getExternalClassForType(rType) =>
ValidationSuccess
case (lType, rType) =>
ValidationFailure(s"Inequality predicate on incompatible types: $lType and $rType")
}
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class GreaterThan(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left > $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.GREATER_THAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class GreaterThanOrEqual(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left >= $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.GREATER_THAN_OR_EQUAL
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class LessThan(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left < $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.LESS_THAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class LessThanOrEqual(left: Expression, right: Expression) extends BinaryComparison {
override def toString = s"$left <= $right"
private[flink] val sqlOperator: SqlOperator = SqlStdOperatorTable.LESS_THAN_OR_EQUAL
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsNull(child: Expression) extends UnaryExpression {
override def toString = s"($child).isNull"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.isNull(child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsNotNull(child: Expression) extends UnaryExpression {
override def toString = s"($child).isNotNull"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.isNotNull(child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsTrue(child: Expression) extends UnaryExpression {
override def toString = s"($child).isTrue"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.call(SqlStdOperatorTable.IS_TRUE, child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsFalse(child: Expression) extends UnaryExpression {
override def toString = s"($child).isFalse"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.call(SqlStdOperatorTable.IS_FALSE, child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsNotTrue(child: Expression) extends UnaryExpression {
override def toString = s"($child).isNotTrue"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.call(SqlStdOperatorTable.IS_NOT_TRUE, child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class IsNotFalse(child: Expression) extends UnaryExpression {
override def toString = s"($child).isNotFalse"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.call(SqlStdOperatorTable.IS_NOT_FALSE, child.toRexNode)
}
override private[flink] def resultType = DataTypes.BOOLEAN
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
abstract class BetweenProperty(
expr: Expression,
lowerBound: Expression,
upperBound: Expression) extends Expression {
override private[flink] def resultType: InternalType = DataTypes.BOOLEAN
override private[flink] def children: Seq[Expression] = Seq(expr, lowerBound, upperBound)
override private[flink] def validateInput(): ValidationResult = {
(expr.resultType, lowerBound.resultType, upperBound.resultType) match {
case (exprType, lowerType, upperType)
if isNumeric(exprType) && isNumeric(lowerType) && isNumeric(upperType)
=> ValidationSuccess
case (exprType, lowerType, upperType)
if isComparable(exprType) && exprType == lowerType && exprType == upperType
=> ValidationSuccess
case (exprType, lowerType, upperType) =>
ValidationFailure(
s"Between is only supported for numeric types and " +
s"identical comparable types, but got $exprType, $lowerType and $upperType"
)
}
}
}
case class Between(
expr: Expression,
lowerBound: Expression,
upperBound: Expression) extends BetweenProperty(expr, lowerBound, upperBound) {
override def toString: String = s"$expr.between($lowerBound, $upperBound)"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.and(
relBuilder.call(
SqlStdOperatorTable.GREATER_THAN_OR_EQUAL,
expr.toRexNode,
lowerBound.toRexNode
),
relBuilder.call(
SqlStdOperatorTable.LESS_THAN_OR_EQUAL,
expr.toRexNode,
upperBound.toRexNode
)
)
}
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
case class NotBetween(
expr: Expression,
lowerBound: Expression,
upperBound: Expression) extends BetweenProperty(expr, lowerBound, upperBound) {
override def toString: String = s"$expr.notBetween($lowerBound, $upperBound)"
override private[flink] def toRexNode(implicit relBuilder: RelBuilder): RexNode = {
relBuilder.or(
relBuilder.call(
SqlStdOperatorTable.LESS_THAN,
expr.toRexNode,
lowerBound.toRexNode
),
relBuilder.call(
SqlStdOperatorTable.GREATER_THAN,
expr.toRexNode,
upperBound.toRexNode
)
)
}
override def accept[T](logicalExprVisitor: LogicalExprVisitor[T]): T =
logicalExprVisitor.visit(this)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy