2018-11-27 00:43:58 +00:00
|
|
|
#include <Interpreters/addMissingDefaults.h>
|
|
|
|
|
|
|
|
#include <Common/typeid_cast.h>
|
|
|
|
#include <DataTypes/NestedUtils.h>
|
|
|
|
#include <DataTypes/DataTypeArray.h>
|
|
|
|
#include <Columns/ColumnArray.h>
|
2020-02-17 15:44:13 +00:00
|
|
|
#include <Interpreters/inplaceBlockConversions.h>
|
2018-11-27 00:43:58 +00:00
|
|
|
#include <Core/Block.h>
|
2020-10-02 12:38:50 +00:00
|
|
|
#include <Storages/ColumnsDescription.h>
|
2021-02-04 20:36:50 +00:00
|
|
|
#include <Interpreters/ExpressionActions.h>
|
2021-02-05 11:41:44 +00:00
|
|
|
#include <Functions/IFunctionAdaptors.h>
|
|
|
|
#include <Functions/materialize.h>
|
2018-11-27 00:43:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
ActionsDAGPtr addMissingDefaults(
|
|
|
|
const Block & header,
|
2020-10-02 12:38:50 +00:00
|
|
|
const NamesAndTypesList & required_columns,
|
|
|
|
const ColumnsDescription & columns,
|
|
|
|
const Context & context)
|
2018-11-27 00:43:58 +00:00
|
|
|
{
|
|
|
|
/// For missing columns of nested structure, you need to create not a column of empty arrays, but a column of arrays of correct lengths.
|
|
|
|
/// First, remember the offset columns for all arrays in the block.
|
2021-02-05 11:41:44 +00:00
|
|
|
std::map<String, Names> nested_groups;
|
2018-11-27 00:43:58 +00:00
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
for (size_t i = 0, size = header.columns(); i < size; ++i)
|
2018-11-27 00:43:58 +00:00
|
|
|
{
|
2021-02-05 11:41:44 +00:00
|
|
|
const auto & elem = header.getByPosition(i);
|
2018-11-27 00:43:58 +00:00
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
if (typeid_cast<const ColumnArray *>(&*elem.column))
|
2018-11-27 00:43:58 +00:00
|
|
|
{
|
|
|
|
String offsets_name = Nested::extractTableName(elem.name);
|
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
auto & group = nested_groups[offsets_name];
|
|
|
|
if (group.empty())
|
|
|
|
group.push_back({});
|
|
|
|
|
|
|
|
group.push_back(elem.name);
|
2018-11-27 00:43:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
auto actions = std::make_shared<ActionsDAG>(header.getColumnsWithTypeAndName());
|
|
|
|
|
2021-02-10 12:45:39 +00:00
|
|
|
FunctionOverloadResolverPtr func_builder_replicate = FunctionFactory::instance().get("replicate", context);
|
2018-11-27 00:43:58 +00:00
|
|
|
|
|
|
|
/// We take given columns from input block and missed columns without default value
|
|
|
|
/// (default and materialized will be computed later).
|
|
|
|
for (const auto & column : required_columns)
|
|
|
|
{
|
2021-02-05 11:41:44 +00:00
|
|
|
if (header.has(column.name))
|
2021-02-05 18:21:48 +00:00
|
|
|
continue;
|
2018-11-27 00:43:58 +00:00
|
|
|
|
2020-10-02 12:38:50 +00:00
|
|
|
if (columns.hasDefault(column.name))
|
2018-11-27 00:43:58 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
String offsets_name = Nested::extractTableName(column.name);
|
2021-02-05 11:41:44 +00:00
|
|
|
if (nested_groups.count(offsets_name))
|
2018-11-27 00:43:58 +00:00
|
|
|
{
|
2021-02-05 11:41:44 +00:00
|
|
|
|
2018-11-27 00:43:58 +00:00
|
|
|
DataTypePtr nested_type = typeid_cast<const DataTypeArray &>(*column.type).getNestedType();
|
2021-02-05 11:41:44 +00:00
|
|
|
ColumnPtr nested_column = nested_type->createColumnConstWithDefaultValue(0);
|
|
|
|
const auto & constant = actions->addColumn({std::move(nested_column), nested_type, column.name}, true);
|
2018-11-27 00:43:58 +00:00
|
|
|
|
2021-02-05 11:41:44 +00:00
|
|
|
auto & group = nested_groups[offsets_name];
|
|
|
|
group[0] = constant.result_name;
|
2021-02-05 19:22:11 +00:00
|
|
|
actions->addFunction(func_builder_replicate, group, constant.result_name, context, true);
|
2021-02-05 11:41:44 +00:00
|
|
|
|
2018-11-27 00:43:58 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** It is necessary to turn a constant column into a full column, since in part of blocks (from other parts),
|
|
|
|
* it can be full (or the interpreter may decide that it is constant everywhere).
|
|
|
|
*/
|
2021-02-05 16:35:21 +00:00
|
|
|
auto new_column = column.type->createColumnConstWithDefaultValue(0);
|
|
|
|
actions->addColumn({std::move(new_column), column.type, column.name}, true, true);
|
2018-11-27 00:43:58 +00:00
|
|
|
}
|
|
|
|
|
2020-10-02 12:38:50 +00:00
|
|
|
/// Computes explicitly specified values by default and materialized columns.
|
2021-02-05 18:19:41 +00:00
|
|
|
if (auto dag = evaluateMissingDefaults(actions->getResultColumns(), required_columns, columns, context))
|
2021-02-05 11:41:44 +00:00
|
|
|
actions = ActionsDAG::merge(std::move(*actions), std::move(*dag));
|
2021-02-09 19:48:34 +00:00
|
|
|
else
|
2021-02-09 19:52:48 +00:00
|
|
|
/// Removes unused columns and reorders result.
|
|
|
|
/// The same is done in evaluateMissingDefaults if not empty dag is returned.
|
2021-02-09 19:48:34 +00:00
|
|
|
actions->removeUnusedActions(required_columns.getNames());
|
2021-02-05 11:41:44 +00:00
|
|
|
|
|
|
|
return actions;
|
2018-11-27 00:43:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|