org.apache.calcite.jdbc.CalciteSchema Maven / Gradle / Ivy
Show all versions of calcite-core Show documentation
/*
* 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.linq4j.function.Experimental;
import org.apache.calcite.linq4j.tree.Expression;
import org.apache.calcite.materialize.Lattice;
import org.apache.calcite.rel.type.RelProtoDataType;
import org.apache.calcite.schema.Function;
import org.apache.calcite.schema.Schema;
import org.apache.calcite.schema.SchemaPlus;
import org.apache.calcite.schema.SchemaVersion;
import org.apache.calcite.schema.Table;
import org.apache.calcite.schema.TableMacro;
import org.apache.calcite.schema.impl.MaterializedViewTable;
import org.apache.calcite.schema.impl.StarTable;
import org.apache.calcite.util.Compatible;
import org.apache.calcite.util.NameMap;
import org.apache.calcite.util.NameMultimap;
import org.apache.calcite.util.NameSet;
import org.apache.calcite.util.Pair;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSortedMap;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.Lists;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.NavigableMap;
import java.util.NavigableSet;
import java.util.Objects;
import java.util.Set;
/**
* Schema.
*
* Wrapper around user-defined schema used internally.
*/
public abstract class CalciteSchema {
private final CalciteSchema parent;
public final Schema schema;
public final String name;
/** Tables explicitly defined in this schema. Does not include tables in
* {@link #schema}. */
protected final NameMap tableMap;
protected final NameMultimap functionMap;
protected final NameMap typeMap;
protected final NameMap latticeMap;
protected final NameSet functionNames;
protected final NameMap nullaryFunctionMap;
protected final NameMap subSchemaMap;
private List extends List> path;
protected CalciteSchema(CalciteSchema parent, Schema schema,
String name, NameMap subSchemaMap,
NameMap tableMap, NameMap latticeMap, NameMap typeMap,
NameMultimap functionMap, NameSet functionNames,
NameMap nullaryFunctionMap,
List extends List> path) {
this.parent = parent;
this.schema = schema;
this.name = name;
if (tableMap == null) {
this.tableMap = new NameMap<>();
} else {
this.tableMap = Objects.requireNonNull(tableMap);
}
if (latticeMap == null) {
this.latticeMap = new NameMap<>();
} else {
this.latticeMap = Objects.requireNonNull(latticeMap);
}
if (subSchemaMap == null) {
this.subSchemaMap = new NameMap<>();
} else {
this.subSchemaMap = Objects.requireNonNull(subSchemaMap);
}
if (functionMap == null) {
this.functionMap = new NameMultimap<>();
this.functionNames = new NameSet();
this.nullaryFunctionMap = new NameMap<>();
} else {
// If you specify functionMap, you must also specify functionNames and
// nullaryFunctionMap.
this.functionMap = Objects.requireNonNull(functionMap);
this.functionNames = Objects.requireNonNull(functionNames);
this.nullaryFunctionMap = Objects.requireNonNull(nullaryFunctionMap);
}
if (typeMap == null) {
this.typeMap = new NameMap<>();
} else {
this.typeMap = Objects.requireNonNull(typeMap);
}
this.path = path;
}
/** Returns a sub-schema with a given name that is defined implicitly
* (that is, by the underlying {@link Schema} object, not explicitly
* by a call to {@link #add(String, Schema)}), or null. */
protected abstract CalciteSchema getImplicitSubSchema(String schemaName,
boolean caseSensitive);
/** Returns a table with a given name that is defined implicitly
* (that is, by the underlying {@link Schema} object, not explicitly
* by a call to {@link #add(String, Table)}), or null. */
protected abstract TableEntry getImplicitTable(String tableName,
boolean caseSensitive);
/** Returns a type with a given name that is defined implicitly
* (that is, by the underlying {@link Schema} object, not explicitly
* by a call to {@link #add(String, RelProtoDataType)}), or null. */
protected abstract TypeEntry getImplicitType(String name,
boolean caseSensitive);
/** Returns table function with a given name and zero arguments that is
* defined implicitly (that is, by the underlying {@link Schema} object,
* not explicitly by a call to {@link #add(String, Function)}), or null. */
protected abstract TableEntry getImplicitTableBasedOnNullaryFunction(String tableName,
boolean caseSensitive);
/** Adds implicit sub-schemas to a builder. */
protected abstract void addImplicitSubSchemaToBuilder(
ImmutableSortedMap.Builder builder);
/** Adds implicit tables to a builder. */
protected abstract void addImplicitTableToBuilder(
ImmutableSortedSet.Builder builder);
/** Adds implicit functions to a builder. */
protected abstract void addImplicitFunctionsToBuilder(
ImmutableList.Builder builder,
String name, boolean caseSensitive);
/** Adds implicit function names to a builder. */
protected abstract void addImplicitFuncNamesToBuilder(
ImmutableSortedSet.Builder builder);
/** Adds implicit type names to a builder. */
protected abstract void addImplicitTypeNamesToBuilder(
ImmutableSortedSet.Builder builder);
/** Adds implicit table functions to a builder. */
protected abstract void addImplicitTablesBasedOnNullaryFunctionsToBuilder(
ImmutableSortedMap.Builder builder);
/** Returns a snapshot representation of this CalciteSchema. */
protected abstract CalciteSchema snapshot(
CalciteSchema parent, SchemaVersion version);
protected abstract boolean isCacheEnabled();
public abstract void setCache(boolean cache);
/** Creates a TableEntryImpl with no SQLs. */
protected TableEntryImpl tableEntry(String name, Table table) {
return new TableEntryImpl(this, name, table, ImmutableList.of());
}
/** Creates a TableEntryImpl with no SQLs. */
protected TypeEntryImpl typeEntry(String name, RelProtoDataType relProtoDataType) {
return new TypeEntryImpl(this, name, relProtoDataType);
}
/** Defines a table within this schema. */
public TableEntry add(String tableName, Table table) {
return add(tableName, table, ImmutableList.of());
}
/** Defines a table within this schema. */
public TableEntry add(String tableName, Table table,
ImmutableList sqls) {
final TableEntryImpl entry =
new TableEntryImpl(this, tableName, table, sqls);
tableMap.put(tableName, entry);
return entry;
}
/** Defines a type within this schema. */
public TypeEntry add(String name, RelProtoDataType type) {
final TypeEntry entry =
new TypeEntryImpl(this, name, type);
typeMap.put(name, entry);
return entry;
}
private FunctionEntry add(String name, Function function) {
final FunctionEntryImpl entry =
new FunctionEntryImpl(this, name, function);
functionMap.put(name, entry);
functionNames.add(name);
if (function.getParameters().isEmpty()) {
nullaryFunctionMap.put(name, entry);
}
return entry;
}
private LatticeEntry add(String name, Lattice lattice) {
if (latticeMap.containsKey(name, false)) {
throw new RuntimeException("Duplicate lattice '" + name + "'");
}
final LatticeEntryImpl entry = new LatticeEntryImpl(this, name, lattice);
latticeMap.put(name, entry);
return entry;
}
public CalciteSchema root() {
for (CalciteSchema schema = this;;) {
if (schema.parent == null) {
return schema;
}
schema = schema.parent;
}
}
/** Returns whether this is a root schema. */
public boolean isRoot() {
return parent == null;
}
/** Returns the path of an object in this schema. */
public List path(String name) {
final List list = new ArrayList<>();
if (name != null) {
list.add(name);
}
for (CalciteSchema s = this; s != null; s = s.parent) {
if (s.parent != null || !s.name.equals("")) {
// Omit the root schema's name from the path if it's the empty string,
// which it usually is.
list.add(s.name);
}
}
return ImmutableList.copyOf(Lists.reverse(list));
}
public final CalciteSchema getSubSchema(String schemaName,
boolean caseSensitive) {
// Check explicit schemas.
//noinspection LoopStatementThatDoesntLoop
for (Map.Entry entry
: subSchemaMap.range(schemaName, caseSensitive).entrySet()) {
return entry.getValue();
}
return getImplicitSubSchema(schemaName, caseSensitive);
}
/** Adds a child schema of this schema. */
public abstract CalciteSchema add(String name, Schema schema);
/** Returns a table that materializes the given SQL statement. */
public final TableEntry getTableBySql(String sql) {
for (TableEntry tableEntry : tableMap.map().values()) {
if (tableEntry.sqls.contains(sql)) {
return tableEntry;
}
}
return null;
}
/** Returns a table with the given name. Does not look for views. */
public final TableEntry getTable(String tableName, boolean caseSensitive) {
// Check explicit tables.
//noinspection LoopStatementThatDoesntLoop
for (Map.Entry entry
: tableMap.range(tableName, caseSensitive).entrySet()) {
return entry.getValue();
}
return getImplicitTable(tableName, caseSensitive);
}
public String getName() {
return name;
}
public SchemaPlus plus() {
return new SchemaPlusImpl();
}
public static CalciteSchema from(SchemaPlus plus) {
return ((SchemaPlusImpl) plus).calciteSchema();
}
/** Returns the default path resolving functions from this schema.
*
* The path consists is a list of lists of strings.
* Each list of strings represents the path of a schema from the root schema.
* For example, [[], [foo], [foo, bar, baz]] represents three schemas: the
* root schema "/" (level 0), "/foo" (level 1) and "/foo/bar/baz" (level 3).
*
* @return Path of this schema; never null, may be empty
*/
public List extends List> getPath() {
if (path != null) {
return path;
}
// Return a path consisting of just this schema.
return ImmutableList.of(path(null));
}
/** Returns a collection of sub-schemas, both explicit (defined using
* {@link #add(String, org.apache.calcite.schema.Schema)}) and implicit
* (defined using {@link org.apache.calcite.schema.Schema#getSubSchemaNames()}
* and {@link Schema#getSubSchema(String)}). */
public final NavigableMap getSubSchemaMap() {
// Build a map of implicit sub-schemas first, then explicit sub-schemas.
// If there are implicit and explicit with the same name, explicit wins.
final ImmutableSortedMap.Builder builder =
new ImmutableSortedMap.Builder<>(NameSet.COMPARATOR);
builder.putAll(subSchemaMap.map());
addImplicitSubSchemaToBuilder(builder);
return Compatible.INSTANCE.navigableMap(builder.build());
}
/** Returns a collection of lattices.
*
* All are explicit (defined using {@link #add(String, Lattice)}). */
public NavigableMap getLatticeMap() {
return ImmutableSortedMap.copyOf(latticeMap.map());
}
/** Returns the set of all table names. Includes implicit and explicit tables
* and functions with zero parameters. */
public final NavigableSet getTableNames() {
final ImmutableSortedSet.Builder builder =
new ImmutableSortedSet.Builder<>(NameSet.COMPARATOR);
// Add explicit tables, case-sensitive.
builder.addAll(tableMap.map().keySet());
// Add implicit tables, case-sensitive.
addImplicitTableToBuilder(builder);
return Compatible.INSTANCE.navigableSet(builder.build());
}
/** Returns the set of all types names. */
public final NavigableSet getTypeNames() {
final ImmutableSortedSet.Builder builder =
new ImmutableSortedSet.Builder<>(NameSet.COMPARATOR);
// Add explicit types.
builder.addAll(typeMap.map().keySet());
// Add implicit types.
addImplicitTypeNamesToBuilder(builder);
return Compatible.INSTANCE.navigableSet(builder.build());
}
/** Returns a type, explicit and implicit, with a given
* name. Never null. */
public final TypeEntry getType(String name, boolean caseSensitive) {
for (Map.Entry entry
: typeMap.range(name, caseSensitive).entrySet()) {
return entry.getValue();
}
return getImplicitType(name, caseSensitive);
}
/** Returns a collection of all functions, explicit and implicit, with a given
* name. Never null. */
public final Collection getFunctions(String name, boolean caseSensitive) {
final ImmutableList.Builder builder = ImmutableList.builder();
// Add explicit functions.
for (FunctionEntry functionEntry
: Pair.right(functionMap.range(name, caseSensitive))) {
builder.add(functionEntry.getFunction());
}
// Add implicit functions.
addImplicitFunctionsToBuilder(builder, name, caseSensitive);
return builder.build();
}
/** Returns the list of function names in this schema, both implicit and
* explicit, never null. */
public final NavigableSet getFunctionNames() {
final ImmutableSortedSet.Builder builder =
new ImmutableSortedSet.Builder<>(NameSet.COMPARATOR);
// Add explicit functions, case-sensitive.
builder.addAll(functionMap.map().keySet());
// Add implicit functions, case-sensitive.
addImplicitFuncNamesToBuilder(builder);
return Compatible.INSTANCE.navigableSet(builder.build());
}
/** Returns tables derived from explicit and implicit functions
* that take zero parameters. */
public final NavigableMap getTablesBasedOnNullaryFunctions() {
ImmutableSortedMap.Builder builder =
new ImmutableSortedMap.Builder<>(NameSet.COMPARATOR);
for (Map.Entry entry
: nullaryFunctionMap.map().entrySet()) {
final Function function = entry.getValue().getFunction();
if (function instanceof TableMacro) {
assert function.getParameters().isEmpty();
final Table table = ((TableMacro) function).apply(ImmutableList.of());
builder.put(entry.getKey(), table);
}
}
// add tables derived from implicit functions
addImplicitTablesBasedOnNullaryFunctionsToBuilder(builder);
return Compatible.INSTANCE.navigableMap(builder.build());
}
/** Returns a tables derived from explicit and implicit functions
* that take zero parameters. */
public final TableEntry getTableBasedOnNullaryFunction(String tableName,
boolean caseSensitive) {
for (Map.Entry entry
: nullaryFunctionMap.range(tableName, caseSensitive).entrySet()) {
final Function function = entry.getValue().getFunction();
if (function instanceof TableMacro) {
assert function.getParameters().isEmpty();
final Table table = ((TableMacro) function).apply(ImmutableList.of());
return tableEntry(tableName, table);
}
}
return getImplicitTableBasedOnNullaryFunction(tableName, caseSensitive);
}
/** Creates a snapshot of this CalciteSchema as of the specified time. All
* explicit objects in this CalciteSchema will be copied into the snapshot
* CalciteSchema, while the contents of the snapshot of the underlying schema
* should not change as specified in {@link Schema#snapshot(SchemaVersion)}.
* Snapshots of explicit sub schemas will be created and copied recursively.
*
* Currently, to accommodate the requirement of creating tables on the fly
* for materializations, the snapshot will still use the same table map and
* lattice map as in the original CalciteSchema instead of making copies.
*
* @param version The current schema version
*
* @return the schema snapshot.
*/
public CalciteSchema createSnapshot(SchemaVersion version) {
Preconditions.checkArgument(this.isRoot(), "must be root schema");
return snapshot(null, version);
}
/** Returns a subset of a map whose keys match the given string
* case-insensitively. */
protected static NavigableMap find(NavigableMap map,
String s) {
assert map.comparator() == NameSet.COMPARATOR;
return map.subMap(s.toUpperCase(Locale.ROOT), true,
s.toLowerCase(Locale.ROOT), true);
}
/** Returns a subset of a set whose values match the given string
* case-insensitively. */
protected static Iterable find(NavigableSet set, String name) {
assert set.comparator() == NameSet.COMPARATOR;
return set.subSet(name.toUpperCase(Locale.ROOT), true,
name.toLowerCase(Locale.ROOT), true);
}
/** Creates a root schema.
*
* When addMetadataSchema
argument is true adds a "metadata"
* schema containing definitions of tables, columns etc. to root schema.
* By default, creates a {@link CachingCalciteSchema}.
*/
public static CalciteSchema createRootSchema(boolean addMetadataSchema) {
return createRootSchema(addMetadataSchema, true);
}
/** Creates a root schema.
*
* @param addMetadataSchema Whether to add a "metadata" schema containing
* definitions of tables, columns etc.
* @param cache If true create {@link CachingCalciteSchema};
* if false create {@link SimpleCalciteSchema}
*/
public static CalciteSchema createRootSchema(boolean addMetadataSchema,
boolean cache) {
return createRootSchema(addMetadataSchema, cache, "");
}
/** Creates a root schema.
*
* @param addMetadataSchema Whether to add a "metadata" schema containing
* definitions of tables, columns etc.
* @param cache If true create {@link CachingCalciteSchema};
* if false create {@link SimpleCalciteSchema}
* @param name Schema name
*/
public static CalciteSchema createRootSchema(boolean addMetadataSchema,
boolean cache, String name) {
CalciteSchema rootSchema;
final Schema schema = new CalciteConnectionImpl.RootSchema();
if (cache) {
rootSchema = new CachingCalciteSchema(null, schema, name);
} else {
rootSchema = new SimpleCalciteSchema(null, schema, name);
}
if (addMetadataSchema) {
rootSchema.add("metadata", MetadataSchema.INSTANCE);
}
return rootSchema;
}
@Experimental
public boolean removeSubSchema(String name) {
return subSchemaMap.remove(name) != null;
}
@Experimental
public boolean removeTable(String name) {
return tableMap.remove(name) != null;
}
@Experimental
public boolean removeFunction(String name) {
final FunctionEntry remove = nullaryFunctionMap.remove(name);
if (remove == null) {
return false;
}
functionMap.remove(name, remove);
return true;
}
@Experimental
public boolean removeType(String name) {
return typeMap.remove(name) != null;
}
/**
* Entry in a schema, such as a table or sub-schema.
*
*
Each object's name is a property of its membership in a schema;
* therefore in principle it could belong to several schemas, or
* even the same schema several times, with different names. In this
* respect, it is like an inode in a Unix file system.
*
* The members of a schema must have unique names.
*/
public abstract static class Entry {
public final CalciteSchema schema;
public final String name;
public Entry(CalciteSchema schema, String name) {
this.schema = Objects.requireNonNull(schema);
this.name = Objects.requireNonNull(name);
}
/** Returns this object's path. For example ["hr", "emps"]. */
public final List path() {
return schema.path(name);
}
}
/** Membership of a table in a schema. */
public abstract static class TableEntry extends Entry {
public final ImmutableList sqls;
public TableEntry(CalciteSchema schema, String name,
ImmutableList sqls) {
super(schema, name);
this.sqls = Objects.requireNonNull(sqls);
}
public abstract Table getTable();
}
/** Membership of a type in a schema. */
public abstract static class TypeEntry extends Entry {
public TypeEntry(CalciteSchema schema, String name) {
super(schema, name);
}
public abstract RelProtoDataType getType();
}
/** Membership of a function in a schema. */
public abstract static class FunctionEntry extends Entry {
public FunctionEntry(CalciteSchema schema, String name) {
super(schema, name);
}
public abstract Function getFunction();
/** Whether this represents a materialized view. (At a given point in time,
* it may or may not be materialized as a table.) */
public abstract boolean isMaterialization();
}
/** Membership of a lattice in a schema. */
public abstract static class LatticeEntry extends Entry {
public LatticeEntry(CalciteSchema schema, String name) {
super(schema, name);
}
public abstract Lattice getLattice();
public abstract TableEntry getStarTable();
}
/** Implementation of {@link SchemaPlus} based on a
* {@link org.apache.calcite.jdbc.CalciteSchema}. */
private class SchemaPlusImpl implements SchemaPlus {
CalciteSchema calciteSchema() {
return CalciteSchema.this;
}
public SchemaPlus getParentSchema() {
return parent == null ? null : parent.plus();
}
public String getName() {
return CalciteSchema.this.getName();
}
public boolean isMutable() {
return schema.isMutable();
}
public void setCacheEnabled(boolean cache) {
CalciteSchema.this.setCache(cache);
}
public boolean isCacheEnabled() {
return CalciteSchema.this.isCacheEnabled();
}
public Schema snapshot(SchemaVersion version) {
throw new UnsupportedOperationException();
}
public Expression getExpression(SchemaPlus parentSchema, String name) {
return schema.getExpression(parentSchema, name);
}
public Table getTable(String name) {
final TableEntry entry = CalciteSchema.this.getTable(name, true);
return entry == null ? null : entry.getTable();
}
public NavigableSet getTableNames() {
return CalciteSchema.this.getTableNames();
}
@Override public RelProtoDataType getType(String name) {
final TypeEntry entry = CalciteSchema.this.getType(name, true);
return entry == null ? null : entry.getType();
}
@Override public Set getTypeNames() {
return CalciteSchema.this.getTypeNames();
}
public Collection getFunctions(String name) {
return CalciteSchema.this.getFunctions(name, true);
}
public NavigableSet getFunctionNames() {
return CalciteSchema.this.getFunctionNames();
}
public SchemaPlus getSubSchema(String name) {
final CalciteSchema subSchema =
CalciteSchema.this.getSubSchema(name, true);
return subSchema == null ? null : subSchema.plus();
}
public Set getSubSchemaNames() {
return CalciteSchema.this.getSubSchemaMap().keySet();
}
public SchemaPlus add(String name, Schema schema) {
final CalciteSchema calciteSchema = CalciteSchema.this.add(name, schema);
return calciteSchema.plus();
}
public T unwrap(Class clazz) {
if (clazz.isInstance(this)) {
return clazz.cast(this);
}
if (clazz.isInstance(CalciteSchema.this)) {
return clazz.cast(CalciteSchema.this);
}
if (clazz.isInstance(CalciteSchema.this.schema)) {
return clazz.cast(CalciteSchema.this.schema);
}
throw new ClassCastException("not a " + clazz);
}
public void setPath(ImmutableList> path) {
CalciteSchema.this.path = path;
}
public void add(String name, Table table) {
CalciteSchema.this.add(name, table);
}
public void add(String name, Function function) {
CalciteSchema.this.add(name, function);
}
public void add(String name, RelProtoDataType type) {
CalciteSchema.this.add(name, type);
}
public void add(String name, Lattice lattice) {
CalciteSchema.this.add(name, lattice);
}
}
/**
* Implementation of {@link CalciteSchema.TableEntry}
* where all properties are held in fields.
*/
public static class TableEntryImpl extends TableEntry {
private final Table table;
/** Creates a TableEntryImpl. */
public TableEntryImpl(CalciteSchema schema, String name, Table table,
ImmutableList sqls) {
super(schema, name, sqls);
assert table != null;
this.table = Objects.requireNonNull(table);
}
public Table getTable() {
return table;
}
}
/**
* Implementation of {@link TypeEntry}
* where all properties are held in fields.
*/
public static class TypeEntryImpl extends TypeEntry {
private final RelProtoDataType protoDataType;
/** Creates a TypeEntryImpl. */
public TypeEntryImpl(CalciteSchema schema, String name, RelProtoDataType protoDataType) {
super(schema, name);
this.protoDataType = protoDataType;
}
public RelProtoDataType getType() {
return protoDataType;
}
}
/**
* Implementation of {@link FunctionEntry}
* where all properties are held in fields.
*/
public static class FunctionEntryImpl extends FunctionEntry {
private final Function function;
/** Creates a FunctionEntryImpl. */
public FunctionEntryImpl(CalciteSchema schema, String name,
Function function) {
super(schema, name);
this.function = function;
}
public Function getFunction() {
return function;
}
public boolean isMaterialization() {
return function
instanceof MaterializedViewTable.MaterializedViewTableMacro;
}
}
/**
* Implementation of {@link LatticeEntry}
* where all properties are held in fields.
*/
public static class LatticeEntryImpl extends LatticeEntry {
private final Lattice lattice;
private final CalciteSchema.TableEntry starTableEntry;
/** Creates a LatticeEntryImpl. */
public LatticeEntryImpl(CalciteSchema schema, String name,
Lattice lattice) {
super(schema, name);
this.lattice = lattice;
// Star table has same name as lattice and is in same schema.
final StarTable starTable = lattice.createStarTable();
starTableEntry = schema.add(name, starTable);
}
public Lattice getLattice() {
return lattice;
}
public TableEntry getStarTable() {
return starTableEntry;
}
}
}
// End CalciteSchema.java