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

com.hazelcast.org.apache.calcite.tools.Frameworks Maven / Gradle / Ivy

There is a newer version: 5.5.0
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 com.hazelcast.org.apache.calcite.tools;

import com.hazelcast.org.apache.calcite.config.CalciteConnectionProperty;
import com.hazelcast.org.apache.calcite.jdbc.CalciteSchema;
import com.hazelcast.org.apache.calcite.jdbc.Driver;
import com.hazelcast.org.apache.calcite.materialize.SqlStatisticProvider;
import com.hazelcast.org.apache.calcite.plan.Context;
import com.hazelcast.org.apache.calcite.plan.Contexts;
import com.hazelcast.org.apache.calcite.plan.RelOptCluster;
import com.hazelcast.org.apache.calcite.plan.RelOptCostFactory;
import com.hazelcast.org.apache.calcite.plan.RelOptSchema;
import com.hazelcast.org.apache.calcite.plan.RelOptTable;
import com.hazelcast.org.apache.calcite.plan.RelTraitDef;
import com.hazelcast.org.apache.calcite.prepare.CalcitePrepareImpl;
import com.hazelcast.org.apache.calcite.prepare.PlannerImpl;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeSystem;
import com.hazelcast.org.apache.calcite.rex.RexExecutor;
import com.hazelcast.org.apache.calcite.schema.SchemaPlus;
import com.hazelcast.org.apache.calcite.server.CalciteServerStatement;
import com.hazelcast.org.apache.calcite.sql.SqlOperatorTable;
import com.hazelcast.org.apache.calcite.sql.fun.SqlStdOperatorTable;
import com.hazelcast.org.apache.calcite.sql.parser.SqlParser;
import com.hazelcast.org.apache.calcite.sql.validate.SqlValidator;
import com.hazelcast.org.apache.calcite.sql2rel.SqlRexConvertletTable;
import com.hazelcast.org.apache.calcite.sql2rel.SqlToRelConverter;
import com.hazelcast.org.apache.calcite.sql2rel.StandardConvertletTable;
import com.hazelcast.org.apache.calcite.statistic.QuerySqlStatisticProvider;
import com.hazelcast.org.apache.calcite.util.Util;

import com.hazelcast.com.google.common.base.Suppliers;
import com.hazelcast.com.google.common.collect.ImmutableList;

import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;

import java.sql.Connection;
import java.util.List;
import java.util.Objects;
import java.util.Properties;
import java.util.function.Supplier;

/**
 * Tools for invoking Calcite functionality without initializing a container /
 * server first.
 */
public class Frameworks {

  /** Caches an instance of the JDBC driver. */
  private static final Supplier DRIVER_SUPPLIER =
      Suppliers.memoize(Driver::new)::get;

  private Frameworks() {
  }

  /**
   * Creates a planner.
   *
   * @param config Planner configuration
   * @return Planner
   */
  public static Planner getPlanner(FrameworkConfig config) {
    return new PlannerImpl(config);
  }

  /** Piece of code to be run in a context where a planner is available. The
   * planner is accessible from the {@code cluster} parameter, as are several
   * other useful objects.
   *
   * @param  result type */
  @FunctionalInterface
  public interface PlannerAction {
    R apply(RelOptCluster cluster, RelOptSchema relOptSchema,
        SchemaPlus rootSchema);
  }

  /** Piece of code to be run in a context where a planner and statement are
   * available. The planner is accessible from the {@code cluster} parameter, as
   * are several other useful objects. The connection and
   * {@link com.hazelcast.org.apache.calcite.DataContext} are accessible from the
   * statement.
   *
   * @param  result type */
  @FunctionalInterface
  public interface BasePrepareAction {
    R apply(RelOptCluster cluster, RelOptSchema relOptSchema,
        SchemaPlus rootSchema, CalciteServerStatement statement);
  }

