#include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int ILLEGAL_TYPE_OF_ARGUMENT; } bool replaceForPositionalArguments(ASTPtr & argument, const ASTSelectQuery * select_query, ASTSelectQuery::Expression expression) { auto columns = select_query->select()->children; const auto * expr_with_alias = dynamic_cast(argument.get()); if (expr_with_alias && !expr_with_alias->alias.empty()) return false; const auto * ast_literal = typeid_cast(argument.get()); if (!ast_literal) return false; auto which = ast_literal->value.getType(); if (which != Field::Types::UInt64) return false; auto pos = ast_literal->value.get(); if (!pos || pos > columns.size()) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Positional argument out of bounds: {} (expected in range [1, {}]", pos, columns.size()); const auto & column = columns[--pos]; if (typeid_cast(column.get()) || typeid_cast(column.get())) { argument = column->clone(); } else if (typeid_cast(column.get())) { std::function throw_if_aggregate_function = [&](ASTPtr node) { if (const auto * function = typeid_cast(node.get())) { auto is_aggregate_function = AggregateUtils::isAggregateFunction(*function); if (is_aggregate_function) { throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal value (aggregate function) for positional argument in {}", ASTSelectQuery::expressionToString(expression)); } else { if (function->arguments) { for (const auto & arg : function->arguments->children) throw_if_aggregate_function(arg); } } } }; if (expression == ASTSelectQuery::Expression::GROUP_BY) throw_if_aggregate_function(column); argument = column->clone(); } else { throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal value for positional argument in {}", ASTSelectQuery::expressionToString(expression)); } return true; } }