mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-15 02:41:59 +00:00
107 lines
4.0 KiB
C++
107 lines
4.0 KiB
C++
/* Copyright (c) 2018 BlackBerry Limited
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License. */
|
|
#include <Core/Settings.h>
|
|
#include <Common/typeid_cast.h>
|
|
#include <Parsers/ASTWatchQuery.h>
|
|
#include <Interpreters/InterpreterWatchQuery.h>
|
|
#include <Interpreters/Context.h>
|
|
#include <Access/Common/AccessFlags.h>
|
|
#include <QueryPipeline/StreamLocalLimits.h>
|
|
#include <QueryPipeline/QueryPipelineBuilder.h>
|
|
#include <Storages/IStorage.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
{
|
|
extern const int UNKNOWN_TABLE;
|
|
extern const int TOO_MANY_COLUMNS;
|
|
extern const int SUPPORT_IS_DISABLED;
|
|
}
|
|
|
|
|
|
BlockIO InterpreterWatchQuery::execute()
|
|
{
|
|
BlockIO res;
|
|
res.pipeline = QueryPipelineBuilder::getPipeline(buildQueryPipeline());
|
|
|
|
/// Constraints on the result, the quota on the result, and also callback for progress.
|
|
{
|
|
const Settings & settings = getContext()->getSettingsRef();
|
|
|
|
StreamLocalLimits limits;
|
|
limits.mode = LimitsMode::LIMITS_CURRENT;
|
|
limits.size_limits.max_rows = settings.max_result_rows;
|
|
limits.size_limits.max_bytes = settings.max_result_bytes;
|
|
limits.size_limits.overflow_mode = settings.result_overflow_mode;
|
|
|
|
res.pipeline.setLimitsAndQuota(limits, getContext()->getQuota());
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
QueryPipelineBuilder InterpreterWatchQuery::buildQueryPipeline()
|
|
{
|
|
const ASTWatchQuery & query = typeid_cast<const ASTWatchQuery &>(*query_ptr);
|
|
auto table_id = getContext()->resolveStorageID(query, Context::ResolveOrdinary);
|
|
|
|
/// Get storage
|
|
storage = DatabaseCatalog::instance().tryGetTable(table_id, getContext());
|
|
|
|
if (!storage)
|
|
throw Exception(ErrorCodes::UNKNOWN_TABLE, "Table {} doesn't exist.", table_id.getNameForLogs());
|
|
|
|
auto storage_name = storage->getName();
|
|
if (storage_name == "LiveView"
|
|
&& !getContext()->getSettingsRef().allow_experimental_live_view)
|
|
throw Exception(ErrorCodes::SUPPORT_IS_DISABLED,
|
|
"Experimental LIVE VIEW feature is not enabled (the setting 'allow_experimental_live_view')");
|
|
else if (storage_name == "WindowView"
|
|
&& !getContext()->getSettingsRef().allow_experimental_window_view)
|
|
throw Exception(ErrorCodes::SUPPORT_IS_DISABLED,
|
|
"Experimental WINDOW VIEW feature is not enabled (the setting 'allow_experimental_window_view')");
|
|
|
|
/// List of columns to read to execute the query.
|
|
Names required_columns = storage->getInMemoryMetadataPtr()->getColumns().getNamesOfPhysical();
|
|
getContext()->checkAccess(AccessType::SELECT, table_id, required_columns);
|
|
|
|
/// Get context settings for this query
|
|
const Settings & settings = getContext()->getSettingsRef();
|
|
|
|
/// Limitation on the number of columns to read.
|
|
if (settings.max_columns_to_read && required_columns.size() > settings.max_columns_to_read)
|
|
throw Exception(ErrorCodes::TOO_MANY_COLUMNS, "Limit for number of columns to read exceeded. "
|
|
"Requested: {}, maximum: {}", required_columns.size(), settings.max_columns_to_read.toString());
|
|
|
|
size_t max_block_size = settings.max_block_size;
|
|
size_t max_streams = 1;
|
|
|
|
/// Define query info
|
|
SelectQueryInfo query_info;
|
|
query_info.query = query_ptr;
|
|
|
|
/// From stage
|
|
QueryProcessingStage::Enum from_stage = QueryProcessingStage::FetchColumns;
|
|
|
|
/// Watch storage
|
|
auto pipe = storage->watch(required_columns, query_info, getContext(), from_stage, max_block_size, max_streams);
|
|
|
|
QueryPipelineBuilder pipeline;
|
|
pipeline.init(std::move(pipe));
|
|
return pipeline;
|
|
}
|
|
|
|
}
|