2017-04-01 09:19:00 +00:00
|
|
|
#include <Common/formatReadable.h>
|
2019-10-07 18:56:03 +00:00
|
|
|
#include <Common/PODArray.h>
|
2017-07-13 20:58:19 +00:00
|
|
|
#include <Common/typeid_cast.h>
|
2020-08-18 21:41:01 +00:00
|
|
|
#include <Common/ThreadProfileEvents.h>
|
2022-01-10 19:39:10 +00:00
|
|
|
#include <Common/MemoryTrackerBlockerInThread.h>
|
2022-11-11 15:26:04 +00:00
|
|
|
#include <Common/SensitiveDataMasker.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
|
2021-09-01 23:18:09 +00:00
|
|
|
#include <Interpreters/AsynchronousInsertQueue.h>
|
2023-06-17 12:27:44 +00:00
|
|
|
#include <Interpreters/Cache/QueryCache.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <IO/WriteBufferFromFile.h>
|
2018-11-30 15:36:41 +00:00
|
|
|
#include <IO/WriteBufferFromVector.h>
|
|
|
|
#include <IO/LimitReadBuffer.h>
|
|
|
|
#include <IO/copyData.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
|
2021-10-15 20:18:20 +00:00
|
|
|
#include <QueryPipeline/BlockIO.h>
|
2021-10-07 08:26:08 +00:00
|
|
|
#include <Processors/Transforms/CountingTransform.h>
|
2021-08-08 03:30:14 +00:00
|
|
|
#include <Processors/Transforms/getSourceFromASTInsertQuery.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
|
2020-12-04 02:15:44 +00:00
|
|
|
#include <Parsers/ASTIdentifier.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTInsertQuery.h>
|
2020-12-04 02:15:44 +00:00
|
|
|
#include <Parsers/ASTLiteral.h>
|
2020-05-21 18:01:25 +00:00
|
|
|
#include <Parsers/ASTSelectQuery.h>
|
2020-12-02 12:08:03 +00:00
|
|
|
#include <Parsers/ASTDropQuery.h>
|
|
|
|
#include <Parsers/ASTCreateQuery.h>
|
|
|
|
#include <Parsers/ASTRenameQuery.h>
|
|
|
|
#include <Parsers/ASTAlterQuery.h>
|
2020-11-02 19:23:26 +00:00
|
|
|
#include <Parsers/ASTSelectWithUnionQuery.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Parsers/ASTShowProcesslistQuery.h>
|
2020-06-15 20:01:58 +00:00
|
|
|
#include <Parsers/ASTWatchQuery.h>
|
2021-05-17 11:14:09 +00:00
|
|
|
#include <Parsers/ASTTransactionControl.h>
|
2022-02-14 19:47:17 +00:00
|
|
|
#include <Parsers/ASTExplainQuery.h>
|
2020-06-15 20:01:58 +00:00
|
|
|
#include <Parsers/Lexer.h>
|
2020-12-04 02:15:44 +00:00
|
|
|
#include <Parsers/parseQuery.h>
|
|
|
|
#include <Parsers/ParserQuery.h>
|
2020-12-02 12:08:03 +00:00
|
|
|
#include <Parsers/queryNormalization.h>
|
2020-12-04 02:15:44 +00:00
|
|
|
#include <Parsers/queryToString.h>
|
2022-01-10 19:01:41 +00:00
|
|
|
#include <Parsers/formatAST.h>
|
2022-06-03 05:26:49 +00:00
|
|
|
#include <Parsers/toOneLineQuery.h>
|
2020-12-04 02:15:44 +00:00
|
|
|
|
2021-07-16 10:10:56 +00:00
|
|
|
#include <Formats/FormatFactory.h>
|
2019-05-28 18:30:10 +00:00
|
|
|
#include <Storages/StorageInput.h>
|
|
|
|
|
2020-03-07 17:37:38 +00:00
|
|
|
#include <Access/EnabledQuota.h>
|
2021-02-25 07:47:08 +00:00
|
|
|
#include <Interpreters/ApplyWithGlobalVisitor.h>
|
|
|
|
#include <Interpreters/Context.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Interpreters/InterpreterFactory.h>
|
2022-01-10 19:01:41 +00:00
|
|
|
#include <Interpreters/InterpreterInsertQuery.h>
|
2022-06-23 13:23:37 +00:00
|
|
|
#include <Interpreters/InterpreterSetQuery.h>
|
|
|
|
#include <Interpreters/InterpreterTransactionControlQuery.h>
|
2021-02-25 07:47:08 +00:00
|
|
|
#include <Interpreters/NormalizeSelectWithUnionQueryVisitor.h>
|
2020-10-22 16:47:20 +00:00
|
|
|
#include <Interpreters/OpenTelemetrySpanLog.h>
|
2021-02-25 07:47:08 +00:00
|
|
|
#include <Interpreters/ProcessList.h>
|
2022-02-06 11:21:05 +00:00
|
|
|
#include <Interpreters/ProcessorsProfileLog.h>
|
2022-06-23 13:23:37 +00:00
|
|
|
#include <Interpreters/QueryLog.h>
|
2019-05-18 21:07:23 +00:00
|
|
|
#include <Interpreters/ReplaceQueryParameterVisitor.h>
|
2022-06-23 13:23:37 +00:00
|
|
|
#include <Interpreters/SelectIntersectExceptQueryVisitor.h>
|
2020-12-18 06:54:38 +00:00
|
|
|
#include <Interpreters/SelectQueryOptions.h>
|
2021-04-09 12:53:51 +00:00
|
|
|
#include <Interpreters/TransactionLog.h>
|
2022-06-23 13:23:37 +00:00
|
|
|
#include <Interpreters/executeQuery.h>
|
2023-04-25 01:11:58 +00:00
|
|
|
#include <Interpreters/DatabaseCatalog.h>
|
2019-07-19 07:44:18 +00:00
|
|
|
#include <Common/ProfileEvents.h>
|
2019-07-19 13:50:11 +00:00
|
|
|
|
2021-09-19 20:15:10 +00:00
|
|
|
#include <IO/CompressionMethod.h>
|
2011-10-30 11:30:52 +00:00
|
|
|
|
2019-03-26 18:28:37 +00:00
|
|
|
#include <Processors/Transforms/LimitsCheckingTransform.h>
|
2019-04-12 13:56:48 +00:00
|
|
|
#include <Processors/Transforms/MaterializingTransform.h>
|
2019-03-26 18:28:37 +00:00
|
|
|
#include <Processors/Formats/IOutputFormat.h>
|
2021-09-15 19:35:48 +00:00
|
|
|
#include <Processors/Executors/CompletedPipelineExecutor.h>
|
2021-08-27 21:29:10 +00:00
|
|
|
#include <Processors/Sources/WaitForAsyncInsertSource.h>
|
2011-10-30 11:30:52 +00:00
|
|
|
|
2021-12-30 09:55:50 +00:00
|
|
|
#include <base/EnumReflection.h>
|
2022-01-27 10:45:20 +00:00
|
|
|
#include <base/demangle.h>
|
2021-12-30 09:55:50 +00:00
|
|
|
|
2022-06-23 13:23:37 +00:00
|
|
|
#include <memory>
|
2021-08-25 14:41:21 +00:00
|
|
|
#include <random>
|
2011-10-30 11:30:52 +00:00
|
|
|
|
2022-06-08 17:14:03 +00:00
|
|
|
#include <Parsers/Kusto/ParserKQLStatement.h>
|
2020-03-28 03:02:26 +00:00
|
|
|
|
2019-06-20 07:17:21 +00:00
|
|
|
namespace ProfileEvents
|
|
|
|
{
|
2020-05-21 18:01:25 +00:00
|
|
|
extern const Event FailedQuery;
|
|
|
|
extern const Event FailedInsertQuery;
|
|
|
|
extern const Event FailedSelectQuery;
|
2020-08-04 16:14:15 +00:00
|
|
|
extern const Event QueryTimeMicroseconds;
|
|
|
|
extern const Event SelectQueryTimeMicroseconds;
|
|
|
|
extern const Event InsertQueryTimeMicroseconds;
|
2022-05-02 15:13:57 +00:00
|
|
|
extern const Event OtherQueryTimeMicroseconds;
|
2019-06-20 07:17:21 +00:00
|
|
|
}
|
|
|
|
|
2011-10-30 11:30:52 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-01-11 19:05:46 +00:00
|
|
|
extern const int INTO_OUTFILE_NOT_ALLOWED;
|
2018-11-01 14:56:37 +00:00
|
|
|
extern const int QUERY_WAS_CANCELLED;
|
2021-05-17 11:14:09 +00:00
|
|
|
extern const int INVALID_TRANSACTION;
|
2021-07-01 16:43:59 +00:00
|
|
|
extern const int LOGICAL_ERROR;
|
2021-09-17 12:05:54 +00:00
|
|
|
extern const int NOT_IMPLEMENTED;
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
2011-10-30 11:30:52 +00:00
|
|
|
|
2018-03-11 00:15:26 +00:00
|
|
|
static void checkASTSizeLimits(const IAST & ast, const Settings & settings)
|
2012-12-26 20:29:28 +00:00
|
|
|
{
|
2018-03-11 00:15:26 +00:00
|
|
|
if (settings.max_ast_depth)
|
|
|
|
ast.checkDepth(settings.max_ast_depth);
|
|
|
|
if (settings.max_ast_elements)
|
|
|
|
ast.checkSize(settings.max_ast_elements);
|
2012-12-26 20:29:28 +00:00
|
|
|
}
|
2014-06-26 00:58:14 +00:00
|
|
|
|
2020-06-15 20:01:58 +00:00
|
|
|
|
2016-08-16 20:33:00 +00:00
|
|
|
/// Log query into text log (not into system table).
|
2022-05-17 11:32:32 +00:00
|
|
|
static void logQuery(const String & query, ContextPtr context, bool internal, QueryProcessingStage::Enum stage)
|
2015-06-17 21:34:15 +00:00
|
|
|
{
|
2019-03-04 18:28:42 +00:00
|
|
|
if (internal)
|
|
|
|
{
|
2022-06-03 05:26:49 +00:00
|
|
|
LOG_DEBUG(&Poco::Logger::get("executeQuery"), "(internal) {} (stage: {})", toOneLineQuery(query), QueryProcessingStage::toString(stage));
|
2019-03-04 18:28:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-04-10 23:33:54 +00:00
|
|
|
const auto & client_info = context->getClientInfo();
|
2020-09-08 13:19:27 +00:00
|
|
|
|
|
|
|
const auto & current_query_id = client_info.current_query_id;
|
|
|
|
const auto & initial_query_id = client_info.initial_query_id;
|
|
|
|
const auto & current_user = client_info.current_user;
|
2019-03-04 18:28:42 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
String comment = context->getSettingsRef().log_comment;
|
|
|
|
size_t max_query_size = context->getSettingsRef().max_query_size;
|
2021-01-25 19:18:23 +00:00
|
|
|
|
|
|
|
if (comment.size() > max_query_size)
|
|
|
|
comment.resize(max_query_size);
|
|
|
|
|
|
|
|
if (!comment.empty())
|
|
|
|
comment = fmt::format(" (comment: {})", comment);
|
|
|
|
|
2021-11-17 18:14:14 +00:00
|
|
|
String transaction_info;
|
|
|
|
if (auto txn = context->getCurrentTransaction())
|
|
|
|
transaction_info = fmt::format(" (TID: {}, TIDH: {})", txn->tid, txn->tid.getHash());
|
|
|
|
|
2022-05-17 11:32:32 +00:00
|
|
|
LOG_DEBUG(&Poco::Logger::get("executeQuery"), "(from {}{}{}){}{} {} (stage: {})",
|
2020-09-08 13:19:27 +00:00
|
|
|
client_info.current_address.toString(),
|
|
|
|
(current_user != "default" ? ", user: " + current_user : ""),
|
2020-05-23 21:41:35 +00:00
|
|
|
(!initial_query_id.empty() && current_query_id != initial_query_id ? ", initial_query_id: " + initial_query_id : std::string()),
|
2021-11-17 18:14:14 +00:00
|
|
|
transaction_info,
|
2021-01-25 19:18:23 +00:00
|
|
|
comment,
|
2022-06-03 05:26:49 +00:00
|
|
|
toOneLineQuery(query),
|
2022-05-17 11:32:32 +00:00
|
|
|
QueryProcessingStage::toString(stage));
|
2020-08-28 19:02:50 +00:00
|
|
|
|
2021-05-04 22:42:14 +00:00
|
|
|
if (client_info.client_trace_context.trace_id != UUID())
|
2020-09-08 13:19:27 +00:00
|
|
|
{
|
|
|
|
LOG_TRACE(&Poco::Logger::get("executeQuery"),
|
2020-11-18 17:43:18 +00:00
|
|
|
"OpenTelemetry traceparent '{}'",
|
|
|
|
client_info.client_trace_context.composeTraceparentHeader());
|
2020-09-08 13:19:27 +00:00
|
|
|
}
|
2019-03-04 18:28:42 +00:00
|
|
|
}
|
2016-08-16 20:33:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Call this inside catch block.
|
|
|
|
static void setExceptionStackTrace(QueryLogElement & elem)
|
|
|
|
{
|
2019-10-10 16:30:33 +00:00
|
|
|
/// Disable memory tracker for stack trace.
|
|
|
|
/// Because if exception is "Memory limit (for query) exceed", then we probably can't allocate another one string.
|
2022-07-26 15:22:00 +00:00
|
|
|
|
|
|
|
LockMemoryExceptionInThread lock(VariableContext::Global);
|
2019-10-10 16:30:33 +00:00
|
|
|
|
2016-08-16 20:33:00 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
throw;
|
|
|
|
}
|
2020-01-02 07:37:13 +00:00
|
|
|
catch (const std::exception & e)
|
2016-08-16 20:33:00 +00:00
|
|
|
{
|
2020-01-02 07:37:13 +00:00
|
|
|
elem.stack_trace = getExceptionStackTraceString(e);
|
2016-08-16 20:33:00 +00:00
|
|
|
}
|
|
|
|
catch (...) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Log exception (with query info) into text log (not into system table).
|
2023-02-25 00:18:34 +00:00
|
|
|
static void logException(ContextPtr context, QueryLogElement & elem, bool log_error = true)
|
2016-08-16 20:33:00 +00:00
|
|
|
{
|
2021-01-25 19:18:23 +00:00
|
|
|
String comment;
|
|
|
|
if (!elem.log_comment.empty())
|
|
|
|
comment = fmt::format(" (comment: {})", elem.log_comment);
|
|
|
|
|
2023-01-18 13:30:20 +00:00
|
|
|
/// Message patterns like "{} (from {}){} (in query: {})" are not really informative,
|
|
|
|
/// so we pass elem.exception_format_string as format string instead.
|
|
|
|
PreformattedMessage message;
|
|
|
|
message.format_string = elem.exception_format_string;
|
|
|
|
|
2023-02-25 00:18:34 +00:00
|
|
|
if (elem.stack_trace.empty() || !log_error)
|
2023-01-25 20:16:42 +00:00
|
|
|
message.text = fmt::format("{} (from {}){} (in query: {})", elem.exception,
|
2023-01-17 19:04:25 +00:00
|
|
|
context->getClientInfo().current_address.toString(),
|
|
|
|
comment,
|
|
|
|
toOneLineQuery(elem.query));
|
2020-05-23 21:41:35 +00:00
|
|
|
else
|
2023-01-25 20:16:42 +00:00
|
|
|
message.text = fmt::format(
|
2023-01-17 19:04:25 +00:00
|
|
|
"{} (from {}){} (in query: {}), Stack trace (when copying this message, always include the lines below):\n\n{}",
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.exception,
|
|
|
|
context->getClientInfo().current_address.toString(),
|
|
|
|
comment,
|
2022-06-03 05:26:49 +00:00
|
|
|
toOneLineQuery(elem.query),
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.stack_trace);
|
2023-01-17 19:04:25 +00:00
|
|
|
|
2023-02-25 00:18:34 +00:00
|
|
|
if (log_error)
|
|
|
|
LOG_ERROR(&Poco::Logger::get("executeQuery"), message);
|
|
|
|
else
|
|
|
|
LOG_INFO(&Poco::Logger::get("executeQuery"), message);
|
2015-06-17 21:34:15 +00:00
|
|
|
}
|
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
static void onExceptionBeforeStart(
|
|
|
|
const String & query_for_logging,
|
|
|
|
ContextPtr context,
|
|
|
|
ASTPtr ast,
|
|
|
|
const std::shared_ptr<OpenTelemetry::SpanHolder> & query_span,
|
|
|
|
UInt64 elapsed_millliseconds)
|
2015-07-01 05:18:54 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
auto query_end_time = std::chrono::system_clock::now();
|
|
|
|
|
2017-01-18 13:53:20 +00:00
|
|
|
/// Exception before the query execution.
|
2021-04-10 23:33:54 +00:00
|
|
|
if (auto quota = context->getQuota())
|
2021-11-18 18:07:35 +00:00
|
|
|
quota->used(QuotaType::ERRORS, 1, /* check_exceeded = */ false);
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
const Settings & settings = context->getSettingsRef();
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
const auto & client_info = context->getClientInfo();
|
|
|
|
|
2017-01-18 13:53:20 +00:00
|
|
|
/// Log the start of query execution into the table if necessary.
|
2018-05-30 16:15:35 +00:00
|
|
|
QueryLogElement elem;
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2020-04-04 21:07:00 +00:00
|
|
|
elem.type = QueryLogElementType::EXCEPTION_BEFORE_START;
|
2022-12-28 20:01:41 +00:00
|
|
|
elem.event_time = timeInSeconds(query_end_time);
|
|
|
|
elem.event_time_microseconds = timeInMicroseconds(query_end_time);
|
|
|
|
elem.query_start_time = client_info.initial_query_start_time;
|
|
|
|
elem.query_start_time_microseconds = client_info.initial_query_start_time_microseconds;
|
|
|
|
elem.query_duration_ms = elapsed_millliseconds;
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.current_database = context->getCurrentDatabase();
|
2019-06-20 07:17:21 +00:00
|
|
|
elem.query = query_for_logging;
|
2021-01-26 14:51:30 +00:00
|
|
|
elem.normalized_query_hash = normalizedQueryHash<false>(query_for_logging);
|
2020-12-02 12:08:03 +00:00
|
|
|
|
2023-02-01 02:31:31 +00:00
|
|
|
// Log query_kind if ast is valid
|
2021-08-04 14:09:23 +00:00
|
|
|
if (ast)
|
2021-08-07 06:26:08 +00:00
|
|
|
{
|
2023-02-01 02:31:31 +00:00
|
|
|
elem.query_kind = ast->getQueryKind();
|
2021-08-07 06:26:08 +00:00
|
|
|
if (settings.log_formatted_queries)
|
|
|
|
elem.formatted_query = queryToString(ast);
|
|
|
|
}
|
2021-08-04 14:09:23 +00:00
|
|
|
|
|
|
|
// We don't calculate databases, tables and columns when the query isn't able to start
|
2020-12-02 12:08:03 +00:00
|
|
|
|
2020-01-22 12:29:30 +00:00
|
|
|
elem.exception_code = getCurrentExceptionCode();
|
2023-01-17 19:04:25 +00:00
|
|
|
auto exception_message = getCurrentExceptionMessageAndPattern(/* with_stacktrace */ false);
|
2023-01-25 20:16:42 +00:00
|
|
|
elem.exception = std::move(exception_message.text);
|
2023-01-17 19:04:25 +00:00
|
|
|
elem.exception_format_string = exception_message.format_string;
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.client_info = context->getClientInfo();
|
2016-10-24 21:40:39 +00:00
|
|
|
|
2021-01-25 19:18:23 +00:00
|
|
|
elem.log_comment = settings.log_comment;
|
|
|
|
if (elem.log_comment.size() > settings.max_query_size)
|
|
|
|
elem.log_comment.resize(settings.max_query_size);
|
2020-12-28 12:57:27 +00:00
|
|
|
|
2021-05-13 18:48:36 +00:00
|
|
|
if (auto txn = context->getCurrentTransaction())
|
2021-03-31 17:55:04 +00:00
|
|
|
elem.tid = txn->tid;
|
|
|
|
|
2018-08-23 01:31:28 +00:00
|
|
|
if (settings.calculate_text_stack_trace)
|
|
|
|
setExceptionStackTrace(elem);
|
2018-05-30 16:15:35 +00:00
|
|
|
logException(context, elem);
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2018-06-19 20:30:35 +00:00
|
|
|
/// Update performance counters before logging to query_log
|
|
|
|
CurrentThread::finalizePerformanceCounters();
|
|
|
|
|
2020-10-29 19:28:46 +00:00
|
|
|
if (settings.log_queries && elem.type >= settings.log_queries_min_type && !settings.log_queries_min_query_duration_ms.totalMilliseconds())
|
2021-04-10 23:33:54 +00:00
|
|
|
if (auto query_log = context->getQueryLog())
|
2018-03-10 19:57:13 +00:00
|
|
|
query_log->add(elem);
|
2020-05-21 18:01:25 +00:00
|
|
|
|
2022-09-01 15:16:05 +00:00
|
|
|
if (query_span)
|
|
|
|
{
|
|
|
|
query_span->addAttribute("clickhouse.exception_code", elem.exception_code);
|
|
|
|
query_span->addAttribute("clickhouse.exception", elem.exception);
|
|
|
|
query_span->addAttribute("db.statement", elem.query);
|
|
|
|
query_span->addAttribute("clickhouse.query_id", elem.client_info.current_query_id);
|
|
|
|
query_span->finish();
|
|
|
|
}
|
2020-08-27 18:44:20 +00:00
|
|
|
|
2020-05-21 18:01:25 +00:00
|
|
|
ProfileEvents::increment(ProfileEvents::FailedQuery);
|
|
|
|
|
|
|
|
if (ast)
|
|
|
|
{
|
|
|
|
if (ast->as<ASTSelectQuery>() || ast->as<ASTSelectWithUnionQuery>())
|
|
|
|
{
|
|
|
|
ProfileEvents::increment(ProfileEvents::FailedSelectQuery);
|
|
|
|
}
|
|
|
|
else if (ast->as<ASTInsertQuery>())
|
|
|
|
{
|
|
|
|
ProfileEvents::increment(ProfileEvents::FailedInsertQuery);
|
|
|
|
}
|
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
}
|
|
|
|
|
2021-05-31 14:49:02 +00:00
|
|
|
static void setQuerySpecificSettings(ASTPtr & ast, ContextMutablePtr context)
|
2020-04-28 15:50:50 +00:00
|
|
|
{
|
2021-03-04 11:10:21 +00:00
|
|
|
if (auto * ast_insert_into = ast->as<ASTInsertQuery>())
|
2020-04-28 15:50:50 +00:00
|
|
|
{
|
|
|
|
if (ast_insert_into->watch)
|
2021-04-10 23:33:54 +00:00
|
|
|
context->setSetting("output_format_enable_streaming", 1);
|
2020-04-28 15:50:50 +00:00
|
|
|
}
|
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2015-06-18 02:11:05 +00:00
|
|
|
static std::tuple<ASTPtr, BlockIO> executeQueryImpl(
|
2017-07-10 03:41:02 +00:00
|
|
|
const char * begin,
|
|
|
|
const char * end,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr context,
|
2015-06-18 02:11:05 +00:00
|
|
|
bool internal,
|
2019-02-08 13:24:24 +00:00
|
|
|
QueryProcessingStage::Enum stage,
|
2020-05-18 13:55:07 +00:00
|
|
|
ReadBuffer * istr)
|
2015-06-18 00:27:25 +00:00
|
|
|
{
|
2022-09-01 15:16:05 +00:00
|
|
|
/// query_span is a special span, when this function exits, it's lifetime is not ended, but ends when the query finishes.
|
|
|
|
/// Some internal queries might call this function recursively by setting 'internal' parameter to 'true',
|
|
|
|
/// to make sure SpanHolders in current stack ends in correct order, we disable this span for these internal queries
|
|
|
|
///
|
|
|
|
/// This does not have impact on the final span logs, because these internal queries are issued by external queries,
|
|
|
|
/// we still have enough span logs for the execution of external queries.
|
|
|
|
std::shared_ptr<OpenTelemetry::SpanHolder> query_span = internal ? nullptr : std::make_shared<OpenTelemetry::SpanHolder>("query");
|
2023-07-06 00:16:38 +00:00
|
|
|
if (query_span && query_span->trace_id != UUID{})
|
|
|
|
LOG_TRACE(&Poco::Logger::get("executeQuery"), "Query span trace_id for opentelemetry log: {}", query_span->trace_id);
|
2022-07-07 10:17:30 +00:00
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
auto query_start_time = std::chrono::system_clock::now();
|
|
|
|
|
|
|
|
/// Used to set the watch in QueryStatus and the output formats. It is not based on query_start_time as that might be based on
|
|
|
|
/// the value passed by the client
|
|
|
|
Stopwatch start_watch{CLOCK_MONOTONIC};
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-05-20 13:21:42 +00:00
|
|
|
auto & client_info = context->getClientInfo();
|
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
if (!internal)
|
2021-05-20 13:21:42 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
// If it's not an internal query and we don't see an initial_query_start_time yet, initialize it
|
|
|
|
// to current time. Internal queries are those executed without an independent client context,
|
|
|
|
// thus should not set initial_query_start_time, because it might introduce data race. It's also
|
|
|
|
// possible to have unset initial_query_start_time for non-internal and non-initial queries. For
|
|
|
|
// example, the query is from an initiator that is running an old version of clickhouse.
|
|
|
|
// On the other hand, if it's initialized then take it as the start of the query
|
|
|
|
if (client_info.initial_query_start_time == 0)
|
|
|
|
{
|
|
|
|
client_info.initial_query_start_time = timeInSeconds(query_start_time);
|
|
|
|
client_info.initial_query_start_time_microseconds = timeInMicroseconds(query_start_time);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
query_start_time = std::chrono::time_point<std::chrono::system_clock>(
|
|
|
|
std::chrono::microseconds{client_info.initial_query_start_time_microseconds});
|
|
|
|
}
|
2021-05-20 13:21:42 +00:00
|
|
|
}
|
|
|
|
|
2021-01-28 13:57:36 +00:00
|
|
|
assert(internal || CurrentThread::get().getQueryContext());
|
|
|
|
assert(internal || CurrentThread::get().getQueryContext()->getCurrentQueryId() == CurrentThread::getQueryId());
|
2018-03-02 05:44:17 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
const Settings & settings = context->getSettingsRef();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-01-28 06:55:43 +00:00
|
|
|
size_t max_query_size = settings.max_query_size;
|
|
|
|
/// Don't limit the size of internal queries or distributed subquery.
|
|
|
|
if (internal || client_info.query_kind == ClientInfo::QueryKind::SECONDARY_QUERY)
|
|
|
|
max_query_size = 0;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
ASTPtr ast;
|
|
|
|
String query;
|
|
|
|
String query_for_logging;
|
2022-11-11 15:26:04 +00:00
|
|
|
size_t log_queries_cut_to_length = context->getSettingsRef().log_queries_cut_to_length;
|
2022-09-19 14:19:21 +00:00
|
|
|
|
|
|
|
/// Parse the query from string.
|
2015-06-18 00:27:25 +00:00
|
|
|
try
|
|
|
|
{
|
2022-09-19 03:25:27 +00:00
|
|
|
if (settings.dialect == Dialect::kusto && !internal)
|
2022-06-08 17:14:03 +00:00
|
|
|
{
|
|
|
|
ParserKQLStatement parser(end, settings.allow_settings_after_format_in_insert);
|
2020-12-04 02:15:44 +00:00
|
|
|
|
2022-09-19 03:25:27 +00:00
|
|
|
/// TODO: parser should fail early when max_query_size limit is reached.
|
2022-06-08 17:14:03 +00:00
|
|
|
ast = parseQuery(parser, begin, end, "", max_query_size, settings.max_parser_depth);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ParserQuery parser(end, settings.allow_settings_after_format_in_insert);
|
|
|
|
|
|
|
|
/// TODO: parser should fail early when max_query_size limit is reached.
|
|
|
|
ast = parseQuery(parser, begin, end, "", max_query_size, settings.max_parser_depth);
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
const char * query_end = end;
|
|
|
|
if (const auto * insert_query = ast->as<ASTInsertQuery>(); insert_query && insert_query->data)
|
|
|
|
query_end = insert_query->data;
|
|
|
|
|
2022-10-21 12:45:28 +00:00
|
|
|
bool is_create_parameterized_view = false;
|
2022-10-19 16:30:03 +00:00
|
|
|
if (const auto * create_query = ast->as<ASTCreateQuery>())
|
2022-10-21 12:45:28 +00:00
|
|
|
is_create_parameterized_view = create_query->isParameterizedView();
|
2022-10-19 16:30:03 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
/// Replace ASTQueryParameter with ASTLiteral for prepared statements.
|
2022-10-21 12:45:28 +00:00
|
|
|
if (!is_create_parameterized_view && context->hasQueryParameters())
|
2022-09-19 14:19:21 +00:00
|
|
|
{
|
|
|
|
ReplaceQueryParameterVisitor visitor(context->getQueryParameters());
|
|
|
|
visitor.visit(ast);
|
|
|
|
query = serializeAST(*ast);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// Copy query into string. It will be written to log and presented in processlist. If an INSERT query, string will not include data to insertion.
|
|
|
|
query.assign(begin, query_end);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Wipe any sensitive information (e.g. passwords) from the query.
|
|
|
|
/// MUST go before any modification (except for prepared statements,
|
|
|
|
/// since it substitute parameters and without them query does not contain
|
|
|
|
/// parameters), to keep query as-is in query_log and server log.
|
2022-11-11 15:26:04 +00:00
|
|
|
if (ast->hasSecretParts())
|
|
|
|
{
|
|
|
|
/// IAST::formatForLogging() wipes secret parts in AST and then calls wipeSensitiveDataAndCutToLength().
|
|
|
|
query_for_logging = ast->formatForLogging(log_queries_cut_to_length);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
query_for_logging = wipeSensitiveDataAndCutToLength(query, log_queries_cut_to_length);
|
|
|
|
}
|
2022-09-19 14:19:21 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
/// Anyway log the query.
|
|
|
|
if (query.empty())
|
|
|
|
query.assign(begin, std::min(end - begin, static_cast<ptrdiff_t>(max_query_size)));
|
|
|
|
|
2022-11-11 15:26:04 +00:00
|
|
|
query_for_logging = wipeSensitiveDataAndCutToLength(query, log_queries_cut_to_length);
|
2022-09-19 14:19:21 +00:00
|
|
|
logQuery(query_for_logging, context, internal, stage);
|
|
|
|
|
|
|
|
if (!internal)
|
2022-12-28 20:01:41 +00:00
|
|
|
onExceptionBeforeStart(query_for_logging, context, ast, query_span, start_watch.elapsedMilliseconds());
|
2022-09-19 14:19:21 +00:00
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
2022-12-01 16:08:20 +00:00
|
|
|
/// Avoid early destruction of process_list_entry if it was not saved to `res` yet (in case of exception)
|
|
|
|
ProcessList::EntryPtr process_list_entry;
|
2022-09-19 14:19:21 +00:00
|
|
|
BlockIO res;
|
2023-03-07 19:17:09 +00:00
|
|
|
auto implicit_txn_control = std::make_shared<bool>(false);
|
2022-09-19 14:19:21 +00:00
|
|
|
String query_database;
|
|
|
|
String query_table;
|
|
|
|
|
2023-03-07 19:17:09 +00:00
|
|
|
auto execute_implicit_tcl_query = [implicit_txn_control](const ContextMutablePtr & query_context, ASTTransactionControl::QueryType tcl_type)
|
|
|
|
{
|
|
|
|
/// Unset the flag on COMMIT and ROLLBACK
|
|
|
|
SCOPE_EXIT({ if (tcl_type != ASTTransactionControl::BEGIN) *implicit_txn_control = false; });
|
|
|
|
|
|
|
|
ASTPtr tcl_ast = std::make_shared<ASTTransactionControl>(tcl_type);
|
|
|
|
InterpreterTransactionControlQuery tc(tcl_ast, query_context);
|
|
|
|
tc.execute();
|
|
|
|
|
|
|
|
/// Set the flag after successful BIGIN
|
|
|
|
if (tcl_type == ASTTransactionControl::BEGIN)
|
|
|
|
*implicit_txn_control = true;
|
|
|
|
};
|
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
try
|
|
|
|
{
|
2021-05-17 11:14:09 +00:00
|
|
|
if (auto txn = context->getCurrentTransaction())
|
|
|
|
{
|
2022-05-20 15:35:29 +00:00
|
|
|
chassert(txn->getState() != MergeTreeTransaction::COMMITTING);
|
|
|
|
chassert(txn->getState() != MergeTreeTransaction::COMMITTED);
|
2022-02-14 19:47:17 +00:00
|
|
|
if (txn->getState() == MergeTreeTransaction::ROLLED_BACK && !ast->as<ASTTransactionControl>() && !ast->as<ASTExplainQuery>())
|
2022-06-23 13:23:37 +00:00
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::INVALID_TRANSACTION,
|
|
|
|
"Cannot execute query because current transaction failed. Expecting ROLLBACK statement");
|
2021-05-17 11:14:09 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 19:23:26 +00:00
|
|
|
/// Interpret SETTINGS clauses as early as possible (before invoking the corresponding interpreter),
|
|
|
|
/// to allow settings to take effect.
|
2023-03-10 15:24:47 +00:00
|
|
|
InterpreterSetQuery::applySettingsFromQuery(ast, context);
|
|
|
|
|
|
|
|
if (auto * insert_query = ast->as<ASTInsertQuery>())
|
2021-02-15 18:57:35 +00:00
|
|
|
insert_query->tail = istr;
|
2019-06-20 07:17:21 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
setQuerySpecificSettings(ast, context);
|
2019-03-04 18:28:42 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
/// There is an option of probabilistic logging of queries.
|
|
|
|
/// If it is used - do the random sampling and "collapse" the settings.
|
|
|
|
/// It allows to consistently log queries with all the subqueries in distributed query processing
|
|
|
|
/// (subqueries on remote nodes will receive these "collapsed" settings)
|
|
|
|
if (!internal && settings.log_queries && settings.log_queries_probability < 1.0)
|
2020-09-15 06:49:02 +00:00
|
|
|
{
|
2022-09-19 14:19:21 +00:00
|
|
|
std::bernoulli_distribution should_write_log{settings.log_queries_probability};
|
2015-06-18 00:27:25 +00:00
|
|
|
|
2022-09-19 14:19:21 +00:00
|
|
|
context->setSetting("log_queries", should_write_log(thread_local_rng));
|
|
|
|
context->setSetting("log_queries_probability", 1.0);
|
2020-10-08 09:06:04 +00:00
|
|
|
}
|
2019-06-15 18:22:48 +00:00
|
|
|
|
2021-09-10 16:35:56 +00:00
|
|
|
if (const auto * query_with_table_output = dynamic_cast<const ASTQueryWithTableAndOutput *>(ast.get()))
|
|
|
|
{
|
|
|
|
query_database = query_with_table_output->getDatabase();
|
|
|
|
query_table = query_with_table_output->getTable();
|
|
|
|
}
|
|
|
|
|
2022-05-17 11:32:32 +00:00
|
|
|
logQuery(query_for_logging, context, internal, stage);
|
2020-10-20 18:10:24 +00:00
|
|
|
|
2020-10-08 09:06:04 +00:00
|
|
|
/// Propagate WITH statement to children ASTSelect.
|
|
|
|
if (settings.enable_global_with_statement)
|
|
|
|
{
|
|
|
|
ApplyWithGlobalVisitor().visit(ast);
|
2020-10-20 18:10:24 +00:00
|
|
|
}
|
2015-06-18 00:27:25 +00:00
|
|
|
|
2021-08-12 11:42:51 +00:00
|
|
|
{
|
2022-08-30 10:09:01 +00:00
|
|
|
SelectIntersectExceptQueryVisitor::Data data{settings.intersect_default_mode, settings.except_default_mode};
|
2021-08-13 09:57:15 +00:00
|
|
|
SelectIntersectExceptQueryVisitor{data}.visit(ast);
|
2021-08-12 11:42:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2021-08-13 09:57:15 +00:00
|
|
|
/// Normalize SelectWithUnionQuery
|
2022-08-30 10:09:01 +00:00
|
|
|
NormalizeSelectWithUnionQueryVisitor::Data data{settings.union_default_mode};
|
2021-08-13 09:57:15 +00:00
|
|
|
NormalizeSelectWithUnionQueryVisitor{data}.visit(ast);
|
2021-08-12 11:42:51 +00:00
|
|
|
}
|
2021-02-25 07:47:08 +00:00
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// Check the limits.
|
2018-03-11 00:15:26 +00:00
|
|
|
checkASTSizeLimits(*ast, settings);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// Put query to process list. But don't put SHOW PROCESSLIST query itself.
|
2019-03-11 13:22:51 +00:00
|
|
|
if (!internal && !ast->as<ASTShowProcesslistQuery>())
|
2015-07-01 05:18:54 +00:00
|
|
|
{
|
2019-06-20 07:17:21 +00:00
|
|
|
/// processlist also has query masked now, to avoid secrets leaks though SHOW PROCESSLIST by other users.
|
2022-12-28 20:01:41 +00:00
|
|
|
process_list_entry = context->getProcessList().insert(query_for_logging, ast.get(), context, start_watch.getStart());
|
2022-10-17 02:21:08 +00:00
|
|
|
context->setProcessListElement(process_list_entry->getQueryStatus());
|
2015-07-01 05:18:54 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-06-01 15:32:27 +00:00
|
|
|
/// Load external tables if they were provided
|
2021-04-10 23:33:54 +00:00
|
|
|
context->initializeExternalTablesIfSet();
|
2018-06-01 15:32:27 +00:00
|
|
|
|
2019-05-28 18:30:10 +00:00
|
|
|
auto * insert_query = ast->as<ASTInsertQuery>();
|
2023-04-25 01:11:58 +00:00
|
|
|
bool async_insert_enabled = settings.async_insert;
|
2021-03-17 14:11:47 +00:00
|
|
|
|
2021-12-28 14:47:19 +00:00
|
|
|
/// Resolve database before trying to use async insert feature - to properly hash the query.
|
|
|
|
if (insert_query)
|
|
|
|
{
|
2021-12-28 21:14:22 +00:00
|
|
|
if (insert_query->table_id)
|
|
|
|
insert_query->table_id = context->resolveStorageID(insert_query->table_id);
|
|
|
|
else if (auto table = insert_query->getTable(); !table.empty())
|
|
|
|
insert_query->table_id = context->resolveStorageID(StorageID{insert_query->getDatabase(), table});
|
2023-04-25 01:11:58 +00:00
|
|
|
|
|
|
|
if (insert_query->table_id)
|
|
|
|
if (auto table = DatabaseCatalog::instance().tryGetTable(insert_query->table_id, context))
|
|
|
|
async_insert_enabled |= table->areAsynchronousInsertsEnabled();
|
2021-12-28 14:47:19 +00:00
|
|
|
}
|
2021-03-17 14:11:47 +00:00
|
|
|
|
2019-05-30 20:12:44 +00:00
|
|
|
if (insert_query && insert_query->select)
|
2019-05-28 18:30:10 +00:00
|
|
|
{
|
2019-05-30 20:12:44 +00:00
|
|
|
/// Prepare Input storage before executing interpreter if we already got a buffer with data.
|
2019-05-28 18:30:10 +00:00
|
|
|
if (istr)
|
|
|
|
{
|
2019-05-30 20:12:44 +00:00
|
|
|
ASTPtr input_function;
|
2019-05-30 21:33:06 +00:00
|
|
|
insert_query->tryFindInputFunction(input_function);
|
2019-05-30 20:12:44 +00:00
|
|
|
if (input_function)
|
|
|
|
{
|
2022-10-14 15:09:35 +00:00
|
|
|
StoragePtr storage = context->executeTableFunction(input_function, insert_query->select->as<ASTSelectQuery>());
|
2019-05-30 20:12:44 +00:00
|
|
|
auto & input_storage = dynamic_cast<StorageInput &>(*storage);
|
2020-06-16 15:51:29 +00:00
|
|
|
auto input_metadata_snapshot = input_storage.getInMemoryMetadataPtr();
|
2021-08-08 03:30:14 +00:00
|
|
|
auto pipe = getSourceFromASTInsertQuery(
|
2021-08-27 03:00:12 +00:00
|
|
|
ast, true, input_metadata_snapshot->getSampleBlock(), context, input_function);
|
2021-07-20 18:18:43 +00:00
|
|
|
input_storage.setPipe(std::move(pipe));
|
2019-05-30 20:12:44 +00:00
|
|
|
}
|
2019-05-28 18:30:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2021-12-27 20:16:53 +00:00
|
|
|
{
|
2019-05-28 18:30:10 +00:00
|
|
|
/// reset Input callbacks if query is not INSERT SELECT
|
2021-04-10 23:33:54 +00:00
|
|
|
context->resetInputCallbacks();
|
2021-12-27 20:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
StreamLocalLimits limits;
|
|
|
|
std::shared_ptr<const EnabledQuota> quota;
|
|
|
|
std::unique_ptr<IInterpreter> interpreter;
|
2019-05-28 18:30:10 +00:00
|
|
|
|
2022-10-16 12:26:38 +00:00
|
|
|
bool async_insert = false;
|
2021-08-08 03:30:14 +00:00
|
|
|
auto * queue = context->getAsynchronousInsertQueue();
|
2023-02-22 21:52:49 +00:00
|
|
|
auto * logger = &Poco::Logger::get("executeQuery");
|
2022-10-16 12:26:38 +00:00
|
|
|
|
2023-04-25 01:11:58 +00:00
|
|
|
if (insert_query && async_insert_enabled)
|
2022-10-16 12:26:38 +00:00
|
|
|
{
|
|
|
|
String reason;
|
|
|
|
|
|
|
|
if (!queue)
|
|
|
|
reason = "asynchronous insert queue is not configured";
|
|
|
|
else if (insert_query->select)
|
|
|
|
reason = "insert query has select";
|
|
|
|
else if (!insert_query->hasInlinedData())
|
|
|
|
reason = "insert query doesn't have inlined data";
|
|
|
|
else
|
|
|
|
async_insert = true;
|
|
|
|
|
|
|
|
if (!async_insert)
|
2023-02-22 21:52:49 +00:00
|
|
|
LOG_DEBUG(logger, "Setting async_insert=1, but INSERT query will be executed synchronously (reason: {})", reason);
|
2022-10-16 12:26:38 +00:00
|
|
|
}
|
2021-08-08 03:30:14 +00:00
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
bool quota_checked = false;
|
|
|
|
std::unique_ptr<ReadBuffer> insert_data_buffer_holder;
|
|
|
|
|
2021-08-27 03:00:12 +00:00
|
|
|
if (async_insert)
|
2021-08-08 03:30:14 +00:00
|
|
|
{
|
2023-02-22 21:52:49 +00:00
|
|
|
if (context->getCurrentTransaction() && settings.throw_on_unsupported_query_inside_transaction)
|
|
|
|
throw Exception(ErrorCodes::NOT_IMPLEMENTED, "Async inserts inside transactions are not supported");
|
|
|
|
if (settings.implicit_transaction && settings.throw_on_unsupported_query_inside_transaction)
|
|
|
|
throw Exception(ErrorCodes::NOT_IMPLEMENTED, "Async inserts with 'implicit_transaction' are not supported");
|
|
|
|
|
2022-03-28 00:43:43 +00:00
|
|
|
quota = context->getQuota();
|
|
|
|
if (quota)
|
|
|
|
{
|
2023-02-22 21:52:49 +00:00
|
|
|
quota_checked = true;
|
2022-03-28 00:43:43 +00:00
|
|
|
quota->used(QuotaType::QUERY_INSERTS, 1);
|
|
|
|
quota->used(QuotaType::QUERIES, 1);
|
|
|
|
quota->checkExceeded(QuotaType::ERRORS);
|
|
|
|
}
|
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
auto result = queue->push(ast, context);
|
2021-08-27 03:00:12 +00:00
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
if (result.status == AsynchronousInsertQueue::PushResult::OK)
|
2021-08-27 21:29:10 +00:00
|
|
|
{
|
2023-02-22 21:52:49 +00:00
|
|
|
if (settings.wait_for_async_insert)
|
|
|
|
{
|
|
|
|
auto timeout = settings.wait_for_async_insert_timeout.totalMilliseconds();
|
|
|
|
auto source = std::make_shared<WaitForAsyncInsertSource>(std::move(result.future), timeout);
|
|
|
|
res.pipeline = QueryPipeline(Pipe(std::move(source)));
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto & table_id = insert_query->table_id;
|
|
|
|
if (!table_id.empty())
|
|
|
|
context->setInsertionTable(table_id);
|
2021-08-27 21:29:10 +00:00
|
|
|
}
|
2023-02-22 21:52:49 +00:00
|
|
|
else if (result.status == AsynchronousInsertQueue::PushResult::TOO_MUCH_DATA)
|
|
|
|
{
|
|
|
|
async_insert = false;
|
|
|
|
insert_data_buffer_holder = std::move(result.insert_data_buffer);
|
2021-08-27 21:29:10 +00:00
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
if (insert_query->data)
|
|
|
|
{
|
|
|
|
/// Reset inlined data because it will be
|
|
|
|
/// available from tail read buffer.
|
|
|
|
insert_query->end = insert_query->data;
|
|
|
|
insert_query->data = nullptr;
|
|
|
|
}
|
2022-01-31 22:27:55 +00:00
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
insert_query->tail = insert_data_buffer_holder.get();
|
|
|
|
LOG_DEBUG(logger, "Setting async_insert=1, but INSERT query will be executed synchronously because it has too much data");
|
|
|
|
}
|
2019-11-11 01:11:32 +00:00
|
|
|
}
|
2023-02-22 21:52:49 +00:00
|
|
|
|
2023-06-17 18:24:20 +00:00
|
|
|
QueryCachePtr query_cache = context->getQueryCache();
|
2023-06-17 20:12:51 +00:00
|
|
|
const bool can_use_query_cache = query_cache != nullptr && settings.use_query_cache && !internal && (ast->as<ASTSelectQuery>() || ast->as<ASTSelectWithUnionQuery>());
|
|
|
|
bool write_into_query_cache = false;
|
2023-03-10 18:29:13 +00:00
|
|
|
|
2023-02-22 21:52:49 +00:00
|
|
|
if (!async_insert)
|
2020-11-09 15:07:38 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
/// If it is a non-internal SELECT, and passive/read use of the query cache is enabled, and the cache knows the query, then set
|
|
|
|
/// a pipeline with a source populated by the query cache.
|
|
|
|
auto get_result_from_query_cache = [&]()
|
2022-06-23 13:23:37 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
if (can_use_query_cache && settings.enable_reads_from_query_cache)
|
2022-06-23 13:23:37 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
QueryCache::Key key(ast, context->getUserName());
|
|
|
|
QueryCache::Reader reader = query_cache->createReader(key);
|
|
|
|
if (reader.hasCacheEntryForKey())
|
|
|
|
{
|
|
|
|
QueryPipeline pipeline;
|
|
|
|
pipeline.readFromQueryCache(reader.getSource(), reader.getSourceTotals(), reader.getSourceExtremes());
|
|
|
|
res.pipeline = std::move(pipeline);
|
|
|
|
return true;
|
|
|
|
}
|
2022-06-23 13:23:37 +00:00
|
|
|
}
|
2023-06-17 20:12:51 +00:00
|
|
|
return false;
|
|
|
|
};
|
2022-01-31 22:27:55 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
if (!get_result_from_query_cache())
|
2021-12-04 13:56:52 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
/// We need to start the (implicit) transaction before getting the interpreter as this will get links to the latest snapshots
|
|
|
|
if (!context->getCurrentTransaction() && settings.implicit_transaction && !ast->as<ASTTransactionControl>())
|
2021-01-26 08:11:46 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
try
|
2021-12-27 20:16:53 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
if (context->isGlobalContext())
|
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Global context cannot create transactions");
|
|
|
|
|
|
|
|
execute_implicit_tcl_query(context, ASTTransactionControl::BEGIN);
|
2021-12-27 20:16:53 +00:00
|
|
|
}
|
2023-06-17 20:12:51 +00:00
|
|
|
catch (Exception & e)
|
2021-12-27 20:16:53 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
e.addMessage("while starting a transaction with 'implicit_transaction'");
|
|
|
|
throw;
|
2021-12-27 20:16:53 +00:00
|
|
|
}
|
2021-01-26 08:11:46 +00:00
|
|
|
}
|
2020-11-09 15:07:38 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
interpreter = InterpreterFactory::get(ast, context, SelectQueryOptions(stage).setInternal(internal));
|
2019-03-26 18:28:37 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
const auto & query_settings = context->getSettingsRef();
|
|
|
|
if (context->getCurrentTransaction() && query_settings.throw_on_unsupported_query_inside_transaction)
|
|
|
|
{
|
|
|
|
if (!interpreter->supportsTransactions())
|
|
|
|
throw Exception(ErrorCodes::NOT_IMPLEMENTED, "Transactions are not supported for this type of query ({})", ast->getID());
|
2023-02-22 21:52:49 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
}
|
2022-03-25 21:00:00 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
if (!interpreter->ignoreQuota() && !quota_checked)
|
2021-12-27 20:16:53 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
quota = context->getQuota();
|
|
|
|
if (quota)
|
|
|
|
{
|
|
|
|
if (ast->as<ASTSelectQuery>() || ast->as<ASTSelectWithUnionQuery>())
|
|
|
|
{
|
|
|
|
quota->used(QuotaType::QUERY_SELECTS, 1);
|
|
|
|
}
|
|
|
|
else if (ast->as<ASTInsertQuery>())
|
|
|
|
{
|
|
|
|
quota->used(QuotaType::QUERY_INSERTS, 1);
|
|
|
|
}
|
|
|
|
quota->used(QuotaType::QUERIES, 1);
|
|
|
|
quota->checkExceeded(QuotaType::ERRORS);
|
|
|
|
}
|
2021-12-27 20:16:53 +00:00
|
|
|
}
|
2022-11-29 13:15:28 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
if (!interpreter->ignoreLimits())
|
|
|
|
{
|
|
|
|
limits.mode = LimitsMode::LIMITS_CURRENT;
|
|
|
|
limits.size_limits = SizeLimits(settings.max_result_rows, settings.max_result_bytes, settings.result_overflow_mode);
|
|
|
|
}
|
2022-11-29 13:15:28 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
if (auto * insert_interpreter = typeid_cast<InterpreterInsertQuery *>(&*interpreter))
|
2022-11-29 13:15:28 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
/// Save insertion table (not table function). TODO: support remote() table function.
|
|
|
|
auto table_id = insert_interpreter->getDatabaseTable();
|
|
|
|
if (!table_id.empty())
|
|
|
|
context->setInsertionTable(std::move(table_id));
|
|
|
|
|
|
|
|
if (insert_data_buffer_holder)
|
|
|
|
insert_interpreter->addBuffer(std::move(insert_data_buffer_holder));
|
2022-11-29 13:15:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
std::unique_ptr<OpenTelemetry::SpanHolder> span;
|
|
|
|
if (OpenTelemetry::CurrentContext().isTraceEnabled())
|
2022-11-29 13:15:28 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
auto * raw_interpreter_ptr = interpreter.get();
|
|
|
|
String class_name(demangle(typeid(*raw_interpreter_ptr).name()));
|
|
|
|
span = std::make_unique<OpenTelemetry::SpanHolder>(class_name + "::execute()");
|
2022-11-29 13:15:28 +00:00
|
|
|
}
|
2023-01-02 10:45:33 +00:00
|
|
|
|
2023-06-17 20:12:51 +00:00
|
|
|
res = interpreter->execute();
|
|
|
|
|
|
|
|
/// If it is a non-internal SELECT query, and active/write use of the query cache is enabled, then add a processor on
|
|
|
|
/// top of the pipeline which stores the result in the query cache.
|
|
|
|
if (can_use_query_cache && settings.enable_writes_to_query_cache
|
|
|
|
&& (!astContainsNonDeterministicFunctions(ast, context) || settings.query_cache_store_results_of_queries_with_nondeterministic_functions))
|
|
|
|
{
|
|
|
|
QueryCache::Key key(
|
|
|
|
ast, res.pipeline.getHeader(),
|
|
|
|
context->getUserName(), settings.query_cache_share_between_users,
|
|
|
|
std::chrono::system_clock::now() + std::chrono::seconds(settings.query_cache_ttl),
|
|
|
|
settings.query_cache_compress_entries);
|
|
|
|
|
|
|
|
const size_t num_query_runs = query_cache->recordQueryRun(key);
|
|
|
|
if (num_query_runs > settings.query_cache_min_query_runs)
|
|
|
|
{
|
|
|
|
auto query_cache_writer = std::make_shared<QueryCache::Writer>(query_cache->createWriter(
|
|
|
|
key,
|
|
|
|
std::chrono::milliseconds(settings.query_cache_min_query_duration.totalMilliseconds()),
|
|
|
|
settings.query_cache_squash_partial_results,
|
|
|
|
settings.max_block_size,
|
|
|
|
settings.query_cache_max_size_in_bytes,
|
|
|
|
settings.query_cache_max_entries));
|
|
|
|
res.pipeline.writeResultIntoQueryCache(query_cache_writer);
|
|
|
|
write_into_query_cache = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-12-27 20:16:53 +00:00
|
|
|
}
|
2019-07-17 18:30:17 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-02-03 16:15:12 +00:00
|
|
|
if (process_list_entry)
|
2018-11-01 14:56:37 +00:00
|
|
|
{
|
|
|
|
/// Query was killed before execution
|
2022-10-17 02:21:08 +00:00
|
|
|
if (process_list_entry->getQueryStatus()->isKilled())
|
|
|
|
throw Exception(ErrorCodes::QUERY_WAS_CANCELLED,
|
|
|
|
"Query '{}' is killed in pending state", process_list_entry->getQueryStatus()->getInfo().client_info.current_query_id);
|
2018-11-01 14:56:37 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-01-24 13:39:39 +00:00
|
|
|
/// Hold element of process list till end of query execution.
|
|
|
|
res.process_list_entry = process_list_entry;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-12-27 20:16:53 +00:00
|
|
|
auto & pipeline = res.pipeline;
|
|
|
|
|
2021-09-21 06:57:55 +00:00
|
|
|
if (pipeline.pulling() || pipeline.completed())
|
2015-07-25 10:38:52 +00:00
|
|
|
{
|
2019-01-23 14:48:50 +00:00
|
|
|
/// Limits on the result, the quota on the result, and also callback for progress.
|
|
|
|
/// Limits apply only to the final result.
|
2021-04-10 23:33:54 +00:00
|
|
|
pipeline.setProgressCallback(context->getProgressCallback());
|
|
|
|
pipeline.setProcessListElement(context->getProcessListElement());
|
2021-09-21 06:57:55 +00:00
|
|
|
if (stage == QueryProcessingStage::Complete && pipeline.pulling())
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.setLimitsAndQuota(limits, quota);
|
2015-07-25 10:38:52 +00:00
|
|
|
}
|
2021-09-15 19:35:48 +00:00
|
|
|
else if (pipeline.pushing())
|
2016-12-06 20:55:13 +00:00
|
|
|
{
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.setProcessListElement(context->getProcessListElement());
|
2016-12-06 20:55:13 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// Everything related to query log.
|
2015-07-01 05:18:54 +00:00
|
|
|
{
|
|
|
|
QueryLogElement elem;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2023-02-19 22:15:09 +00:00
|
|
|
elem.type = QueryLogElementType::QUERY_START;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
elem.event_time = timeInSeconds(query_start_time);
|
|
|
|
elem.event_time_microseconds = timeInMicroseconds(query_start_time);
|
|
|
|
elem.query_start_time = timeInSeconds(query_start_time);
|
|
|
|
elem.query_start_time_microseconds = timeInMicroseconds(query_start_time);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.current_database = context->getCurrentDatabase();
|
2019-06-20 07:17:21 +00:00
|
|
|
elem.query = query_for_logging;
|
2021-08-07 06:26:08 +00:00
|
|
|
if (settings.log_formatted_queries)
|
|
|
|
elem.formatted_query = queryToString(ast);
|
2021-01-26 14:51:30 +00:00
|
|
|
elem.normalized_query_hash = normalizedQueryHash<false>(query_for_logging);
|
2023-02-01 02:31:31 +00:00
|
|
|
elem.query_kind = ast->getQueryKind();
|
2020-12-02 12:08:03 +00:00
|
|
|
|
2021-05-20 13:21:42 +00:00
|
|
|
elem.client_info = client_info;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-05-13 18:48:36 +00:00
|
|
|
if (auto txn = context->getCurrentTransaction())
|
2021-03-31 17:55:04 +00:00
|
|
|
elem.tid = txn->tid;
|
|
|
|
|
2015-09-24 19:25:18 +00:00
|
|
|
bool log_queries = settings.log_queries && !internal;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// Log into system table start of query execution, if need.
|
2020-08-14 21:27:05 +00:00
|
|
|
if (log_queries)
|
2018-03-10 19:57:13 +00:00
|
|
|
{
|
2021-09-19 18:53:36 +00:00
|
|
|
/// This check is not obvious, but without it 01220_scalar_optimization_in_alter fails.
|
|
|
|
if (pipeline.initialized())
|
2020-12-18 06:54:38 +00:00
|
|
|
{
|
2021-04-10 23:33:54 +00:00
|
|
|
const auto & info = context->getQueryAccessInfo();
|
2020-12-18 06:54:38 +00:00
|
|
|
elem.query_databases = info.databases;
|
|
|
|
elem.query_tables = info.tables;
|
|
|
|
elem.query_columns = info.columns;
|
2023-05-08 10:01:24 +00:00
|
|
|
elem.query_partitions = info.partitions;
|
2021-02-10 14:12:49 +00:00
|
|
|
elem.query_projections = info.projections;
|
2021-06-24 16:54:04 +00:00
|
|
|
elem.query_views = info.views;
|
2020-12-18 06:54:38 +00:00
|
|
|
}
|
|
|
|
|
2021-12-28 14:47:19 +00:00
|
|
|
if (async_insert)
|
|
|
|
InterpreterInsertQuery::extendQueryLogElemImpl(elem, context);
|
|
|
|
else if (interpreter)
|
2021-12-27 20:16:53 +00:00
|
|
|
interpreter->extendQueryLogElem(elem, ast, context, query_database, query_table);
|
2020-12-18 06:54:38 +00:00
|
|
|
|
2018-05-17 16:01:41 +00:00
|
|
|
if (settings.log_query_settings)
|
2021-04-10 23:33:54 +00:00
|
|
|
elem.query_settings = std::make_shared<Settings>(context->getSettingsRef());
|
2018-05-17 16:01:41 +00:00
|
|
|
|
2021-01-25 19:18:23 +00:00
|
|
|
elem.log_comment = settings.log_comment;
|
|
|
|
if (elem.log_comment.size() > settings.max_query_size)
|
|
|
|
elem.log_comment.resize(settings.max_query_size);
|
2020-12-28 02:38:16 +00:00
|
|
|
|
2020-10-29 19:28:46 +00:00
|
|
|
if (elem.type >= settings.log_queries_min_type && !settings.log_queries_min_query_duration_ms.totalMilliseconds())
|
2020-08-14 21:27:05 +00:00
|
|
|
{
|
2021-04-10 23:33:54 +00:00
|
|
|
if (auto query_log = context->getQueryLog())
|
2020-08-14 21:27:05 +00:00
|
|
|
query_log->add(elem);
|
|
|
|
}
|
2018-03-10 19:57:13 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-08-04 16:14:15 +00:00
|
|
|
/// Common code for finish and exception callbacks
|
2022-12-28 20:01:41 +00:00
|
|
|
auto status_info_to_query_log
|
|
|
|
= [](QueryLogElement & element, const QueryStatusInfo & info, const ASTPtr query_ast, const ContextPtr context_ptr) mutable
|
2020-08-04 16:14:15 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
const auto time_now = std::chrono::system_clock::now();
|
|
|
|
UInt64 elapsed_microseconds = info.elapsed_microseconds;
|
|
|
|
element.event_time = timeInSeconds(time_now);
|
|
|
|
element.event_time_microseconds = timeInMicroseconds(time_now);
|
|
|
|
element.query_duration_ms = elapsed_microseconds / 1000;
|
|
|
|
|
|
|
|
ProfileEvents::increment(ProfileEvents::QueryTimeMicroseconds, elapsed_microseconds);
|
Fix gcc10 build by reducing storage of the lambdas in executeQuery
There is no need to capture query AST for the status_info_to_query_log,
since callers already captured it anyway.
gcc10 reports:
../src/Interpreters/executeQuery.cpp: In member function ‘void std::__1::function<_Rp(_ArgTypes ...)>::swap(std::__1::function<_Rp(_ArgTypes ...)>&) [with _Rp = void; _ArgTypes = {DB::IBlockInputStream*, DB::IBlockOutputStream*, DB::QueryPipeline*}]’:
../src/Interpreters/executeQuery.cpp:490:49: error: array subscript 35 is outside array bounds of ‘std::__1::aligned_storage<32, 16>::type [1]’ [-Werror=array-bounds]
490 | auto status_info_to_query_log = [ast](QueryLogElement &element, const QueryStatusInfo &info) mutable
| ^
In file included from ../contrib/libcxx/include/algorithm:644,
from ../contrib/libcxx/include/__string:57,
from ../contrib/libcxx/include/string_view:175,
from ../contrib/libcxx/include/string:504,
from ../src/Common/formatReadable.h:3, from ../src/Interpreters/executeQuery.cpp:1:
../contrib/libcxx/include/functional:1877:60: note: while referencing ‘__tempbuf’
1877 | typename aligned_storage<sizeof(__buf_)>::type __tempbuf;
| ^~~~~~~~~
2020-08-19 18:17:33 +00:00
|
|
|
if (query_ast->as<ASTSelectQuery>() || query_ast->as<ASTSelectWithUnionQuery>())
|
2020-08-04 16:14:15 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
ProfileEvents::increment(ProfileEvents::SelectQueryTimeMicroseconds, elapsed_microseconds);
|
2020-08-04 16:14:15 +00:00
|
|
|
}
|
Fix gcc10 build by reducing storage of the lambdas in executeQuery
There is no need to capture query AST for the status_info_to_query_log,
since callers already captured it anyway.
gcc10 reports:
../src/Interpreters/executeQuery.cpp: In member function ‘void std::__1::function<_Rp(_ArgTypes ...)>::swap(std::__1::function<_Rp(_ArgTypes ...)>&) [with _Rp = void; _ArgTypes = {DB::IBlockInputStream*, DB::IBlockOutputStream*, DB::QueryPipeline*}]’:
../src/Interpreters/executeQuery.cpp:490:49: error: array subscript 35 is outside array bounds of ‘std::__1::aligned_storage<32, 16>::type [1]’ [-Werror=array-bounds]
490 | auto status_info_to_query_log = [ast](QueryLogElement &element, const QueryStatusInfo &info) mutable
| ^
In file included from ../contrib/libcxx/include/algorithm:644,
from ../contrib/libcxx/include/__string:57,
from ../contrib/libcxx/include/string_view:175,
from ../contrib/libcxx/include/string:504,
from ../src/Common/formatReadable.h:3, from ../src/Interpreters/executeQuery.cpp:1:
../contrib/libcxx/include/functional:1877:60: note: while referencing ‘__tempbuf’
1877 | typename aligned_storage<sizeof(__buf_)>::type __tempbuf;
| ^~~~~~~~~
2020-08-19 18:17:33 +00:00
|
|
|
else if (query_ast->as<ASTInsertQuery>())
|
2020-08-04 16:14:15 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
ProfileEvents::increment(ProfileEvents::InsertQueryTimeMicroseconds, elapsed_microseconds);
|
2020-08-04 16:14:15 +00:00
|
|
|
}
|
2022-05-02 15:13:57 +00:00
|
|
|
else
|
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
ProfileEvents::increment(ProfileEvents::OtherQueryTimeMicroseconds, elapsed_microseconds);
|
2022-05-02 15:13:57 +00:00
|
|
|
}
|
2020-08-04 16:14:15 +00:00
|
|
|
|
|
|
|
element.read_rows = info.read_rows;
|
|
|
|
element.read_bytes = info.read_bytes;
|
|
|
|
|
|
|
|
element.written_rows = info.written_rows;
|
|
|
|
element.written_bytes = info.written_bytes;
|
|
|
|
|
|
|
|
element.memory_usage = info.peak_memory_usage > 0 ? info.peak_memory_usage : 0;
|
|
|
|
|
2022-03-02 17:22:12 +00:00
|
|
|
element.thread_ids = info.thread_ids;
|
|
|
|
element.profile_counters = info.profile_counters;
|
2021-06-25 08:22:39 +00:00
|
|
|
|
|
|
|
/// We need to refresh the access info since dependent views might have added extra information, either during
|
2022-05-09 19:13:02 +00:00
|
|
|
/// creation of the view (PushingToViews chain) or while executing its internal SELECT
|
2021-06-25 08:22:39 +00:00
|
|
|
const auto & access_info = context_ptr->getQueryAccessInfo();
|
2021-06-28 09:05:31 +00:00
|
|
|
element.query_databases.insert(access_info.databases.begin(), access_info.databases.end());
|
|
|
|
element.query_tables.insert(access_info.tables.begin(), access_info.tables.end());
|
|
|
|
element.query_columns.insert(access_info.columns.begin(), access_info.columns.end());
|
2023-05-08 10:01:24 +00:00
|
|
|
element.query_partitions.insert(access_info.partitions.begin(), access_info.partitions.end());
|
2021-06-28 09:05:31 +00:00
|
|
|
element.query_projections.insert(access_info.projections.begin(), access_info.projections.end());
|
|
|
|
element.query_views.insert(access_info.views.begin(), access_info.views.end());
|
2021-08-05 15:24:10 +00:00
|
|
|
|
2021-08-04 18:44:18 +00:00
|
|
|
const auto & factories_info = context_ptr->getQueryFactoriesInfo();
|
|
|
|
element.used_aggregate_functions = factories_info.aggregate_functions;
|
|
|
|
element.used_aggregate_function_combinators = factories_info.aggregate_function_combinators;
|
|
|
|
element.used_database_engines = factories_info.database_engines;
|
|
|
|
element.used_data_type_families = factories_info.data_type_families;
|
|
|
|
element.used_dictionaries = factories_info.dictionaries;
|
|
|
|
element.used_formats = factories_info.formats;
|
|
|
|
element.used_functions = factories_info.functions;
|
|
|
|
element.used_storages = factories_info.storages;
|
|
|
|
element.used_table_functions = factories_info.table_functions;
|
2023-02-07 17:50:31 +00:00
|
|
|
|
|
|
|
element.async_read_counters = context_ptr->getAsyncReadCounters();
|
2020-08-04 16:14:15 +00:00
|
|
|
};
|
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// Also make possible for caller to log successful query finish and exception during execution.
|
2022-06-23 13:23:37 +00:00
|
|
|
auto finish_callback = [elem,
|
|
|
|
context,
|
|
|
|
ast,
|
2023-06-17 20:12:51 +00:00
|
|
|
write_into_query_cache,
|
2022-06-23 13:23:37 +00:00
|
|
|
log_queries,
|
|
|
|
log_queries_min_type = settings.log_queries_min_type,
|
|
|
|
log_queries_min_query_duration_ms = settings.log_queries_min_query_duration_ms.totalMilliseconds(),
|
|
|
|
log_processors_profiles = settings.log_processors_profiles,
|
|
|
|
status_info_to_query_log,
|
|
|
|
implicit_txn_control,
|
2023-03-07 19:17:09 +00:00
|
|
|
execute_implicit_tcl_query,
|
2022-07-07 09:41:58 +00:00
|
|
|
pulling_pipeline = pipeline.pulling(),
|
|
|
|
query_span](QueryPipeline & query_pipeline) mutable
|
2015-07-01 05:18:54 +00:00
|
|
|
{
|
2023-06-17 20:12:51 +00:00
|
|
|
if (write_into_query_cache)
|
|
|
|
/// Trigger the actual write of the buffered query result into the query cache. This is done explicitly to prevent
|
|
|
|
/// partial/garbage results in case of exceptions during query execution.
|
2023-01-26 22:23:10 +00:00
|
|
|
query_pipeline.finalizeWriteInQueryCache();
|
2023-01-04 17:08:51 +00:00
|
|
|
|
2022-10-17 02:21:08 +00:00
|
|
|
QueryStatusPtr process_list_elem = context->getProcessListElement();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
if (process_list_elem)
|
|
|
|
{
|
|
|
|
/// Update performance counters before logging to query_log
|
|
|
|
CurrentThread::finalizePerformanceCounters();
|
2018-03-09 23:04:26 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
QueryStatusInfo info = process_list_elem->getInfo(true, context->getSettingsRef().log_profile_events);
|
|
|
|
elem.type = QueryLogElementType::QUERY_FINISH;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
status_info_to_query_log(elem, info, ast, context);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
if (pulling_pipeline)
|
|
|
|
{
|
|
|
|
query_pipeline.tryGetResultRowsAndBytes(elem.result_rows, elem.result_bytes);
|
|
|
|
}
|
|
|
|
else /// will be used only for ordinary INSERT queries
|
|
|
|
{
|
|
|
|
auto progress_out = process_list_elem->getProgressOut();
|
|
|
|
elem.result_rows = progress_out.written_rows;
|
|
|
|
elem.result_bytes = progress_out.written_bytes;
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
auto progress_callback = context->getProgressCallback();
|
|
|
|
if (progress_callback)
|
|
|
|
{
|
2022-10-05 14:54:56 +00:00
|
|
|
Progress p;
|
2022-09-13 10:28:21 +00:00
|
|
|
p.incrementPiecewiseAtomically(Progress{ResultProgress{elem.result_rows, elem.result_bytes}});
|
|
|
|
progress_callback(p);
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
if (elem.read_rows != 0)
|
2022-02-06 11:21:05 +00:00
|
|
|
{
|
2022-12-28 20:01:41 +00:00
|
|
|
double elapsed_seconds = static_cast<double>(info.elapsed_microseconds) / 1000000.0;
|
|
|
|
double rows_per_second = static_cast<double>(elem.read_rows) / elapsed_seconds;
|
2023-02-27 23:16:34 +00:00
|
|
|
LOG_DEBUG(
|
2022-12-28 20:01:41 +00:00
|
|
|
&Poco::Logger::get("executeQuery"),
|
|
|
|
"Read {} rows, {} in {} sec., {} rows/sec., {}/sec.",
|
|
|
|
elem.read_rows,
|
|
|
|
ReadableSize(elem.read_bytes),
|
|
|
|
elapsed_seconds,
|
|
|
|
rows_per_second,
|
2022-09-13 10:28:21 +00:00
|
|
|
ReadableSize(elem.read_bytes / elapsed_seconds));
|
|
|
|
}
|
2022-02-06 11:21:05 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
if (log_queries && elem.type >= log_queries_min_type && static_cast<Int64>(elem.query_duration_ms) >= log_queries_min_query_duration_ms)
|
|
|
|
{
|
|
|
|
if (auto query_log = context->getQueryLog())
|
|
|
|
query_log->add(elem);
|
|
|
|
}
|
|
|
|
if (log_processors_profiles)
|
|
|
|
{
|
|
|
|
if (auto processors_profile_log = context->getProcessorsProfileLog())
|
2022-02-27 17:07:34 +00:00
|
|
|
{
|
2022-09-13 10:28:21 +00:00
|
|
|
ProcessorProfileLogElement processor_elem;
|
2022-12-28 20:01:41 +00:00
|
|
|
processor_elem.event_time = elem.event_time;
|
|
|
|
processor_elem.event_time_microseconds = elem.event_time_microseconds;
|
2023-05-11 07:44:15 +00:00
|
|
|
processor_elem.initial_query_id = elem.client_info.initial_query_id;
|
2022-09-13 10:28:21 +00:00
|
|
|
processor_elem.query_id = elem.client_info.current_query_id;
|
2022-02-27 17:07:34 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
auto get_proc_id = [](const IProcessor & proc) -> UInt64
|
2022-02-27 17:07:34 +00:00
|
|
|
{
|
2022-09-13 10:28:21 +00:00
|
|
|
return reinterpret_cast<std::uintptr_t>(&proc);
|
|
|
|
};
|
2022-08-11 11:02:41 +00:00
|
|
|
|
2022-09-13 10:28:21 +00:00
|
|
|
for (const auto & processor : query_pipeline.getProcessors())
|
|
|
|
{
|
|
|
|
std::vector<UInt64> parents;
|
|
|
|
for (const auto & port : processor->getOutputs())
|
|
|
|
{
|
|
|
|
if (!port.isConnected())
|
|
|
|
continue;
|
|
|
|
const IProcessor & next = port.getInputPort().getProcessor();
|
|
|
|
parents.push_back(get_proc_id(next));
|
|
|
|
}
|
|
|
|
|
|
|
|
processor_elem.id = get_proc_id(*processor);
|
|
|
|
processor_elem.parent_ids = std::move(parents);
|
|
|
|
|
|
|
|
processor_elem.plan_step = reinterpret_cast<std::uintptr_t>(processor->getQueryPlanStep());
|
|
|
|
processor_elem.plan_group = processor->getQueryPlanStepGroup();
|
|
|
|
|
|
|
|
processor_elem.processor_name = processor->getName();
|
|
|
|
|
2022-10-07 10:46:45 +00:00
|
|
|
/// NOTE: convert this to UInt64
|
|
|
|
processor_elem.elapsed_us = static_cast<UInt32>(processor->getElapsedUs());
|
|
|
|
processor_elem.input_wait_elapsed_us = static_cast<UInt32>(processor->getInputWaitElapsedUs());
|
|
|
|
processor_elem.output_wait_elapsed_us = static_cast<UInt32>(processor->getOutputWaitElapsedUs());
|
2022-09-13 10:28:21 +00:00
|
|
|
|
|
|
|
auto stats = processor->getProcessorDataStats();
|
|
|
|
processor_elem.input_rows = stats.input_rows;
|
|
|
|
processor_elem.input_bytes = stats.input_bytes;
|
|
|
|
processor_elem.output_rows = stats.output_rows;
|
|
|
|
processor_elem.output_bytes = stats.output_bytes;
|
|
|
|
|
|
|
|
processors_profile_log->add(processor_elem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-11 11:02:41 +00:00
|
|
|
|
2023-03-07 19:17:09 +00:00
|
|
|
if (*implicit_txn_control)
|
|
|
|
execute_implicit_tcl_query(context, ASTTransactionControl::COMMIT);
|
2022-02-05 16:33:42 +00:00
|
|
|
}
|
2020-08-20 20:59:40 +00:00
|
|
|
|
2022-09-01 15:16:05 +00:00
|
|
|
if (query_span)
|
|
|
|
{
|
|
|
|
query_span->addAttribute("db.statement", elem.query);
|
|
|
|
query_span->addAttribute("clickhouse.query_id", elem.client_info.current_query_id);
|
|
|
|
query_span->addAttribute("clickhouse.query_status", "QueryFinish");
|
|
|
|
query_span->addAttributeIfNotEmpty("clickhouse.tracestate", OpenTelemetry::CurrentContext().tracestate);
|
|
|
|
query_span->addAttributeIfNotZero("clickhouse.read_rows", elem.read_rows);
|
|
|
|
query_span->addAttributeIfNotZero("clickhouse.read_bytes", elem.read_bytes);
|
2022-09-13 10:28:21 +00:00
|
|
|
query_span->addAttributeIfNotZero("clickhouse.written_rows", elem.written_rows);
|
2022-09-01 15:16:05 +00:00
|
|
|
query_span->addAttributeIfNotZero("clickhouse.written_bytes", elem.written_bytes);
|
|
|
|
query_span->addAttributeIfNotZero("clickhouse.memory_usage", elem.memory_usage);
|
|
|
|
query_span->finish();
|
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
};
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-12-28 20:01:41 +00:00
|
|
|
auto exception_callback = [start_watch,
|
|
|
|
elem,
|
2022-06-23 13:23:37 +00:00
|
|
|
context,
|
|
|
|
ast,
|
|
|
|
log_queries,
|
|
|
|
log_queries_min_type = settings.log_queries_min_type,
|
|
|
|
log_queries_min_query_duration_ms = settings.log_queries_min_query_duration_ms.totalMilliseconds(),
|
2023-05-13 00:57:31 +00:00
|
|
|
my_quota(quota),
|
2022-06-23 13:23:37 +00:00
|
|
|
status_info_to_query_log,
|
2022-07-07 09:41:58 +00:00
|
|
|
implicit_txn_control,
|
2023-03-07 19:17:09 +00:00
|
|
|
execute_implicit_tcl_query,
|
2023-02-25 00:18:34 +00:00
|
|
|
query_span](bool log_error) mutable
|
2015-07-01 05:18:54 +00:00
|
|
|
{
|
2023-03-07 19:17:09 +00:00
|
|
|
if (*implicit_txn_control)
|
|
|
|
execute_implicit_tcl_query(context, ASTTransactionControl::ROLLBACK);
|
2022-06-23 13:23:37 +00:00
|
|
|
else if (auto txn = context->getCurrentTransaction())
|
2021-04-09 12:53:51 +00:00
|
|
|
txn->onException();
|
|
|
|
|
2023-05-13 00:57:31 +00:00
|
|
|
if (my_quota)
|
|
|
|
my_quota->used(QuotaType::ERRORS, 1, /* check_exceeded = */ false);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-04-04 21:07:00 +00:00
|
|
|
elem.type = QueryLogElementType::EXCEPTION_WHILE_PROCESSING;
|
2020-01-22 12:29:30 +00:00
|
|
|
elem.exception_code = getCurrentExceptionCode();
|
2023-01-17 19:04:25 +00:00
|
|
|
auto exception_message = getCurrentExceptionMessageAndPattern(/* with_stacktrace */ false);
|
2023-01-25 20:16:42 +00:00
|
|
|
elem.exception = std::move(exception_message.text);
|
2023-01-17 19:04:25 +00:00
|
|
|
elem.exception_format_string = exception_message.format_string;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-10-17 02:21:08 +00:00
|
|
|
QueryStatusPtr process_list_elem = context->getProcessListElement();
|
2021-04-10 23:33:54 +00:00
|
|
|
const Settings & current_settings = context->getSettingsRef();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-06-13 19:01:07 +00:00
|
|
|
/// Update performance counters before logging to query_log
|
2018-06-19 20:30:35 +00:00
|
|
|
CurrentThread::finalizePerformanceCounters();
|
2022-12-28 20:01:41 +00:00
|
|
|
const auto time_now = std::chrono::system_clock::now();
|
|
|
|
elem.event_time = timeInSeconds(time_now);
|
|
|
|
elem.event_time_microseconds = timeInMicroseconds(time_now);
|
2018-06-13 19:01:07 +00:00
|
|
|
|
2015-07-01 05:18:54 +00:00
|
|
|
if (process_list_elem)
|
|
|
|
{
|
2018-08-27 18:16:32 +00:00
|
|
|
QueryStatusInfo info = process_list_elem->getInfo(true, current_settings.log_profile_events, false);
|
2021-06-25 08:22:39 +00:00
|
|
|
status_info_to_query_log(elem, info, ast, context);
|
2015-07-01 05:18:54 +00:00
|
|
|
}
|
2022-12-28 20:01:41 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
elem.query_duration_ms = start_watch.elapsedMilliseconds();
|
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2023-02-25 00:18:34 +00:00
|
|
|
if (current_settings.calculate_text_stack_trace && log_error)
|
2018-08-23 01:31:28 +00:00
|
|
|
setExceptionStackTrace(elem);
|
2023-02-25 00:18:34 +00:00
|
|
|
logException(context, elem, log_error);
|
2015-07-01 05:18:54 +00:00
|
|
|
|
2018-05-17 16:01:41 +00:00
|
|
|
/// In case of exception we log internal queries also
|
2022-04-18 08:18:31 +00:00
|
|
|
if (log_queries && elem.type >= log_queries_min_type && static_cast<Int64>(elem.query_duration_ms) >= log_queries_min_query_duration_ms)
|
2018-03-10 19:57:13 +00:00
|
|
|
{
|
2021-04-10 23:33:54 +00:00
|
|
|
if (auto query_log = context->getQueryLog())
|
2018-03-10 19:57:13 +00:00
|
|
|
query_log->add(elem);
|
|
|
|
}
|
2020-05-21 18:01:25 +00:00
|
|
|
|
|
|
|
ProfileEvents::increment(ProfileEvents::FailedQuery);
|
2020-05-22 23:37:14 +00:00
|
|
|
if (ast->as<ASTSelectQuery>() || ast->as<ASTSelectWithUnionQuery>())
|
|
|
|
{
|
2020-05-21 18:01:25 +00:00
|
|
|
ProfileEvents::increment(ProfileEvents::FailedSelectQuery);
|
|
|
|
}
|
2020-05-22 23:37:14 +00:00
|
|
|
else if (ast->as<ASTInsertQuery>())
|
|
|
|
{
|
|
|
|
ProfileEvents::increment(ProfileEvents::FailedInsertQuery);
|
|
|
|
}
|
2022-07-07 09:41:58 +00:00
|
|
|
|
2022-09-01 15:16:05 +00:00
|
|
|
if (query_span)
|
|
|
|
{
|
|
|
|
query_span->addAttribute("db.statement", elem.query);
|
|
|
|
query_span->addAttribute("clickhouse.query_id", elem.client_info.current_query_id);
|
|
|
|
query_span->addAttribute("clickhouse.exception", elem.exception);
|
|
|
|
query_span->addAttribute("clickhouse.exception_code", elem.exception_code);
|
|
|
|
query_span->finish();
|
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
};
|
|
|
|
|
2019-04-05 10:52:07 +00:00
|
|
|
res.finish_callback = std::move(finish_callback);
|
|
|
|
res.exception_callback = std::move(exception_callback);
|
2015-06-29 23:54:33 +00:00
|
|
|
}
|
2021-02-26 12:04:11 +00:00
|
|
|
}
|
2015-07-01 05:18:54 +00:00
|
|
|
catch (...)
|
|
|
|
{
|
2023-03-07 19:17:09 +00:00
|
|
|
if (*implicit_txn_control)
|
|
|
|
execute_implicit_tcl_query(context, ASTTransactionControl::ROLLBACK);
|
2022-06-23 13:23:37 +00:00
|
|
|
else if (auto txn = context->getCurrentTransaction())
|
2021-04-09 12:53:51 +00:00
|
|
|
txn->onException();
|
|
|
|
|
2015-11-12 02:14:28 +00:00
|
|
|
if (!internal)
|
2022-12-28 20:01:41 +00:00
|
|
|
onExceptionBeforeStart(query_for_logging, context, ast, query_span, start_watch.elapsedMilliseconds());
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2015-07-01 05:18:54 +00:00
|
|
|
throw;
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-02-27 15:40:11 +00:00
|
|
|
return std::make_tuple(ast, std::move(res));
|
2015-06-18 02:11:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BlockIO executeQuery(
|
|
|
|
const String & query,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr context,
|
2015-06-18 02:11:05 +00:00
|
|
|
bool internal,
|
2021-02-15 18:57:35 +00:00
|
|
|
QueryProcessingStage::Enum stage)
|
2015-06-18 02:11:05 +00:00
|
|
|
{
|
2019-11-04 03:53:26 +00:00
|
|
|
ASTPtr ast;
|
2015-06-18 02:11:05 +00:00
|
|
|
BlockIO streams;
|
2021-02-15 18:57:35 +00:00
|
|
|
std::tie(ast, streams) = executeQueryImpl(query.data(), query.data() + query.size(), context, internal, stage, nullptr);
|
2020-01-23 12:53:32 +00:00
|
|
|
|
|
|
|
if (const auto * ast_query_with_output = dynamic_cast<const ASTQueryWithOutput *>(ast.get()))
|
2019-11-04 03:53:26 +00:00
|
|
|
{
|
2020-01-23 12:59:01 +00:00
|
|
|
String format_name = ast_query_with_output->format
|
|
|
|
? getIdentifierName(ast_query_with_output->format)
|
2021-04-10 23:33:54 +00:00
|
|
|
: context->getDefaultFormat();
|
2020-01-23 12:59:01 +00:00
|
|
|
|
2019-11-04 03:53:26 +00:00
|
|
|
if (format_name == "Null")
|
|
|
|
streams.null_format = true;
|
|
|
|
}
|
2020-01-23 12:53:32 +00:00
|
|
|
|
2015-06-18 02:11:05 +00:00
|
|
|
return streams;
|
2015-06-18 00:27:25 +00:00
|
|
|
}
|
|
|
|
|
2020-05-19 14:06:33 +00:00
|
|
|
BlockIO executeQuery(
|
2021-06-28 22:51:01 +00:00
|
|
|
bool allow_processors,
|
2020-11-02 19:23:26 +00:00
|
|
|
const String & query,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr context,
|
2020-11-02 19:23:26 +00:00
|
|
|
bool internal,
|
2021-06-28 22:51:01 +00:00
|
|
|
QueryProcessingStage::Enum stage)
|
2020-05-19 14:06:33 +00:00
|
|
|
{
|
2021-09-15 19:35:48 +00:00
|
|
|
if (!allow_processors)
|
|
|
|
throw Exception(ErrorCodes::NOT_IMPLEMENTED, "Flag allow_processors is deprecated for executeQuery");
|
2020-05-19 14:06:33 +00:00
|
|
|
|
2021-09-17 12:38:11 +00:00
|
|
|
return executeQuery(query, context, internal, stage);
|
2020-05-19 14:06:33 +00:00
|
|
|
}
|
|
|
|
|
2015-06-18 00:27:25 +00:00
|
|
|
|
2011-10-30 11:30:52 +00:00
|
|
|
void executeQuery(
|
|
|
|
ReadBuffer & istr,
|
|
|
|
WriteBuffer & ostr,
|
2017-01-11 19:05:46 +00:00
|
|
|
bool allow_into_outfile,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr context,
|
2021-08-01 22:12:15 +00:00
|
|
|
SetResultDetailsFunc set_result_details,
|
2021-08-19 11:07:47 +00:00
|
|
|
const std::optional<FormatSettings> & output_format_settings)
|
2011-10-30 11:30:52 +00:00
|
|
|
{
|
2014-04-02 18:38:17 +00:00
|
|
|
PODArray<char> parse_buf;
|
2011-10-30 11:30:52 +00:00
|
|
|
const char * begin;
|
|
|
|
const char * end;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-02-15 18:57:35 +00:00
|
|
|
istr.nextIfAtEnd();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
size_t max_query_size = context->getSettingsRef().max_query_size;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-03-26 19:41:55 +00:00
|
|
|
if (istr.buffer().end() - istr.position() > static_cast<ssize_t>(max_query_size))
|
2011-10-30 11:30:52 +00:00
|
|
|
{
|
2016-08-17 04:53:14 +00:00
|
|
|
/// If remaining buffer space in 'istr' is enough to parse query up to 'max_query_size' bytes, then parse inplace.
|
2011-10-30 11:30:52 +00:00
|
|
|
begin = istr.position();
|
|
|
|
end = istr.buffer().end();
|
|
|
|
istr.position() += end - begin;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-06-28 20:43:37 +00:00
|
|
|
/// FIXME: this is an extra copy not required for async insertion.
|
|
|
|
|
2016-08-17 04:53:14 +00:00
|
|
|
/// If not - copy enough data into 'parse_buf'.
|
2018-11-30 15:36:41 +00:00
|
|
|
WriteBufferFromVector<PODArray<char>> out(parse_buf);
|
2023-02-22 16:54:35 +00:00
|
|
|
LimitReadBuffer limit(istr, max_query_size + 1, /* trow_exception */ false, /* exact_limit */ {});
|
2018-11-30 15:36:41 +00:00
|
|
|
copyData(limit, out);
|
2020-01-10 21:42:26 +00:00
|
|
|
out.finalize();
|
2018-11-30 15:36:41 +00:00
|
|
|
|
2018-09-02 03:00:04 +00:00
|
|
|
begin = parse_buf.data();
|
2011-10-30 11:30:52 +00:00
|
|
|
end = begin + parse_buf.size();
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2023-02-13 16:44:54 +00:00
|
|
|
QueryResultDetails result_details
|
|
|
|
{
|
|
|
|
.query_id = context->getClientInfo().current_query_id,
|
|
|
|
.timezone = DateLUT::instance().getTimeZone(),
|
|
|
|
};
|
|
|
|
|
2023-03-22 06:31:16 +00:00
|
|
|
/// Set the result details in case of any exception raised during query execution
|
2023-03-21 06:10:01 +00:00
|
|
|
SCOPE_EXIT({
|
2023-03-22 06:31:16 +00:00
|
|
|
if (set_result_details == nullptr)
|
|
|
|
/// Either the result_details have been set in the flow below or the caller of this function does not provide this callback
|
|
|
|
return;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2023-03-21 06:10:01 +00:00
|
|
|
set_result_details(result_details);
|
2023-03-22 06:31:16 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
2023-03-22 08:39:14 +00:00
|
|
|
/// This exception can be ignored.
|
|
|
|
/// because if the code goes here, it means there's already an exception raised during query execution,
|
2023-03-22 09:46:09 +00:00
|
|
|
/// and that exception will be propagated to outer caller,
|
2023-03-22 06:35:58 +00:00
|
|
|
/// there's no need to report the exception thrown here.
|
2023-03-22 06:31:16 +00:00
|
|
|
}
|
2023-03-21 06:10:01 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
ASTPtr ast;
|
|
|
|
BlockIO streams;
|
|
|
|
|
|
|
|
std::tie(ast, streams) = executeQueryImpl(begin, end, context, false, QueryProcessingStage::Complete, &istr);
|
|
|
|
auto & pipeline = streams.pipeline;
|
|
|
|
|
2021-09-20 09:48:53 +00:00
|
|
|
std::unique_ptr<WriteBuffer> compressed_buffer;
|
2015-06-29 21:35:35 +00:00
|
|
|
try
|
2013-08-12 00:36:18 +00:00
|
|
|
{
|
2021-09-15 19:35:48 +00:00
|
|
|
if (pipeline.pushing())
|
2015-06-29 21:35:35 +00:00
|
|
|
{
|
2021-09-17 17:52:26 +00:00
|
|
|
auto pipe = getSourceFromASTInsertQuery(ast, true, pipeline.getHeader(), context, nullptr);
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.complete(std::move(pipe));
|
2015-06-29 21:35:35 +00:00
|
|
|
}
|
2021-09-17 11:40:03 +00:00
|
|
|
else if (pipeline.pulling())
|
2015-06-29 21:35:35 +00:00
|
|
|
{
|
2021-09-15 19:35:48 +00:00
|
|
|
const ASTQueryWithOutput * ast_query_with_output = dynamic_cast<const ASTQueryWithOutput *>(ast.get());
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-01-11 19:05:46 +00:00
|
|
|
WriteBuffer * out_buf = &ostr;
|
|
|
|
if (ast_query_with_output && ast_query_with_output->out_file)
|
|
|
|
{
|
|
|
|
if (!allow_into_outfile)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INTO_OUTFILE_NOT_ALLOWED, "INTO OUTFILE is not allowed");
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-09-15 19:35:48 +00:00
|
|
|
const auto & out_file = typeid_cast<const ASTLiteral &>(*ast_query_with_output->out_file).value.safeGet<std::string>();
|
2021-09-08 15:46:48 +00:00
|
|
|
|
2021-08-03 11:33:52 +00:00
|
|
|
std::string compression_method;
|
|
|
|
if (ast_query_with_output->compression)
|
|
|
|
{
|
|
|
|
const auto & compression_method_node = ast_query_with_output->compression->as<ASTLiteral &>();
|
|
|
|
compression_method = compression_method_node.value.safeGet<std::string>();
|
|
|
|
}
|
|
|
|
|
2021-08-03 11:54:37 +00:00
|
|
|
compressed_buffer = wrapWriteBufferWithCompressionMethod(
|
|
|
|
std::make_unique<WriteBufferFromFile>(out_file, DBMS_DEFAULT_BUFFER_SIZE, O_WRONLY | O_EXCL | O_CREAT),
|
2021-08-03 11:33:52 +00:00
|
|
|
chooseCompressionMethod(out_file, compression_method),
|
2021-08-03 11:54:37 +00:00
|
|
|
/* compression level = */ 3
|
|
|
|
);
|
2017-01-11 19:05:46 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-01-11 19:05:46 +00:00
|
|
|
String format_name = ast_query_with_output && (ast_query_with_output->format != nullptr)
|
2021-09-15 19:35:48 +00:00
|
|
|
? getIdentifierName(ast_query_with_output->format)
|
|
|
|
: context->getDefaultFormat();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-09-15 19:35:48 +00:00
|
|
|
auto out = FormatFactory::instance().getOutputFormatParallelIfPossible(
|
2021-08-03 11:54:37 +00:00
|
|
|
format_name,
|
|
|
|
compressed_buffer ? *compressed_buffer : *out_buf,
|
2021-09-19 20:15:10 +00:00
|
|
|
materializeBlock(pipeline.getHeader()),
|
2021-08-03 11:54:37 +00:00
|
|
|
context,
|
|
|
|
output_format_settings);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-09-15 19:35:48 +00:00
|
|
|
out->setAutoFlush();
|
|
|
|
|
2019-01-23 14:48:50 +00:00
|
|
|
/// Save previous progress callback if any. TODO Do it more conveniently.
|
2021-04-10 23:33:54 +00:00
|
|
|
auto previous_progress_callback = context->getProgressCallback();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-01-23 14:48:50 +00:00
|
|
|
/// NOTE Progress callback takes shared ownership of 'out'.
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.setProgressCallback([out, previous_progress_callback] (const Progress & progress)
|
2019-01-23 14:48:50 +00:00
|
|
|
{
|
|
|
|
if (previous_progress_callback)
|
|
|
|
previous_progress_callback(progress);
|
|
|
|
out->onProgress(progress);
|
|
|
|
});
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2023-02-13 16:44:54 +00:00
|
|
|
result_details.content_type = out->getContentType();
|
|
|
|
result_details.format = format_name;
|
2019-02-02 12:24:26 +00:00
|
|
|
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.complete(std::move(out));
|
2015-06-29 21:35:35 +00:00
|
|
|
}
|
2021-09-15 19:35:48 +00:00
|
|
|
else
|
2019-03-26 18:28:37 +00:00
|
|
|
{
|
2021-09-15 19:35:48 +00:00
|
|
|
pipeline.setProgressCallback(context->getProgressCallback());
|
2019-03-26 18:28:37 +00:00
|
|
|
}
|
2019-04-12 13:56:48 +00:00
|
|
|
|
2023-02-13 16:44:54 +00:00
|
|
|
if (set_result_details)
|
2023-03-21 07:27:13 +00:00
|
|
|
{
|
2023-03-22 06:31:16 +00:00
|
|
|
/// The call of set_result_details itself might throw exception,
|
|
|
|
/// in such case there's no need to call this function again in the SCOPE_EXIT defined above.
|
|
|
|
/// So the callback is cleared before its execution.
|
|
|
|
auto set_result_details_copy = set_result_details;
|
2023-03-21 06:10:01 +00:00
|
|
|
set_result_details = nullptr;
|
2023-03-22 06:31:16 +00:00
|
|
|
|
|
|
|
set_result_details_copy(result_details);
|
2023-03-21 06:10:01 +00:00
|
|
|
}
|
2023-02-13 16:44:54 +00:00
|
|
|
|
2021-09-17 11:40:03 +00:00
|
|
|
if (pipeline.initialized())
|
|
|
|
{
|
|
|
|
CompletedPipelineExecutor executor(pipeline);
|
|
|
|
executor.execute();
|
2019-03-26 18:28:37 +00:00
|
|
|
}
|
2021-02-17 14:21:32 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/// It's possible to have queries without input and output.
|
|
|
|
}
|
2015-06-18 02:11:05 +00:00
|
|
|
}
|
2015-06-29 21:35:35 +00:00
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
streams.onException();
|
2015-07-01 05:18:54 +00:00
|
|
|
throw;
|
2015-06-29 21:35:35 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2015-07-01 05:18:54 +00:00
|
|
|
streams.onFinish();
|
2015-06-18 02:11:05 +00:00
|
|
|
}
|
2012-03-11 08:52:56 +00:00
|
|
|
|
2021-07-01 13:21:38 +00:00
|
|
|
void executeTrivialBlockIO(BlockIO & streams, ContextPtr context)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
if (!streams.pipeline.initialized())
|
|
|
|
return;
|
|
|
|
|
2021-09-15 19:35:48 +00:00
|
|
|
if (!streams.pipeline.completed())
|
2021-07-01 13:21:38 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Query pipeline requires output, but no output buffer provided, it's a bug");
|
|
|
|
|
|
|
|
streams.pipeline.setProgressCallback(context->getProgressCallback());
|
2021-09-15 19:35:48 +00:00
|
|
|
CompletedPipelineExecutor executor(streams.pipeline);
|
|
|
|
executor.execute();
|
2015-06-18 02:11:05 +00:00
|
|
|
}
|
2015-06-29 21:35:35 +00:00
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
streams.onException();
|
2015-07-01 05:18:54 +00:00
|
|
|
throw;
|
2015-06-29 21:35:35 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2015-07-01 05:18:54 +00:00
|
|
|
streams.onFinish();
|
2015-06-18 02:11:05 +00:00
|
|
|
}
|
2012-03-11 08:52:56 +00:00
|
|
|
|
2011-10-30 11:30:52 +00:00
|
|
|
}
|