  /** As {@link BasePrepareAction} but with a {@link FrameworkConfig} included.
   * Deprecated because a functional interface is more convenient.
   *
   * @param  result type */
  @Deprecated // to be removed before 2.0
  public abstract static class PrepareAction
      implements BasePrepareAction {
    private final FrameworkConfig config;
    protected PrepareAction() {
      this.config = newConfigBuilder()
          .defaultSchema(Frameworks.createRootSchema(true)).build();
    }

    protected PrepareAction(FrameworkConfig config) {
      this.config = config;
    }

    public FrameworkConfig getConfig() {
      return config;
    }
  }

  /**
   * Initializes a container then calls user-specified code with a planner.
   *
   * @param action Callback containing user-specified code
   * @param config FrameworkConfig to use for planner action.
   * @return Return value from action
   */
  public static  R withPlanner(final PlannerAction action,
      final FrameworkConfig config) {
    return withPrepare(config,
        (cluster, relOptSchema, rootSchema, statement) -> {
          final CalciteSchema schema =
              CalciteSchema.from(
                  Util.first(config.getDefaultSchema(), rootSchema));
          return action.apply(cluster, relOptSchema, schema.root().plus());
        });
  }

  /**
   * Initializes a container then calls user-specified code with a planner.
   *
   * @param action Callback containing user-specified code
   * @return Return value from action
   */
  public static  R withPlanner(final PlannerAction action) {
    FrameworkConfig config = newConfigBuilder() //
        .defaultSchema(Frameworks.createRootSchema(true)).build();
    return withPlanner(action, config);
  }

  @Deprecated // to be removed before 2.0
  public static  R withPrepare(PrepareAction action) {
    return withPrepare(action.getConfig(), action);
  }

  /** As {@link #withPrepare(FrameworkConfig, BasePrepareAction)} but using a
   * default configuration. */
  public static  R withPrepare(BasePrepareAction action) {
    final FrameworkConfig config = newConfigBuilder()
        .defaultSchema(Frameworks.createRootSchema(true)).build();
    return withPrepare(config, action);
  }

  /**
   * Initializes a container then calls user-specified code with a planner
   * and statement.
   *
   * @param action Callback containing user-specified code
   * @return Return value from action
   */
  public static  R withPrepare(FrameworkConfig config,
      BasePrepareAction action) {
    try {
      final Properties info = new Properties();
      if (config.getTypeSystem() != RelDataTypeSystem.DEFAULT) {
        info.setProperty(CalciteConnectionProperty.TYPE_SYSTEM.camelName(),
            config.getTypeSystem().getClass().getName());
      }
      // Connect via a Driver instance. Don't use DriverManager because driver
      // auto-loading can get broken by shading and jar-repacking.
      Connection connection =
          DRIVER_SUPPLIER.get().connect("jdbc:calcite:", info);
      final CalciteServerStatement statement =
          connection.createStatement()
              .unwrap(CalciteServerStatement.class);
      return new CalcitePrepareImpl().perform(statement, config, action);
    } catch (Exception e) {
      throw new RuntimeException(e);
    }
  }

  /**
   * Creates a root schema.
   *
   * @param addMetadataSchema Whether to add "metadata" schema containing
   *    definitions of tables, columns etc.
   */
  public static SchemaPlus createRootSchema(boolean addMetadataSchema) {
    return CalciteSchema.createRootSchema(addMetadataSchema).plus();
  }

  /** Creates a config builder with each setting initialized to its default
   * value. */
  public static ConfigBuilder newConfigBuilder() {
    return new ConfigBuilder();
  }

