ClickHouse/src/Planner/PlannerCollectSets.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

105 lines
3.6 KiB
C++
Raw Normal View History

#include <Planner/PlannerCollectSets.h>
#include <Interpreters/Context.h>
2022-08-27 14:32:50 +00:00
#include <Storages/StorageSet.h>
#include <Analyzer/Utils.h>
#include <Analyzer/SetUtils.h>
#include <Analyzer/InDepthQueryTreeVisitor.h>
#include <Analyzer/ConstantNode.h>
#include <Analyzer/FunctionNode.h>
2022-08-27 14:32:50 +00:00
#include <Analyzer/TableNode.h>
namespace DB
{
namespace
{
class CollectSetsMatcher
{
public:
using Visitor = ConstInDepthQueryTreeVisitor<CollectSetsMatcher, true, false>;
struct Data
{
const PlannerContext & planner_context;
};
static void visit(const QueryTreeNodePtr & node, Data & data)
{
auto * function_node = node->as<FunctionNode>();
if (!function_node || !isNameOfInFunction(function_node->getFunctionName()))
return;
auto in_first_argument = function_node->getArguments().getNodes().at(0);
auto in_second_argument = function_node->getArguments().getNodes().at(1);
auto in_second_argument_node_type = in_second_argument->getNodeType();
const auto & planner_context = data.planner_context;
const auto & global_planner_context = planner_context.getGlobalPlannerContext();
const auto & settings = planner_context.getQueryContext()->getSettingsRef();
2022-08-31 15:21:17 +00:00
String set_key = global_planner_context->getSetKey(in_second_argument);
auto prepared_set = global_planner_context->getSetOrNull(set_key);
if (prepared_set)
return;
2022-08-27 14:32:50 +00:00
/// Tables and table functions are replaced with subquery at Analysis stage, except special Set table.
auto * second_argument_table = in_second_argument->as<TableNode>();
StorageSet * storage_set = second_argument_table != nullptr ? dynamic_cast<StorageSet *>(second_argument_table->getStorage().get()) : nullptr;
if (storage_set)
{
global_planner_context->registerSet(set_key, storage_set->getSet());
}
2022-08-31 15:21:17 +00:00
else if (auto constant_value = in_second_argument->getConstantValueOrNull())
{
auto set = makeSetForConstantValue(
in_first_argument->getResultType(),
constant_value->getValue(),
constant_value->getType(),
settings);
global_planner_context->registerSet(set_key, std::move(set));
}
2022-08-27 14:32:50 +00:00
else if (in_second_argument_node_type == QueryTreeNodeType::QUERY ||
in_second_argument_node_type == QueryTreeNodeType::UNION)
{
SizeLimits size_limits_for_set = {settings.max_rows_in_set, settings.max_bytes_in_set, settings.set_overflow_mode};
bool tranform_null_in = settings.transform_null_in;
auto set = std::make_shared<Set>(size_limits_for_set, false /*fill_set_elements*/, tranform_null_in);
global_planner_context->registerSet(set_key, set);
global_planner_context->registerSubqueryNodeForSet(set_key, SubqueryNodeForSet{in_second_argument, set});
}
else
{
throw Exception(ErrorCodes::UNSUPPORTED_METHOD,
"Function IN is supported only if second argument is constant or table expression");
}
}
static bool needChildVisit(const QueryTreeNodePtr &, const QueryTreeNodePtr & child_node)
{
2022-08-31 15:21:17 +00:00
return !(child_node->getNodeType() == QueryTreeNodeType::QUERY ||
child_node->getNodeType() == QueryTreeNodeType::UNION);
}
};
using CollectSetsVisitor = CollectSetsMatcher::Visitor;
}
void collectSets(const QueryTreeNodePtr & node, const PlannerContext & planner_context)
{
CollectSetsVisitor::Data data {planner_context};
CollectSetsVisitor visitor(data);
visitor.visit(node);
}
}