diff --git a/core/src/main/codegen/templates/Parser.jj b/core/src/main/codegen/templates/Parser.jj index 917b92fb7035..19cba6b17bf9 100644 --- a/core/src/main/codegen/templates/Parser.jj +++ b/core/src/main/codegen/templates/Parser.jj @@ -1579,15 +1579,37 @@ List SelectList() : SqlNode SelectItem() : { SqlNode e; - final SqlIdentifier id; + SqlIdentifier id; + final List ids = new ArrayList(); + final Span s = span(); } { e = SelectExpression() [ - [ ] - id = SimpleIdentifier() { - e = SqlStdOperatorTable.AS.createCall(span().end(e), e, id); - } + ( + LOOKAHEAD(2) ( + [ ] + id = SimpleIdentifier() + { + e = SqlStdOperatorTable.AS.createCall(s.end(e), e, id); + } + ) + | + ( + + id = SimpleIdentifier() { + ids.add(id); + } + ( id = SimpleIdentifier() { ids.add(id);} )* + { + if (!this.conformance.allowSelectTableFunction()) { + throw SqlUtil.newContextException(getPos(), + RESOURCE.notAllowTableFunctionInSelect()); + } + e = SqlStdOperatorTable.AS.createCall(s.end(e), e, new SqlNodeList(ids, s.end(e))); + } + ) + ) ] { return e; diff --git a/core/src/main/java/org/apache/calcite/runtime/CalciteResource.java b/core/src/main/java/org/apache/calcite/runtime/CalciteResource.java index 3c394226b9b3..439bc2fab7ae 100644 --- a/core/src/main/java/org/apache/calcite/runtime/CalciteResource.java +++ b/core/src/main/java/org/apache/calcite/runtime/CalciteResource.java @@ -894,6 +894,18 @@ ExInst invalidTypesForComparison(String clazzName0, String op, @BaseMessage("Not a valid input for REGEXP_REPLACE: ''{0}''") ExInst invalidInputForRegexpReplace(String value); + + @BaseMessage("Table function is not allowed in select list in current SQL conformance level") + ExInst notAllowTableFunctionInSelect(); + + @BaseMessage("''{0}'' should be a table function") + ExInst exceptTableFunction(String name); + + @BaseMessage("Only one table function is allowed in select list") + ExInst onlyOneTableFunctionAllowedInSelect(); + + @BaseMessage("Table function is not allowed in aggregate statement") + ExInst notAllowTableFunctionInAggregate(); } // End CalciteResource.java diff --git a/core/src/main/java/org/apache/calcite/sql/SqlAsOperator.java b/core/src/main/java/org/apache/calcite/sql/SqlAsOperator.java index 4d9ea95faaa7..e48821aa74f5 100644 --- a/core/src/main/java/org/apache/calcite/sql/SqlAsOperator.java +++ b/core/src/main/java/org/apache/calcite/sql/SqlAsOperator.java @@ -30,6 +30,7 @@ import org.apache.calcite.sql.validate.SqlValidatorScope; import org.apache.calcite.util.Util; +import java.util.ArrayList; import java.util.List; import static org.apache.calcite.util.Static.RESOURCE; @@ -102,12 +103,26 @@ public void validateCall( // we don't want to validate the identifier. final List operands = call.getOperandList(); assert operands.size() == 2; - assert operands.get(1) instanceof SqlIdentifier; operands.get(0).validateExpr(validator, scope); - SqlIdentifier id = (SqlIdentifier) operands.get(1); - if (!id.isSimple()) { - throw validator.newValidationError(id, - RESOURCE.aliasMustBeSimpleIdentifier()); + + SqlNode asIdentifier = operands.get(1); + assert asIdentifier instanceof SqlIdentifier + || (asIdentifier instanceof SqlNodeList + && validator.getConformance().allowSelectTableFunction()); + + List ids = new ArrayList<>(); + if (asIdentifier instanceof SqlIdentifier) { + ids.add(operands.get(1)); + } else { + ids.addAll(((SqlNodeList) operands.get(1)).getList()); + } + for (int i = 0; i < ids.size(); i++) { + assert ids.get(i) instanceof SqlIdentifier; + SqlIdentifier id = (SqlIdentifier) ids.get(i); + if (!id.isSimple()) { + throw validator.newValidationError(id, + RESOURCE.aliasMustBeSimpleIdentifier()); + } } } diff --git a/core/src/main/java/org/apache/calcite/sql/SqlUtil.java b/core/src/main/java/org/apache/calcite/sql/SqlUtil.java index 469f83d2dd3b..69f5271fb941 100644 --- a/core/src/main/java/org/apache/calcite/sql/SqlUtil.java +++ b/core/src/main/java/org/apache/calcite/sql/SqlUtil.java @@ -1069,6 +1069,21 @@ private void visitChild(SqlNode node) { return check(type); } } + + /** + * Whether the selectItem is a table function node in the select. + * eg. "select table_func(1) as (f0,f1)" + * + * @param selectItem select item + * @return true if this selectItem is a table function call + */ + public static boolean isTableFunctionInSelect(SqlNode selectItem) { + if (selectItem.getKind() == SqlKind.AS) { + SqlBasicCall call = (SqlBasicCall) selectItem; + return call.getOperands()[1] instanceof SqlNodeList; + } + return false; + } } // End SqlUtil.java diff --git a/core/src/main/java/org/apache/calcite/sql/validate/AliasNamespace.java b/core/src/main/java/org/apache/calcite/sql/validate/AliasNamespace.java index c7afbcaffd39..fb74c07c6dad 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/AliasNamespace.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/AliasNamespace.java @@ -22,6 +22,7 @@ import org.apache.calcite.sql.SqlIdentifier; import org.apache.calcite.sql.SqlNode; import org.apache.calcite.sql.SqlNodeList; +import org.apache.calcite.sql.SqlUtil; import org.apache.calcite.sql.fun.SqlStdOperatorTable; import org.apache.calcite.sql.parser.SqlParserPos; import org.apache.calcite.util.Util; @@ -69,7 +70,12 @@ protected RelDataType validateImpl(RelDataType targetRowType) { final SqlValidatorNamespace childNs = validator.getNamespace(operands.get(0)); final RelDataType rowType = childNs.getRowTypeSansSystemColumns(); - final List columnNames = Util.skip(operands, 2); + List columnNames; + if (SqlUtil.isTableFunctionInSelect(call)) { + columnNames = ((SqlNodeList) operands.get(1)).getList(); + } else { + columnNames = Util.skip(operands, 2); + } for (final SqlNode operand : columnNames) { String name = ((SqlIdentifier) operand).getSimple(); if (nameList.contains(name)) { diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlAbstractConformance.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlAbstractConformance.java index f6764191dc9e..e6f8a134c860 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlAbstractConformance.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlAbstractConformance.java @@ -102,6 +102,10 @@ public boolean shouldConvertRaggedUnionTypesToVarying() { public boolean allowExtendedTrim() { return SqlConformanceEnum.DEFAULT.allowExtendedTrim(); } + + public boolean allowSelectTableFunction() { + return SqlConformanceEnum.DEFAULT.allowSelectTableFunction(); + } } // End SqlAbstractConformance.java diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlConformance.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlConformance.java index 644886f30858..4309d92b6cb6 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlConformance.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlConformance.java @@ -61,6 +61,7 @@ public interface SqlConformance { @Deprecated // to be removed before 2.0 SqlConformanceEnum PRAGMATIC_2003 = SqlConformanceEnum.PRAGMATIC_2003; + SqlConformanceEnum HIVE = SqlConformanceEnum.HIVE; /** * Whether this dialect supports features from a wide variety of * dialects. This is enabled for the Babel parser, disabled otherwise. @@ -393,6 +394,13 @@ public interface SqlConformance { * false otherwise. */ boolean allowExtendedTrim(); + + /** + * Whether SELECT can contain a table function. + *

For example, consider the query + *

 SELECT SPLIT(col) AS (F0, F1) FROM A 
+ */ + boolean allowSelectTableFunction(); } // End SqlConformance.java diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlConformanceEnum.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlConformanceEnum.java index 5d9c44e96043..89893e37953e 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlConformanceEnum.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlConformanceEnum.java @@ -69,7 +69,11 @@ public enum SqlConformanceEnum implements SqlConformance { /** Conformance value that instructs Calcite to use SQL semantics * consistent with Microsoft SQL Server version 2008. */ - SQL_SERVER_2008; + SQL_SERVER_2008, + + /** Conformance value that instructs Calcite to use SQL semantics + * consistent with Hive version. */ + HIVE; public boolean isLiberal() { switch (this) { @@ -304,6 +308,14 @@ public boolean allowExtendedTrim() { } } + public boolean allowSelectTableFunction() { + switch (this) { + case HIVE: + return true; + default: + return false; + } + } } // End SqlConformanceEnum.java diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlDelegatingConformance.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlDelegatingConformance.java index 680cc607e403..fef4d62ff3b8 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlDelegatingConformance.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlDelegatingConformance.java @@ -73,6 +73,9 @@ protected SqlDelegatingConformance(SqlConformance delegate) { return delegate.allowNiladicParentheses(); } + @Override public boolean allowSelectTableFunction() { + return delegate.allowSelectTableFunction(); + } } // End SqlDelegatingConformance.java diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlValidator.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlValidator.java index 5c2bb13b281b..fb5a9b8bd69b 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlValidator.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlValidator.java @@ -23,6 +23,7 @@ import org.apache.calcite.runtime.CalciteContextException; import org.apache.calcite.runtime.CalciteException; import org.apache.calcite.runtime.Resources; +import org.apache.calcite.sql.SqlBasicCall; import org.apache.calcite.sql.SqlCall; import org.apache.calcite.sql.SqlDataTypeSpec; import org.apache.calcite.sql.SqlDelete; @@ -748,6 +749,14 @@ CalciteException handleUnresolvedFunction(SqlCall call, */ SqlValidatorScope getOverScope(SqlNode node); + /** + * Returns the table function SqlBasicCall in SqlSelect. + * + * @param select The select node + * @return The table function node associate with the select node + */ + SqlBasicCall getTableFunctionInSelect(SqlSelect select); + /** * Validates that a query is capable of producing a return of given modality * (relational or streaming). diff --git a/core/src/main/java/org/apache/calcite/sql/validate/SqlValidatorImpl.java b/core/src/main/java/org/apache/calcite/sql/validate/SqlValidatorImpl.java index 3035052c3aad..1556d65575d5 100644 --- a/core/src/main/java/org/apache/calcite/sql/validate/SqlValidatorImpl.java +++ b/core/src/main/java/org/apache/calcite/sql/validate/SqlValidatorImpl.java @@ -160,6 +160,11 @@ public class SqlValidatorImpl implements SqlValidatorWithHints { */ public static final String UPDATE_ANON_PREFIX = "SYS$ANON"; + /** + * Prefix for table function. + */ + private static final String TABLE_FUNCTION_PREFIX = "_table_function_"; + //~ Instance fields -------------------------------------------------------- private final SqlOperatorTable opTab; @@ -284,6 +289,9 @@ public class SqlValidatorImpl implements SqlValidatorWithHints { private final SqlValidatorImpl.ValidationErrorFunction validationErrorFunction = new SqlValidatorImpl.ValidationErrorFunction(); + // Maping the table function and the select node. + private final Map selectTableFunctions = + new IdentityHashMap<>(); //~ Constructors ----------------------------------------------------------- /** @@ -1052,6 +1060,10 @@ public SqlValidatorScope getOverScope(SqlNode node) { return scopes.get(node); } + public SqlBasicCall getTableFunctionInSelect(SqlSelect select) { + return selectTableFunctions.get(select); + } + private SqlValidatorNamespace getNamespace(SqlNode node, SqlValidatorScope scope) { if (node instanceof SqlIdentifier && scope instanceof DelegatingScope) { @@ -4064,6 +4076,14 @@ protected RelDataType validateSelectList( expandedSelectItems, aliases, fieldList); + } else if (SqlUtil.isTableFunctionInSelect(selectItem)) { + handleTableFunctionInSelect( + select, + (SqlBasicCall) selectItem, + expandedSelectItems, + aliases, + fieldList + ); } else { expandSelectItem( selectItem, @@ -4174,6 +4194,114 @@ private void handleScalarSubQuery( fieldList.add(Pair.of(alias, nodeType)); } + /** + * Process table function found in select list.Checks that is + * actually a table function and validate the table function + * count in Select list. + * + * @param parentSelect Base SqlSelect + * @param selectItem Child select items from select list + * @param expandedSelectItems Select items after processing + * @param aliases Built from user or system values + * @param fields Built up entries for each select list entry + */ + private void handleTableFunctionInSelect( + SqlSelect parentSelect, + SqlBasicCall selectItem, + List expandedSelectItems, + Set aliases, + List> fields) { + SqlBasicCall functionCall = (SqlBasicCall) selectItem.getOperands()[0]; + SqlFunction function = (SqlFunction) functionCall.getOperator(); + // Check whether there are more than one table function in select list. + for (SqlNode item : parentSelect.getSelectList()) { + if (SqlUtil.isTableFunctionInSelect(item) + && item != selectItem) { + throw newValidationError(parentSelect.getSelectList(), + RESOURCE.onlyOneTableFunctionAllowedInSelect()); + } + } + + // Change the function category to USER_DEFINED_TABLE_FUNCTION. + // It is because that in sql-select list, the SqlFunctionCategory is USER_DEFINED_FUNCTION + // for a SqlUnresolvedFunction,so we should do this change. + if (function instanceof SqlUnresolvedFunction) { + if (!function.getFunctionType().isTableFunction()) { + SqlFunction newFunction = + new SqlUnresolvedFunction(function.getNameAsId(), + function.getReturnTypeInference(), + function.getOperandTypeInference(), + function.getOperandTypeChecker(), + function.getParamTypes(), + SqlFunctionCategory.USER_DEFINED_TABLE_FUNCTION); + functionCall.setOperator(newFunction); + function = newFunction; + } + } + // Check functionCall whether is a table function + List overloads = new ArrayList<>(); + opTab.lookupOperatorOverloads(function.getNameAsId(), + function.getFunctionType(), + function.getSyntax(), overloads, catalogReader.nameMatcher()); + if (overloads.size() == 0) { + throw newValidationError(functionCall, + RESOURCE.exceptTableFunction(function.getName())); + } + // Check the parent select whether is a aggregate statement + if (isAggregate(parentSelect)) { + throw newValidationError(functionCall, + RESOURCE.notAllowTableFunctionInAggregate()); + } + SqlNodeList aliasNodes + = (SqlNodeList) selectItem.getOperands()[1]; + List aliasList = new ArrayList<>(); + for (SqlNode aliasNode : aliasNodes) { + aliasList.add(deriveAlias(aliasNode, aliasList.size())); + } + aliases.addAll(aliasList); + + String tableAlias = TABLE_FUNCTION_PREFIX + nextGeneratedId++; + // Expand the table function alias + for (String alias : aliasList) { + SqlIdentifier id = new SqlIdentifier(Lists.newArrayList + (tableAlias, alias), SqlParserPos.ZERO); + expandedSelectItems.add(id); + } + + // Register namespace for table function + SqlValidatorScope fromScope = getFromScope(parentSelect); + ProcedureNamespace tableNs = new ProcedureNamespace(this, + fromScope, functionCall, selectItem); + tableNs.validateImpl(unknownType); + registerNamespace(null, null, + tableNs, false); + AliasNamespace aliasNs = new AliasNamespace(this, + selectItem, parentSelect); + aliasNs.validateImpl(unknownType); + registerNamespace(getSelectScope(parentSelect), + tableAlias, aliasNs, false); + + // Create a table scope for table function + TableScope tableScope = new TableScope(fromScope, parentSelect); + if (fromScope instanceof ListScope) { + for (ScopeChild child : ((ListScope) fromScope).children) { + tableScope.addChild(child.namespace, child.name, child.nullable); + } + } + scopes.put(functionCall, tableScope); + // Associate the select with the table function + selectTableFunctions.put(parentSelect, functionCall); + + RelDataType type = aliasNs.getRowType(); + setValidatedNodeType(selectItem, type); + for (int i = 0; i < aliasList.size(); i++) { + fields.add( + Pair.of( + aliasList.get(i), type.getFieldList() + .get(i).getType())); + } + } + /** * Derives a row-type for INSERT and UPDATE operations. * diff --git a/core/src/main/java/org/apache/calcite/sql2rel/SqlToRelConverter.java b/core/src/main/java/org/apache/calcite/sql2rel/SqlToRelConverter.java index a9675836d62d..b7ddbdb62d69 100644 --- a/core/src/main/java/org/apache/calcite/sql2rel/SqlToRelConverter.java +++ b/core/src/main/java/org/apache/calcite/sql2rel/SqlToRelConverter.java @@ -3904,6 +3904,8 @@ private void convertSelectList( SqlNodeList selectList = select.getSelectList(); selectList = validator.expandStar(selectList, select, false); + convertTableFunctionInSelect(bb, select); + replaceSubQueries(bb, selectList, RelOptUtil.Logic.TRUE_FALSE_UNKNOWN); List fieldNames = new ArrayList<>(); @@ -3952,6 +3954,31 @@ private void convertSelectList( } } + /** + * Convert the table function in select to Join or TableFunctionScan. + * + * @param bb Blackboard + * @param select SqlSelect + * */ + private void convertTableFunctionInSelect(Blackboard bb, SqlSelect select) { + SqlBasicCall tableFunction = validator.getTableFunctionInSelect(select); + // rewrite the table function if select list contain one + if (tableFunction != null) { + SqlValidatorScope tableScope = validator.getJoinScope(tableFunction); + final Blackboard rightBb = + createBlackboard(tableScope, null, false); + convertCollectionTable(rightBb, tableFunction); + + if (select.getFrom() != null) { + RelNode join = createJoin(bb, bb.root, rightBb.root, + rexBuilder.makeLiteral(true), JoinRelType.INNER); + bb.setRoot(join, false); + } else { + bb.setRoot(rightBb.root, false); + } + } + } + /** * Adds extra select items. The default implementation adds nothing; derived * classes may add columns to exprList, nameList, aliasList and diff --git a/core/src/main/resources/org/apache/calcite/runtime/CalciteResource.properties b/core/src/main/resources/org/apache/calcite/runtime/CalciteResource.properties index 1b670b6c8cd4..c6fa901b1597 100644 --- a/core/src/main/resources/org/apache/calcite/runtime/CalciteResource.properties +++ b/core/src/main/resources/org/apache/calcite/runtime/CalciteResource.properties @@ -291,4 +291,8 @@ InvalidInputForJsonKeys=Not a valid input for JSON_KEYS: ''{0}'' InvalidInputForJsonRemove=Invalid input for JSON_REMOVE: document: ''{0}'', jsonpath expressions: ''{1}'' InvalidInputForJsonStorageSize=Not a valid input for JSON_STORAGE_SIZE: ''{0}'' InvalidInputForRegexpReplace=Not a valid input for REGEXP_REPLACE: ''{0}'' +NotAllowTableFunctionInSelect=Table function is not allowed in select list in current SQL conformance level +ExceptTableFunction=''{0}'' should be a table function +OnlyOneTableFunctionAllowedInSelect=Only one table function is allowed in select list +NotAllowTableFunctionInAggregate=Table function is not allowed in aggregate statement # End CalciteResource.properties diff --git a/core/src/test/java/org/apache/calcite/sql/parser/SqlParserTest.java b/core/src/test/java/org/apache/calcite/sql/parser/SqlParserTest.java index c9f8cb5e1772..076efc8cf1db 100644 --- a/core/src/test/java/org/apache/calcite/sql/parser/SqlParserTest.java +++ b/core/src/test/java/org/apache/calcite/sql/parser/SqlParserTest.java @@ -3476,6 +3476,26 @@ void checkPeriodPredicate(Checker checker) { .ok(expected2 + " AS `T` (`X`)"); } + @Test public void testTableFunctionInSelect() { + conformance = SqlConformance.HIVE; + sql("select n,c from (select TABLE_FUNC(1) as (n,c) from emp)") + .ok("SELECT `N`, `C`\n" + + "FROM (SELECT `TABLE_FUNC`(1) AS (`N`, `C`)\n" + + "FROM `EMP`)"); + sql("select TABLE_FUNC(1) as (c) from emp") + .ok("SELECT `TABLE_FUNC`(1) AS (`C`)\n" + + "FROM `EMP`"); + sql("select char_length(c)*2 from" + + " (select TABLE_FUNC(1) as (n,c) from emp) where char_length(c) > 1") + .ok("SELECT (CHAR_LENGTH(`C`) * 2)\n" + + "FROM (SELECT `TABLE_FUNC`(1) AS (`N`, `C`)\n" + + "FROM `EMP`)\n" + + "WHERE (CHAR_LENGTH(`C`) > 1)"); + conformance = SqlConformanceEnum.DEFAULT; + sql("select TABLE_FUNC(1) as (n,c^)^ from emp") + .fails("(.*)Table function is not allowed in select list(.*)"); + } + @Test public void testCollectionTableWithLateral() { final String sql = "select * from dept, lateral table(ramp(dept.deptno))"; final String expected = "SELECT *\n" diff --git a/core/src/test/java/org/apache/calcite/test/MockSqlOperatorTable.java b/core/src/test/java/org/apache/calcite/test/MockSqlOperatorTable.java index 13c8253777ef..fc96fb2acd90 100644 --- a/core/src/test/java/org/apache/calcite/test/MockSqlOperatorTable.java +++ b/core/src/test/java/org/apache/calcite/test/MockSqlOperatorTable.java @@ -18,6 +18,8 @@ import org.apache.calcite.rel.type.RelDataType; import org.apache.calcite.rel.type.RelDataTypeFactory; +import org.apache.calcite.schema.FunctionParameter; +import org.apache.calcite.schema.TableFunction; import org.apache.calcite.sql.SqlAggFunction; import org.apache.calcite.sql.SqlFunction; import org.apache.calcite.sql.SqlFunctionCategory; @@ -33,10 +35,14 @@ import org.apache.calcite.sql.type.SqlTypeName; import org.apache.calcite.sql.util.ChainedSqlOperatorTable; import org.apache.calcite.sql.util.ListSqlOperatorTable; +import org.apache.calcite.sql.validate.SqlUserDefinedTableFunction; import org.apache.calcite.util.Optionality; import com.google.common.collect.ImmutableList; +import java.lang.reflect.Type; +import java.util.List; + /** * Mock operator table for testing purposes. Contains the standard SQL operator * table, plus a list of operators. @@ -69,6 +75,7 @@ public static void addRamp(MockSqlOperatorTable opTab) { opTab.addOperator(new DedupFunction()); opTab.addOperator(new MyFunction()); opTab.addOperator(new MyAvgAggFunction()); + opTab.addOperator(new MyTableFunction()); } /** "RAMP" user-defined function. */ @@ -144,6 +151,58 @@ public MyAvgAggFunction() { return false; } } + + /*** "TABLE_FUNC" User-Defined table function */ + public static class MyTableFunction extends SqlUserDefinedTableFunction { + public MyTableFunction() { + super( + new SqlIdentifier("TABLE_FUNC", SqlParserPos.ZERO), + ReturnTypes.CURSOR, + null, + OperandTypes.NUMERIC, + null, + new MyTableFunctionImpl()); + } + } + /** TableFunctionImpl */ + private static class MyTableFunctionImpl implements TableFunction { + @Override public List getParameters() { + FunctionParameter parameter = new FunctionParameter() { + @Override public int getOrdinal() { + return 0; + } + + @Override public String getName() { + return "i0"; + } + + @Override public RelDataType getType(RelDataTypeFactory typeFactory) { + return typeFactory.createSqlType(SqlTypeName.BIGINT); + } + + @Override public boolean isOptional() { + return false; + } + }; + return ImmutableList.of(parameter); + } + + + @Override public RelDataType getRowType(RelDataTypeFactory typeFactory, + List arguments) { + RelDataType type0 = typeFactory.createSqlType(SqlTypeName.BIGINT); + RelDataType type1 = typeFactory.createSqlType(SqlTypeName.VARCHAR); + + return typeFactory.createStructType( + ImmutableList.of(type0, type1), + ImmutableList.of("f0", "f1")); + } + + + @Override public Type getElementType(List arguments) { + return Object[].class; + } + } } // End MockSqlOperatorTable.java diff --git a/core/src/test/java/org/apache/calcite/test/SqlToRelConverterTest.java b/core/src/test/java/org/apache/calcite/test/SqlToRelConverterTest.java index 7ead746deb5f..2e3e107116da 100644 --- a/core/src/test/java/org/apache/calcite/test/SqlToRelConverterTest.java +++ b/core/src/test/java/org/apache/calcite/test/SqlToRelConverterTest.java @@ -1071,6 +1071,57 @@ public void testJoinUsingDynamicTable() { sql("select * from dept, lateral table(ramp(dept.deptno))").ok(); } + @Test public void testTableFunctionInSelect() { + sql("select table_func(dept.deptno) as (f0, f1) from dept") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithUnion() { + sql("select * from (" + + "select a.deptno, table_func(a.deptno) as (f0, f1)" + + " from (select * from dept) a " + + " union all " + + " select 1, table_func(b.deptno) as (f0, f1) from dept b" + + ")") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithoutFrom() { + sql("select table_func(1) as (f0,f1)") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithOrderBy() { + sql("select table_func(dept.deptno) as (f0, f1) from" + + " dept order by deptno") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithJoin() { + sql("select table_func(a.deptno) as (f0, f1) from " + + "dept a join dept b on a.deptno = b.deptno") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithInQuery() { + sql("select * from dept where deptno in " + + "(select f0 from (select table_func(a.deptno) as (f0, f1) from dept a))") + .conformance(SqlConformance.HIVE) + .ok(); + } + + @Test public void testTableFunctionInSelectWithUnnest() { + sql("select table_func(e.empno) as (f0, f1) " + + "from dept_nested as d, UNNEST(d.employees) e") + .conformance(SqlConformance.HIVE) + .ok(); + } + @Test public void testCollectionTableWithLateral2() { sql("select * from dept, lateral table(ramp(deptno))").ok(); } diff --git a/core/src/test/java/org/apache/calcite/test/SqlValidatorTest.java b/core/src/test/java/org/apache/calcite/test/SqlValidatorTest.java index 5fe0a66276ee..b3c7cb87999a 100644 --- a/core/src/test/java/org/apache/calcite/test/SqlValidatorTest.java +++ b/core/src/test/java/org/apache/calcite/test/SqlValidatorTest.java @@ -11352,6 +11352,35 @@ private void checkCustomColumnResolving(String table) { } } + @Test public void testTableFunction() { + tester = tester.withConformance(SqlConformance.HIVE); + sql("select n,c from (select TABLE_FUNC(1) as (n,c) from emp)") + .ok(); + sql("select char_length(c)*2 from" + + " (select TABLE_FUNC(1) as (n,c) from emp) where char_length(c) > 1") + .ok(); + sql("select char_length(c)*2, f0 from" + + " (select 1 as f0, TABLE_FUNC(1) as (n,c) from emp) where char_length(c) > 1") + .ok(); + sql("select TABLE_FUNC(1) as (n,c)") + .ok(); + + sql("select ^test_udf('1')^ as (n,c) from emp") + .fails("'TEST_UDF' should be a table function"); + + sql("^select TABLE_FUNC(1) as (n1,c1), TABLE_FUNC(2)^ as (n2,c2) from emp") + .fails("Only one table function is allowed in select list"); + + sql("select ^TABLE_FUNC(1)^ as (n1,c1) , count(1) from emp having count(1) > 0") + .fails("Table function is not allowed in aggregate statement"); + + sql("select ^TABLE_FUNC(1)^ as (n1,c1) from emp group by empno") + .fails("Table function is not allowed in aggregate statement"); + + tester = tester.withConformance(SqlConformanceEnum.DEFAULT); + sql("select TABLE_FUNC(1) as (n,c) from emp") + .fails("(.*)Table function is not allowed in select list(.*)"); + } } // End SqlValidatorTest.java diff --git a/core/src/test/java/org/apache/calcite/test/TableFunctionTest.java b/core/src/test/java/org/apache/calcite/test/TableFunctionTest.java index 98cc725edd7d..848f8854efdc 100644 --- a/core/src/test/java/org/apache/calcite/test/TableFunctionTest.java +++ b/core/src/test/java/org/apache/calcite/test/TableFunctionTest.java @@ -408,6 +408,23 @@ private Connection getConnectionWithMultiplyFunction() with().query(q).returnsUnordered("C=7"); } + @Test public void testTableFunctionInSelect() { + final String q = "select * from (" + + "select f0, \"s\".\"fibonacci2\"(20) as (c)" + + " from (select 1 as f0)" + + ")"; + with() + .with(CalciteConnectionProperty.CONFORMANCE, + SqlConformanceEnum.HIVE) + .query(q).returnsUnordered("F0=1; C=1", + "F0=1; C=1", + "F0=1; C=13", + "F0=1; C=2", + "F0=1; C=3", + "F0=1; C=5", + "F0=1; C=8"); + } + @Test public void testCrossApply() { final String q1 = "select *\n" + "from (values 2, 5) as t (c)\n" diff --git a/core/src/test/resources/org/apache/calcite/test/SqlToRelConverterTest.xml b/core/src/test/resources/org/apache/calcite/test/SqlToRelConverterTest.xml index f8cace1338a0..b39ed2e2bbae 100644 --- a/core/src/test/resources/org/apache/calcite/test/SqlToRelConverterTest.xml +++ b/core/src/test/resources/org/apache/calcite/test/SqlToRelConverterTest.xml @@ -236,6 +236,120 @@ LogicalProject(DEPTNO=[$0], NAME=[$1], I=[$2]) LogicalCorrelate(correlation=[$cor0], joinType=[inner], requiredColumns=[{0}]) LogicalTableScan(table=[[CATALOG, SALES, DEPT]]) LogicalTableFunctionScan(invocation=[RAMP($cor0.DEPTNO)], rowType=[RecordType(INTEGER I)]) +]]> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/site/_docs/reference.md b/site/_docs/reference.md index c5efae80af73..6e24900636d3 100644 --- a/site/_docs/reference.md +++ b/site/_docs/reference.md @@ -213,6 +213,7 @@ selectWithoutFrom: projectItem: expression [ [ AS ] columnAlias ] + | expression AS '(' columnAlias [, columnAlias ]* ')' | tableAlias . * tableExpression: @@ -288,6 +289,9 @@ the nth item in the SELECT clause. In *query*, *count* and *start* may each be either an unsigned integer literal or a dynamic parameter whose value is an integer. +In *projectItem*, *expression* followed by a *As* column alias list is only allowed in +certain [conformance levels] ({{ site.apiRoot }}/org/apache/calcite/sql/validate/SqlConformance.html#allowSelectTableFunction--); +in those same conformance levels, table function can be used in the select list. An aggregate query is a query that contains a GROUP BY or a HAVING clause, or aggregate functions in the SELECT clause. In the SELECT, HAVING and ORDER BY clauses of an aggregate query, all expressions