com.hazelcast.org.apache.calcite.sql.validate.AliasNamespace 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.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeField;
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.fun.SqlStdOperatorTable;
import com.hazelcast.org.apache.calcite.sql.parser.SqlParserPos;
import com.hazelcast.org.apache.calcite.util.Util;
import java.util.ArrayList;
import java.util.List;
import static com.hazelcast.org.apache.calcite.util.Static.RESOURCE;
/**
* Namespace for an AS t(c1, c2, ...)
clause.
*
* A namespace is necessary only if there is a column list, in order to
* re-map column names; a relation AS t
clause just uses the same
* namespace as relation
.
*/
public class AliasNamespace extends AbstractNamespace {
//~ Instance fields --------------------------------------------------------
protected final SqlCall call;
//~ Constructors -----------------------------------------------------------
/**
* Creates an AliasNamespace.
*
* @param validator Validator
* @param call Call to AS operator
* @param enclosingNode Enclosing node
*/
protected AliasNamespace(
SqlValidatorImpl validator,
SqlCall call,
SqlNode enclosingNode) {
super(validator, enclosingNode);
this.call = call;
assert call.getOperator() == SqlStdOperatorTable.AS;
}
//~ Methods ----------------------------------------------------------------
protected RelDataType validateImpl(RelDataType targetRowType) {
final List nameList = new ArrayList<>();
final List operands = call.getOperandList();
final SqlValidatorNamespace childNs =
validator.getNamespace(operands.get(0));
final RelDataType rowType = childNs.getRowTypeSansSystemColumns();
final List columnNames = Util.skip(operands, 2);
for (final SqlNode operand : columnNames) {
String name = ((SqlIdentifier) operand).getSimple();
if (nameList.contains(name)) {
throw validator.newValidationError(operand,
RESOURCE.aliasListDuplicate(name));
}
nameList.add(name);
}
if (nameList.size() != rowType.getFieldCount()) {
// Position error over all column names
final SqlNode node = operands.size() == 3
? operands.get(2)
: new SqlNodeList(columnNames, SqlParserPos.sum(columnNames));
throw validator.newValidationError(node,
RESOURCE.aliasListDegree(rowType.getFieldCount(), getString(rowType),
nameList.size()));
}
final List typeList = new ArrayList<>();
for (RelDataTypeField field : rowType.getFieldList()) {
typeList.add(field.getType());
}
return validator.getTypeFactory().createStructType(
typeList,
nameList);
}
private String getString(RelDataType rowType) {
StringBuilder buf = new StringBuilder();
buf.append("(");
for (RelDataTypeField field : rowType.getFieldList()) {
if (field.getIndex() > 0) {
buf.append(", ");
}
buf.append("'");
buf.append(field.getName());
buf.append("'");
}
buf.append(")");
return buf.toString();
}
public SqlNode getNode() {
return call;
}
public String translate(String name) {
final RelDataType underlyingRowType =
validator.getValidatedNodeType(call.operand(0));
int i = 0;
for (RelDataTypeField field : rowType.getFieldList()) {
if (field.getName().equals(name)) {
return underlyingRowType.getFieldList().get(i).getName();
}
++i;
}
throw new AssertionError("unknown field '" + name
+ "' in rowtype " + underlyingRowType);
}
}