2015-09-04 21:39:10 +00:00
|
|
|
#include <Poco/Util/Application.h>
|
2016-11-20 12:43:20 +00:00
|
|
|
#include <Poco/String.h>
|
2015-09-04 21:39:10 +00:00
|
|
|
|
2018-10-18 15:03:14 +00:00
|
|
|
#include <Core/Block.h>
|
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTFunction.h>
|
|
|
|
#include <Parsers/ASTIdentifier.h>
|
|
|
|
#include <Parsers/ASTLiteral.h>
|
2017-06-19 03:05:05 +00:00
|
|
|
#include <Parsers/ASTQualifiedAsterisk.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTExpressionList.h>
|
|
|
|
#include <Parsers/ASTSelectQuery.h>
|
2018-02-26 09:05:06 +00:00
|
|
|
#include <Parsers/ASTSelectWithUnionQuery.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTSubquery.h>
|
|
|
|
#include <Parsers/ASTOrderByElement.h>
|
2018-02-05 14:03:01 +00:00
|
|
|
#include <Parsers/formatAST.h>
|
2018-10-09 14:32:11 +00:00
|
|
|
#include <Parsers/DumpASTNode.h>
|
2013-05-24 10:49:19 +00:00
|
|
|
|
2017-12-25 00:38:45 +00:00
|
|
|
#include <DataTypes/DataTypeNullable.h>
|
2017-12-25 18:58:39 +00:00
|
|
|
#include <DataTypes/NestedUtils.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <DataTypes/DataTypesNumber.h>
|
2019-04-05 16:45:41 +00:00
|
|
|
#include <DataTypes/DataTypeLowCardinality.h>
|
2018-08-28 14:40:07 +00:00
|
|
|
|
2018-10-12 15:41:28 +00:00
|
|
|
#include <Columns/IColumn.h>
|
2013-05-24 10:49:19 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Interpreters/ExpressionAnalyzer.h>
|
|
|
|
#include <Interpreters/ExpressionActions.h>
|
2019-08-14 16:53:30 +00:00
|
|
|
#include <Interpreters/ActionsVisitor.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Interpreters/InJoinSubqueriesPreprocessor.h>
|
|
|
|
#include <Interpreters/LogicalExpressionsOptimizer.h>
|
2018-03-04 16:15:31 +00:00
|
|
|
#include <Interpreters/PredicateExpressionsOptimizer.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Interpreters/ExternalDictionaries.h>
|
|
|
|
#include <Interpreters/Set.h>
|
|
|
|
#include <Interpreters/Join.h>
|
2013-05-24 10:49:19 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <AggregateFunctions/AggregateFunctionFactory.h>
|
2017-07-10 23:30:17 +00:00
|
|
|
#include <AggregateFunctions/parseAggregateFunctionParameters.h>
|
2015-04-16 06:12:35 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Storages/StorageDistributed.h>
|
|
|
|
#include <Storages/StorageJoin.h>
|
2014-03-14 14:52:48 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <DataStreams/copyData.h>
|
2019-05-17 14:34:25 +00:00
|
|
|
#include <DataStreams/IBlockInputStream.h>
|
2014-03-14 14:52:48 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Dictionaries/IDictionary.h>
|
2015-05-13 16:11:07 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Common/typeid_cast.h>
|
2018-01-15 19:07:47 +00:00
|
|
|
#include <Common/StringUtils/StringUtils.h>
|
2014-06-26 00:58:14 +00:00
|
|
|
|
2017-06-06 17:18:32 +00:00
|
|
|
#include <ext/range.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <DataTypes/DataTypeFactory.h>
|
2018-02-06 19:34:53 +00:00
|
|
|
#include <Functions/FunctionsMiscellaneous.h>
|
2018-07-26 12:58:23 +00:00
|
|
|
#include <Parsers/ExpressionListParsers.h>
|
|
|
|
#include <Parsers/parseQuery.h>
|
2018-07-27 16:21:43 +00:00
|
|
|
#include <Parsers/queryToString.h>
|
2018-10-12 15:41:28 +00:00
|
|
|
#include <Interpreters/interpretSubquery.h>
|
2018-10-30 16:31:21 +00:00
|
|
|
#include <Interpreters/DatabaseAndTableWithAlias.h>
|
2018-09-12 05:41:09 +00:00
|
|
|
#include <Interpreters/QueryNormalizer.h>
|
2018-10-16 19:00:05 +00:00
|
|
|
|
2018-10-12 15:41:28 +00:00
|
|
|
#include <Interpreters/ActionsVisitor.h>
|
2018-10-16 19:00:05 +00:00
|
|
|
#include <Interpreters/ExternalTablesVisitor.h>
|
|
|
|
#include <Interpreters/GlobalSubqueriesVisitor.h>
|
2019-08-12 19:27:09 +00:00
|
|
|
#include <Interpreters/GetAggregatesVisitor.h>
|
2013-05-24 10:49:19 +00:00
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2018-10-10 16:23:27 +00:00
|
|
|
using LogAST = DebugASTLog<false>; /// set to true to enable logs
|
2018-10-09 14:32:11 +00:00
|
|
|
|
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
extern const int UNKNOWN_IDENTIFIER;
|
2018-08-29 15:15:42 +00:00
|
|
|
extern const int EXPECTED_ALL_OR_ANY;
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
2017-01-14 09:00:19 +00:00
|
|
|
ExpressionAnalyzer::ExpressionAnalyzer(
|
2018-08-27 17:58:43 +00:00
|
|
|
const ASTPtr & query_,
|
2018-11-08 15:43:14 +00:00
|
|
|
const SyntaxAnalyzerResultPtr & syntax_analyzer_result_,
|
2017-04-01 07:20:54 +00:00
|
|
|
const Context & context_,
|
2019-01-30 15:51:39 +00:00
|
|
|
const NameSet & required_result_columns_,
|
2017-04-01 07:20:54 +00:00
|
|
|
size_t subquery_depth_,
|
2019-08-13 13:48:09 +00:00
|
|
|
bool do_global)
|
|
|
|
: ExpressionAnalyzerData(required_result_columns_)
|
2018-11-08 15:43:14 +00:00
|
|
|
, query(query_), context(context_), settings(context.getSettings())
|
2019-08-13 13:48:09 +00:00
|
|
|
, subquery_depth(subquery_depth_)
|
2018-11-09 17:23:48 +00:00
|
|
|
, syntax(syntax_analyzer_result_)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-04-02 17:37:49 +00:00
|
|
|
/// external_tables, subqueries_for_sets for global subqueries.
|
|
|
|
/// Replaces global subqueries with the generated names of temporary tables that will be sent to remote servers.
|
2019-08-13 13:48:09 +00:00
|
|
|
initGlobalSubqueriesAndExternalTables(do_global);
|
2015-10-15 11:07:39 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// has_aggregation, aggregation_keys, aggregate_descriptions, aggregated_columns.
|
2017-04-02 17:37:49 +00:00
|
|
|
/// This analysis should be performed after processing global subqueries, because otherwise,
|
|
|
|
/// if the aggregate function contains a global subquery, then `analyzeAggregation` method will save
|
|
|
|
/// in `aggregate_descriptions` the information about the parameters of this aggregate function, among which
|
|
|
|
/// global subquery. Then, when you call `initGlobalSubqueriesAndExternalTables` method, this
|
|
|
|
/// the global subquery will be replaced with a temporary table, resulting in aggregate_descriptions
|
|
|
|
/// will contain out-of-date information, which will lead to an error when the query is executed.
|
2017-04-01 07:20:54 +00:00
|
|
|
analyzeAggregation();
|
2014-07-04 19:13:38 +00:00
|
|
|
}
|
|
|
|
|
2018-10-17 11:06:46 +00:00
|
|
|
bool ExpressionAnalyzer::isRemoteStorage() const
|
2018-10-12 15:41:28 +00:00
|
|
|
{
|
2019-08-09 14:50:04 +00:00
|
|
|
return storage() && storage()->isRemote();
|
2018-10-12 15:41:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-04 19:13:38 +00:00
|
|
|
void ExpressionAnalyzer::analyzeAggregation()
|
|
|
|
{
|
2017-04-02 17:37:49 +00:00
|
|
|
/** Find aggregation keys (aggregation_keys), information about aggregate functions (aggregate_descriptions),
|
|
|
|
* as well as a set of columns obtained after the aggregation, if any,
|
|
|
|
* or after all the actions that are usually performed before aggregation (aggregated_columns).
|
2017-04-01 07:20:54 +00:00
|
|
|
*
|
2017-04-02 17:37:49 +00:00
|
|
|
* Everything below (compiling temporary ExpressionActions) - only for the purpose of query analysis (type output).
|
2017-04-01 07:20:54 +00:00
|
|
|
*/
|
|
|
|
|
2019-03-11 13:22:51 +00:00
|
|
|
auto * select_query = query->as<ASTSelectQuery>();
|
2019-03-11 12:49:39 +00:00
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
ExpressionActionsPtr temp_actions = std::make_shared<ExpressionActions>(sourceColumns(), context);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (select_query)
|
|
|
|
{
|
2018-12-19 13:13:51 +00:00
|
|
|
bool is_array_join_left;
|
|
|
|
ASTPtr array_join_expression_list = select_query->array_join_expression_list(is_array_join_left);
|
|
|
|
if (array_join_expression_list)
|
|
|
|
{
|
|
|
|
getRootActions(array_join_expression_list, true, temp_actions);
|
|
|
|
addMultipleArrayJoinAction(temp_actions, is_array_join_left);
|
2019-07-08 20:06:17 +00:00
|
|
|
|
|
|
|
array_join_columns.clear();
|
|
|
|
for (auto & column : temp_actions->getSampleBlock().getNamesAndTypesList())
|
|
|
|
if (syntax->array_join_result_to_source.count(column.name))
|
|
|
|
array_join_columns.emplace_back(column);
|
2018-12-19 13:13:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
const ASTTablesInSelectQueryElement * join = select_query->join();
|
|
|
|
if (join)
|
|
|
|
{
|
2019-03-15 16:14:13 +00:00
|
|
|
const auto & table_join = join->table_join->as<ASTTableJoin &>();
|
|
|
|
if (table_join.using_expression_list)
|
|
|
|
getRootActions(table_join.using_expression_list, true, temp_actions);
|
|
|
|
if (table_join.on_expression)
|
2018-11-09 17:23:48 +00:00
|
|
|
for (const auto & key_ast : analyzedJoin().key_asts_left)
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(key_ast, true, temp_actions);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
addJoinAction(temp_actions, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-13 12:39:03 +00:00
|
|
|
has_aggregation = makeAggregateDescriptions(temp_actions);
|
|
|
|
if (select_query && (select_query->groupBy() || select_query->having()))
|
|
|
|
has_aggregation = true;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (has_aggregation)
|
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
getSelectQuery(); /// assertSelect()
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Find out aggregation keys.
|
2019-04-09 14:22:35 +00:00
|
|
|
if (select_query->groupBy())
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
NameSet unique_keys;
|
2019-04-09 14:22:35 +00:00
|
|
|
ASTs & group_asts = select_query->groupBy()->children;
|
2017-04-01 07:20:54 +00:00
|
|
|
for (ssize_t i = 0; i < ssize_t(group_asts.size()); ++i)
|
|
|
|
{
|
|
|
|
ssize_t size = group_asts.size();
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(group_asts[i], true, temp_actions);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
const auto & column_name = group_asts[i]->getColumnName();
|
|
|
|
const auto & block = temp_actions->getSampleBlock();
|
|
|
|
|
|
|
|
if (!block.has(column_name))
|
|
|
|
throw Exception("Unknown identifier (in GROUP BY): " + column_name, ErrorCodes::UNKNOWN_IDENTIFIER);
|
|
|
|
|
|
|
|
const auto & col = block.getByName(column_name);
|
|
|
|
|
|
|
|
/// Constant expressions have non-null column pointer at this stage.
|
2019-06-27 19:28:52 +00:00
|
|
|
if (col.column && isColumnConst(*col.column))
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
/// But don't remove last key column if no aggregate functions, otherwise aggregation will not work.
|
|
|
|
if (!aggregate_descriptions.empty() || size > 1)
|
|
|
|
{
|
|
|
|
if (i + 1 < static_cast<ssize_t>(size))
|
|
|
|
group_asts[i] = std::move(group_asts.back());
|
|
|
|
|
|
|
|
group_asts.pop_back();
|
|
|
|
|
|
|
|
--i;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
NameAndTypePair key{column_name, col.type};
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Aggregation keys are uniqued.
|
|
|
|
if (!unique_keys.count(key.name))
|
|
|
|
{
|
|
|
|
unique_keys.insert(key.name);
|
2018-04-04 18:56:30 +00:00
|
|
|
aggregation_keys.push_back(key);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Key is no longer needed, therefore we can save a little by moving it.
|
2018-04-04 18:56:30 +00:00
|
|
|
aggregated_columns.push_back(std::move(key));
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (group_asts.empty())
|
|
|
|
{
|
2019-04-09 14:22:35 +00:00
|
|
|
select_query->setExpression(ASTSelectQuery::Expression::GROUP_BY, {});
|
|
|
|
has_aggregation = select_query->having() || aggregate_descriptions.size();
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < aggregate_descriptions.size(); ++i)
|
|
|
|
{
|
|
|
|
AggregateDescription & desc = aggregate_descriptions[i];
|
2018-04-04 18:56:30 +00:00
|
|
|
aggregated_columns.emplace_back(desc.column_name, desc.function->getReturnType());
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
2018-04-04 18:56:30 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
aggregated_columns = temp_actions->getSampleBlock().getNamesAndTypesList();
|
|
|
|
}
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-13 13:48:09 +00:00
|
|
|
void ExpressionAnalyzer::initGlobalSubqueriesAndExternalTables(bool do_global)
|
2014-07-04 20:30:06 +00:00
|
|
|
{
|
2017-04-02 17:37:49 +00:00
|
|
|
/// Adds existing external tables (not subqueries) to the external_tables dictionary.
|
2018-12-10 13:02:45 +00:00
|
|
|
ExternalTablesVisitor::Data tables_data{context, external_tables};
|
2018-12-07 15:14:50 +00:00
|
|
|
ExternalTablesVisitor(tables_data).visit(query);
|
2014-07-04 20:30:06 +00:00
|
|
|
|
2018-10-17 10:59:05 +00:00
|
|
|
if (do_global)
|
|
|
|
{
|
2018-12-10 13:02:45 +00:00
|
|
|
GlobalSubqueriesVisitor::Data subqueries_data(context, subquery_depth, isRemoteStorage(),
|
2018-10-17 10:59:05 +00:00
|
|
|
external_tables, subqueries_for_sets, has_global_subqueries);
|
2018-12-07 15:36:54 +00:00
|
|
|
GlobalSubqueriesVisitor(subqueries_data).visit(query);
|
2018-10-17 10:59:05 +00:00
|
|
|
}
|
2015-11-04 22:02:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-02 19:37:04 +00:00
|
|
|
void ExpressionAnalyzer::tryMakeSetForIndexFromSubquery(const ASTPtr & subquery_or_table_name)
|
2018-01-21 07:30:07 +00:00
|
|
|
{
|
2019-01-22 12:33:56 +00:00
|
|
|
auto set_key = PreparedSetKey::forSubquery(*subquery_or_table_name);
|
|
|
|
if (prepared_sets.count(set_key))
|
|
|
|
return; /// Already prepared.
|
2018-01-21 07:30:07 +00:00
|
|
|
|
2019-01-22 12:33:56 +00:00
|
|
|
auto interpreter_subquery = interpretSubquery(subquery_or_table_name, context, subquery_depth + 1, {});
|
|
|
|
BlockIO res = interpreter_subquery->execute();
|
2018-01-21 07:30:07 +00:00
|
|
|
|
2019-01-22 12:33:56 +00:00
|
|
|
SetPtr set = std::make_shared<Set>(settings.size_limits_for_set, true);
|
2018-04-19 21:34:04 +00:00
|
|
|
set->setHeader(res.in->getHeader());
|
2019-01-22 12:33:56 +00:00
|
|
|
|
2019-07-11 13:51:54 +00:00
|
|
|
res.in->readPrefix();
|
2018-01-21 07:30:07 +00:00
|
|
|
while (Block block = res.in->read())
|
|
|
|
{
|
|
|
|
/// If the limits have been exceeded, give up and let the default subquery processing actions take place.
|
2018-07-02 18:57:14 +00:00
|
|
|
if (!set->insertFromBlock(block))
|
2018-01-21 07:30:07 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-07-11 13:51:54 +00:00
|
|
|
res.in->readSuffix();
|
2018-01-21 07:30:07 +00:00
|
|
|
|
2019-01-22 12:33:56 +00:00
|
|
|
prepared_sets[set_key] = std::move(set);
|
2018-01-21 07:30:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-12 19:27:09 +00:00
|
|
|
/// Perfomance optimisation for IN() if storage supports it.
|
|
|
|
void ExpressionAnalyzer::makeSetsForIndex(const ASTPtr & node)
|
2014-03-31 14:49:43 +00:00
|
|
|
{
|
2019-08-12 19:27:09 +00:00
|
|
|
if (!node || !storage() || !storage()->supportsIndexForIn())
|
|
|
|
return;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (auto & child : node->children)
|
2018-02-14 17:39:16 +00:00
|
|
|
{
|
2019-01-22 12:33:56 +00:00
|
|
|
/// Don't descend into subqueries.
|
2019-03-11 13:22:51 +00:00
|
|
|
if (child->as<ASTSubquery>())
|
2018-03-16 06:33:23 +00:00
|
|
|
continue;
|
|
|
|
|
2019-01-22 12:33:56 +00:00
|
|
|
/// Don't descend into lambda functions
|
2019-03-11 13:22:51 +00:00
|
|
|
const auto * func = child->as<ASTFunction>();
|
2018-03-16 06:33:23 +00:00
|
|
|
if (func && func->name == "lambda")
|
|
|
|
continue;
|
|
|
|
|
2019-08-12 19:27:09 +00:00
|
|
|
makeSetsForIndex(child);
|
2018-02-14 17:39:16 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-03-11 13:22:51 +00:00
|
|
|
const auto * func = node->as<ASTFunction>();
|
2018-02-24 01:31:42 +00:00
|
|
|
if (func && functionIsInOperator(func->name))
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-07-14 00:33:37 +00:00
|
|
|
const IAST & args = *func->arguments;
|
2019-08-12 19:27:09 +00:00
|
|
|
const ASTPtr & left_in_operand = args.children.at(0);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-08-12 19:27:09 +00:00
|
|
|
if (storage()->mayBenefitFromIndexForIn(left_in_operand, context))
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2018-03-16 06:39:32 +00:00
|
|
|
const ASTPtr & arg = args.children.at(1);
|
2019-03-11 13:22:51 +00:00
|
|
|
if (arg->as<ASTSubquery>() || arg->as<ASTIdentifier>())
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2019-01-22 12:33:56 +00:00
|
|
|
if (settings.use_index_for_in_with_subqueries)
|
|
|
|
tryMakeSetForIndexFromSubquery(arg);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-09 14:50:04 +00:00
|
|
|
NamesAndTypesList temp_columns = sourceColumns();
|
2019-01-22 12:33:56 +00:00
|
|
|
temp_columns.insert(temp_columns.end(), array_join_columns.begin(), array_join_columns.end());
|
2019-08-12 19:27:09 +00:00
|
|
|
temp_columns.insert(temp_columns.end(), columnsAddedByJoin().begin(), columnsAddedByJoin().end());
|
|
|
|
|
2019-01-22 12:33:56 +00:00
|
|
|
ExpressionActionsPtr temp_actions = std::make_shared<ExpressionActions>(temp_columns, context);
|
2019-08-12 19:27:09 +00:00
|
|
|
getRootActions(left_in_operand, true, temp_actions);
|
2019-01-22 12:33:56 +00:00
|
|
|
|
|
|
|
Block sample_block_with_calculated_columns = temp_actions->getSampleBlock();
|
2019-08-12 19:27:09 +00:00
|
|
|
if (sample_block_with_calculated_columns.has(left_in_operand->getColumnName()))
|
2019-01-22 12:33:56 +00:00
|
|
|
makeExplicitSet(func, sample_block_with_calculated_columns, true, context,
|
|
|
|
settings.size_limits_for_set, prepared_sets);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-01 13:29:32 +00:00
|
|
|
}
|
|
|
|
|
2018-10-16 12:34:20 +00:00
|
|
|
|
|
|
|
void ExpressionAnalyzer::getRootActions(const ASTPtr & ast, bool no_subqueries, ExpressionActionsPtr & actions, bool only_consts)
|
2018-07-24 12:41:35 +00:00
|
|
|
{
|
2018-10-16 12:34:20 +00:00
|
|
|
LogAST log;
|
2018-11-26 01:39:04 +00:00
|
|
|
ActionsVisitor actions_visitor(context, settings.size_limits_for_set, subquery_depth,
|
2019-08-09 14:50:04 +00:00
|
|
|
sourceColumns(), actions, prepared_sets, subqueries_for_sets,
|
2018-10-17 11:06:46 +00:00
|
|
|
no_subqueries, only_consts, !isRemoteStorage(), log.stream());
|
2019-08-14 16:53:30 +00:00
|
|
|
actions_visitor.visit(ast, actions);
|
2018-10-16 12:34:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-13 12:39:03 +00:00
|
|
|
bool ExpressionAnalyzer::makeAggregateDescriptions(ExpressionActionsPtr & actions)
|
2013-05-24 10:49:19 +00:00
|
|
|
{
|
2019-08-13 12:39:03 +00:00
|
|
|
for (const ASTFunction * node : aggregates())
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
AggregateDescription aggregate;
|
|
|
|
aggregate.column_name = node->getColumnName();
|
|
|
|
|
|
|
|
const ASTs & arguments = node->arguments->children;
|
|
|
|
aggregate.argument_names.resize(arguments.size());
|
|
|
|
DataTypes types(arguments.size());
|
|
|
|
|
|
|
|
for (size_t i = 0; i < arguments.size(); ++i)
|
|
|
|
{
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(arguments[i], true, actions);
|
2017-04-01 07:20:54 +00:00
|
|
|
const std::string & name = arguments[i]->getColumnName();
|
2019-08-01 18:22:38 +00:00
|
|
|
types[i] = actions->getSampleBlock().getByName(name).type;
|
2017-04-01 07:20:54 +00:00
|
|
|
aggregate.argument_names[i] = name;
|
|
|
|
}
|
|
|
|
|
2017-07-10 23:30:17 +00:00
|
|
|
aggregate.parameters = (node->parameters) ? getAggregateFunctionParametersArray(node->parameters) : Array();
|
|
|
|
aggregate.function = AggregateFunctionFactory::instance().get(node->name, types, aggregate.parameters);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
aggregate_descriptions.push_back(aggregate);
|
|
|
|
}
|
2019-08-13 12:39:03 +00:00
|
|
|
|
|
|
|
return !aggregates().empty();
|
2016-03-05 02:30:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-13 13:48:09 +00:00
|
|
|
const ASTSelectQuery * ExpressionAnalyzer::getSelectQuery() const
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-03-11 13:22:51 +00:00
|
|
|
const auto * select_query = query->as<ASTSelectQuery>();
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!select_query)
|
|
|
|
throw Exception("Not a select query", ErrorCodes::LOGICAL_ERROR);
|
2019-08-13 13:48:09 +00:00
|
|
|
return select_query;
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
2013-06-14 16:38:54 +00:00
|
|
|
|
2019-08-13 13:48:09 +00:00
|
|
|
const ASTSelectQuery * ExpressionAnalyzer::getAggregatingQuery() const
|
2013-05-24 10:49:19 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!has_aggregation)
|
|
|
|
throw Exception("No aggregation", ErrorCodes::LOGICAL_ERROR);
|
2019-08-13 13:48:09 +00:00
|
|
|
return getSelectQuery();
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
2013-06-14 16:38:54 +00:00
|
|
|
|
2017-12-25 21:57:29 +00:00
|
|
|
void ExpressionAnalyzer::initChain(ExpressionActionsChain & chain, const NamesAndTypesList & columns) const
|
2013-05-28 14:24:20 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (chain.steps.empty())
|
|
|
|
{
|
2018-08-30 16:31:20 +00:00
|
|
|
chain.steps.emplace_back(std::make_shared<ExpressionActions>(columns, context));
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2013-05-28 14:24:20 +00:00
|
|
|
}
|
2013-05-28 11:54:37 +00:00
|
|
|
|
2016-07-23 02:25:09 +00:00
|
|
|
/// "Big" ARRAY JOIN.
|
2018-12-19 13:13:51 +00:00
|
|
|
void ExpressionAnalyzer::addMultipleArrayJoinAction(ExpressionActionsPtr & actions, bool array_join_is_left) const
|
2013-10-17 13:32:32 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
NameSet result_columns;
|
2018-11-08 15:43:14 +00:00
|
|
|
for (const auto & result_source : syntax->array_join_result_to_source)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
/// Assign new names to columns, if needed.
|
|
|
|
if (result_source.first != result_source.second)
|
|
|
|
actions->add(ExpressionAction::copyColumn(result_source.second, result_source.first));
|
2015-10-22 20:56:52 +00:00
|
|
|
|
2017-04-02 17:37:49 +00:00
|
|
|
/// Make ARRAY JOIN (replace arrays with their insides) for the columns in these new names.
|
2017-04-01 07:20:54 +00:00
|
|
|
result_columns.insert(result_source.first);
|
|
|
|
}
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2018-12-19 13:13:51 +00:00
|
|
|
actions->add(ExpressionAction::arrayJoin(result_columns, array_join_is_left, context));
|
2013-10-17 13:32:32 +00:00
|
|
|
}
|
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
bool ExpressionAnalyzer::appendArrayJoin(ExpressionActionsChain & chain, bool only_types)
|
2013-07-26 16:33:05 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2018-12-19 13:13:51 +00:00
|
|
|
bool is_array_join_left;
|
|
|
|
ASTPtr array_join_expression_list = select_query->array_join_expression_list(is_array_join_left);
|
|
|
|
if (!array_join_expression_list)
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2017-04-01 07:20:54 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2018-12-19 13:13:51 +00:00
|
|
|
getRootActions(array_join_expression_list, only_types, step.actions);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2018-12-19 13:13:51 +00:00
|
|
|
addMultipleArrayJoinAction(step.actions, is_array_join_left);
|
2013-10-21 11:33:25 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return true;
|
2013-07-26 16:33:05 +00:00
|
|
|
}
|
|
|
|
|
2015-04-18 22:30:43 +00:00
|
|
|
void ExpressionAnalyzer::addJoinAction(ExpressionActionsPtr & actions, bool only_types) const
|
2014-06-13 02:05:05 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
if (only_types)
|
2019-08-09 14:50:04 +00:00
|
|
|
actions->add(ExpressionAction::ordinaryJoin(nullptr, analyzedJoin().key_names_left, columnsAddedByJoin()));
|
2017-04-01 07:20:54 +00:00
|
|
|
else
|
|
|
|
for (auto & subquery_for_set : subqueries_for_sets)
|
|
|
|
if (subquery_for_set.second.join)
|
2018-11-09 17:23:48 +00:00
|
|
|
actions->add(ExpressionAction::ordinaryJoin(subquery_for_set.second.join, analyzedJoin().key_names_left,
|
2019-08-09 14:50:04 +00:00
|
|
|
columnsAddedByJoin()));
|
2014-06-13 02:05:05 +00:00
|
|
|
}
|
|
|
|
|
2019-05-05 10:49:12 +00:00
|
|
|
static void appendRequiredColumns(
|
2019-08-01 19:56:29 +00:00
|
|
|
NameSet & required_columns, const Block & sample, const Names & key_names_right, const NamesAndTypesList & columns_added_by_join)
|
2019-02-06 14:48:05 +00:00
|
|
|
{
|
2019-05-05 10:49:12 +00:00
|
|
|
for (auto & column : key_names_right)
|
2019-02-06 14:48:05 +00:00
|
|
|
if (!sample.has(column))
|
2019-08-01 19:56:29 +00:00
|
|
|
required_columns.insert(column);
|
2019-02-06 14:48:05 +00:00
|
|
|
|
2019-05-05 10:49:12 +00:00
|
|
|
for (auto & column : columns_added_by_join)
|
2019-07-30 18:39:37 +00:00
|
|
|
if (!sample.has(column.name))
|
2019-08-01 19:56:29 +00:00
|
|
|
required_columns.insert(column.name);
|
2019-02-06 14:48:05 +00:00
|
|
|
}
|
|
|
|
|
2014-06-13 02:05:05 +00:00
|
|
|
bool ExpressionAnalyzer::appendJoin(ExpressionActionsChain & chain, bool only_types)
|
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (!select_query->join())
|
|
|
|
return false;
|
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2017-04-01 07:20:54 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
|
|
|
|
2019-03-15 16:14:13 +00:00
|
|
|
const auto & join_element = select_query->join()->as<ASTTablesInSelectQueryElement &>();
|
|
|
|
auto & join_params = join_element.table_join->as<ASTTableJoin &>();
|
2018-08-28 13:57:31 +00:00
|
|
|
|
2019-03-15 16:14:13 +00:00
|
|
|
if (join_params.strictness == ASTTableJoin::Strictness::Unspecified && join_params.kind != ASTTableJoin::Kind::Cross)
|
2018-08-28 13:57:31 +00:00
|
|
|
{
|
2018-10-18 15:03:14 +00:00
|
|
|
if (settings.join_default_strictness == "ANY")
|
2019-03-15 16:14:13 +00:00
|
|
|
join_params.strictness = ASTTableJoin::Strictness::Any;
|
2018-10-18 15:03:14 +00:00
|
|
|
else if (settings.join_default_strictness == "ALL")
|
2019-03-15 16:14:13 +00:00
|
|
|
join_params.strictness = ASTTableJoin::Strictness::All;
|
2018-08-28 13:57:31 +00:00
|
|
|
else
|
2018-08-29 15:15:42 +00:00
|
|
|
throw Exception("Expected ANY or ALL in JOIN section, because setting (join_default_strictness) is empty", DB::ErrorCodes::EXPECTED_ALL_OR_ANY);
|
2018-08-28 13:57:31 +00:00
|
|
|
}
|
|
|
|
|
2019-08-14 15:34:32 +00:00
|
|
|
if (join_params.using_expression_list)
|
|
|
|
{
|
|
|
|
getRootActions(join_params.using_expression_list, only_types, step.actions);
|
|
|
|
}
|
|
|
|
else if (join_params.on_expression)
|
|
|
|
{
|
|
|
|
auto list = std::make_shared<ASTExpressionList>();
|
|
|
|
list->children = analyzedJoin().key_asts_left;
|
|
|
|
getRootActions(list, only_types, step.actions);
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-08-14 15:34:32 +00:00
|
|
|
const auto & table_to_join = join_element.table_expression->as<ASTTableExpression &>();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-04-02 17:37:49 +00:00
|
|
|
/// Two JOINs are not supported with the same subquery, but different USINGs.
|
2019-03-15 16:14:13 +00:00
|
|
|
auto join_hash = join_element.getTreeHash();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-02-23 08:05:21 +00:00
|
|
|
SubqueryForSet & subquery_for_set = subqueries_for_sets[toString(join_hash.first) + "_" + toString(join_hash.second)];
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-04-02 17:37:49 +00:00
|
|
|
/// Special case - if table name is specified on the right of JOIN, then the table has the type Join (the previously prepared mapping).
|
|
|
|
/// TODO This syntax does not support specifying a database name.
|
2019-03-15 16:14:13 +00:00
|
|
|
if (table_to_join.database_and_table_name)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2019-03-15 16:14:13 +00:00
|
|
|
DatabaseAndTableWithAlias database_table(table_to_join.database_and_table_name);
|
2018-10-30 16:31:21 +00:00
|
|
|
StoragePtr table = context.tryGetTable(database_table.database, database_table.table);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (table)
|
|
|
|
{
|
2019-03-11 12:49:39 +00:00
|
|
|
auto * storage_join = dynamic_cast<StorageJoin *>(table.get());
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
if (storage_join)
|
|
|
|
{
|
2019-03-15 16:14:13 +00:00
|
|
|
storage_join->assertCompatible(join_params.kind, join_params.strictness);
|
2017-04-02 17:37:49 +00:00
|
|
|
/// TODO Check the set of keys.
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
JoinPtr & join = storage_join->getJoin();
|
|
|
|
subquery_for_set.join = join;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!subquery_for_set.join)
|
|
|
|
{
|
2019-02-06 14:48:05 +00:00
|
|
|
auto & analyzed_join = analyzedJoin();
|
|
|
|
/// Actions which need to be calculated on joined block.
|
|
|
|
ExpressionActionsPtr joined_block_actions =
|
2019-08-09 14:50:04 +00:00
|
|
|
analyzed_join.createJoinedBlockActions(columnsAddedByJoin(), select_query, context);
|
2019-02-06 14:48:05 +00:00
|
|
|
|
2017-04-02 17:37:49 +00:00
|
|
|
/** For GLOBAL JOINs (in the case, for example, of the push method for executing GLOBAL subqueries), the following occurs
|
|
|
|
* - in the addExternalStorage function, the JOIN (SELECT ...) subquery is replaced with JOIN _data1,
|
|
|
|
* in the subquery_for_set object this subquery is exposed as source and the temporary table _data1 as the `table`.
|
|
|
|
* - this function shows the expression JOIN _data1.
|
2017-04-01 07:20:54 +00:00
|
|
|
*/
|
|
|
|
if (!subquery_for_set.source)
|
|
|
|
{
|
|
|
|
ASTPtr table;
|
2018-07-30 13:57:50 +00:00
|
|
|
|
2019-03-15 16:14:13 +00:00
|
|
|
if (table_to_join.subquery)
|
|
|
|
table = table_to_join.subquery;
|
|
|
|
else if (table_to_join.table_function)
|
|
|
|
table = table_to_join.table_function;
|
|
|
|
else if (table_to_join.database_and_table_name)
|
|
|
|
table = table_to_join.database_and_table_name;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-08-01 19:56:29 +00:00
|
|
|
Names action_columns = joined_block_actions->getRequiredColumns();
|
|
|
|
NameSet required_columns(action_columns.begin(), action_columns.end());
|
2019-02-06 14:48:05 +00:00
|
|
|
|
2019-05-05 10:49:12 +00:00
|
|
|
appendRequiredColumns(
|
2019-08-09 14:50:04 +00:00
|
|
|
required_columns, joined_block_actions->getSampleBlock(), analyzed_join.key_names_right, columnsAddedByJoin());
|
2019-01-30 12:01:00 +00:00
|
|
|
|
2019-08-01 19:56:29 +00:00
|
|
|
auto original_map = analyzed_join.getOriginalColumnsMap(required_columns);
|
|
|
|
Names original_columns;
|
|
|
|
for (auto & pr : original_map)
|
|
|
|
original_columns.push_back(pr.second);
|
|
|
|
|
2018-10-01 14:18:47 +00:00
|
|
|
auto interpreter = interpretSubquery(table, context, subquery_depth, original_columns);
|
2019-07-30 18:39:37 +00:00
|
|
|
|
2019-08-01 19:56:29 +00:00
|
|
|
subquery_for_set.makeSource(interpreter, original_map);
|
2018-07-31 11:31:18 +00:00
|
|
|
}
|
|
|
|
|
2019-01-30 12:01:00 +00:00
|
|
|
Block sample_block = subquery_for_set.renamedSampleBlock();
|
2018-11-08 09:00:25 +00:00
|
|
|
joined_block_actions->execute(sample_block);
|
2018-10-02 18:14:37 +00:00
|
|
|
|
2017-04-02 17:37:49 +00:00
|
|
|
/// TODO You do not need to set this up when JOIN is only needed on remote servers.
|
2019-01-30 12:01:00 +00:00
|
|
|
subquery_for_set.join = std::make_shared<Join>(analyzedJoin().key_names_right, settings.join_use_nulls,
|
2019-03-15 16:14:13 +00:00
|
|
|
settings.size_limits_for_join, join_params.kind, join_params.strictness);
|
2018-07-30 13:57:50 +00:00
|
|
|
subquery_for_set.join->setSampleBlock(sample_block);
|
2018-11-08 09:00:25 +00:00
|
|
|
subquery_for_set.joined_block_actions = joined_block_actions;
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
addJoinAction(step.actions, false);
|
|
|
|
|
|
|
|
return true;
|
2014-06-13 02:05:05 +00:00
|
|
|
}
|
|
|
|
|
2018-11-08 16:39:43 +00:00
|
|
|
bool ExpressionAnalyzer::appendPrewhere(
|
|
|
|
ExpressionActionsChain & chain, bool only_types, const Names & additional_required_columns)
|
2018-04-06 13:58:06 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2018-04-06 13:58:06 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->prewhere())
|
2018-04-06 13:58:06 +00:00
|
|
|
return false;
|
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2018-04-12 09:45:24 +00:00
|
|
|
auto & step = chain.getLastStep();
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->prewhere(), only_types, step.actions);
|
|
|
|
String prewhere_column_name = select_query->prewhere()->getColumnName();
|
2018-04-12 09:45:24 +00:00
|
|
|
step.required_output.push_back(prewhere_column_name);
|
2018-06-29 11:42:44 +00:00
|
|
|
step.can_remove_required_output.push_back(true);
|
2018-04-12 09:45:24 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
/// Remove unused source_columns from prewhere actions.
|
2019-08-09 14:50:04 +00:00
|
|
|
auto tmp_actions = std::make_shared<ExpressionActions>(sourceColumns(), context);
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->prewhere(), only_types, tmp_actions);
|
2018-04-12 09:45:24 +00:00
|
|
|
tmp_actions->finalize({prewhere_column_name});
|
|
|
|
auto required_columns = tmp_actions->getRequiredColumns();
|
|
|
|
NameSet required_source_columns(required_columns.begin(), required_columns.end());
|
|
|
|
|
2018-10-04 08:58:19 +00:00
|
|
|
/// Add required columns to required output in order not to remove them after prewhere execution.
|
|
|
|
/// TODO: add sampling and final execution to common chain.
|
2018-11-08 16:39:43 +00:00
|
|
|
for (const auto & column : additional_required_columns)
|
2018-09-10 10:06:12 +00:00
|
|
|
{
|
|
|
|
if (required_source_columns.count(column))
|
|
|
|
{
|
|
|
|
step.required_output.push_back(column);
|
|
|
|
step.can_remove_required_output.push_back(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-12 09:45:24 +00:00
|
|
|
auto names = step.actions->getSampleBlock().getNames();
|
|
|
|
NameSet name_set(names.begin(), names.end());
|
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
for (const auto & column : sourceColumns())
|
2018-04-12 09:45:24 +00:00
|
|
|
if (required_source_columns.count(column.name) == 0)
|
|
|
|
name_set.erase(column.name);
|
|
|
|
|
|
|
|
Names required_output(name_set.begin(), name_set.end());
|
|
|
|
step.actions->finalize(required_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
/// Add empty action with input = {prewhere actions output} + {unused source columns}
|
|
|
|
/// Reasons:
|
|
|
|
/// 1. Remove remove source columns which are used only in prewhere actions during prewhere actions execution.
|
|
|
|
/// Example: select A prewhere B > 0. B can be removed at prewhere step.
|
|
|
|
/// 2. Store side columns which were calculated during prewhere actions execution if they are used.
|
|
|
|
/// Example: select F(A) prewhere F(A) > 0. F(A) can be saved from prewhere step.
|
2018-06-25 13:08:35 +00:00
|
|
|
/// 3. Check if we can remove filter column at prewhere step. If we can, action will store single REMOVE_COLUMN.
|
2018-04-12 09:45:24 +00:00
|
|
|
ColumnsWithTypeAndName columns = step.actions->getSampleBlock().getColumnsWithTypeAndName();
|
|
|
|
auto required_columns = step.actions->getRequiredColumns();
|
|
|
|
NameSet prewhere_input_names(required_columns.begin(), required_columns.end());
|
|
|
|
NameSet unused_source_columns;
|
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
for (const auto & column : sourceColumns())
|
2018-04-12 09:45:24 +00:00
|
|
|
{
|
|
|
|
if (prewhere_input_names.count(column.name) == 0)
|
|
|
|
{
|
|
|
|
columns.emplace_back(column.type, column.name);
|
|
|
|
unused_source_columns.emplace(column.name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-30 16:31:20 +00:00
|
|
|
chain.steps.emplace_back(std::make_shared<ExpressionActions>(std::move(columns), context));
|
2018-04-12 09:45:24 +00:00
|
|
|
chain.steps.back().additional_input = std::move(unused_source_columns);
|
|
|
|
}
|
2018-04-06 13:58:06 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2016-07-22 20:39:28 +00:00
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
bool ExpressionAnalyzer::appendWhere(ExpressionActionsChain & chain, bool only_types)
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->where())
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2017-04-01 07:20:54 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
step.required_output.push_back(select_query->where()->getColumnName());
|
2018-06-29 11:42:44 +00:00
|
|
|
step.can_remove_required_output = {true};
|
2018-04-23 19:05:46 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->where(), only_types, step.actions);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return true;
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
bool ExpressionAnalyzer::appendGroupBy(ExpressionActionsChain & chain, bool only_types)
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getAggregatingQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->groupBy())
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2017-04-01 07:20:54 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
ASTs asts = select_query->groupBy()->children;
|
2017-04-01 07:20:54 +00:00
|
|
|
for (size_t i = 0; i < asts.size(); ++i)
|
|
|
|
{
|
|
|
|
step.required_output.push_back(asts[i]->getColumnName());
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(asts[i], only_types, step.actions);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return true;
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
void ExpressionAnalyzer::appendAggregateFunctionsArguments(ExpressionActionsChain & chain, bool only_types)
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getAggregatingQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2017-04-01 07:20:54 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
for (size_t i = 0; i < aggregate_descriptions.size(); ++i)
|
|
|
|
{
|
|
|
|
for (size_t j = 0; j < aggregate_descriptions[i].argument_names.size(); ++j)
|
|
|
|
{
|
|
|
|
step.required_output.push_back(aggregate_descriptions[i].argument_names[j]);
|
|
|
|
}
|
|
|
|
}
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-08-13 12:39:03 +00:00
|
|
|
/// Collect aggregates removing duplicates by node.getColumnName()
|
|
|
|
/// It's not clear why we recollect aggregates (for query parts) while we're able to use previously collected ones (for entire query)
|
|
|
|
/// @note The original recollection logic didn't remove duplicates.
|
|
|
|
GetAggregatesVisitor::Data data;
|
|
|
|
GetAggregatesVisitor(data).visit(select_query->select());
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (select_query->having())
|
2019-08-13 12:39:03 +00:00
|
|
|
GetAggregatesVisitor(data).visit(select_query->having());
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (select_query->orderBy())
|
2019-08-13 12:39:03 +00:00
|
|
|
GetAggregatesVisitor(data).visit(select_query->orderBy());
|
|
|
|
|
|
|
|
/// TODO: data.aggregates -> aggregates()
|
|
|
|
for (const ASTFunction * node : data.aggregates)
|
|
|
|
for (auto & argument : node->arguments->children)
|
|
|
|
getRootActions(argument, only_types, step.actions);
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
bool ExpressionAnalyzer::appendHaving(ExpressionActionsChain & chain, bool only_types)
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getAggregatingQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->having())
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
initChain(chain, aggregated_columns);
|
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
step.required_output.push_back(select_query->having()->getColumnName());
|
|
|
|
getRootActions(select_query->having(), only_types, step.actions);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return true;
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
void ExpressionAnalyzer::appendSelect(ExpressionActionsChain & chain, bool only_types)
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
initChain(chain, aggregated_columns);
|
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->select(), only_types, step.actions);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
for (const auto & child : select_query->select()->children)
|
2018-03-01 01:25:06 +00:00
|
|
|
step.required_output.push_back(child->getColumnName());
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
2013-05-24 10:49:19 +00:00
|
|
|
|
2014-03-28 12:13:58 +00:00
|
|
|
bool ExpressionAnalyzer::appendOrderBy(ExpressionActionsChain & chain, bool only_types)
|
2013-05-24 10:49:19 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->orderBy())
|
2017-04-01 07:20:54 +00:00
|
|
|
return false;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
initChain(chain, aggregated_columns);
|
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->orderBy(), only_types, step.actions);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
for (auto & child : select_query->orderBy()->children)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2019-04-09 14:22:35 +00:00
|
|
|
const auto * ast = child->as<ASTOrderByElement>();
|
2017-04-01 07:20:54 +00:00
|
|
|
if (!ast || ast->children.size() < 1)
|
|
|
|
throw Exception("Bad order expression AST", ErrorCodes::UNKNOWN_TYPE_OF_AST_NODE);
|
|
|
|
ASTPtr order_expression = ast->children.at(0);
|
|
|
|
step.required_output.push_back(order_expression->getColumnName());
|
|
|
|
}
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
return true;
|
2013-05-28 11:54:37 +00:00
|
|
|
}
|
|
|
|
|
2018-03-01 05:24:56 +00:00
|
|
|
bool ExpressionAnalyzer::appendLimitBy(ExpressionActionsChain & chain, bool only_types)
|
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2018-03-01 05:24:56 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
if (!select_query->limitBy())
|
2018-03-01 05:24:56 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
initChain(chain, aggregated_columns);
|
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
getRootActions(select_query->limitBy(), only_types, step.actions);
|
2018-03-01 05:24:56 +00:00
|
|
|
|
2019-05-23 16:25:53 +00:00
|
|
|
NameSet aggregated_names;
|
|
|
|
for (const auto & column : aggregated_columns)
|
|
|
|
{
|
2019-05-25 10:51:48 +00:00
|
|
|
step.required_output.push_back(column.name);
|
|
|
|
aggregated_names.insert(column.name);
|
2019-05-23 16:25:53 +00:00
|
|
|
}
|
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
for (const auto & child : select_query->limitBy()->children)
|
2019-05-25 10:51:48 +00:00
|
|
|
{
|
|
|
|
auto child_name = child->getColumnName();
|
|
|
|
if (!aggregated_names.count(child_name))
|
|
|
|
step.required_output.push_back(std::move(child_name));
|
|
|
|
}
|
2018-03-01 05:24:56 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-12-01 21:13:25 +00:00
|
|
|
void ExpressionAnalyzer::appendProjectResult(ExpressionActionsChain & chain) const
|
2013-05-28 11:54:37 +00:00
|
|
|
{
|
2019-08-13 13:48:09 +00:00
|
|
|
const auto * select_query = getSelectQuery();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
initChain(chain, aggregated_columns);
|
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
NamesWithAliases result_columns;
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2019-04-09 14:22:35 +00:00
|
|
|
ASTs asts = select_query->select()->children;
|
2017-04-01 07:20:54 +00:00
|
|
|
for (size_t i = 0; i < asts.size(); ++i)
|
|
|
|
{
|
2018-02-26 09:05:06 +00:00
|
|
|
String result_name = asts[i]->getAliasOrColumnName();
|
2019-01-30 15:51:39 +00:00
|
|
|
if (required_result_columns.empty() || required_result_columns.count(result_name))
|
2018-02-26 09:05:06 +00:00
|
|
|
{
|
|
|
|
result_columns.emplace_back(asts[i]->getColumnName(), result_name);
|
|
|
|
step.required_output.push_back(result_columns.back().second);
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
step.actions->add(ExpressionAction::project(result_columns));
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-04 13:45:39 +00:00
|
|
|
void ExpressionAnalyzer::appendExpression(ExpressionActionsChain & chain, const ASTPtr & expr, bool only_types)
|
2018-09-03 13:36:58 +00:00
|
|
|
{
|
2019-08-09 14:50:04 +00:00
|
|
|
initChain(chain, sourceColumns());
|
2018-09-03 13:36:58 +00:00
|
|
|
ExpressionActionsChain::Step & step = chain.steps.back();
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(expr, only_types, step.actions);
|
2018-09-03 13:36:58 +00:00
|
|
|
step.required_output.push_back(expr->getColumnName());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-03 17:24:46 +00:00
|
|
|
ExpressionActionsPtr ExpressionAnalyzer::getActions(bool add_aliases, bool project_result)
|
2013-05-24 10:49:19 +00:00
|
|
|
{
|
2019-08-09 14:50:04 +00:00
|
|
|
ExpressionActionsPtr actions = std::make_shared<ExpressionActions>(sourceColumns(), context);
|
2017-04-01 07:20:54 +00:00
|
|
|
NamesWithAliases result_columns;
|
|
|
|
Names result_names;
|
|
|
|
|
|
|
|
ASTs asts;
|
|
|
|
|
2019-03-11 13:22:51 +00:00
|
|
|
if (const auto * node = query->as<ASTExpressionList>())
|
2017-04-01 07:20:54 +00:00
|
|
|
asts = node->children;
|
|
|
|
else
|
2018-08-27 17:58:43 +00:00
|
|
|
asts = ASTs(1, query);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
for (size_t i = 0; i < asts.size(); ++i)
|
|
|
|
{
|
|
|
|
std::string name = asts[i]->getColumnName();
|
|
|
|
std::string alias;
|
2018-09-03 17:24:46 +00:00
|
|
|
if (add_aliases)
|
2017-04-01 07:20:54 +00:00
|
|
|
alias = asts[i]->getAliasOrColumnName();
|
|
|
|
else
|
|
|
|
alias = name;
|
|
|
|
result_columns.emplace_back(name, alias);
|
|
|
|
result_names.push_back(alias);
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(asts[i], false, actions);
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-09-03 17:24:46 +00:00
|
|
|
if (add_aliases)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2018-09-03 17:24:46 +00:00
|
|
|
if (project_result)
|
|
|
|
actions->add(ExpressionAction::project(result_columns));
|
|
|
|
else
|
|
|
|
actions->add(ExpressionAction::addAliases(result_columns));
|
2017-04-01 07:20:54 +00:00
|
|
|
}
|
2018-09-05 13:04:28 +00:00
|
|
|
|
|
|
|
if (!(add_aliases && project_result))
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2017-04-02 17:37:49 +00:00
|
|
|
/// We will not delete the original columns.
|
2019-08-09 14:50:04 +00:00
|
|
|
for (const auto & column_name_type : sourceColumns())
|
2017-04-01 07:20:54 +00:00
|
|
|
result_names.push_back(column_name_type.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
actions->finalize(result_names);
|
|
|
|
|
|
|
|
return actions;
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ExpressionActionsPtr ExpressionAnalyzer::getConstActions()
|
|
|
|
{
|
2018-08-30 16:31:20 +00:00
|
|
|
ExpressionActionsPtr actions = std::make_shared<ExpressionActions>(NamesAndTypesList(), context);
|
2013-10-17 13:32:32 +00:00
|
|
|
|
2018-10-16 12:34:20 +00:00
|
|
|
getRootActions(query, true, actions, true);
|
2017-04-01 07:20:54 +00:00
|
|
|
return actions;
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
2015-04-18 22:30:43 +00:00
|
|
|
void ExpressionAnalyzer::getAggregateInfo(Names & key_names, AggregateDescriptions & aggregates) const
|
2013-05-24 10:49:19 +00:00
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
for (const auto & name_and_type : aggregation_keys)
|
|
|
|
key_names.emplace_back(name_and_type.name);
|
2015-03-12 02:22:55 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
aggregates = aggregate_descriptions;
|
2013-05-24 10:49:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|