2018-08-16 02:55:52 +00:00
|
|
|
#include <Interpreters/InterpreterSelectQuery.h>
|
2018-02-25 06:34:20 +00:00
|
|
|
#include <Interpreters/InterpreterSelectWithUnionQuery.h>
|
2020-05-20 20:16:32 +00:00
|
|
|
#include <Interpreters/Context.h>
|
2021-08-05 15:38:52 +00:00
|
|
|
#include <DataTypes/DataTypeLowCardinality.h>
|
2019-07-21 02:13:42 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTCreateQuery.h>
|
2021-11-26 17:21:54 +00:00
|
|
|
#include <Parsers/ASTFunction.h>
|
2021-11-26 15:49:40 +00:00
|
|
|
#include <Parsers/ASTIdentifier.h>
|
|
|
|
#include <Parsers/ASTSelectWithUnionQuery.h>
|
2018-08-16 02:55:52 +00:00
|
|
|
#include <Parsers/ASTSubquery.h>
|
2019-01-25 15:42:24 +00:00
|
|
|
#include <Parsers/ASTTablesInSelectQuery.h>
|
2013-11-08 17:43:03 +00:00
|
|
|
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Storages/StorageView.h>
|
2017-12-30 00:36:06 +00:00
|
|
|
#include <Storages/StorageFactory.h>
|
2020-06-05 11:54:54 +00:00
|
|
|
#include <Storages/SelectQueryDescription.h>
|
2013-11-08 17:43:03 +00:00
|
|
|
|
2018-08-16 02:55:52 +00:00
|
|
|
#include <Common/typeid_cast.h>
|
2019-07-21 02:13:42 +00:00
|
|
|
|
2021-10-16 14:03:50 +00:00
|
|
|
#include <QueryPipeline/Pipe.h>
|
2020-02-17 15:27:51 +00:00
|
|
|
#include <Processors/Transforms/MaterializingTransform.h>
|
2021-09-08 18:29:38 +00:00
|
|
|
#include <Processors/QueryPlan/QueryPlan.h>
|
2020-11-17 17:16:55 +00:00
|
|
|
#include <Processors/QueryPlan/ExpressionStep.h>
|
2021-03-04 17:38:12 +00:00
|
|
|
#include <Processors/QueryPlan/BuildQueryPipelineSettings.h>
|
|
|
|
#include <Processors/QueryPlan/Optimizations/QueryPlanOptimizationSettings.h>
|
2013-10-30 13:52:02 +00:00
|
|
|
|
2022-09-23 11:35:22 +00:00
|
|
|
#include <Interpreters/ReplaceQueryParameterVisitor.h>
|
|
|
|
|
2013-10-30 13:52:02 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-09-19 19:18:34 +00:00
|
|
|
extern const int INCORRECT_QUERY;
|
2018-08-16 02:55:52 +00:00
|
|
|
extern const int LOGICAL_ERROR;
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
2021-08-05 15:38:52 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
bool isNullableOrLcNullable(DataTypePtr type)
|
|
|
|
{
|
|
|
|
if (type->isNullable())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (const auto * lc_type = typeid_cast<const DataTypeLowCardinality *>(type.get()))
|
|
|
|
return lc_type->getDictionaryType()->isNullable();
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns `true` if there are nullable column in src but corresponding column in dst is not
|
|
|
|
bool changedNullabilityOneWay(const Block & src_block, const Block & dst_block)
|
|
|
|
{
|
|
|
|
std::unordered_map<String, bool> src_nullable;
|
|
|
|
for (const auto & col : src_block)
|
|
|
|
src_nullable[col.name] = isNullableOrLcNullable(col.type);
|
|
|
|
|
|
|
|
for (const auto & col : dst_block)
|
|
|
|
{
|
|
|
|
if (!isNullableOrLcNullable(col.type) && src_nullable[col.name])
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-06 10:48:47 +00:00
|
|
|
bool hasJoin(const ASTSelectQuery & select)
|
|
|
|
{
|
|
|
|
const auto & tables = select.tables();
|
|
|
|
if (!tables || tables->children.size() < 2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const auto & joined_table = tables->children[1]->as<ASTTablesInSelectQueryElement &>();
|
|
|
|
return joined_table.table_join != nullptr;
|
|
|
|
}
|
|
|
|
|
2021-08-06 09:15:09 +00:00
|
|
|
bool hasJoin(const ASTSelectWithUnionQuery & ast)
|
|
|
|
{
|
|
|
|
for (const auto & child : ast.list_of_selects->children)
|
|
|
|
{
|
2021-08-06 10:48:47 +00:00
|
|
|
if (const auto * select = child->as<ASTSelectQuery>(); select && hasJoin(*select))
|
2021-08-06 09:15:09 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-05 15:38:52 +00:00
|
|
|
}
|
2013-10-30 13:52:02 +00:00
|
|
|
|
2014-09-30 03:08:47 +00:00
|
|
|
StorageView::StorageView(
|
2019-12-04 16:06:55 +00:00
|
|
|
const StorageID & table_id_,
|
2017-09-17 18:49:43 +00:00
|
|
|
const ASTCreateQuery & query,
|
2021-05-13 14:21:55 +00:00
|
|
|
const ColumnsDescription & columns_,
|
2021-06-22 14:32:02 +00:00
|
|
|
const String & comment)
|
2019-12-04 16:06:55 +00:00
|
|
|
: IStorage(table_id_)
|
2013-11-08 17:43:03 +00:00
|
|
|
{
|
2020-06-19 15:39:41 +00:00
|
|
|
StorageInMemoryMetadata storage_metadata;
|
|
|
|
storage_metadata.setColumns(columns_);
|
2021-04-23 12:18:23 +00:00
|
|
|
storage_metadata.setComment(comment);
|
2019-08-24 21:20:20 +00:00
|
|
|
|
2017-09-19 19:18:34 +00:00
|
|
|
if (!query.select)
|
|
|
|
throw Exception("SELECT query is not specified for " + getName(), ErrorCodes::INCORRECT_QUERY);
|
2020-06-05 11:54:54 +00:00
|
|
|
SelectQueryDescription description;
|
|
|
|
|
|
|
|
description.inner_query = query.select->ptr();
|
2022-09-23 11:35:22 +00:00
|
|
|
is_parameterized_view = query.isParameterizedView();
|
2020-06-19 15:39:41 +00:00
|
|
|
storage_metadata.setSelectQuery(description);
|
|
|
|
setInMemoryMetadata(storage_metadata);
|
2013-11-08 17:43:03 +00:00
|
|
|
}
|
|
|
|
|
2020-09-17 13:22:24 +00:00
|
|
|
void StorageView::read(
|
|
|
|
QueryPlan & query_plan,
|
|
|
|
const Names & column_names,
|
2021-07-09 03:15:41 +00:00
|
|
|
const StorageSnapshotPtr & storage_snapshot,
|
2020-11-10 12:02:22 +00:00
|
|
|
SelectQueryInfo & query_info,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context,
|
2020-09-17 13:22:24 +00:00
|
|
|
QueryProcessingStage::Enum /*processed_stage*/,
|
|
|
|
const size_t /*max_block_size*/,
|
2022-10-07 10:46:45 +00:00
|
|
|
const size_t /*num_streams*/)
|
2020-09-17 13:22:24 +00:00
|
|
|
{
|
2021-07-09 03:15:41 +00:00
|
|
|
ASTPtr current_inner_query = storage_snapshot->metadata->getSelectQuery().inner_query;
|
2020-09-17 13:22:24 +00:00
|
|
|
|
|
|
|
if (query_info.view_query)
|
|
|
|
{
|
|
|
|
if (!query_info.view_query->as<ASTSelectWithUnionQuery>())
|
|
|
|
throw Exception("Unexpected optimized VIEW query", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
current_inner_query = query_info.view_query->clone();
|
|
|
|
}
|
|
|
|
|
2022-01-11 12:19:41 +00:00
|
|
|
auto options = SelectQueryOptions(QueryProcessingStage::Complete, 0, false, query_info.settings_limit_offset_done);
|
|
|
|
InterpreterSelectWithUnionQuery interpreter(current_inner_query, context, options, column_names);
|
2022-05-31 14:43:38 +00:00
|
|
|
interpreter.addStorageLimits(*query_info.storage_limits);
|
2020-09-17 13:22:24 +00:00
|
|
|
interpreter.buildQueryPlan(query_plan);
|
|
|
|
|
|
|
|
/// It's expected that the columns read from storage are not constant.
|
|
|
|
/// Because method 'getSampleBlockForColumns' is used to obtain a structure of result in InterpreterSelectQuery.
|
2021-02-18 11:15:16 +00:00
|
|
|
auto materializing_actions = std::make_shared<ActionsDAG>(query_plan.getCurrentDataStream().header.getColumnsWithTypeAndName());
|
|
|
|
materializing_actions->addMaterializingOutputActions();
|
|
|
|
|
2021-03-04 17:38:12 +00:00
|
|
|
auto materializing = std::make_unique<ExpressionStep>(query_plan.getCurrentDataStream(), std::move(materializing_actions));
|
2020-09-17 13:22:24 +00:00
|
|
|
materializing->setStepDescription("Materialize constants after VIEW subquery");
|
|
|
|
query_plan.addStep(std::move(materializing));
|
|
|
|
|
|
|
|
/// And also convert to expected structure.
|
2021-08-20 13:33:30 +00:00
|
|
|
const auto & expected_header = storage_snapshot->getSampleBlockForColumns(column_names);
|
2021-08-05 15:38:52 +00:00
|
|
|
const auto & header = query_plan.getCurrentDataStream().header;
|
|
|
|
|
2021-08-06 09:15:09 +00:00
|
|
|
const auto * select_with_union = current_inner_query->as<ASTSelectWithUnionQuery>();
|
|
|
|
if (select_with_union && hasJoin(*select_with_union) && changedNullabilityOneWay(header, expected_header))
|
2021-08-05 15:38:52 +00:00
|
|
|
{
|
|
|
|
throw DB::Exception(ErrorCodes::INCORRECT_QUERY,
|
2021-08-06 09:15:09 +00:00
|
|
|
"Query from view {} returned Nullable column having not Nullable type in structure. "
|
2021-09-22 18:33:15 +00:00
|
|
|
"If query from view has JOIN, it may be cause by different values of 'join_use_nulls' setting. "
|
|
|
|
"You may explicitly specify 'join_use_nulls' in 'CREATE VIEW' query to avoid this error",
|
2021-08-06 09:15:09 +00:00
|
|
|
getStorageID().getFullTableName());
|
2021-08-05 15:38:52 +00:00
|
|
|
}
|
2021-08-06 09:15:09 +00:00
|
|
|
|
2020-11-17 17:16:55 +00:00
|
|
|
auto convert_actions_dag = ActionsDAG::makeConvertingActions(
|
|
|
|
header.getColumnsWithTypeAndName(),
|
2021-08-05 15:38:52 +00:00
|
|
|
expected_header.getColumnsWithTypeAndName(),
|
2020-11-17 17:16:55 +00:00
|
|
|
ActionsDAG::MatchColumnsMode::Name);
|
|
|
|
|
2021-03-04 17:38:12 +00:00
|
|
|
auto converting = std::make_unique<ExpressionStep>(query_plan.getCurrentDataStream(), convert_actions_dag);
|
2020-09-17 13:22:24 +00:00
|
|
|
converting->setStepDescription("Convert VIEW subquery result to VIEW table structure");
|
|
|
|
query_plan.addStep(std::move(converting));
|
|
|
|
}
|
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
static ASTTableExpression * getFirstTableExpression(ASTSelectQuery & select_query)
|
2020-03-18 00:57:00 +00:00
|
|
|
{
|
2021-02-25 14:43:58 +00:00
|
|
|
if (!select_query.tables() || select_query.tables()->children.empty())
|
|
|
|
throw Exception("Logical error: no table expression in view select AST", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
auto * select_element = select_query.tables()->children[0]->as<ASTTablesInSelectQueryElement>();
|
2020-03-18 00:57:00 +00:00
|
|
|
|
|
|
|
if (!select_element->table_expression)
|
|
|
|
throw Exception("Logical error: incorrect table expression", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
return select_element->table_expression->as<ASTTableExpression>();
|
|
|
|
}
|
|
|
|
|
2022-09-27 14:30:59 +00:00
|
|
|
void StorageView::replaceQueryParametersIfParametrizedView(ASTPtr & outer_query, const NameToNameMap & parameter_values) const
|
2022-09-23 11:35:22 +00:00
|
|
|
{
|
|
|
|
if (is_parameterized_view)
|
|
|
|
{
|
|
|
|
ReplaceQueryParameterVisitor visitor(parameter_values);
|
|
|
|
visitor.visit(outer_query);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-19 16:30:03 +00:00
|
|
|
void StorageView::replaceWithSubquery(ASTSelectQuery & outer_query, ASTPtr view_query, ASTPtr & view_name, bool parameterized_view)
|
2020-06-15 12:36:10 +00:00
|
|
|
{
|
|
|
|
ASTTableExpression * table_expression = getFirstTableExpression(outer_query);
|
2020-03-18 00:57:00 +00:00
|
|
|
|
|
|
|
if (!table_expression->database_and_table_name)
|
2020-08-28 14:07:14 +00:00
|
|
|
{
|
2022-10-19 16:30:03 +00:00
|
|
|
/// If it's a view or merge table function, add a fake db.table name.
|
|
|
|
/// For parameterized view, the function name is the db.view name, so add the function name
|
2022-07-09 15:51:59 +00:00
|
|
|
if (table_expression->table_function)
|
|
|
|
{
|
|
|
|
auto table_function_name = table_expression->table_function->as<ASTFunction>()->name;
|
2022-09-01 10:20:00 +00:00
|
|
|
if (table_function_name == "view" || table_function_name == "viewIfPermitted")
|
2022-07-09 15:51:59 +00:00
|
|
|
table_expression->database_and_table_name = std::make_shared<ASTTableIdentifier>("__view");
|
2022-09-23 11:35:22 +00:00
|
|
|
else if (table_function_name == "merge")
|
2022-09-01 10:20:00 +00:00
|
|
|
table_expression->database_and_table_name = std::make_shared<ASTTableIdentifier>("__merge");
|
2022-10-19 16:30:03 +00:00
|
|
|
else if (parameterized_view)
|
2022-09-23 11:35:22 +00:00
|
|
|
table_expression->database_and_table_name = std::make_shared<ASTTableIdentifier>(table_function_name);
|
|
|
|
|
2022-07-09 15:51:59 +00:00
|
|
|
}
|
|
|
|
if (!table_expression->database_and_table_name)
|
2020-08-28 14:07:14 +00:00
|
|
|
throw Exception("Logical error: incorrect table expression", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
2020-03-18 00:57:00 +00:00
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
DatabaseAndTableWithAlias db_table(table_expression->database_and_table_name);
|
|
|
|
String alias = db_table.alias.empty() ? db_table.table : db_table.alias;
|
|
|
|
|
|
|
|
view_name = table_expression->database_and_table_name;
|
2020-03-18 00:57:00 +00:00
|
|
|
table_expression->database_and_table_name = {};
|
|
|
|
table_expression->subquery = std::make_shared<ASTSubquery>();
|
2020-06-15 12:36:10 +00:00
|
|
|
table_expression->subquery->children.push_back(view_query);
|
|
|
|
table_expression->subquery->setAlias(alias);
|
2020-03-18 00:57:00 +00:00
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
for (auto & child : table_expression->children)
|
|
|
|
if (child.get() == view_name.get())
|
|
|
|
child = view_query;
|
2022-09-27 14:30:59 +00:00
|
|
|
else if (child.get()
|
|
|
|
&& child->as<ASTFunction>()
|
|
|
|
&& table_expression->table_function
|
|
|
|
&& table_expression->table_function->as<ASTFunction>()
|
|
|
|
&& child->as<ASTFunction>()->name == table_expression->table_function->as<ASTFunction>()->name)
|
2022-09-23 11:35:22 +00:00
|
|
|
child = view_query;
|
2020-06-15 12:36:10 +00:00
|
|
|
}
|
2020-03-18 00:57:00 +00:00
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
ASTPtr StorageView::restoreViewName(ASTSelectQuery & select_query, const ASTPtr & view_name)
|
2020-02-05 08:22:25 +00:00
|
|
|
{
|
2020-06-15 12:36:10 +00:00
|
|
|
ASTTableExpression * table_expression = getFirstTableExpression(select_query);
|
2020-02-05 08:22:25 +00:00
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
if (!table_expression->subquery)
|
|
|
|
throw Exception("Logical error: incorrect table expression", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
|
|
|
|
ASTPtr subquery = table_expression->subquery;
|
|
|
|
table_expression->subquery = {};
|
|
|
|
table_expression->database_and_table_name = view_name;
|
2020-02-05 08:22:25 +00:00
|
|
|
|
2020-06-15 12:36:10 +00:00
|
|
|
for (auto & child : table_expression->children)
|
|
|
|
if (child.get() == subquery.get())
|
|
|
|
child = view_name;
|
|
|
|
return subquery->children[0];
|
2020-02-05 08:22:25 +00:00
|
|
|
}
|
|
|
|
|
2017-12-30 00:36:06 +00:00
|
|
|
void registerStorageView(StorageFactory & factory)
|
|
|
|
{
|
|
|
|
factory.registerStorage("View", [](const StorageFactory::Arguments & args)
|
|
|
|
{
|
|
|
|
if (args.query.storage)
|
|
|
|
throw Exception("Specifying ENGINE is not allowed for a View", ErrorCodes::INCORRECT_QUERY);
|
|
|
|
|
2022-04-19 20:47:29 +00:00
|
|
|
return std::make_shared<StorageView>(args.table_id, args.query, args.columns, args.comment);
|
2017-12-30 00:36:06 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-10-30 13:52:02 +00:00
|
|
|
}
|