All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.calcite.prepare.CalcitePrepareImpl Maven / Gradle / Ivy

There is a newer version: 1.21.0.265
Show newest version
/*
 * 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.calcite.prepare;

import org.apache.calcite.adapter.enumerable.EnumerableCalc;
import org.apache.calcite.adapter.enumerable.EnumerableConvention;
import org.apache.calcite.adapter.enumerable.EnumerableInterpretable;
import org.apache.calcite.adapter.enumerable.EnumerableRel;
import org.apache.calcite.adapter.enumerable.EnumerableRules;
import org.apache.calcite.adapter.enumerable.RexToLixTranslator;
import org.apache.calcite.adapter.java.JavaTypeFactory;
import org.apache.calcite.avatica.AvaticaParameter;
import org.apache.calcite.avatica.ColumnMetaData;
import org.apache.calcite.avatica.Meta;
import org.apache.calcite.config.CalciteConnectionConfig;
import org.apache.calcite.config.CalciteSystemProperty;
import org.apache.calcite.interpreter.BindableConvention;
import org.apache.calcite.interpreter.Interpreters;
import org.apache.calcite.jdbc.CalcitePrepare;
import org.apache.calcite.jdbc.CalciteSchema;
import org.apache.calcite.jdbc.CalciteSchema.LatticeEntry;
import org.apache.calcite.linq4j.Linq4j;
import org.apache.calcite.linq4j.Ord;
import org.apache.calcite.linq4j.Queryable;
import org.apache.calcite.linq4j.function.Function1;
import org.apache.calcite.linq4j.tree.BinaryExpression;
import org.apache.calcite.linq4j.tree.BlockStatement;
import org.apache.calcite.linq4j.tree.Blocks;
import org.apache.calcite.linq4j.tree.ConstantExpression;
import org.apache.calcite.linq4j.tree.Expression;
import org.apache.calcite.linq4j.tree.Expressions;
import org.apache.calcite.linq4j.tree.MemberExpression;
import org.apache.calcite.linq4j.tree.MethodCallExpression;
import org.apache.calcite.linq4j.tree.NewExpression;
import org.apache.calcite.linq4j.tree.ParameterExpression;
import org.apache.calcite.materialize.MaterializationService;
import org.apache.calcite.plan.Contexts;
import org.apache.calcite.plan.Convention;
import org.apache.calcite.plan.ConventionTraitDef;
import org.apache.calcite.plan.RelOptCluster;
import org.apache.calcite.plan.RelOptCostFactory;
import org.apache.calcite.plan.RelOptPlanner;
import org.apache.calcite.plan.RelOptRule;
import org.apache.calcite.plan.RelOptTable;
import org.apache.calcite.plan.RelOptUtil;
import org.apache.calcite.plan.hep.HepPlanner;
import org.apache.calcite.plan.hep.HepProgramBuilder;
import org.apache.calcite.plan.volcano.VolcanoPlanner;
import org.apache.calcite.rel.RelCollation;
import org.apache.calcite.rel.RelCollationTraitDef;
import org.apache.calcite.rel.RelCollations;
import org.apache.calcite.rel.RelNode;
import org.apache.calcite.rel.RelRoot;
import org.apache.calcite.rel.core.Filter;
import org.apache.calcite.rel.core.Project;
import org.apache.calcite.rel.core.Sort;
import org.apache.calcite.rel.core.TableScan;
import org.apache.calcite.rel.type.RelDataType;
import org.apache.calcite.rel.type.RelDataTypeFactory;
import org.apache.calcite.rel.type.RelDataTypeField;
import org.apache.calcite.rex.RexBuilder;
import org.apache.calcite.rex.RexInputRef;
import org.apache.calcite.rex.RexNode;
import org.apache.calcite.rex.RexProgram;
import org.apache.calcite.runtime.Bindable;
import org.apache.calcite.runtime.Hook;
import org.apache.calcite.runtime.Typed;
import org.apache.calcite.schema.Schemas;
import org.apache.calcite.schema.Table;
import org.apache.calcite.server.CalciteServerStatement;
import org.apache.calcite.sql.SqlBinaryOperator;
import org.apache.calcite.sql.SqlExecutableStatement;
import org.apache.calcite.sql.SqlExplainFormat;
import org.apache.calcite.sql.SqlExplainLevel;
import org.apache.calcite.sql.SqlKind;
import org.apache.calcite.sql.SqlNode;
import org.apache.calcite.sql.SqlOperator;
import org.apache.calcite.sql.SqlOperatorTable;
import org.apache.calcite.sql.SqlUtil;
import org.apache.calcite.sql.fun.SqlStdOperatorTable;
import org.apache.calcite.sql.parser.SqlParseException;
import org.apache.calcite.sql.parser.SqlParser;
import org.apache.calcite.sql.parser.SqlParserImplFactory;
import org.apache.calcite.sql.type.ExtraSqlTypes;
import org.apache.calcite.sql.type.SqlTypeName;
import org.apache.calcite.sql.util.ChainedSqlOperatorTable;
import org.apache.calcite.sql.validate.SqlConformance;
import org.apache.calcite.sql.validate.SqlValidator;
import org.apache.calcite.sql2rel.SqlRexConvertletTable;
import org.apache.calcite.sql2rel.SqlToRelConverter;
import org.apache.calcite.sql2rel.StandardConvertletTable;
import org.apache.calcite.tools.FrameworkConfig;
import org.apache.calcite.tools.Frameworks;
import org.apache.calcite.util.ImmutableIntList;
import org.apache.calcite.util.Pair;
import org.apache.calcite.util.Util;

import com.google.common.base.Supplier;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;

import java.lang.reflect.Type;
import java.math.BigDecimal;
import java.sql.DatabaseMetaData;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import static org.apache.calcite.util.Static.RESOURCE;

/**
 * Shit just got real.
 *
 * 

This class is public so that projects that create their own JDBC driver * and server can fine-tune preferences. However, this class and its methods are * subject to change without notice.

*/ public class CalcitePrepareImpl implements CalcitePrepare { @Deprecated // to be removed before 2.0 public static final boolean ENABLE_ENUMERABLE = CalciteSystemProperty.ENABLE_ENUMERABLE.value(); @Deprecated // to be removed before 2.0 public static final boolean ENABLE_STREAM = CalciteSystemProperty.ENABLE_STREAM.value(); @Deprecated // to be removed before 2.0 public static final List ENUMERABLE_RULES = EnumerableRules.ENUMERABLE_RULES; /** Whether the bindable convention should be the root convention of any * plan. If not, enumerable convention is the default. */ public final boolean enableBindable = Hook.ENABLE_BINDABLE.get(false); private static final Set SIMPLE_SQLS = ImmutableSet.of( "SELECT 1", "select 1", "SELECT 1 FROM DUAL", "select 1 from dual", "values 1", "VALUES 1"); public CalcitePrepareImpl() { } public ParseResult parse( Context context, String sql) { return parse_(context, sql, false, false, false); } public ConvertResult convert(Context context, String sql) { return (ConvertResult) parse_(context, sql, true, false, false); } public AnalyzeViewResult analyzeView(Context context, String sql, boolean fail) { return (AnalyzeViewResult) parse_(context, sql, true, true, fail); } /** Shared implementation for {@link #parse}, {@link #convert} and * {@link #analyzeView}. */ private ParseResult parse_(Context context, String sql, boolean convert, boolean analyze, boolean fail) { final JavaTypeFactory typeFactory = context.getTypeFactory(); CalciteCatalogReader catalogReader = new CalciteCatalogReader( context.getRootSchema(), context.getDefaultSchemaPath(), typeFactory, context.config()); SqlParser parser = createParser(sql); SqlNode sqlNode; try { sqlNode = parser.parseStmt(); } catch (SqlParseException e) { throw new RuntimeException("parse failed", e); } final SqlValidator validator = createSqlValidator(context, catalogReader); SqlNode sqlNode1 = validator.validate(sqlNode); if (convert) { return convert_( context, sql, analyze, fail, catalogReader, validator, sqlNode1); } return new ParseResult(this, validator, sql, sqlNode1, validator.getValidatedNodeType(sqlNode1)); } private ParseResult convert_(Context context, String sql, boolean analyze, boolean fail, CalciteCatalogReader catalogReader, SqlValidator validator, SqlNode sqlNode1) { final JavaTypeFactory typeFactory = context.getTypeFactory(); final Convention resultConvention = enableBindable ? BindableConvention.INSTANCE : EnumerableConvention.INSTANCE; final HepPlanner planner = new HepPlanner(new HepProgramBuilder().build()); planner.addRelTraitDef(ConventionTraitDef.INSTANCE); final SqlToRelConverter.ConfigBuilder configBuilder = SqlToRelConverter.configBuilder().withTrimUnusedFields(true); if (analyze) { configBuilder.withConvertTableAccess(false); } final CalcitePreparingStmt preparingStmt = new CalcitePreparingStmt(this, context, catalogReader, typeFactory, context.getRootSchema(), null, planner, resultConvention, createConvertletTable()); final SqlToRelConverter converter = preparingStmt.getSqlToRelConverter(validator, catalogReader, configBuilder.build()); final RelRoot root = converter.convertQuery(sqlNode1, false, true); if (analyze) { return analyze_(validator, sql, sqlNode1, root, fail); } return new ConvertResult(this, validator, sql, sqlNode1, validator.getValidatedNodeType(sqlNode1), root); } private AnalyzeViewResult analyze_(SqlValidator validator, String sql, SqlNode sqlNode, RelRoot root, boolean fail) { final RexBuilder rexBuilder = root.rel.getCluster().getRexBuilder(); RelNode rel = root.rel; final RelNode viewRel = rel; Project project; if (rel instanceof Project) { project = (Project) rel; rel = project.getInput(); } else { project = null; } Filter filter; if (rel instanceof Filter) { filter = (Filter) rel; rel = filter.getInput(); } else { filter = null; } TableScan scan; if (rel instanceof TableScan) { scan = (TableScan) rel; } else { scan = null; } if (scan == null) { if (fail) { throw validator.newValidationError(sqlNode, RESOURCE.modifiableViewMustBeBasedOnSingleTable()); } return new AnalyzeViewResult(this, validator, sql, sqlNode, validator.getValidatedNodeType(sqlNode), root, null, null, null, null, false); } final RelOptTable targetRelTable = scan.getTable(); final RelDataType targetRowType = targetRelTable.getRowType(); final Table table = targetRelTable.unwrap(Table.class); final List tablePath = targetRelTable.getQualifiedName(); assert table != null; List columnMapping; final Map projectMap = new HashMap<>(); if (project == null) { columnMapping = ImmutableIntList.range(0, targetRowType.getFieldCount()); } else { columnMapping = new ArrayList<>(); for (Ord node : Ord.zip(project.getProjects())) { if (node.e instanceof RexInputRef) { RexInputRef rexInputRef = (RexInputRef) node.e; int index = rexInputRef.getIndex(); if (projectMap.get(index) != null) { if (fail) { throw validator.newValidationError(sqlNode, RESOURCE.moreThanOneMappedColumn( targetRowType.getFieldList().get(index).getName(), Util.last(tablePath))); } return new AnalyzeViewResult(this, validator, sql, sqlNode, validator.getValidatedNodeType(sqlNode), root, null, null, null, null, false); } projectMap.put(index, rexBuilder.makeInputRef(viewRel, node.i)); columnMapping.add(index); } else { columnMapping.add(-1); } } } final RexNode constraint; if (filter != null) { constraint = filter.getCondition(); } else { constraint = rexBuilder.makeLiteral(true); } final List filters = new ArrayList<>(); // If we put a constraint in projectMap above, then filters will not be empty despite // being a modifiable view. final List filters2 = new ArrayList<>(); boolean retry = false; RelOptUtil.inferViewPredicates(projectMap, filters, constraint); if (fail && !filters.isEmpty()) { final Map projectMap2 = new HashMap<>(); RelOptUtil.inferViewPredicates(projectMap2, filters2, constraint); if (!filters2.isEmpty()) { throw validator.newValidationError(sqlNode, RESOURCE.modifiableViewMustHaveOnlyEqualityPredicates()); } retry = true; } // Check that all columns that are not projected have a constant value for (RelDataTypeField field : targetRowType.getFieldList()) { final int x = columnMapping.indexOf(field.getIndex()); if (x >= 0) { assert Util.skip(columnMapping, x + 1).indexOf(field.getIndex()) < 0 : "column projected more than once; should have checked above"; continue; // target column is projected } if (projectMap.get(field.getIndex()) != null) { continue; // constant expression } if (field.getType().isNullable()) { continue; // don't need expression for nullable columns; NULL suffices } if (fail) { throw validator.newValidationError(sqlNode, RESOURCE.noValueSuppliedForViewColumn(field.getName(), Util.last(tablePath))); } return new AnalyzeViewResult(this, validator, sql, sqlNode, validator.getValidatedNodeType(sqlNode), root, null, null, null, null, false); } final boolean modifiable = filters.isEmpty() || retry && filters2.isEmpty(); return new AnalyzeViewResult(this, validator, sql, sqlNode, validator.getValidatedNodeType(sqlNode), root, modifiable ? table : null, ImmutableList.copyOf(tablePath), constraint, ImmutableIntList.copyOf(columnMapping), modifiable); } @Override public void executeDdl(Context context, SqlNode node) { if (node instanceof SqlExecutableStatement) { SqlExecutableStatement statement = (SqlExecutableStatement) node; statement.execute(context); return; } throw new UnsupportedOperationException(); } /** Factory method for default SQL parser. */ protected SqlParser createParser(String sql) { return createParser(sql, createParserConfig()); } /** Factory method for SQL parser with a given configuration. */ protected SqlParser createParser(String sql, SqlParser.ConfigBuilder parserConfig) { return SqlParser.create(sql, parserConfig.build()); } /** Factory method for SQL parser configuration. */ protected SqlParser.ConfigBuilder createParserConfig() { return SqlParser.configBuilder(); } /** Factory method for default convertlet table. */ protected SqlRexConvertletTable createConvertletTable() { return StandardConvertletTable.INSTANCE; } /** Factory method for cluster. */ protected RelOptCluster createCluster(RelOptPlanner planner, RexBuilder rexBuilder) { return RelOptCluster.create(planner, rexBuilder); } /** Creates a collection of planner factories. * *

The collection must have at least one factory, and each factory must * create a planner. If the collection has more than one planner, Calcite will * try each planner in turn.

* *

One of the things you can do with this mechanism is to try a simpler, * faster, planner with a smaller rule set first, then fall back to a more * complex planner for complex and costly queries.

* *

The default implementation returns a factory that calls * {@link #createPlanner(org.apache.calcite.jdbc.CalcitePrepare.Context)}.

*/ protected List> createPlannerFactories() { return Collections.singletonList( context -> createPlanner(context, null, null)); } /** Creates a query planner and initializes it with a default set of * rules. */ protected RelOptPlanner createPlanner(CalcitePrepare.Context prepareContext) { return createPlanner(prepareContext, null, null); } /** Creates a query planner and initializes it with a default set of * rules. */ protected RelOptPlanner createPlanner( final CalcitePrepare.Context prepareContext, org.apache.calcite.plan.Context externalContext, RelOptCostFactory costFactory) { if (externalContext == null) { externalContext = Contexts.of(prepareContext.config()); } final VolcanoPlanner planner = new VolcanoPlanner(costFactory, externalContext); planner.addRelTraitDef(ConventionTraitDef.INSTANCE); if (CalciteSystemProperty.ENABLE_COLLATION_TRAIT.value()) { planner.addRelTraitDef(RelCollationTraitDef.INSTANCE); } RelOptUtil.registerDefaultRules(planner, prepareContext.config().materializationsEnabled(), enableBindable); final CalcitePrepare.SparkHandler spark = prepareContext.spark(); if (spark.enabled()) { spark.registerRules( new SparkHandler.RuleSetBuilder() { public void addRule(RelOptRule rule) { // TODO: } public void removeRule(RelOptRule rule) { // TODO: } }); } Hook.PLANNER.run(planner); // allow test to add or remove rules return planner; } public CalciteSignature prepareQueryable( Context context, Queryable queryable) { return prepare_(context, Query.of(queryable), queryable.getElementType(), -1); } public CalciteSignature prepareSql( Context context, Query query, Type elementType, long maxRowCount) { return prepare_(context, query, elementType, maxRowCount); } CalciteSignature prepare_( Context context, Query query, Type elementType, long maxRowCount) { if (SIMPLE_SQLS.contains(query.sql)) { return simplePrepare(context, query.sql); } final JavaTypeFactory typeFactory = context.getTypeFactory(); CalciteCatalogReader catalogReader = new CalciteCatalogReader( context.getRootSchema(), context.getDefaultSchemaPath(), typeFactory, context.config()); final List> plannerFactories = createPlannerFactories(); if (plannerFactories.isEmpty()) { throw new AssertionError("no planner factories"); } RuntimeException exception = Util.FoundOne.NULL; for (Function1 plannerFactory : plannerFactories) { final RelOptPlanner planner = plannerFactory.apply(context); if (planner == null) { throw new AssertionError("factory returned null planner"); } try { return prepare2_(context, query, elementType, maxRowCount, catalogReader, planner); } catch (RelOptPlanner.CannotPlanException e) { exception = e; } } throw exception; } /** Quickly prepares a simple SQL statement, circumventing the usual * preparation process. */ private CalciteSignature simplePrepare(Context context, String sql) { final JavaTypeFactory typeFactory = context.getTypeFactory(); final RelDataType x = typeFactory.builder() .add(SqlUtil.deriveAliasFromOrdinal(0), SqlTypeName.INTEGER) .build(); @SuppressWarnings("unchecked") final List list = (List) ImmutableList.of(1); final List origin = null; final List> origins = Collections.nCopies(x.getFieldCount(), origin); final List columns = getColumnMetaDataList(typeFactory, x, x, origins); final Meta.CursorFactory cursorFactory = Meta.CursorFactory.deduce(columns, null); return new CalciteSignature<>( sql, ImmutableList.of(), ImmutableMap.of(), x, columns, cursorFactory, context.getRootSchema(), ImmutableList.of(), -1, dataContext -> Linq4j.asEnumerable(list), Meta.StatementType.SELECT); } /** * Deduces the broad type of statement. * Currently returns SELECT for most statement types, but this may change. * * @param kind Kind of statement */ private Meta.StatementType getStatementType(SqlKind kind) { switch (kind) { case INSERT: case DELETE: case UPDATE: return Meta.StatementType.IS_DML; default: return Meta.StatementType.SELECT; } } /** * Deduces the broad type of statement for a prepare result. * Currently returns SELECT for most statement types, but this may change. * * @param preparedResult Prepare result */ private Meta.StatementType getStatementType(Prepare.PreparedResult preparedResult) { if (preparedResult.isDml()) { return Meta.StatementType.IS_DML; } else { return Meta.StatementType.SELECT; } } CalciteSignature prepare2_( Context context, Query query, Type elementType, long maxRowCount, CalciteCatalogReader catalogReader, RelOptPlanner planner) { final JavaTypeFactory typeFactory = context.getTypeFactory(); final EnumerableRel.Prefer prefer; if (elementType == Object[].class) { prefer = EnumerableRel.Prefer.ARRAY; } else { prefer = EnumerableRel.Prefer.CUSTOM; } final Convention resultConvention = enableBindable ? BindableConvention.INSTANCE : EnumerableConvention.INSTANCE; final CalcitePreparingStmt preparingStmt = new CalcitePreparingStmt(this, context, catalogReader, typeFactory, context.getRootSchema(), prefer, planner, resultConvention, createConvertletTable()); final RelDataType x; final Prepare.PreparedResult preparedResult; final Meta.StatementType statementType; if (query.sql != null) { final CalciteConnectionConfig config = context.config(); final SqlParser.ConfigBuilder parserConfig = createParserConfig() .setQuotedCasing(config.quotedCasing()) .setUnquotedCasing(config.unquotedCasing()) .setQuoting(config.quoting()) .setConformance(config.conformance()) .setCaseSensitive(config.caseSensitive()); final SqlParserImplFactory parserFactory = config.parserFactory(SqlParserImplFactory.class, null); if (parserFactory != null) { parserConfig.setParserFactory(parserFactory); } SqlParser parser = createParser(query.sql, parserConfig); SqlNode sqlNode; try { sqlNode = parser.parseStmt(); statementType = getStatementType(sqlNode.getKind()); } catch (SqlParseException e) { throw new RuntimeException( "parse failed: " + e.getMessage(), e); } Hook.PARSE_TREE.run(new Object[] {query.sql, sqlNode}); if (sqlNode.getKind().belongsTo(SqlKind.DDL)) { executeDdl(context, sqlNode); return new CalciteSignature<>(query.sql, ImmutableList.of(), ImmutableMap.of(), null, ImmutableList.of(), Meta.CursorFactory.OBJECT, null, ImmutableList.of(), -1, null, Meta.StatementType.OTHER_DDL); } final SqlValidator validator = createSqlValidator(context, catalogReader); validator.setIdentifierExpansion(true); validator.setDefaultNullCollation(config.defaultNullCollation()); preparedResult = preparingStmt.prepareSql( sqlNode, Object.class, validator, true); switch (sqlNode.getKind()) { case INSERT: case DELETE: case UPDATE: case EXPLAIN: // FIXME: getValidatedNodeType is wrong for DML x = RelOptUtil.createDmlRowType(sqlNode.getKind(), typeFactory); break; default: x = validator.getValidatedNodeType(sqlNode); } } else if (query.queryable != null) { x = context.getTypeFactory().createType(elementType); preparedResult = preparingStmt.prepareQueryable(query.queryable, x); statementType = getStatementType(preparedResult); } else { assert query.rel != null; x = query.rel.getRowType(); preparedResult = preparingStmt.prepareRel(query.rel); statementType = getStatementType(preparedResult); } final List parameters = new ArrayList<>(); final RelDataType parameterRowType = preparedResult.getParameterRowType(); for (RelDataTypeField field : parameterRowType.getFieldList()) { RelDataType type = field.getType(); parameters.add( new AvaticaParameter( false, getPrecision(type), getScale(type), getTypeOrdinal(type), getTypeName(type), getClassName(type), field.getName())); } RelDataType jdbcType = makeStruct(typeFactory, x); final List> originList = preparedResult.getFieldOrigins(); final List columns = getColumnMetaDataList(typeFactory, x, jdbcType, originList); Class resultClazz = null; if (preparedResult instanceof Typed) { resultClazz = (Class) ((Typed) preparedResult).getElementType(); } final Meta.CursorFactory cursorFactory = preparingStmt.resultConvention == BindableConvention.INSTANCE ? Meta.CursorFactory.ARRAY : Meta.CursorFactory.deduce(columns, resultClazz); //noinspection unchecked final Bindable bindable = preparedResult.getBindable(cursorFactory); return new CalciteSignature<>( query.sql, parameters, preparingStmt.internalParameters, jdbcType, columns, cursorFactory, context.getRootSchema(), preparedResult instanceof Prepare.PreparedResultImpl ? ((Prepare.PreparedResultImpl) preparedResult).collations : ImmutableList.of(), maxRowCount, bindable, statementType); } private SqlValidator createSqlValidator(Context context, CalciteCatalogReader catalogReader) { final SqlOperatorTable opTab0 = context.config().fun(SqlOperatorTable.class, SqlStdOperatorTable.instance()); final SqlOperatorTable opTab = ChainedSqlOperatorTable.of(opTab0, catalogReader); final JavaTypeFactory typeFactory = context.getTypeFactory(); final SqlConformance conformance = context.config().conformance(); return new CalciteSqlValidator(opTab, catalogReader, typeFactory, conformance); } private List getColumnMetaDataList( JavaTypeFactory typeFactory, RelDataType x, RelDataType jdbcType, List> originList) { final List columns = new ArrayList<>(); for (Ord pair : Ord.zip(jdbcType.getFieldList())) { final RelDataTypeField field = pair.e; final RelDataType type = field.getType(); final RelDataType fieldType = x.isStruct() ? x.getFieldList().get(pair.i).getType() : type; columns.add( metaData(typeFactory, columns.size(), field.getName(), type, fieldType, originList.get(pair.i))); } return columns; } private ColumnMetaData metaData(JavaTypeFactory typeFactory, int ordinal, String fieldName, RelDataType type, RelDataType fieldType, List origins) { final ColumnMetaData.AvaticaType avaticaType = avaticaType(typeFactory, type, fieldType); return new ColumnMetaData( ordinal, false, true, false, false, type.isNullable() ? DatabaseMetaData.columnNullable : DatabaseMetaData.columnNoNulls, true, type.getPrecision(), fieldName, origin(origins, 0), origin(origins, 2), getPrecision(type), getScale(type), origin(origins, 1), null, avaticaType, true, false, false, avaticaType.columnClassName()); } private ColumnMetaData.AvaticaType avaticaType(JavaTypeFactory typeFactory, RelDataType type, RelDataType fieldType) { final String typeName = getTypeName(type); if (type.getComponentType() != null) { final ColumnMetaData.AvaticaType componentType = avaticaType(typeFactory, type.getComponentType(), null); final Type clazz = typeFactory.getJavaClass(type.getComponentType()); final ColumnMetaData.Rep rep = ColumnMetaData.Rep.of(clazz); assert rep != null; return ColumnMetaData.array(componentType, typeName, rep); } else { int typeOrdinal = getTypeOrdinal(type); switch (typeOrdinal) { case Types.STRUCT: final List columns = new ArrayList<>(); for (RelDataTypeField field : type.getFieldList()) { columns.add( metaData(typeFactory, field.getIndex(), field.getName(), field.getType(), null, null)); } return ColumnMetaData.struct(columns); case ExtraSqlTypes.GEOMETRY: typeOrdinal = Types.VARCHAR; // fall through default: final Type clazz = typeFactory.getJavaClass(Util.first(fieldType, type)); final ColumnMetaData.Rep rep = ColumnMetaData.Rep.of(clazz); assert rep != null; return ColumnMetaData.scalar(typeOrdinal, typeName, rep); } } } private static String origin(List origins, int offsetFromEnd) { return origins == null || offsetFromEnd >= origins.size() ? null : origins.get(origins.size() - 1 - offsetFromEnd); } private int getTypeOrdinal(RelDataType type) { return type.getSqlTypeName().getJdbcOrdinal(); } private static String getClassName(RelDataType type) { return Object.class.getName(); // CALCITE-2613 } private static int getScale(RelDataType type) { return type.getScale() == RelDataType.SCALE_NOT_SPECIFIED ? 0 : type.getScale(); } private static int getPrecision(RelDataType type) { return type.getPrecision() == RelDataType.PRECISION_NOT_SPECIFIED ? 0 : type.getPrecision(); } /** Returns the type name in string form. Does not include precision, scale * or whether nulls are allowed. Example: "DECIMAL" not "DECIMAL(7, 2)"; * "INTEGER" not "JavaType(int)". */ private static String getTypeName(RelDataType type) { final SqlTypeName sqlTypeName = type.getSqlTypeName(); switch (sqlTypeName) { case ARRAY: case MULTISET: case MAP: case ROW: return type.toString(); // e.g. "INTEGER ARRAY" case INTERVAL_YEAR_MONTH: return "INTERVAL_YEAR_TO_MONTH"; case INTERVAL_DAY_HOUR: return "INTERVAL_DAY_TO_HOUR"; case INTERVAL_DAY_MINUTE: return "INTERVAL_DAY_TO_MINUTE"; case INTERVAL_DAY_SECOND: return "INTERVAL_DAY_TO_SECOND"; case INTERVAL_HOUR_MINUTE: return "INTERVAL_HOUR_TO_MINUTE"; case INTERVAL_HOUR_SECOND: return "INTERVAL_HOUR_TO_SECOND"; case INTERVAL_MINUTE_SECOND: return "INTERVAL_MINUTE_TO_SECOND"; default: return sqlTypeName.getName(); // e.g. "DECIMAL", "INTERVAL_YEAR_MONTH" } } protected void populateMaterializations(Context context, RelOptPlanner planner, Prepare.Materialization materialization) { // REVIEW: initialize queryRel and tableRel inside MaterializationService, // not here? try { final CalciteSchema schema = materialization.materializedTable.schema; CalciteCatalogReader catalogReader = new CalciteCatalogReader( schema.root(), materialization.viewSchemaPath, context.getTypeFactory(), context.config()); final CalciteMaterializer materializer = new CalciteMaterializer(this, context, catalogReader, schema, planner, createConvertletTable()); materializer.populate(materialization); } catch (Exception e) { throw new RuntimeException("While populating materialization " + materialization.materializedTable.path(), e); } } private static RelDataType makeStruct( RelDataTypeFactory typeFactory, RelDataType type) { if (type.isStruct()) { return type; } return typeFactory.builder().add("$0", type).build(); } @Deprecated // to be removed before 2.0 public R perform(CalciteServerStatement statement, Frameworks.PrepareAction action) { return perform(statement, action.getConfig(), action); } /** Executes a prepare action. */ public R perform(CalciteServerStatement statement, FrameworkConfig config, Frameworks.BasePrepareAction action) { final CalcitePrepare.Context prepareContext = statement.createPrepareContext(); final JavaTypeFactory typeFactory = prepareContext.getTypeFactory(); final CalciteSchema schema = config.getDefaultSchema() != null ? CalciteSchema.from(config.getDefaultSchema()) : prepareContext.getRootSchema(); CalciteCatalogReader catalogReader = new CalciteCatalogReader(schema.root(), schema.path(null), typeFactory, prepareContext.config()); final RexBuilder rexBuilder = new RexBuilder(typeFactory); final RelOptPlanner planner = createPlanner(prepareContext, config.getContext(), config.getCostFactory()); final RelOptCluster cluster = createCluster(planner, rexBuilder); return action.apply(cluster, catalogReader, prepareContext.getRootSchema().plus(), statement); } /** Holds state for the process of preparing a SQL statement. */ static class CalcitePreparingStmt extends Prepare implements RelOptTable.ViewExpander { protected final RelOptPlanner planner; protected final RexBuilder rexBuilder; protected final CalcitePrepareImpl prepare; protected final CalciteSchema schema; protected final RelDataTypeFactory typeFactory; protected final SqlRexConvertletTable convertletTable; private final EnumerableRel.Prefer prefer; private final Map internalParameters = new LinkedHashMap<>(); private int expansionDepth; private SqlValidator sqlValidator; CalcitePreparingStmt(CalcitePrepareImpl prepare, Context context, CatalogReader catalogReader, RelDataTypeFactory typeFactory, CalciteSchema schema, EnumerableRel.Prefer prefer, RelOptPlanner planner, Convention resultConvention, SqlRexConvertletTable convertletTable) { super(context, catalogReader, resultConvention); this.prepare = prepare; this.schema = schema; this.prefer = prefer; this.planner = planner; this.typeFactory = typeFactory; this.convertletTable = convertletTable; this.rexBuilder = new RexBuilder(typeFactory); } @Override protected void init(Class runtimeContextClass) { } public PreparedResult prepareQueryable( final Queryable queryable, RelDataType resultType) { return prepare_(() -> { final RelOptCluster cluster = prepare.createCluster(planner, rexBuilder); return new LixToRelTranslator(cluster, CalcitePreparingStmt.this) .translate(queryable); }, resultType); } public PreparedResult prepareRel(final RelNode rel) { return prepare_(() -> rel, rel.getRowType()); } private PreparedResult prepare_(Supplier fn, RelDataType resultType) { Class runtimeContextClass = Object.class; init(runtimeContextClass); final RelNode rel = fn.get(); final RelDataType rowType = rel.getRowType(); final List> fields = Pair.zip(ImmutableIntList.identity(rowType.getFieldCount()), rowType.getFieldNames()); final RelCollation collation = rel instanceof Sort ? ((Sort) rel).collation : RelCollations.EMPTY; RelRoot root = new RelRoot(rel, resultType, SqlKind.SELECT, fields, collation); if (timingTracer != null) { timingTracer.traceTime("end sql2rel"); } final RelDataType jdbcType = makeStruct(rexBuilder.getTypeFactory(), resultType); fieldOrigins = Collections.nCopies(jdbcType.getFieldCount(), null); parameterRowType = rexBuilder.getTypeFactory().builder().build(); // Structured type flattening, view expansion, and plugging in // physical storage. root = root.withRel(flattenTypes(root.rel, true)); // Trim unused fields. root = trimUnusedFields(root); final List materializations = ImmutableList.of(); final List lattices = ImmutableList.of(); root = optimize(root, materializations, lattices); if (timingTracer != null) { timingTracer.traceTime("end optimization"); } return implement(root); } @Override protected SqlToRelConverter getSqlToRelConverter( SqlValidator validator, CatalogReader catalogReader, SqlToRelConverter.Config config) { final RelOptCluster cluster = prepare.createCluster(planner, rexBuilder); return new SqlToRelConverter(this, validator, catalogReader, cluster, convertletTable, config); } @Override public RelNode flattenTypes( RelNode rootRel, boolean restructure) { final SparkHandler spark = context.spark(); if (spark.enabled()) { return spark.flattenTypes(planner, rootRel, restructure); } return rootRel; } @Override protected RelNode decorrelate(SqlToRelConverter sqlToRelConverter, SqlNode query, RelNode rootRel) { return sqlToRelConverter.decorrelate(query, rootRel); } @Override public RelRoot expandView(RelDataType rowType, String queryString, List schemaPath, List viewPath) { expansionDepth++; SqlParser parser = prepare.createParser(queryString); SqlNode sqlNode; try { sqlNode = parser.parseQuery(); } catch (SqlParseException e) { throw new RuntimeException("parse failed", e); } // View may have different schema path than current connection. final CatalogReader catalogReader = this.catalogReader.withSchemaPath(schemaPath); SqlValidator validator = createSqlValidator(catalogReader); final SqlToRelConverter.Config config = SqlToRelConverter.configBuilder() .withTrimUnusedFields(true).build(); SqlToRelConverter sqlToRelConverter = getSqlToRelConverter(validator, catalogReader, config); RelRoot root = sqlToRelConverter.convertQuery(sqlNode, true, false); --expansionDepth; return root; } protected SqlValidator createSqlValidator(CatalogReader catalogReader) { return prepare.createSqlValidator(context, (CalciteCatalogReader) catalogReader); } @Override protected SqlValidator getSqlValidator() { if (sqlValidator == null) { sqlValidator = createSqlValidator(catalogReader); } return sqlValidator; } @Override protected PreparedResult createPreparedExplanation( RelDataType resultType, RelDataType parameterRowType, RelRoot root, SqlExplainFormat format, SqlExplainLevel detailLevel) { return new CalcitePreparedExplain(resultType, parameterRowType, root, format, detailLevel); } @Override protected PreparedResult implement(RelRoot root) { RelDataType resultType = root.rel.getRowType(); boolean isDml = root.kind.belongsTo(SqlKind.DML); final Bindable bindable; if (resultConvention == BindableConvention.INSTANCE) { bindable = Interpreters.bindable(root.rel); } else { EnumerableRel enumerable = (EnumerableRel) root.rel; if (!root.isRefTrivial()) { final List projects = new ArrayList<>(); final RexBuilder rexBuilder = enumerable.getCluster().getRexBuilder(); for (int field : Pair.left(root.fields)) { projects.add(rexBuilder.makeInputRef(enumerable, field)); } RexProgram program = RexProgram.create(enumerable.getRowType(), projects, null, root.validatedRowType, rexBuilder); enumerable = EnumerableCalc.create(enumerable, program); } try { CatalogReader.THREAD_LOCAL.set(catalogReader); final SqlConformance conformance = context.config().conformance(); internalParameters.put("_conformance", conformance); bindable = EnumerableInterpretable.toBindable(internalParameters, context.spark(), enumerable, prefer); } finally { CatalogReader.THREAD_LOCAL.remove(); } } if (timingTracer != null) { timingTracer.traceTime("end codegen"); } if (timingTracer != null) { timingTracer.traceTime("end compilation"); } return new PreparedResultImpl( resultType, parameterRowType, fieldOrigins, root.collation.getFieldCollations().isEmpty() ? ImmutableList.of() : ImmutableList.of(root.collation), root.rel, mapTableModOp(isDml, root.kind), isDml) { public String getCode() { throw new UnsupportedOperationException(); } public Bindable getBindable(Meta.CursorFactory cursorFactory) { return bindable; } public Type getElementType() { return ((Typed) bindable).getElementType(); } }; } @Override protected List getMaterializations() { final List materializations = context.config().materializationsEnabled() ? MaterializationService.instance().query(schema) : ImmutableList.of(); for (Prepare.Materialization materialization : materializations) { prepare.populateMaterializations(context, planner, materialization); } return materializations; } @Override protected List getLattices() { return Schemas.getLatticeEntries(schema); } } /** An {@code EXPLAIN} statement, prepared and ready to execute. */ private static class CalcitePreparedExplain extends Prepare.PreparedExplain { CalcitePreparedExplain( RelDataType resultType, RelDataType parameterRowType, RelRoot root, SqlExplainFormat format, SqlExplainLevel detailLevel) { super(resultType, parameterRowType, root, format, detailLevel); } public Bindable getBindable(final Meta.CursorFactory cursorFactory) { final String explanation = getCode(); return dataContext -> { switch (cursorFactory.style) { case ARRAY: return Linq4j.singletonEnumerable(new String[] {explanation}); case OBJECT: default: return Linq4j.singletonEnumerable(explanation); } }; } } /** Translator from Java AST to {@link RexNode}. */ interface ScalarTranslator { RexNode toRex(BlockStatement statement); List toRexList(BlockStatement statement); RexNode toRex(Expression expression); ScalarTranslator bind(List parameterList, List values); } /** Basic translator. */ static class EmptyScalarTranslator implements ScalarTranslator { private final RexBuilder rexBuilder; EmptyScalarTranslator(RexBuilder rexBuilder) { this.rexBuilder = rexBuilder; } public static ScalarTranslator empty(RexBuilder builder) { return new EmptyScalarTranslator(builder); } public List toRexList(BlockStatement statement) { final List simpleList = simpleList(statement); final List list = new ArrayList<>(); for (Expression expression1 : simpleList) { list.add(toRex(expression1)); } return list; } public RexNode toRex(BlockStatement statement) { return toRex(Blocks.simple(statement)); } private static List simpleList(BlockStatement statement) { Expression simple = Blocks.simple(statement); if (simple instanceof NewExpression) { NewExpression newExpression = (NewExpression) simple; return newExpression.arguments; } else { return Collections.singletonList(simple); } } public RexNode toRex(Expression expression) { switch (expression.getNodeType()) { case MemberAccess: // Case-sensitive name match because name was previously resolved. return rexBuilder.makeFieldAccess( toRex( ((MemberExpression) expression).expression), ((MemberExpression) expression).field.getName(), true); case GreaterThan: return binary(expression, SqlStdOperatorTable.GREATER_THAN); case LessThan: return binary(expression, SqlStdOperatorTable.LESS_THAN); case Parameter: return parameter((ParameterExpression) expression); case Call: MethodCallExpression call = (MethodCallExpression) expression; SqlOperator operator = RexToLixTranslator.JAVA_TO_SQL_METHOD_MAP.get(call.method); if (operator != null) { return rexBuilder.makeCall( type(call), operator, toRex( Expressions.list() .appendIfNotNull(call.targetExpression) .appendAll(call.expressions))); } throw new RuntimeException( "Could translate call to method " + call.method); case Constant: final ConstantExpression constant = (ConstantExpression) expression; Object value = constant.value; if (value instanceof Number) { Number number = (Number) value; if (value instanceof Double || value instanceof Float) { return rexBuilder.makeApproxLiteral( BigDecimal.valueOf(number.doubleValue())); } else if (value instanceof BigDecimal) { return rexBuilder.makeExactLiteral((BigDecimal) value); } else { return rexBuilder.makeExactLiteral( BigDecimal.valueOf(number.longValue())); } } else if (value instanceof Boolean) { return rexBuilder.makeLiteral((Boolean) value); } else { return rexBuilder.makeLiteral(constant.toString()); } default: throw new UnsupportedOperationException( "unknown expression type " + expression.getNodeType() + " " + expression); } } private RexNode binary(Expression expression, SqlBinaryOperator op) { BinaryExpression call = (BinaryExpression) expression; return rexBuilder.makeCall(type(call), op, toRex(ImmutableList.of(call.expression0, call.expression1))); } private List toRex(List expressions) { final List list = new ArrayList<>(); for (Expression expression : expressions) { list.add(toRex(expression)); } return list; } protected RelDataType type(Expression expression) { final Type type = expression.getType(); return ((JavaTypeFactory) rexBuilder.getTypeFactory()).createType(type); } public ScalarTranslator bind( List parameterList, List values) { return new LambdaScalarTranslator( rexBuilder, parameterList, values); } public RexNode parameter(ParameterExpression param) { throw new RuntimeException("unknown parameter " + param); } } /** Translator that looks for parameters. */ private static class LambdaScalarTranslator extends EmptyScalarTranslator { private final List parameterList; private final List values; LambdaScalarTranslator( RexBuilder rexBuilder, List parameterList, List values) { super(rexBuilder); this.parameterList = parameterList; this.values = values; } public RexNode parameter(ParameterExpression param) { int i = parameterList.indexOf(param); if (i >= 0) { return values.get(i); } throw new RuntimeException("unknown parameter " + param); } } } // End CalcitePrepareImpl.java




© 2015 - 2025 Weber Informatics LLC | Privacy Policy