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

org.apache.calcite.jdbc.CalciteConnectionImpl 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.jdbc;

import org.apache.calcite.DataContext;
import org.apache.calcite.adapter.java.JavaTypeFactory;
import org.apache.calcite.avatica.AvaticaConnection;
import org.apache.calcite.avatica.AvaticaFactory;
import org.apache.calcite.avatica.AvaticaSite;
import org.apache.calcite.avatica.AvaticaStatement;
import org.apache.calcite.avatica.Helper;
import org.apache.calcite.avatica.InternalProperty;
import org.apache.calcite.avatica.Meta;
import org.apache.calcite.avatica.MetaImpl;
import org.apache.calcite.avatica.NoSuchStatementException;
import org.apache.calcite.avatica.UnregisteredDriver;
import org.apache.calcite.avatica.remote.TypedValue;
import org.apache.calcite.config.CalciteConnectionConfig;
import org.apache.calcite.config.CalciteConnectionConfigImpl;
import org.apache.calcite.jdbc.CalcitePrepare.Context;
import org.apache.calcite.linq4j.BaseQueryable;
import org.apache.calcite.linq4j.Enumerable;
import org.apache.calcite.linq4j.Enumerator;
import org.apache.calcite.linq4j.Ord;
import org.apache.calcite.linq4j.QueryProvider;
import org.apache.calcite.linq4j.Queryable;
import org.apache.calcite.linq4j.function.Function0;
import org.apache.calcite.linq4j.tree.Expression;
import org.apache.calcite.linq4j.tree.Expressions;
import org.apache.calcite.materialize.Lattice;
import org.apache.calcite.materialize.MaterializationService;
import org.apache.calcite.prepare.CalciteCatalogReader;
import org.apache.calcite.rel.type.DelegatingTypeSystem;
import org.apache.calcite.rel.type.RelDataTypeSystem;
import org.apache.calcite.runtime.Hook;
import org.apache.calcite.schema.SchemaPlus;
import org.apache.calcite.schema.SchemaVersion;
import org.apache.calcite.schema.Schemas;
import org.apache.calcite.schema.impl.AbstractSchema;
import org.apache.calcite.schema.impl.LongSchemaVersion;
import org.apache.calcite.server.CalciteServer;
import org.apache.calcite.server.CalciteServerStatement;
import org.apache.calcite.sql.advise.SqlAdvisor;
import org.apache.calcite.sql.advise.SqlAdvisorValidator;
import org.apache.calcite.sql.fun.SqlStdOperatorTable;
import org.apache.calcite.sql.parser.SqlParser;
import org.apache.calcite.sql.validate.SqlConformanceEnum;
import org.apache.calcite.sql.validate.SqlValidatorWithHints;
import org.apache.calcite.tools.RelRunner;
import org.apache.calcite.util.BuiltInMethod;
import org.apache.calcite.util.Holder;
import org.apache.calcite.util.Util;

import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;

import java.io.Serializable;
import java.lang.reflect.Type;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Properties;
import java.util.TimeZone;
import java.util.concurrent.atomic.AtomicBoolean;

