#pragma once #include #include #include #include namespace DB { class ASTIdentifier; class ASTQualifiedAsterisk; struct ASTTableJoin; class ASTSelectQuery; class ASTExpressionList; class ASTFunction; /// Visit one node for names qualification. @sa InDepthNodeVisitor. class TranslateQualifiedNamesMatcher { public: using Visitor = InDepthNodeVisitor; struct Data { const NameSet source_columns; const std::vector & tables; std::unordered_set join_using_columns; bool has_columns; Data(const NameSet & source_columns_, const std::vector & tables_, bool has_columns_ = true) : source_columns(source_columns_) , tables(tables_) , has_columns(has_columns_) {} bool hasColumn(const String & name) const { return source_columns.count(name); } bool hasTable() const { return !tables.empty(); } bool processAsterisks() const { return hasTable() && has_columns; } bool unknownColumn(size_t table_pos, const ASTIdentifier & identifier) const; }; static void visit(ASTPtr & ast, Data & data); static bool needChildVisit(ASTPtr & node, const ASTPtr & child); private: static void visit(ASTIdentifier & identifier, ASTPtr & ast, Data &); static void visit(const ASTQualifiedAsterisk & node, const ASTPtr & ast, Data &); static void visit(ASTTableJoin & join, const ASTPtr & ast, Data &); static void visit(ASTSelectQuery & select, const ASTPtr & ast, Data &); static void visit(ASTExpressionList &, const ASTPtr &, Data &); static void visit(ASTFunction &, const ASTPtr &, Data &); static void extractJoinUsingColumns(const ASTPtr ast, Data & data); }; /// Visits AST for names qualification. /// It finds columns and translate their names to the normal form. Expand asterisks and qualified asterisks with column names. using TranslateQualifiedNamesVisitor = TranslateQualifiedNamesMatcher::Visitor; /// Restore ASTIdentifiers to long form, change table name in case of distributed. struct RestoreQualifiedNamesMatcher { struct Data { DatabaseAndTableWithAlias distributed_table; DatabaseAndTableWithAlias remote_table; bool rename = false; void changeTable(ASTIdentifier & identifier) const; }; static bool needChildVisit(ASTPtr & node, const ASTPtr & child); static void visit(ASTPtr & ast, Data & data); static void visit(ASTIdentifier & identifier, ASTPtr & ast, Data & data); }; using RestoreQualifiedNamesVisitor = InDepthNodeVisitor; }