This is an automated email from the ASF dual-hosted git repository.

danny0405 pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/hudi.git


The following commit(s) were added to refs/heads/master by this push:
     new ad817728bb5 [HUDI-6071] Flink Hive Catalog throws exception through 
SQL Client when table type contains smallint (#8449)
ad817728bb5 is described below

commit ad817728bb52979ff297672d94f9dee77cb3d16e
Author: ChestnutQiang <[email protected]>
AuthorDate: Tue Apr 18 19:25:42 2023 +0800

    [HUDI-6071] Flink Hive Catalog throws exception through SQL Client when 
table type contains smallint (#8449)
    
    Co-authored-by: lizhiqiang <wujuan.dtstack.com>
---
 .../java/org/apache/hudi/util/ExpressionUtils.java |   4 +
 .../org/apache/hudi/util/TestExpressionUtils.java  | 151 +++++++++++++++++++++
 2 files changed, 155 insertions(+)

diff --git 
a/hudi-flink-datasource/hudi-flink/src/main/java/org/apache/hudi/util/ExpressionUtils.java
 
b/hudi-flink-datasource/hudi-flink/src/main/java/org/apache/hudi/util/ExpressionUtils.java
index 3931e296440..78245fb80a0 100644
--- 
a/hudi-flink-datasource/hudi-flink/src/main/java/org/apache/hudi/util/ExpressionUtils.java
+++ 
b/hudi-flink-datasource/hudi-flink/src/main/java/org/apache/hudi/util/ExpressionUtils.java
@@ -166,9 +166,13 @@ public class ExpressionUtils {
       case BOOLEAN:
         return expr.getValueAs(Boolean.class).orElse(null);
       case TINYINT:
+        return expr.getValueAs(Byte.class).orElse(null);
       case SMALLINT:
+        return expr.getValueAs(Short.class).orElse(null);
       case INTEGER:
         return expr.getValueAs(Integer.class).orElse(null);
+      case BIGINT:
+        return expr.getValueAs(Long.class).orElse(null);
       case FLOAT:
         return expr.getValueAs(Float.class).orElse(null);
       case DOUBLE:
diff --git 
a/hudi-flink-datasource/hudi-flink/src/test/java/org/apache/hudi/util/TestExpressionUtils.java
 
b/hudi-flink-datasource/hudi-flink/src/test/java/org/apache/hudi/util/TestExpressionUtils.java
new file mode 100644
index 00000000000..50816a298de
--- /dev/null
+++ 
b/hudi-flink-datasource/hudi-flink/src/test/java/org/apache/hudi/util/TestExpressionUtils.java
@@ -0,0 +1,151 @@
+/*
+ * 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.hudi.util;
+
+import org.apache.flink.table.api.DataTypes;
+import org.apache.flink.table.expressions.CallExpression;
+import org.apache.flink.table.expressions.Expression;
+import org.apache.flink.table.expressions.FieldReferenceExpression;
+import org.apache.flink.table.expressions.ValueLiteralExpression;
+import org.apache.flink.table.functions.BuiltInFunctionDefinitions;
+import org.apache.flink.table.types.DataType;
+import org.apache.flink.table.types.logical.RowType;
+import org.junit.jupiter.api.Test;
+
+import java.math.BigDecimal;
+import java.nio.charset.StandardCharsets;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.ZoneOffset;
+import java.time.temporal.ChronoField;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+
+import static org.junit.jupiter.api.Assertions.assertEquals;
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
+class TestExpressionUtils {
+
+  private static final DataType ROW_DATA_TYPE = DataTypes.ROW(
+      DataTypes.FIELD("f_tinyint", DataTypes.TINYINT()),
+      DataTypes.FIELD("f_smallint", DataTypes.SMALLINT()),
+      DataTypes.FIELD("f_int", DataTypes.INT()),
+      DataTypes.FIELD("f_long", DataTypes.BIGINT()),
+      DataTypes.FIELD("f_float", DataTypes.FLOAT()),
+      DataTypes.FIELD("f_double", DataTypes.DOUBLE()),
+      DataTypes.FIELD("f_boolean", DataTypes.BOOLEAN()),
+      DataTypes.FIELD("f_decimal", DataTypes.DECIMAL(10, 2)),
+      DataTypes.FIELD("f_bytes", DataTypes.VARBINARY(10)),
+      DataTypes.FIELD("f_string", DataTypes.VARCHAR(10)),
+      DataTypes.FIELD("f_time", DataTypes.TIME(3)),
+      DataTypes.FIELD("f_date", DataTypes.DATE()),
+      DataTypes.FIELD("f_timestamp", DataTypes.TIMESTAMP(3))
+  ).notNull();
+
+  private static final DataType ROW_DATA_TYPE_FIELD_NON_NULL = DataTypes.ROW(
+      DataTypes.FIELD("f_tinyint", DataTypes.TINYINT().notNull()),
+      DataTypes.FIELD("f_smallint", DataTypes.SMALLINT().notNull()),
+      DataTypes.FIELD("f_int", DataTypes.INT().notNull()),
+      DataTypes.FIELD("f_long", DataTypes.BIGINT().notNull()),
+      DataTypes.FIELD("f_float", DataTypes.FLOAT().notNull()),
+      DataTypes.FIELD("f_double", DataTypes.DOUBLE().notNull()),
+      DataTypes.FIELD("f_boolean", DataTypes.BOOLEAN().notNull()),
+      DataTypes.FIELD("f_decimal", DataTypes.DECIMAL(10, 2).notNull()),
+      DataTypes.FIELD("f_bytes", DataTypes.VARBINARY(10).notNull()),
+      DataTypes.FIELD("f_string", DataTypes.VARCHAR(10).notNull()),
+      DataTypes.FIELD("f_time", DataTypes.TIME(3).notNull()),
+      DataTypes.FIELD("f_date", DataTypes.DATE().notNull()),
+      DataTypes.FIELD("f_timestamp", DataTypes.TIMESTAMP(3).notNull())
+  ).notNull();
+  
+  @Test
+  void getValueFromLiteralForNull() {
+    List<RowType.RowField> fields = ((RowType) 
ROW_DATA_TYPE.getLogicalType()).getFields();
+    List<DataType> dataTypes = ROW_DATA_TYPE.getChildren();
+    CallExpression callExpression;
+    for (int i = 0; i < fields.size(); i++) {
+      // 1. Build all types
+      callExpression = new CallExpression(
+          BuiltInFunctionDefinitions.IS_NOT_NULL,
+          Arrays.asList(new FieldReferenceExpression(fields.get(i).getName(),
+              dataTypes.get(i),
+              2,
+              2), new ValueLiteralExpression(null, dataTypes.get(i))),
+          DataTypes.BOOLEAN());
+      List<Expression> childExprs = callExpression.getChildren();
+
+      // 2. Parse each type
+      boolean hasNullLiteral =
+          childExprs.stream().anyMatch(e ->
+              e instanceof ValueLiteralExpression
+                  && 
ExpressionUtils.getValueFromLiteral((ValueLiteralExpression) e) == null);
+      assertTrue(hasNullLiteral);
+    }
+
+  }
+
+  @Test
+  void getValueFromLiteralForNonNull() {
+    List<RowType.RowField> fields = ((RowType) 
ROW_DATA_TYPE_FIELD_NON_NULL.getLogicalType()).getFields();
+    List<DataType> dataTypes = ROW_DATA_TYPE_FIELD_NON_NULL.getChildren();
+    // tests for non-null literals
+    List<Object> dataList = new ArrayList<>(fields.size());
+    dataList.add(new Byte("1")); // f_tinyint
+    dataList.add(new Short("2")); // f_smallint
+    dataList.add(new Integer("3")); // f_int
+    dataList.add(new Long("4")); // f_long
+    dataList.add(new Float(5.0)); // f_float
+    dataList.add(new Double(6.0)); // f_double
+    dataList.add(new Boolean(true)); // f_boolean
+    dataList.add(new BigDecimal(3.0)); // f_decimal
+    dataList.add("hudi".getBytes(StandardCharsets.UTF_8)); // f_bytes
+    dataList.add("hudi ok"); // f_string
+    dataList.add(LocalTime.of(1, 11, 11)); // f_time
+    dataList.add(LocalDate.of(2023, 1, 2)); // f_date
+    dataList.add(LocalDateTime.of(2023, 1, 2, 3, 4)); // f_timestamp
+    CallExpression callExpression;
+    for (int i = 0; i < fields.size(); i++) {
+      // 1. Build all types
+      callExpression = new CallExpression(
+          BuiltInFunctionDefinitions.IS_NOT_NULL,
+          Arrays.asList(
+              new FieldReferenceExpression(
+                  fields.get(i).getName(),
+                  dataTypes.get(i),
+                  i,
+                  i),
+              new ValueLiteralExpression(dataList.get(i), dataTypes.get(i))),
+          DataTypes.BOOLEAN());
+      List<Expression> childExprs = callExpression.getChildren();
+      // 2. Parse each type
+      if (dataList.get(i) instanceof LocalTime) {
+        assertEquals(((LocalTime) 
dataList.get(i)).get(ChronoField.MILLI_OF_DAY), 
ExpressionUtils.getValueFromLiteral((ValueLiteralExpression) 
childExprs.get(1)));
+      } else if (dataList.get(i) instanceof LocalDate) {
+        assertEquals(((LocalDate) dataList.get(i)).toEpochDay(), 
ExpressionUtils.getValueFromLiteral((ValueLiteralExpression) 
childExprs.get(1)));
+      } else if (dataList.get(i) instanceof LocalDateTime) {
+        assertEquals(((LocalDateTime) 
dataList.get(i)).toInstant(ZoneOffset.UTC).toEpochMilli(), 
ExpressionUtils.getValueFromLiteral((ValueLiteralExpression) 
childExprs.get(1)));
+      } else {
+        assertEquals(dataList.get(i), 
ExpressionUtils.getValueFromLiteral((ValueLiteralExpression) 
childExprs.get(1)));
+      }
+    }
+  }
+}
\ No newline at end of file

Reply via email to