/**
 * Implementation of JDBC connection
 * in the Calcite engine.
 *
 * 

Abstract to allow newer versions of JDBC to add methods.

*/ abstract class CalciteConnectionImpl extends AvaticaConnection implements CalciteConnection, QueryProvider { public final JavaTypeFactory typeFactory; final CalciteSchema rootSchema; final Function0 prepareFactory; final CalciteServer server = new CalciteServerImpl(); // must be package-protected static final Trojan TROJAN = createTrojan(); /** * Creates a CalciteConnectionImpl. * *

Not public; method is called only from the driver.

* * @param driver Driver * @param factory Factory for JDBC objects * @param url Server URL * @param info Other connection properties * @param rootSchema Root schema, or null * @param typeFactory Type factory, or null */ protected CalciteConnectionImpl(Driver driver, AvaticaFactory factory, String url, Properties info, CalciteSchema rootSchema, JavaTypeFactory typeFactory) { super(driver, factory, url, info); CalciteConnectionConfig cfg = new CalciteConnectionConfigImpl(info); this.prepareFactory = driver.prepareFactory; if (typeFactory != null) { this.typeFactory = typeFactory; } else { RelDataTypeSystem typeSystem = cfg.typeSystem(RelDataTypeSystem.class, RelDataTypeSystem.DEFAULT); if (cfg.conformance().shouldConvertRaggedUnionTypesToVarying()) { typeSystem = new DelegatingTypeSystem(typeSystem) { @Override public boolean shouldConvertRaggedUnionTypesToVarying() { return true; } }; } this.typeFactory = new JavaTypeFactoryImpl(typeSystem); } this.rootSchema = Objects.requireNonNull(rootSchema != null ? rootSchema : CalciteSchema.createRootSchema(true)); Preconditions.checkArgument(this.rootSchema.isRoot(), "must be root schema"); this.properties.put(InternalProperty.CASE_SENSITIVE, cfg.caseSensitive()); this.properties.put(InternalProperty.UNQUOTED_CASING, cfg.unquotedCasing()); this.properties.put(InternalProperty.QUOTED_CASING, cfg.quotedCasing()); this.properties.put(InternalProperty.QUOTING, cfg.quoting()); } CalciteMetaImpl meta() { return (CalciteMetaImpl) meta; } public CalciteConnectionConfig config() { return new CalciteConnectionConfigImpl(info); } public Context createPrepareContext() { return new ContextImpl(this); } /** Called after the constructor has completed and the model has been * loaded. */ void init() { final MaterializationService service = MaterializationService.instance(); for (CalciteSchema.LatticeEntry e : Schemas.getLatticeEntries(rootSchema)) { final Lattice lattice = e.getLattice(); for (Lattice.Tile tile : lattice.computeTiles()) { service.defineTile(lattice, tile.bitSet(), tile.measures, e.schema, true, true); } } } @Override public T unwrap(Class iface) throws SQLException { if (iface == RelRunner.class) { return iface.cast((RelRunner) rel -> { try { return prepareStatement_(CalcitePrepare.Query.of(rel), ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_READ_ONLY, getHoldability()); } catch (SQLException e) { throw new RuntimeException(e); } }); } return super.unwrap(iface); } @Override public CalciteStatement createStatement(int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { return (CalciteStatement) super.createStatement(resultSetType, resultSetConcurrency, resultSetHoldability); } @Override public CalcitePreparedStatement prepareStatement( String sql, int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { final CalcitePrepare.Query query = CalcitePrepare.Query.of(sql); return prepareStatement_(query, resultSetType, resultSetConcurrency, resultSetHoldability); } private CalcitePreparedStatement prepareStatement_( CalcitePrepare.Query query, int resultSetType, int resultSetConcurrency, int resultSetHoldability) throws SQLException { try { final Meta.Signature signature = parseQuery(query, createPrepareContext(), -1); final CalcitePreparedStatement calcitePreparedStatement = (CalcitePreparedStatement) factory.newPreparedStatement(this, null, signature, resultSetType, resultSetConcurrency, resultSetHoldability); server.getStatement(calcitePreparedStatement.handle).setSignature(signature); return calcitePreparedStatement; } catch (Exception e) { throw Helper.INSTANCE.createException( "Error while preparing statement [" + query.sql + "]", e); } } CalcitePrepare.CalciteSignature parseQuery( CalcitePrepare.Query query, CalcitePrepare.Context prepareContext, long maxRowCount) { CalcitePrepare.Dummy.push(prepareContext); try { final CalcitePrepare prepare = prepareFactory.apply(); return prepare.prepareSql(prepareContext, query, Object[].class, maxRowCount); } finally { CalcitePrepare.Dummy.pop(prepareContext); } } @Override public AtomicBoolean getCancelFlag(Meta.StatementHandle handle) throws NoSuchStatementException { final CalciteServerStatement serverStatement = server.getStatement(handle); return ((CalciteServerStatementImpl) serverStatement).cancelFlag; } // CalciteConnection methods public SchemaPlus getRootSchema() { return rootSchema.plus(); } public JavaTypeFactory getTypeFactory() { return typeFactory; } public Properties getProperties() { return info; } // QueryProvider methods public Queryable createQuery( Expression expression, Class rowType) { return new CalciteQueryable<>(this, rowType, expression); } public Queryable createQuery(Expression expression, Type rowType) { return new CalciteQueryable<>(this, rowType, expression); } public T execute(Expression expression, Type type) { return null; // TODO: } public T execute(Expression expression, Class type) { return null; // TODO: } public Enumerator executeQuery(Queryable queryable) { try { CalciteStatement statement = (CalciteStatement) createStatement(); CalcitePrepare.CalciteSignature signature = statement.prepare(queryable); return enumerable(statement.handle, signature).enumerator(); } catch (SQLException e) { throw new RuntimeException(e); } } public Enumerable enumerable(Meta.StatementHandle handle, CalcitePrepare.CalciteSignature signature) throws SQLException { Map map = new LinkedHashMap<>(); AvaticaStatement statement = lookupStatement(handle); final List parameterValues = TROJAN.getParameterValues(statement); if (MetaImpl.checkParameterValueHasNull(parameterValues)) { throw new SQLException("exception while executing query: unbound parameter"); } for (Ord o : Ord.zip(parameterValues)) { map.put("?" + o.i, o.e.toLocal()); } map.putAll(signature.internalParameters); final AtomicBoolean cancelFlag; try { cancelFlag = getCancelFlag(handle); } catch (NoSuchStatementException e) { throw new RuntimeException(e); } map.put(DataContext.Variable.CANCEL_FLAG.camelName, cancelFlag); int queryTimeout = statement.getQueryTimeout(); // Avoid overflow if (queryTimeout > 0 && queryTimeout < Integer.MAX_VALUE / 1000) { map.put(DataContext.Variable.TIMEOUT.camelName, queryTimeout * 1000L); } final DataContext dataContext = createDataContext(map, signature.rootSchema); return signature.enumerable(dataContext); } public DataContext createDataContext(Map parameterValues, CalciteSchema rootSchema) { if (config().spark()) { return new SlimDataContext(); } return new DataContextImpl(this, parameterValues, rootSchema); } // do not make public UnregisteredDriver getDriver() { return driver; } // do not make public AvaticaFactory getFactory() { return factory; } /** Implementation of Queryable. * * @param element type */ static class CalciteQueryable extends BaseQueryable { CalciteQueryable(CalciteConnection connection, Type elementType, Expression expression) { super(connection, elementType, expression); } public CalciteConnection getConnection() { return (CalciteConnection) provider; } } /** Implementation of Server. */ private static class CalciteServerImpl implements CalciteServer { final Map statementMap = new HashMap<>(); public void removeStatement(Meta.StatementHandle h) { statementMap.remove(h.id); } public void addStatement(CalciteConnection connection, Meta.StatementHandle h) { final CalciteConnectionImpl c = (CalciteConnectionImpl) connection; final CalciteServerStatement previous = statementMap.put(h.id, new CalciteServerStatementImpl(c)); if (previous != null) { throw new AssertionError(); } } public CalciteServerStatement getStatement(Meta.StatementHandle h) throws NoSuchStatementException { CalciteServerStatement statement = statementMap.get(h.id); if (statement == null) { throw new NoSuchStatementException(h); } return statement; } } /** Schema that has no parents. */ static class RootSchema extends AbstractSchema { RootSchema() { super(); } @Override public Expression getExpression(SchemaPlus parentSchema, String name) { return Expressions.call( DataContext.ROOT, BuiltInMethod.DATA_CONTEXT_GET_ROOT_SCHEMA.method); } } /** Implementation of DataContext. */ static class DataContextImpl implements DataContext { private final ImmutableMap map; private final CalciteSchema rootSchema; private final QueryProvider queryProvider; private final JavaTypeFactory typeFactory; DataContextImpl(CalciteConnectionImpl connection, Map parameters, CalciteSchema rootSchema) { this.queryProvider = connection; this.typeFactory = connection.getTypeFactory(); this.rootSchema = rootSchema; // Store the time at which the query started executing. The SQL // standard says that functions such as CURRENT_TIMESTAMP return the // same value throughout the query. final Holder timeHolder = Holder.of(System.currentTimeMillis()); // Give a hook chance to alter the clock. Hook.CURRENT_TIME.run(timeHolder); final long time = timeHolder.get(); final TimeZone timeZone = connection.getTimeZone(); final long localOffset = timeZone.getOffset(time); final long currentOffset = localOffset; final String user = "sa"; final String systemUser = System.getProperty("user.name"); final String localeName = connection.config().locale(); final Locale locale = localeName != null ? Util.parseLocale(localeName) : Locale.ROOT; // Give a hook chance to alter standard input, output, error streams. final Holder streamHolder = Holder.of(new Object[] {System.in, System.out, System.err}); Hook.STANDARD_STREAMS.run(streamHolder); ImmutableMap.Builder builder = ImmutableMap.builder(); builder.put(Variable.UTC_TIMESTAMP.camelName, time) .put(Variable.CURRENT_TIMESTAMP.camelName, time + currentOffset) .put(Variable.LOCAL_TIMESTAMP.camelName, time + localOffset) .put(Variable.TIME_ZONE.camelName, timeZone) .put(Variable.USER.camelName, user) .put(Variable.SYSTEM_USER.camelName, systemUser) .put(Variable.LOCALE.camelName, locale) .put(Variable.STDIN.camelName, streamHolder.get()[0]) .put(Variable.STDOUT.camelName, streamHolder.get()[1]) .put(Variable.STDERR.camelName, streamHolder.get()[2]); for (Map.Entry entry : parameters.entrySet()) { Object e = entry.getValue(); if (e == null) { e = AvaticaSite.DUMMY_VALUE; } builder.put(entry.getKey(), e); } map = builder.build(); } public synchronized Object get(String name) { Object o = map.get(name); if (o == AvaticaSite.DUMMY_VALUE) { return null; } if (o == null && Variable.SQL_ADVISOR.camelName.equals(name)) { return getSqlAdvisor(); } return o; } private SqlAdvisor getSqlAdvisor() { final CalciteConnectionImpl con = (CalciteConnectionImpl) queryProvider; final String schemaName; try { schemaName = con.getSchema(); } catch (SQLException e) { throw new RuntimeException(e); } final List schemaPath = schemaName == null ? ImmutableList.of() : ImmutableList.of(schemaName); final SqlValidatorWithHints validator = new SqlAdvisorValidator(SqlStdOperatorTable.instance(), new CalciteCatalogReader(rootSchema, schemaPath, typeFactory, con.config()), typeFactory, SqlConformanceEnum.DEFAULT); final CalciteConnectionConfig config = con.config(); // This duplicates org.apache.calcite.prepare.CalcitePrepareImpl.prepare2_ final SqlParser.Config parserConfig = SqlParser.configBuilder() .setQuotedCasing(config.quotedCasing()) .setUnquotedCasing(config.unquotedCasing()) .setQuoting(config.quoting()) .setConformance(config.conformance()) .setCaseSensitive(config.caseSensitive()) .build(); return new SqlAdvisor(validator, parserConfig); } public SchemaPlus getRootSchema() { return rootSchema == null ? null : rootSchema.plus(); } public JavaTypeFactory getTypeFactory() { return typeFactory; } public QueryProvider getQueryProvider() { return queryProvider; } } /** Implementation of Context. */ static class ContextImpl implements CalcitePrepare.Context { private final CalciteConnectionImpl connection; private final CalciteSchema mutableRootSchema; private final CalciteSchema rootSchema; ContextImpl(CalciteConnectionImpl connection) { this.connection = Objects.requireNonNull(connection); long now = System.currentTimeMillis(); SchemaVersion schemaVersion = new LongSchemaVersion(now); this.mutableRootSchema = connection.rootSchema; this.rootSchema = mutableRootSchema.createSnapshot(schemaVersion); } public JavaTypeFactory getTypeFactory() { return connection.typeFactory; } public CalciteSchema getRootSchema() { return rootSchema; } public CalciteSchema getMutableRootSchema() { return mutableRootSchema; } public List getDefaultSchemaPath() { final String schemaName; try { schemaName = connection.getSchema(); } catch (SQLException e) { throw new RuntimeException(e); } return schemaName == null ? ImmutableList.of() : ImmutableList.of(schemaName); } public List getObjectPath() { return null; } public CalciteConnectionConfig config() { return connection.config(); } public DataContext getDataContext() { return connection.createDataContext(ImmutableMap.of(), rootSchema); } public RelRunner getRelRunner() { final RelRunner runner; try { runner = connection.unwrap(RelRunner.class); } catch (SQLException e) { throw new RuntimeException(e); } if (runner == null) { throw new UnsupportedOperationException(); } return runner; } public CalcitePrepare.SparkHandler spark() { final boolean enable = config().spark(); return CalcitePrepare.Dummy.getSparkHandler(enable); } } /** Implementation of {@link DataContext} that has few variables and is * {@link Serializable}. For Spark. */ private static class SlimDataContext implements DataContext, Serializable { public SchemaPlus getRootSchema() { return null; } public JavaTypeFactory getTypeFactory() { return null; } public QueryProvider getQueryProvider() { return null; } public Object get(String name) { return null; } } /** Implementation of {@link CalciteServerStatement}. */ static class CalciteServerStatementImpl implements CalciteServerStatement { private final CalciteConnectionImpl connection; private Iterator iterator; private Meta.Signature signature; private final AtomicBoolean cancelFlag = new AtomicBoolean(); CalciteServerStatementImpl(CalciteConnectionImpl connection) { this.connection = Objects.requireNonNull(connection); } public Context createPrepareContext() { return connection.createPrepareContext(); } public CalciteConnection getConnection() { return connection; } public void setSignature(Meta.Signature signature) { this.signature = signature; } public Meta.Signature getSignature() { return signature; } public Iterator getResultSet() { return iterator; } public void setResultSet(Iterator iterator) { this.iterator = iterator; } } } // End CalciteConnectionImpl.java