com.hazelcast.org.apache.calcite.sql.validate.DelegatingScope Maven / Gradle / Ivy
/*
* 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.sql.validate;
import com.hazelcast.org.apache.calcite.prepare.Prepare;
import com.hazelcast.org.apache.calcite.rel.type.DynamicRecordType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeField;
import com.hazelcast.org.apache.calcite.rel.type.StructKind;
import com.hazelcast.org.apache.calcite.schema.CustomColumnResolvingTable;
import com.hazelcast.org.apache.calcite.schema.Table;
import com.hazelcast.org.apache.calcite.sql.SqlCall;
import com.hazelcast.org.apache.calcite.sql.SqlIdentifier;
import com.hazelcast.org.apache.calcite.sql.SqlNode;
import com.hazelcast.org.apache.calcite.sql.SqlNodeList;
import com.hazelcast.org.apache.calcite.sql.SqlSelect;
import com.hazelcast.org.apache.calcite.sql.SqlWindow;
import com.hazelcast.org.apache.calcite.sql.parser.SqlParserPos;
import com.hazelcast.org.apache.calcite.util.Pair;
import com.hazelcast.org.apache.calcite.util.Util;
import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import static com.hazelcast.org.apache.calcite.util.Static.RESOURCE;
import static java.util.Objects.requireNonNull;
/**
* A scope which delegates all requests to its parent scope. Use this as a base
* class for defining nested scopes.
*/
public abstract class DelegatingScope implements SqlValidatorScope {
//~ Instance fields --------------------------------------------------------
/**
* Parent scope. This is where to look next to resolve an identifier; it is
* not always the parent object in the parse tree.
*
* This is never null: at the top of the tree, it is an
* {@link EmptyScope}.
*/
protected final SqlValidatorScope parent;
protected final SqlValidatorImpl validator;
//~ Constructors -----------------------------------------------------------
/**
* Creates a DelegatingScope
.
*
* @param parent Parent scope
*/
DelegatingScope(SqlValidatorScope parent) {
super();
assert parent != null;
this.validator = (SqlValidatorImpl) parent.getValidator();
this.parent = parent;
}
//~ Methods ----------------------------------------------------------------
@Override public void addChild(SqlValidatorNamespace ns, String alias,
boolean nullable) {
// By default, you cannot add to a scope. Derived classes can
// override.
throw new UnsupportedOperationException();
}
@Override public void resolve(List names, SqlNameMatcher nameMatcher,
boolean deep, Resolved resolved) {
parent.resolve(names, nameMatcher, deep, resolved);
}
/** If a record type allows implicit references to fields, recursively looks
* into the fields. Otherwise returns immediately. */
void resolveInNamespace(SqlValidatorNamespace ns, boolean nullable,
List names, SqlNameMatcher nameMatcher, Path path,
Resolved resolved) {
if (names.isEmpty()) {
resolved.found(ns, nullable, this, path, null);
return;
}
final RelDataType rowType = ns.getRowType();
if (rowType.isStruct()) {
SqlValidatorTable validatorTable = ns.getTable();
if (validatorTable instanceof Prepare.PreparingTable) {
Table t = ((Prepare.PreparingTable) validatorTable).unwrap(Table.class);
if (t instanceof CustomColumnResolvingTable) {
final List>> entries =
((CustomColumnResolvingTable) t).resolveColumn(
rowType, validator.getTypeFactory(), names);
for (Pair> entry : entries) {
final RelDataTypeField field = entry.getKey();
final List remainder = entry.getValue();
final SqlValidatorNamespace ns2 =
new FieldNamespace(validator, field.getType());
final Step path2 = path.plus(rowType, field.getIndex(),
field.getName(), StructKind.FULLY_QUALIFIED);
resolveInNamespace(ns2, nullable, remainder, nameMatcher, path2,
resolved);
}
return;
}
}
final String name = names.get(0);
final RelDataTypeField field0 = nameMatcher.field(rowType, name);
if (field0 != null) {
final SqlValidatorNamespace ns2 = requireNonNull(
ns.lookupChild(field0.getName()),
() -> "field " + field0.getName() + " is not found in " + ns);
final Step path2 = path.plus(rowType, field0.getIndex(),
field0.getName(), StructKind.FULLY_QUALIFIED);
resolveInNamespace(ns2, nullable, names.subList(1, names.size()),
nameMatcher, path2, resolved);
} else {
for (RelDataTypeField field : rowType.getFieldList()) {
switch (field.getType().getStructKind()) {
case PEEK_FIELDS:
case PEEK_FIELDS_DEFAULT:
case PEEK_FIELDS_NO_EXPAND:
final Step path2 = path.plus(rowType, field.getIndex(),
field.getName(), field.getType().getStructKind());
final SqlValidatorNamespace ns2 = requireNonNull(
ns.lookupChild(field.getName()),
() -> "field " + field.getName() + " is not found in " + ns);
resolveInNamespace(ns2, nullable, names, nameMatcher, path2,
resolved);
break;
default:
break;
}
}
}
}
}
protected void addColumnNames(
SqlValidatorNamespace ns,
List colNames) {
final RelDataType rowType;
try {
rowType = ns.getRowType();
} catch (Error e) {
// namespace is not good - bail out.
return;
}
for (RelDataTypeField field : rowType.getFieldList()) {
colNames.add(
new SqlMonikerImpl(
field.getName(),
SqlMonikerType.COLUMN));
}
}
@Override public void findAllColumnNames(List result) {
parent.findAllColumnNames(result);
}
@Override public void findAliases(Collection result) {
parent.findAliases(result);
}
@SuppressWarnings("deprecation")
@Override public Pair findQualifyingTableName(
String columnName, SqlNode ctx) {
//noinspection deprecation
return parent.findQualifyingTableName(columnName, ctx);
}
@Override public Map findQualifyingTableNames(String columnName,
SqlNode ctx, SqlNameMatcher nameMatcher) {
return parent.findQualifyingTableNames(columnName, ctx, nameMatcher);
}
@Override public @Nullable RelDataType resolveColumn(String name, SqlNode ctx) {
return parent.resolveColumn(name, ctx);
}
@Override public RelDataType nullifyType(SqlNode node, RelDataType type) {
return parent.nullifyType(node, type);
}
@SuppressWarnings("deprecation")
@Override public @Nullable SqlValidatorNamespace getTableNamespace(List names) {
return parent.getTableNamespace(names);
}
@Override public void resolveTable(List names, SqlNameMatcher nameMatcher,
Path path, Resolved resolved) {
parent.resolveTable(names, nameMatcher, path, resolved);
}
@Override public SqlValidatorScope getOperandScope(SqlCall call) {
if (call instanceof SqlSelect) {
return validator.getSelectScope((SqlSelect) call);
}
return this;
}
@Override public SqlValidator getValidator() {
return validator;
}
/**
* Converts an identifier into a fully-qualified identifier. For example,
* the "empno" in "select empno from emp natural join dept" becomes
* "emp.empno".
*
* If the identifier cannot be resolved, throws. Never returns null.
*/
@Override public SqlQualified fullyQualify(SqlIdentifier identifier) {
if (identifier.isStar()) {
return SqlQualified.create(this, 1, null, identifier);
}
final SqlIdentifier previous = identifier;
final SqlNameMatcher nameMatcher = validator.catalogReader.nameMatcher();
String columnName;
final String tableName;
final SqlValidatorNamespace namespace;
switch (identifier.names.size()) {
case 1: {
columnName = identifier.names.get(0);
final Map map =
findQualifyingTableNames(columnName, identifier, nameMatcher);
switch (map.size()) {
case 0:
if (nameMatcher.isCaseSensitive()) {
final SqlNameMatcher liberalMatcher = SqlNameMatchers.liberal();
final Map map2 =
findQualifyingTableNames(columnName, identifier, liberalMatcher);
if (!map2.isEmpty()) {
final List list = new ArrayList<>();
for (ScopeChild entry : map2.values()) {
final RelDataTypeField field =
liberalMatcher.field(entry.namespace.getRowType(),
columnName);
if (field == null) {
continue;
}
list.add(field.getName());
}
Collections.sort(list);
throw validator.newValidationError(identifier,
RESOURCE.columnNotFoundDidYouMean(columnName,
Util.sepList(list, "', '")));
}
}
throw validator.newValidationError(identifier,
RESOURCE.columnNotFound(columnName));
case 1:
tableName = map.keySet().iterator().next();
namespace = map.get(tableName).namespace;
break;
default:
throw validator.newValidationError(identifier,
RESOURCE.columnAmbiguous(columnName));
}
final ResolvedImpl resolved = new ResolvedImpl();
resolveInNamespace(namespace, false, identifier.names, nameMatcher,
Path.EMPTY, resolved);
final RelDataTypeField field =
nameMatcher.field(namespace.getRowType(), columnName);
if (field != null) {
if (hasAmbiguousField(namespace.getRowType(), field,
columnName, nameMatcher)) {
throw validator.newValidationError(identifier,
RESOURCE.columnAmbiguous(columnName));
}
columnName = field.getName(); // use resolved field name
}
// todo: do implicit collation here
final SqlParserPos pos = identifier.getParserPosition();
identifier =
new SqlIdentifier(ImmutableList.of(tableName, columnName), null,
pos, ImmutableList.of(SqlParserPos.ZERO, pos));
}
// fall through
default: {
SqlValidatorNamespace fromNs = null;
Path fromPath = null;
RelDataType fromRowType = null;
final ResolvedImpl resolved = new ResolvedImpl();
int size = identifier.names.size();
int i = size - 1;
for (; i > 0; i--) {
final SqlIdentifier prefix = identifier.getComponent(0, i);
resolved.clear();
resolve(prefix.names, nameMatcher, false, resolved);
if (resolved.count() == 1) {
final Resolve resolve = resolved.only();
fromNs = resolve.namespace;
fromPath = resolve.path;
fromRowType = resolve.rowType();
break;
}
// Look for a table alias that is the wrong case.
if (nameMatcher.isCaseSensitive()) {
final SqlNameMatcher liberalMatcher = SqlNameMatchers.liberal();
resolved.clear();
resolve(prefix.names, liberalMatcher, false, resolved);
if (resolved.count() == 1) {
final Step lastStep = Util.last(resolved.only().path.steps());
throw validator.newValidationError(prefix,
RESOURCE.tableNameNotFoundDidYouMean(prefix.toString(),
lastStep.name));
}
}
}
if (fromNs == null || fromNs instanceof SchemaNamespace) {
// Look for a column not qualified by a table alias.
columnName = identifier.names.get(0);
final Map map =
findQualifyingTableNames(columnName, identifier, nameMatcher);
switch (map.size()) {
default:
final SqlIdentifier prefix1 = identifier.skipLast(1);
throw validator.newValidationError(prefix1,
RESOURCE.tableNameNotFound(prefix1.toString()));
case 1: {
final Map.Entry entry =
map.entrySet().iterator().next();
final String tableName2 = map.keySet().iterator().next();
fromNs = entry.getValue().namespace;
fromPath = Path.EMPTY;
// Adding table name is for RecordType column with StructKind.PEEK_FIELDS or
// StructKind.PEEK_FIELDS only. Access to a field in a RecordType column of
// other StructKind should always be qualified with table name.
final RelDataTypeField field =
nameMatcher.field(fromNs.getRowType(), columnName);
if (field != null) {
switch (field.getType().getStructKind()) {
case PEEK_FIELDS:
case PEEK_FIELDS_DEFAULT:
case PEEK_FIELDS_NO_EXPAND:
columnName = field.getName(); // use resolved field name
resolve(ImmutableList.of(tableName2), nameMatcher, false,
resolved);
if (resolved.count() == 1) {
final Resolve resolve = resolved.only();
fromNs = resolve.namespace;
fromPath = resolve.path;
fromRowType = resolve.rowType();
identifier = identifier
.setName(0, columnName)
.add(0, tableName2, SqlParserPos.ZERO);
++i;
++size;
}
break;
default:
// Throw an error if the table was not found.
// If one or more of the child namespaces allows peeking
// (e.g. if they are Phoenix column families) then we relax the SQL
// standard requirement that record fields are qualified by table alias.
final SqlIdentifier prefix = identifier.skipLast(1);
throw validator.newValidationError(prefix,
RESOURCE.tableNameNotFound(prefix.toString()));
}
}
}
}
}
// If a table alias is part of the identifier, make sure that the table
// alias uses the same case as it was defined. For example, in
//
// SELECT e.empno FROM Emp as E
//
// change "e.empno" to "E.empno".
if (fromNs.getEnclosingNode() != null
&& !(this instanceof MatchRecognizeScope)) {
String alias =
SqlValidatorUtil.getAlias(fromNs.getEnclosingNode(), -1);
if (alias != null
&& i > 0
&& !alias.equals(identifier.names.get(i - 1))) {
identifier = identifier.setName(i - 1, alias);
}
}
if (requireNonNull(fromPath, "fromPath").stepCount() > 1) {
assert fromRowType != null;
for (Step p : fromPath.steps()) {
fromRowType = fromRowType.getFieldList().get(p.i).getType();
}
++i;
}
final SqlIdentifier suffix = identifier.getComponent(i, size);
resolved.clear();
resolveInNamespace(fromNs, false, suffix.names, nameMatcher, Path.EMPTY,
resolved);
final Path path;
switch (resolved.count()) {
case 0:
// Maybe the last component was correct, just wrong case
if (nameMatcher.isCaseSensitive()) {
SqlNameMatcher liberalMatcher = SqlNameMatchers.liberal();
resolved.clear();
resolveInNamespace(fromNs, false, suffix.names, liberalMatcher,
Path.EMPTY, resolved);
if (resolved.count() > 0) {
int k = size - 1;
final SqlIdentifier prefix = identifier.getComponent(0, i);
final SqlIdentifier suffix3 = identifier.getComponent(i, k + 1);
final Step step = Util.last(resolved.resolves.get(0).path.steps());
throw validator.newValidationError(suffix3,
RESOURCE.columnNotFoundInTableDidYouMean(suffix3.toString(),
prefix.toString(), step.name));
}
}
// Find the shortest suffix that also fails. Suppose we cannot resolve
// "a.b.c"; we find we cannot resolve "a.b" but can resolve "a". So,
// the error will be "Column 'a.b' not found".
int k = size - 1;
for (; k > i; --k) {
SqlIdentifier suffix2 = identifier.getComponent(i, k);
resolved.clear();
resolveInNamespace(fromNs, false, suffix2.names, nameMatcher,
Path.EMPTY, resolved);
if (resolved.count() > 0) {
break;
}
}
final SqlIdentifier prefix = identifier.getComponent(0, i);
final SqlIdentifier suffix3 = identifier.getComponent(i, k + 1);
throw validator.newValidationError(suffix3,
RESOURCE.columnNotFoundInTable(suffix3.toString(), prefix.toString()));
case 1:
path = resolved.only().path;
break;
default:
final Comparator c =
new Comparator() {
@Override public int compare(Resolve o1, Resolve o2) {
// Name resolution that uses fewer implicit steps wins.
int c = Integer.compare(worstKind(o1.path), worstKind(o2.path));
if (c != 0) {
return c;
}
// Shorter path wins
return Integer.compare(o1.path.stepCount(), o2.path.stepCount());
}
private int worstKind(Path path) {
int kind = -1;
for (Step step : path.steps()) {
kind = Math.max(kind, step.kind.ordinal());
}
return kind;
}
};
resolved.resolves.sort(c);
if (c.compare(resolved.resolves.get(0), resolved.resolves.get(1)) == 0) {
throw validator.newValidationError(suffix,
RESOURCE.columnAmbiguous(suffix.toString()));
}
path = resolved.resolves.get(0).path;
}
// Normalize case to match definition, make elided fields explicit,
// and check that references to dynamic stars ("**") are unambiguous.
int k = i;
for (Step step : path.steps()) {
final String name = identifier.names.get(k);
if (step.i < 0) {
throw validator.newValidationError(
identifier, RESOURCE.columnNotFound(name));
}
final RelDataTypeField field0 =
requireNonNull(
step.rowType,
() -> "rowType of step " + step.name
).getFieldList().get(step.i);
final String fieldName = field0.getName();
switch (step.kind) {
case PEEK_FIELDS:
case PEEK_FIELDS_DEFAULT:
case PEEK_FIELDS_NO_EXPAND:
identifier = identifier.add(k, fieldName, SqlParserPos.ZERO);
break;
default:
if (!fieldName.equals(name)) {
identifier = identifier.setName(k, fieldName);
}
if (hasAmbiguousField(step.rowType, field0, name, nameMatcher)) {
throw validator.newValidationError(identifier,
RESOURCE.columnAmbiguous(name));
}
}
++k;
}
// Multiple name components may have been resolved as one step by
// CustomResolvingTable.
if (identifier.names.size() > k) {
identifier = identifier.getComponent(0, k);
}
if (i > 1) {
// Simplify overqualified identifiers.
// For example, schema.emp.deptno becomes emp.deptno.
//
// It is safe to convert schema.emp or database.schema.emp to emp
// because it would not have resolved if the FROM item had an alias. The
// following query is invalid:
// SELECT schema.emp.deptno FROM schema.emp AS e
identifier = identifier.getComponent(i - 1, identifier.names.size());
}
if (!previous.equals(identifier)) {
validator.setOriginal(identifier, previous);
}
return SqlQualified.create(this, i, fromNs, identifier);
}
}
}
@Override public void validateExpr(SqlNode expr) {
// Do not delegate to parent. An expression valid in this scope may not
// be valid in the parent scope.
}
@Override public @Nullable SqlWindow lookupWindow(String name) {
return parent.lookupWindow(name);
}
@Override public SqlMonotonicity getMonotonicity(SqlNode expr) {
return parent.getMonotonicity(expr);
}
@Override public @Nullable SqlNodeList getOrderList() {
return parent.getOrderList();
}
/** Returns whether {@code rowType} contains more than one star column or
* fields with the same name, which implies ambiguous column. */
private static boolean hasAmbiguousField(RelDataType rowType,
RelDataTypeField field, String columnName, SqlNameMatcher nameMatcher) {
if (field.isDynamicStar()
&& !DynamicRecordType.isDynamicStarColName(columnName)) {
int count = 0;
for (RelDataTypeField possibleStar : rowType.getFieldList()) {
if (possibleStar.isDynamicStar()) {
if (++count > 1) {
return true;
}
}
}
} else { // check if there are fields with the same name
int count = 0;
for (RelDataTypeField f : rowType.getFieldList()) {
if (Util.matches(nameMatcher.isCaseSensitive(), f.getName(), columnName)) {
count++;
}
}
if (count > 1) {
return true;
}
}
return false;
}
/**
* Returns the parent scope of this DelegatingScope
.
*/
public SqlValidatorScope getParent() {
return parent;
}
}