mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-12 01:12:12 +00:00
204 lines
7.8 KiB
C++
204 lines
7.8 KiB
C++
#pragma once
|
|
|
|
#include <memory>
|
|
|
|
#include <Core/QueryProcessingStage.h>
|
|
#include <Parsers/ASTSelectQuery.h>
|
|
#include <DataStreams/IBlockStream_fwd.h>
|
|
#include <Interpreters/ExpressionActions.h>
|
|
#include <Interpreters/ExpressionAnalyzer.h>
|
|
#include <Interpreters/IInterpreter.h>
|
|
#include <Interpreters/SelectQueryOptions.h>
|
|
#include <Storages/SelectQueryInfo.h>
|
|
#include <Storages/TableLockHolder.h>
|
|
#include <Storages/ReadInOrderOptimizer.h>
|
|
#include <Interpreters/StorageID.h>
|
|
|
|
#include <Columns/FilterDescription.h>
|
|
|
|
namespace Poco { class Logger; }
|
|
|
|
namespace DB
|
|
{
|
|
|
|
struct SubqueryForSet;
|
|
class InterpreterSelectWithUnionQuery;
|
|
class Context;
|
|
class QueryPlan;
|
|
|
|
struct TreeRewriterResult;
|
|
using TreeRewriterResultPtr = std::shared_ptr<const TreeRewriterResult>;
|
|
|
|
|
|
/** Interprets the SELECT query. Returns the stream of blocks with the results of the query before `to_stage` stage.
|
|
*/
|
|
class InterpreterSelectQuery : public IInterpreter
|
|
{
|
|
public:
|
|
/**
|
|
* query_ptr
|
|
* - A query AST to interpret.
|
|
*
|
|
* required_result_column_names
|
|
* - don't calculate all columns except the specified ones from the query
|
|
* - it is used to remove calculation (and reading) of unnecessary columns from subqueries.
|
|
* empty means - use all columns.
|
|
*/
|
|
|
|
InterpreterSelectQuery(
|
|
const ASTPtr & query_ptr_,
|
|
const Context & context_,
|
|
const SelectQueryOptions &,
|
|
const Names & required_result_column_names_ = Names{});
|
|
|
|
/// Read data not from the table specified in the query, but from the prepared source `input`.
|
|
InterpreterSelectQuery(
|
|
const ASTPtr & query_ptr_,
|
|
const Context & context_,
|
|
const BlockInputStreamPtr & input_,
|
|
const SelectQueryOptions & = {});
|
|
|
|
/// Read data not from the table specified in the query, but from the prepared pipe `input`.
|
|
InterpreterSelectQuery(
|
|
const ASTPtr & query_ptr_,
|
|
const Context & context_,
|
|
Pipe input_pipe_,
|
|
const SelectQueryOptions & = {});
|
|
|
|
/// Read data not from the table specified in the query, but from the specified `storage_`.
|
|
InterpreterSelectQuery(
|
|
const ASTPtr & query_ptr_,
|
|
const Context & context_,
|
|
const StoragePtr & storage_,
|
|
const StorageMetadataPtr & metadata_snapshot_ = nullptr,
|
|
const SelectQueryOptions & = {});
|
|
|
|
~InterpreterSelectQuery() override;
|
|
|
|
/// Execute a query. Get the stream of blocks to read.
|
|
BlockIO execute() override;
|
|
|
|
/// Builds QueryPlan for current query.
|
|
void buildQueryPlan(QueryPlan & query_plan);
|
|
|
|
bool ignoreLimits() const override { return options.ignore_limits; }
|
|
bool ignoreQuota() const override { return options.ignore_quota; }
|
|
|
|
Block getSampleBlock();
|
|
|
|
void ignoreWithTotals();
|
|
|
|
ASTPtr getQuery() const { return query_ptr; }
|
|
|
|
size_t getMaxStreams() const { return max_streams; }
|
|
|
|
const SelectQueryInfo & getQueryInfo() const { return query_info; }
|
|
|
|
static void addEmptySourceToQueryPlan(QueryPlan & query_plan, const Block & source_header, const SelectQueryInfo & query_info);
|
|
|
|
private:
|
|
InterpreterSelectQuery(
|
|
const ASTPtr & query_ptr_,
|
|
const Context & context_,
|
|
const BlockInputStreamPtr & input_,
|
|
std::optional<Pipe> input_pipe,
|
|
const StoragePtr & storage_,
|
|
const SelectQueryOptions &,
|
|
const Names & required_result_column_names = {},
|
|
const StorageMetadataPtr & metadata_snapshot_= nullptr);
|
|
|
|
ASTSelectQuery & getSelectQuery() { return query_ptr->as<ASTSelectQuery &>(); }
|
|
|
|
Block getSampleBlockImpl();
|
|
|
|
void executeImpl(QueryPlan & query_plan, const BlockInputStreamPtr & prepared_input, std::optional<Pipe> prepared_pipe);
|
|
|
|
/// Different stages of query execution.
|
|
|
|
void executeFetchColumns(
|
|
QueryProcessingStage::Enum processing_stage,
|
|
QueryPlan & query_plan,
|
|
const PrewhereInfoPtr & prewhere_info,
|
|
const Names & columns_to_remove_after_prewhere);
|
|
|
|
void executeWhere(QueryPlan & query_plan, const ExpressionActionsPtr & expression, bool remove_filter);
|
|
void executeAggregation(QueryPlan & query_plan, const ExpressionActionsPtr & expression, bool overflow_row, bool final, InputOrderInfoPtr group_by_info);
|
|
void executeMergeAggregated(QueryPlan & query_plan, bool overflow_row, bool final);
|
|
void executeTotalsAndHaving(QueryPlan & query_plan, bool has_having, const ExpressionActionsPtr & expression, bool overflow_row, bool final);
|
|
void executeHaving(QueryPlan & query_plan, const ExpressionActionsPtr & expression);
|
|
static void executeExpression(QueryPlan & query_plan, const ExpressionActionsPtr & expression, const std::string & description);
|
|
void executeOrder(QueryPlan & query_plan, InputOrderInfoPtr sorting_info);
|
|
void executeOrderOptimized(QueryPlan & query_plan, InputOrderInfoPtr sorting_info, UInt64 limit, SortDescription & output_order_descr);
|
|
void executeWithFill(QueryPlan & query_plan);
|
|
void executeMergeSorted(QueryPlan & query_plan, const std::string & description);
|
|
void executePreLimit(QueryPlan & query_plan, bool do_not_skip_offset);
|
|
void executeLimitBy(QueryPlan & query_plan);
|
|
void executeLimit(QueryPlan & query_plan);
|
|
void executeOffset(QueryPlan & query_plan);
|
|
static void executeProjection(QueryPlan & query_plan, const ExpressionActionsPtr & expression);
|
|
void executeDistinct(QueryPlan & query_plan, bool before_order, Names columns, bool pre_distinct);
|
|
void executeExtremes(QueryPlan & query_plan);
|
|
void executeSubqueriesInSetsAndJoins(QueryPlan & query_plan, std::unordered_map<String, SubqueryForSet> & subqueries_for_sets);
|
|
void executeMergeSorted(QueryPlan & query_plan, const SortDescription & sort_description, UInt64 limit, const std::string & description);
|
|
|
|
String generateFilterActions(
|
|
ExpressionActionsPtr & actions, const ASTPtr & row_policy_filter, const Names & prerequisite_columns = {}) const;
|
|
|
|
enum class Modificator
|
|
{
|
|
ROLLUP = 0,
|
|
CUBE = 1
|
|
};
|
|
|
|
void executeRollupOrCube(QueryPlan & query_plan, Modificator modificator);
|
|
|
|
/** If there is a SETTINGS section in the SELECT query, then apply settings from it.
|
|
*
|
|
* Section SETTINGS - settings for a specific query.
|
|
* Normally, the settings can be passed in other ways, not inside the query.
|
|
* But the use of this section is justified if you need to set the settings for one subquery.
|
|
*/
|
|
void initSettings();
|
|
|
|
SelectQueryOptions options;
|
|
ASTPtr query_ptr;
|
|
std::shared_ptr<Context> context;
|
|
TreeRewriterResultPtr syntax_analyzer_result;
|
|
std::unique_ptr<SelectQueryExpressionAnalyzer> query_analyzer;
|
|
SelectQueryInfo query_info;
|
|
|
|
/// Is calculated in getSampleBlock. Is used later in readImpl.
|
|
ExpressionAnalysisResult analysis_result;
|
|
/// For row-level security.
|
|
FilterInfoPtr filter_info;
|
|
|
|
QueryProcessingStage::Enum from_stage = QueryProcessingStage::FetchColumns;
|
|
|
|
/// How many streams we ask for storage to produce, and in how many threads we will do further processing.
|
|
size_t max_streams = 1;
|
|
|
|
/// List of columns to read to execute the query.
|
|
Names required_columns;
|
|
/// Structure of query source (table, subquery, etc).
|
|
Block source_header;
|
|
/// Structure of query result.
|
|
Block result_header;
|
|
|
|
/// The subquery interpreter, if the subquery
|
|
std::unique_ptr<InterpreterSelectWithUnionQuery> interpreter_subquery;
|
|
|
|
/// Table from where to read data, if not subquery.
|
|
StoragePtr storage;
|
|
StorageID table_id = StorageID::createEmpty(); /// Will be initialized if storage is not nullptr
|
|
TableLockHolder table_lock;
|
|
|
|
/// Used when we read from prepared input, not table or subquery.
|
|
BlockInputStreamPtr input;
|
|
std::optional<Pipe> input_pipe;
|
|
|
|
Poco::Logger * log;
|
|
StorageMetadataPtr metadata_snapshot;
|
|
};
|
|
|
|
}
|