blob: a2f6fbbcaa6f019fe634ffcef79b2680ea33fc7a [file] [log] [blame]
/*
* 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.hadoop.hive.ql.parse;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.hive.ql.metadata.Table;
import org.apache.hadoop.hive.ql.plan.CreateTableDesc;
import org.apache.hadoop.hive.ql.plan.CreateViewDesc;
/**
* Implementation of the query block.
*
**/
public class QB {
private static final Logger LOG = LoggerFactory.getLogger("hive.ql.parse.QB");
private final int numJoins = 0;
private final int numGbys = 0;
private int numSels = 0;
private int numSelDi = 0;
private HashMap<String, String> aliasToTabs;
private HashMap<String, QBExpr> aliasToSubq;
private HashMap<String, Table> viewAliasToViewSchema;
private HashMap<String, Map<String, String>> aliasToProps;
private List<String> aliases;
private QBParseInfo qbp;
private QBMetaData qbm;
private QBJoinTree qbjoin;
private String id;
private boolean isQuery;
private boolean isAnalyzeRewrite;
private CreateTableDesc tblDesc = null; // table descriptor of the final
private CreateTableDesc directoryDesc = null ;
private boolean insideView;
private Set<String> aliasInsideView;
// If this is a materialized view, this stores the view descriptor
private CreateViewDesc viewDesc;
// used by PTFs
/*
* This map maintains the PTFInvocationSpec for each PTF chain invocation in this QB.
*/
private HashMap<ASTNode, PTFInvocationSpec> ptfNodeToSpec;
/*
* the WindowingSpec used for windowing clauses in this QB.
*/
private HashMap<String, WindowingSpec> destToWindowingSpec;
/*
* If this QB represents a SubQuery predicate then this will point to the SubQuery object.
*/
private QBSubQuery subQueryPredicateDef;
/*
* used to give a unique name to each SubQuery QB Currently there can be at
* most 2 SubQueries in a Query: 1 in the Where clause, and 1 in the Having
* clause.
*/
private int numSubQueryPredicates;
/*
* for now a top level QB can have 1 where clause SQ predicate.
*/
private QBSubQuery whereClauseSubQueryPredicate;
/*
* for now a top level QB can have 1 where clause SQ predicate.
*/
private QBSubQuery havingClauseSubQueryPredicate;
// results
public void print(String msg) {
LOG.info(msg + "alias=" + qbp.getAlias());
for (String alias : getSubqAliases()) {
QBExpr qbexpr = getSubqForAlias(alias);
LOG.info(msg + "start subquery " + alias);
qbexpr.print(msg + " ");
LOG.info(msg + "end subquery " + alias);
}
}
public QB() {
}
public QB(String outer_id, String alias, boolean isSubQ) {
// Must be deterministic order maps - see HIVE-8707
aliasToTabs = new LinkedHashMap<String, String>();
aliasToSubq = new LinkedHashMap<String, QBExpr>();
viewAliasToViewSchema = new LinkedHashMap<String, Table>();
aliasToProps = new LinkedHashMap<String, Map<String, String>>();
aliases = new ArrayList<String>();
if (alias != null) {
alias = alias.toLowerCase();
}
qbp = new QBParseInfo(alias, isSubQ);
qbm = new QBMetaData();
// Must be deterministic order maps - see HIVE-8707
ptfNodeToSpec = new LinkedHashMap<ASTNode, PTFInvocationSpec>();
destToWindowingSpec = new LinkedHashMap<String, WindowingSpec>();
id = getAppendedAliasFromId(outer_id, alias);
aliasInsideView = new HashSet<>();
}
// For sub-queries, the id. and alias should be appended since same aliases can be re-used
// within different sub-queries.
// For a query like:
// select ...
// (select * from T1 a where ...) subq1
// join
// (select * from T2 a where ...) subq2
// ..
// the alias is modified to subq1:a and subq2:a from a, to identify the right sub-query.
public static String getAppendedAliasFromId(String outer_id, String alias) {
return (outer_id == null ? alias : outer_id + ":" + alias);
}
public String getAlias() {
return qbp.getAlias();
}
public QBParseInfo getParseInfo() {
return qbp;
}
public QBMetaData getMetaData() {
return qbm;
}
public void setQBParseInfo(QBParseInfo qbp) {
this.qbp = qbp;
}
public void countSelDi() {
numSelDi++;
}
public void countSel() {
numSels++;
}
public boolean exists(String alias) {
alias = alias.toLowerCase();
if (aliasToTabs.get(alias) != null || aliasToSubq.get(alias) != null) {
return true;
}
return false;
}
public void setTabAlias(String alias, String tabName) {
aliasToTabs.put(alias.toLowerCase(), tabName);
}
public void setSubqAlias(String alias, QBExpr qbexpr) {
aliasToSubq.put(alias.toLowerCase(), qbexpr);
}
public void setTabProps(String alias, Map<String, String> props) {
aliasToProps.put(alias.toLowerCase(), props);
}
public void addAlias(String alias) {
if (!aliases.contains(alias.toLowerCase())) {
aliases.add(alias.toLowerCase());
}
}
public String getId() {
return id;
}
public int getNumGbys() {
return numGbys;
}
public int getNumSelDi() {
return numSelDi;
}
public int getNumSels() {
return numSels;
}
public int getNumJoins() {
return numJoins;
}
public Set<String> getSubqAliases() {
return aliasToSubq.keySet();
}
public Set<String> getTabAliases() {
return aliasToTabs.keySet();
}
public List<String> getAliases() {
return aliases;
}
public QBExpr getSubqForAlias(String alias) {
return aliasToSubq.get(alias.toLowerCase());
}
public String getTabNameForAlias(String alias) {
return aliasToTabs.get(alias.toLowerCase());
}
public Map<String, String> getTabPropsForAlias(String alias) {
return aliasToProps.get(alias.toLowerCase());
}
public void rewriteViewToSubq(String alias, String viewName, QBExpr qbexpr, Table tab) {
alias = alias.toLowerCase();
String tableName = aliasToTabs.remove(alias);
assert (viewName.equals(tableName));
aliasToSubq.put(alias, qbexpr);
if (tab != null) {
viewAliasToViewSchema.put(alias, tab);
}
}
public void rewriteCTEToSubq(String alias, String cteName, QBExpr qbexpr) {
rewriteViewToSubq(alias, cteName, qbexpr, null);
}
public QBJoinTree getQbJoinTree() {
return qbjoin;
}
public void setQbJoinTree(QBJoinTree qbjoin) {
this.qbjoin = qbjoin;
}
public void setIsQuery(boolean isQuery) {
this.isQuery = isQuery;
}
/**
* Set to true in SemanticAnalyzer.getMetadataForDestFile,
* if destination is a file and query is not CTAS
* @return
*/
public boolean getIsQuery() {
return isQuery;
}
// to decide whether to rewrite RR of subquery
public boolean isTopLevelSelectStarQuery() {
return !isCTAS() && qbp.isTopLevelSimpleSelectStarQuery();
}
// to find target for fetch task conversion optimizer (not allows subqueries)
public boolean isSimpleSelectQuery() {
if (!qbp.isSimpleSelectQuery() || isCTAS() || qbp.isAnalyzeCommand()) {
return false;
}
for (QBExpr qbexpr : aliasToSubq.values()) {
if (!qbexpr.isSimpleSelectQuery()) {
return false;
}
}
return true;
}
public boolean hasTableSample(String alias) {
return qbp.getTabSample(alias) != null;
}
public CreateTableDesc getTableDesc() {
return tblDesc;
}
public void setTableDesc(CreateTableDesc desc) {
tblDesc = desc;
}
public CreateTableDesc getDirectoryDesc() {
return directoryDesc;
}
public void setDirectoryDesc(CreateTableDesc directoryDesc) {
this.directoryDesc = directoryDesc;
}
/**
* Whether this QB is for a CREATE-TABLE-AS-SELECT.
*/
public boolean isCTAS() {
return tblDesc != null;
}
/**
* Retrieve skewed column name for a table.
* @param alias table alias
* @return
*/
public List<String> getSkewedColumnNames(String alias) {
List<String> skewedColNames = null;
if (null != qbm &&
null != qbm.getAliasToTable() &&
qbm.getAliasToTable().size() > 0) {
Table tbl = getMetaData().getTableForAlias(alias);
skewedColNames = tbl.getSkewedColNames();
}
return skewedColNames;
}
public boolean isAnalyzeRewrite() {
return isAnalyzeRewrite;
}
public void setAnalyzeRewrite(boolean isAnalyzeRewrite) {
this.isAnalyzeRewrite = isAnalyzeRewrite;
}
public PTFInvocationSpec getPTFInvocationSpec(ASTNode node) {
return ptfNodeToSpec == null ? null : ptfNodeToSpec.get(node);
}
public void addPTFNodeToSpec(ASTNode node, PTFInvocationSpec spec) {
// Must be deterministic order map - see HIVE-8707
ptfNodeToSpec = ptfNodeToSpec == null ? new LinkedHashMap<ASTNode, PTFInvocationSpec>() : ptfNodeToSpec;
ptfNodeToSpec.put(node, spec);
}
public HashMap<ASTNode, PTFInvocationSpec> getPTFNodeToSpec() {
return ptfNodeToSpec;
}
public WindowingSpec getWindowingSpec(String dest) {
return destToWindowingSpec.get(dest);
}
public void addDestToWindowingSpec(String dest, WindowingSpec windowingSpec) {
destToWindowingSpec.put(dest, windowingSpec);
}
public boolean hasWindowingSpec(String dest) {
return destToWindowingSpec.get(dest) != null;
}
public HashMap<String, WindowingSpec> getAllWindowingSpecs() {
return destToWindowingSpec;
}
protected void setSubQueryDef(QBSubQuery subQueryPredicateDef) {
this.subQueryPredicateDef = subQueryPredicateDef;
}
protected QBSubQuery getSubQueryPredicateDef() {
return subQueryPredicateDef;
}
protected int getNumSubQueryPredicates() {
return numSubQueryPredicates;
}
protected int incrNumSubQueryPredicates() {
return ++numSubQueryPredicates;
}
void setWhereClauseSubQueryPredicate(QBSubQuery sq) {
whereClauseSubQueryPredicate = sq;
}
public QBSubQuery getWhereClauseSubQueryPredicate() {
return whereClauseSubQueryPredicate;
}
void setHavingClauseSubQueryPredicate(QBSubQuery sq) {
havingClauseSubQueryPredicate = sq;
}
public QBSubQuery getHavingClauseSubQueryPredicate() {
return havingClauseSubQueryPredicate;
}
public CreateViewDesc getViewDesc() {
return viewDesc;
}
public void setViewDesc(CreateViewDesc viewDesc) {
this.viewDesc = viewDesc;
}
public boolean isMaterializedView() {
return viewDesc != null && viewDesc.isMaterialized();
}
public boolean isView() {
return viewDesc != null && !viewDesc.isMaterialized();
}
public boolean isMultiDestQuery() {
return qbp != null && qbp.getClauseNamesForDest() != null && qbp.getClauseNamesForDest().size() > 1;
}
public HashMap<String, Table> getViewToTabSchema() {
return viewAliasToViewSchema;
}
public boolean isInsideView() {
return insideView;
}
public void setInsideView(boolean insideView) {
this.insideView = insideView;
}
public Set<String> getAliasInsideView() {
return aliasInsideView;
}
/**
* returns true, if the query block contains any query, or subquery without a source table
* Like select current_user(), select current_database()
* @return true, if the query block contains any query without a source table
*/
public boolean containsQueryWithoutSourceTable() {
for (QBExpr qbexpr : aliasToSubq.values()) {
if (qbexpr.containsQueryWithoutSourceTable()) {
return true;
}
}
return aliasToTabs.size()==0 && aliasToSubq.size()==0;
}
}