#include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int SYNTAX_ERROR; } bool ParserTableExpression::parseImpl(Pos & pos, ASTPtr & node, Expected & expected) { auto res = std::make_shared(); if (!ParserWithOptionalAlias(std::make_unique(), true).parse(pos, res->subquery, expected) && !ParserWithOptionalAlias(std::make_unique(true, true), true).parse(pos, res->table_function, expected) && !ParserWithOptionalAlias(std::make_unique(true, true), true) .parse(pos, res->database_and_table_name, expected)) return false; /// FINAL if (ParserKeyword("FINAL").ignore(pos, expected)) res->final = true; /// SAMPLE number if (ParserKeyword("SAMPLE").ignore(pos, expected)) { ParserSampleRatio ratio; if (!ratio.parse(pos, res->sample_size, expected)) return false; /// OFFSET number if (ParserKeyword("OFFSET").ignore(pos, expected)) { if (!ratio.parse(pos, res->sample_offset, expected)) return false; } } if (res->database_and_table_name) res->children.emplace_back(res->database_and_table_name); if (res->table_function) res->children.emplace_back(res->table_function); if (res->subquery) res->children.emplace_back(res->subquery); if (res->sample_size) res->children.emplace_back(res->sample_size); if (res->sample_offset) res->children.emplace_back(res->sample_offset); assert(res->database_and_table_name || res->table_function || res->subquery); node = res; return true; } bool ParserArrayJoin::parseImpl(Pos & pos, ASTPtr & node, Expected & expected) { auto res = std::make_shared(); /// [LEFT] ARRAY JOIN expr list Pos saved_pos = pos; bool has_array_join = false; if (ParserKeyword("LEFT ARRAY JOIN").ignore(pos, expected)) { res->kind = ASTArrayJoin::Kind::Left; has_array_join = true; } else { pos = saved_pos; /// INNER may be specified explicitly, otherwise it is assumed as default. ParserKeyword("INNER").ignore(pos, expected); if (ParserKeyword("ARRAY JOIN").ignore(pos, expected)) { res->kind = ASTArrayJoin::Kind::Inner; has_array_join = true; } } if (!has_array_join) return false; if (!ParserExpressionList(false).parse(pos, res->expression_list, expected)) return false; if (res->expression_list) res->children.emplace_back(res->expression_list); node = res; return true; } void ParserTablesInSelectQueryElement::parseJoinStrictness(Pos & pos, ASTTableJoin & table_join) { if (ParserKeyword("ANY").ignore(pos)) table_join.strictness = ASTTableJoin::Strictness::Any; else if (ParserKeyword("ALL").ignore(pos)) table_join.strictness = ASTTableJoin::Strictness::All; else if (ParserKeyword("ASOF").ignore(pos)) table_join.strictness = ASTTableJoin::Strictness::Asof; else if (ParserKeyword("SEMI").ignore(pos)) table_join.strictness = ASTTableJoin::Strictness::Semi; else if (ParserKeyword("ANTI").ignore(pos) || ParserKeyword("ONLY").ignore(pos)) table_join.strictness = ASTTableJoin::Strictness::Anti; } bool ParserTablesInSelectQueryElement::parseImpl(Pos & pos, ASTPtr & node, Expected & expected) { auto res = std::make_shared(); if (is_first) { if (!ParserTableExpression().parse(pos, res->table_expression, expected)) return false; } else if (ParserArrayJoin().parse(pos, res->array_join, expected)) { } else { auto table_join = std::make_shared(); if (pos->type == TokenType::Comma) { ++pos; table_join->kind = ASTTableJoin::Kind::Comma; } else { if (ParserKeyword("GLOBAL").ignore(pos)) table_join->locality = ASTTableJoin::Locality::Global; else if (ParserKeyword("LOCAL").ignore(pos)) table_join->locality = ASTTableJoin::Locality::Local; table_join->strictness = ASTTableJoin::Strictness::Unspecified; /// Legacy: allow JOIN type before JOIN kind parseJoinStrictness(pos, *table_join); bool no_kind = false; if (ParserKeyword("INNER").ignore(pos)) table_join->kind = ASTTableJoin::Kind::Inner; else if (ParserKeyword("LEFT").ignore(pos)) table_join->kind = ASTTableJoin::Kind::Left; else if (ParserKeyword("RIGHT").ignore(pos)) table_join->kind = ASTTableJoin::Kind::Right; else if (ParserKeyword("FULL").ignore(pos)) table_join->kind = ASTTableJoin::Kind::Full; else if (ParserKeyword("CROSS").ignore(pos)) table_join->kind = ASTTableJoin::Kind::Cross; else no_kind = true; /// Standard position: JOIN type after JOIN kind parseJoinStrictness(pos, *table_join); /// Optional OUTER keyword for outer joins. if (table_join->kind == ASTTableJoin::Kind::Left || table_join->kind == ASTTableJoin::Kind::Right || table_join->kind == ASTTableJoin::Kind::Full) { ParserKeyword("OUTER").ignore(pos); } if (no_kind) { /// Use INNER by default as in another DBMS. if (table_join->strictness == ASTTableJoin::Strictness::Semi || table_join->strictness == ASTTableJoin::Strictness::Anti) table_join->kind = ASTTableJoin::Kind::Left; else table_join->kind = ASTTableJoin::Kind::Inner; } if (table_join->strictness != ASTTableJoin::Strictness::Unspecified && table_join->kind == ASTTableJoin::Kind::Cross) throw Exception("You must not specify ANY or ALL for CROSS JOIN.", ErrorCodes::SYNTAX_ERROR); if ((table_join->strictness == ASTTableJoin::Strictness::Semi || table_join->strictness == ASTTableJoin::Strictness::Anti) && (table_join->kind != ASTTableJoin::Kind::Left && table_join->kind != ASTTableJoin::Kind::Right)) throw Exception("SEMI|ANTI JOIN should be LEFT or RIGHT.", ErrorCodes::SYNTAX_ERROR); if (!ParserKeyword("JOIN").ignore(pos, expected)) return false; } if (!ParserTableExpression().parse(pos, res->table_expression, expected)) return false; if (table_join->kind != ASTTableJoin::Kind::Comma && table_join->kind != ASTTableJoin::Kind::Cross) { if (ParserKeyword("USING").ignore(pos, expected)) { /// Expression for USING could be in parentheses or not. bool in_parens = pos->type == TokenType::OpeningRoundBracket; if (in_parens) ++pos; if (!ParserExpressionList(false).parse(pos, table_join->using_expression_list, expected)) return false; if (in_parens) { if (pos->type != TokenType::ClosingRoundBracket) return false; ++pos; } } else if (ParserKeyword("ON").ignore(pos, expected)) { /// OR is operator with lowest priority, so start parsing from it. if (!ParserLogicalOrExpression().parse(pos, table_join->on_expression, expected)) return false; } else { return false; } } if (table_join->using_expression_list) table_join->children.emplace_back(table_join->using_expression_list); if (table_join->on_expression) table_join->children.emplace_back(table_join->on_expression); res->table_join = table_join; } if (res->table_expression) res->children.emplace_back(res->table_expression); if (res->table_join) res->children.emplace_back(res->table_join); if (res->array_join) res->children.emplace_back(res->array_join); node = res; return true; } bool ParserTablesInSelectQuery::parseImpl(Pos & pos, ASTPtr & node, Expected & expected) { auto res = std::make_shared(); ASTPtr child; if (ParserTablesInSelectQueryElement(true).parse(pos, child, expected)) res->children.emplace_back(child); else return false; while (ParserTablesInSelectQueryElement(false).parse(pos, child, expected)) res->children.emplace_back(child); node = res; return true; } }