#include #include #include #include #include #include #include #include #include #include namespace DB { bool ParserPartition::parseImpl(Pos & pos, ASTPtr & node, Expected & expected) { ParserKeyword s_id(Keyword::ID); ParserKeyword s_all(Keyword::ALL); ParserStringLiteral parser_string_literal; ParserSubstitution parser_substitution; ParserExpression parser_expr; auto partition = std::make_shared(); if (s_id.ignore(pos, expected)) { ASTPtr partition_id; if (!parser_string_literal.parse(pos, partition_id, expected) && !parser_substitution.parse(pos, partition_id, expected)) return false; if (auto * partition_id_literal = partition_id->as(); partition_id_literal != nullptr) partition->setPartitionID(partition_id); else if (auto * partition_id_query_parameter = partition_id->as(); partition_id_query_parameter != nullptr) partition->setPartitionID(partition_id); else return false; } else if (s_all.ignore(pos, expected)) { partition->all = true; } else { ASTPtr value; std::optional fields_count; if (parser_substitution.parse(pos, value, expected)) { /// It can be tuple substitution fields_count = std::nullopt; } else if (parser_expr.parse(pos, value, expected)) { if (const auto * tuple_ast = value->as(); tuple_ast) { if (tuple_ast->name != "tuple") return false; const auto * arguments_ast = tuple_ast->arguments->as(); if (arguments_ast) fields_count = arguments_ast->children.size(); else fields_count = 0; } else if (const auto* literal_ast = value->as(); literal_ast) { if (literal_ast->value.getType() == Field::Types::Tuple) { fields_count = literal_ast->value.get().size(); } else { fields_count = 1; } } else return false; } else { return false; } partition->setPartitionValue(value); partition->fields_count = fields_count; } node = partition; return true; } }