  /** Creates a config builder initializing each setting from an existing
   * config.
   *
   * 

So, {@code newConfigBuilder(config).build()} will return a * value equal to {@code config}. */ public static ConfigBuilder newConfigBuilder(FrameworkConfig config) { return new ConfigBuilder(config); } /** * A builder to help you build a {@link FrameworkConfig} using defaults * where values aren't required. */ public static class ConfigBuilder { private SqlRexConvertletTable convertletTable; private SqlOperatorTable operatorTable; private ImmutableList programs; private Context context; private @Nullable ImmutableList traitDefs; private SqlParser.Config parserConfig; private SqlValidator.Config sqlValidatorConfig; private SqlToRelConverter.Config sqlToRelConverterConfig; private @Nullable SchemaPlus defaultSchema; private @Nullable RexExecutor executor; private @Nullable RelOptCostFactory costFactory; private RelDataTypeSystem typeSystem; private boolean evolveLattice; private SqlStatisticProvider statisticProvider; private RelOptTable.@Nullable ViewExpander viewExpander; /** Creates a ConfigBuilder, initializing to defaults. */ private ConfigBuilder() { convertletTable = StandardConvertletTable.INSTANCE; operatorTable = SqlStdOperatorTable.instance(); programs = ImmutableList.of(); context = Contexts.empty(); parserConfig = SqlParser.Config.DEFAULT; sqlValidatorConfig = SqlValidator.Config.DEFAULT; sqlToRelConverterConfig = SqlToRelConverter.config(); typeSystem = RelDataTypeSystem.DEFAULT; evolveLattice = false; statisticProvider = QuerySqlStatisticProvider.SILENT_CACHING_INSTANCE; } /** Creates a ConfigBuilder, initializing from an existing config. */ private ConfigBuilder(FrameworkConfig config) { convertletTable = config.getConvertletTable(); operatorTable = config.getOperatorTable(); programs = config.getPrograms(); context = config.getContext(); traitDefs = config.getTraitDefs(); parserConfig = config.getParserConfig(); sqlValidatorConfig = config.getSqlValidatorConfig(); sqlToRelConverterConfig = config.getSqlToRelConverterConfig(); defaultSchema = config.getDefaultSchema(); executor = config.getExecutor(); costFactory = config.getCostFactory(); typeSystem = config.getTypeSystem(); evolveLattice = config.isEvolveLattice(); statisticProvider = config.getStatisticProvider(); } public FrameworkConfig build() { return new StdFrameworkConfig(context, convertletTable, operatorTable, programs, traitDefs, parserConfig, sqlValidatorConfig, sqlToRelConverterConfig, defaultSchema, costFactory, typeSystem, executor, evolveLattice, statisticProvider, viewExpander); } public ConfigBuilder context(Context c) { this.context = Objects.requireNonNull(c, "c"); return this; } public ConfigBuilder executor(RexExecutor executor) { this.executor = Objects.requireNonNull(executor, "executor"); return this; } public ConfigBuilder convertletTable( SqlRexConvertletTable convertletTable) { this.convertletTable = Objects.requireNonNull(convertletTable, "convertletTable"); return this; } public ConfigBuilder operatorTable(SqlOperatorTable operatorTable) { this.operatorTable = Objects.requireNonNull(operatorTable, "operatorTable"); return this; } public ConfigBuilder traitDefs(@Nullable List traitDefs) { if (traitDefs == null) { this.traitDefs = null; } else { this.traitDefs = ImmutableList.copyOf(traitDefs); } return this; } public ConfigBuilder traitDefs(RelTraitDef... traitDefs) { this.traitDefs = ImmutableList.copyOf(traitDefs); return this; } public ConfigBuilder parserConfig(SqlParser.Config parserConfig) { this.parserConfig = Objects.requireNonNull(parserConfig, "parserConfig"); return this; } public ConfigBuilder sqlValidatorConfig(SqlValidator.Config sqlValidatorConfig) { this.sqlValidatorConfig = Objects.requireNonNull(sqlValidatorConfig, "sqlValidatorConfig"); return this; } public ConfigBuilder sqlToRelConverterConfig( SqlToRelConverter.Config sqlToRelConverterConfig) { this.sqlToRelConverterConfig = Objects.requireNonNull(sqlToRelConverterConfig, "sqlToRelConverterConfig"); return this; } public ConfigBuilder defaultSchema(SchemaPlus defaultSchema) { this.defaultSchema = defaultSchema; return this; } public ConfigBuilder costFactory(RelOptCostFactory costFactory) { this.costFactory = costFactory; return this; } public ConfigBuilder ruleSets(RuleSet... ruleSets) { return programs(Programs.listOf(ruleSets)); } public ConfigBuilder ruleSets(List ruleSets) { return programs(Programs.listOf(Objects.requireNonNull(ruleSets, "ruleSets"))); } public ConfigBuilder programs(List programs) { this.programs = ImmutableList.copyOf(programs); return this; } public ConfigBuilder programs(Program... programs) { this.programs = ImmutableList.copyOf(programs); return this; } public ConfigBuilder typeSystem(RelDataTypeSystem typeSystem) { this.typeSystem = Objects.requireNonNull(typeSystem, "typeSystem"); return this; } public ConfigBuilder evolveLattice(boolean evolveLattice) { this.evolveLattice = evolveLattice; return this; } public ConfigBuilder statisticProvider( SqlStatisticProvider statisticProvider) { this.statisticProvider = Objects.requireNonNull(statisticProvider, "statisticProvider"); return this; } public ConfigBuilder viewExpander(RelOptTable.ViewExpander viewExpander) { this.viewExpander = viewExpander; return this; } } /** * An implementation of {@link FrameworkConfig} that uses standard Calcite * classes to provide basic planner functionality. */ static class StdFrameworkConfig implements FrameworkConfig { private final Context context; private final SqlRexConvertletTable convertletTable; private final SqlOperatorTable operatorTable; private final ImmutableList programs; private final @Nullable ImmutableList traitDefs; private final SqlParser.Config parserConfig; private final SqlValidator.Config sqlValidatorConfig; private final SqlToRelConverter.Config sqlToRelConverterConfig; private final @Nullable SchemaPlus defaultSchema; private final @Nullable RelOptCostFactory costFactory; private final RelDataTypeSystem typeSystem; private final @Nullable RexExecutor executor; private final boolean evolveLattice; private final SqlStatisticProvider statisticProvider; private final RelOptTable.@Nullable ViewExpander viewExpander; StdFrameworkConfig(Context context, SqlRexConvertletTable convertletTable, SqlOperatorTable operatorTable, ImmutableList programs, @Nullable ImmutableList traitDefs, SqlParser.Config parserConfig, SqlValidator.Config sqlValidatorConfig, SqlToRelConverter.Config sqlToRelConverterConfig, @Nullable SchemaPlus defaultSchema, @Nullable RelOptCostFactory costFactory, RelDataTypeSystem typeSystem, @Nullable RexExecutor executor, boolean evolveLattice, SqlStatisticProvider statisticProvider, RelOptTable.@Nullable ViewExpander viewExpander) { this.context = context; this.convertletTable = convertletTable; this.operatorTable = operatorTable; this.programs = programs; this.traitDefs = traitDefs; this.parserConfig = parserConfig; this.sqlValidatorConfig = sqlValidatorConfig; this.sqlToRelConverterConfig = sqlToRelConverterConfig; this.defaultSchema = defaultSchema; this.costFactory = costFactory; this.typeSystem = typeSystem; this.executor = executor; this.evolveLattice = evolveLattice; this.statisticProvider = statisticProvider; this.viewExpander = viewExpander; } @Override public SqlParser.Config getParserConfig() { return parserConfig; } @Override public SqlValidator.Config getSqlValidatorConfig() { return sqlValidatorConfig; } @Override public SqlToRelConverter.Config getSqlToRelConverterConfig() { return sqlToRelConverterConfig; } @Override public @Nullable SchemaPlus getDefaultSchema() { return defaultSchema; } @Override public @Nullable RexExecutor getExecutor() { return executor; } @Override public ImmutableList getPrograms() { return programs; } @Override public @Nullable RelOptCostFactory getCostFactory() { return costFactory; } @Override public @Nullable ImmutableList getTraitDefs() { return traitDefs; } @Override public SqlRexConvertletTable getConvertletTable() { return convertletTable; } @Override public Context getContext() { return context; } @Override public SqlOperatorTable getOperatorTable() { return operatorTable; } @Override public RelDataTypeSystem getTypeSystem() { return typeSystem; } @Override public boolean isEvolveLattice() { return evolveLattice; } @Override public SqlStatisticProvider getStatisticProvider() { return statisticProvider; } @Override public RelOptTable.@Nullable ViewExpander getViewExpander() { return viewExpander; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy