#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { /// Visitors consist of functions with unified interface 'void visit(Casted & x, ASTPtr & y)', there x is y, successfully casted to Casted. /// Both types and function could have const specifiers. The second argument is used by visitor to replaces AST node (y) if needed. /// Visits AST nodes, add default database to tables if not set. There's different logic for DDLs and selects. class AddDefaultDatabaseVisitor { public: explicit AddDefaultDatabaseVisitor( ContextPtr context_, const String & database_name_, bool only_replace_current_database_function_ = false, bool only_replace_in_join_ = false) : context(context_) , database_name(database_name_) , only_replace_current_database_function(only_replace_current_database_function_) , only_replace_in_join(only_replace_in_join_) { if (!context->isGlobalContext()) { for (const auto & [table_name, _ /* storage */] : context->getExternalTables()) { external_tables.insert(table_name); } } } void visitDDL(ASTPtr & ast) const { visitDDLChildren(ast); if (!tryVisitDynamicCast(ast) && !tryVisitDynamicCast(ast) && !tryVisitDynamicCast(ast) && !tryVisitDynamicCast(ast)) {} } void visit(ASTPtr & ast) const { if (!tryVisit(ast) && !tryVisit(ast) && !tryVisit(ast)) visitChildren(*ast); } void visit(ASTSelectQuery & select) const { ASTPtr unused; visit(select, unused); } void visit(ASTSelectWithUnionQuery & select) const { ASTPtr unused; visit(select, unused); } void visit(ASTColumns & columns) const { for (auto & child : columns.children) visit(child); } void visit(ASTRefreshStrategy & refresh) const { ASTPtr unused; visit(refresh, unused); } private: ContextPtr context; const String database_name; std::set external_tables; mutable std::unordered_set with_aliases; bool only_replace_current_database_function = false; bool only_replace_in_join = false; void visit(ASTSelectWithUnionQuery & select, ASTPtr &) const { for (auto & child : select.list_of_selects->children) { if (child->as()) tryVisit(child); else if (child->as()) tryVisit(child); } } void visit(ASTSelectQuery & select, ASTPtr &) const { if (select.recursive_with) for (const auto & child : select.with()->children) { if (typeid_cast(child.get())) with_aliases.insert(child->as()->name); } if (select.tables()) tryVisit(select.refTables()); visitChildren(select); } void visit(ASTSelectIntersectExceptQuery & select, ASTPtr &) const { for (auto & child : select.getListOfSelects()) { if (child->as()) tryVisit(child); else if (child->as()) tryVisit(child); else if (child->as()) tryVisit(child); } } void visit(ASTTablesInSelectQuery & tables, ASTPtr &) const { for (auto & child : tables.children) tryVisit(child); } void visit(ASTTablesInSelectQueryElement & tables_element, ASTPtr &) const { if (only_replace_in_join && !tables_element.table_join) return; if (tables_element.table_expression) tryVisit(tables_element.table_expression); } void visit(ASTTableExpression & table_expression, ASTPtr &) const { if (table_expression.database_and_table_name) tryVisit(table_expression.database_and_table_name); } void visit(const ASTTableIdentifier & identifier, ASTPtr & ast) const { /// Already has database. if (identifier.compound()) return; /// There is temporary table with such name, should not be rewritten. if (external_tables.contains(identifier.shortName())) return; /// This is WITH RECURSIVE alias. if (with_aliases.contains(identifier.name())) return; auto qualified_identifier = std::make_shared(database_name, identifier.name()); if (!identifier.alias.empty()) qualified_identifier->setAlias(identifier.alias); ast = qualified_identifier; } void visit(ASTFunction & function, ASTPtr &) const { bool is_operator_in = functionIsInOrGlobalInOperator(function.name); bool is_dict_get = functionIsDictGet(function.name); for (auto & child : function.children) { if (child.get() == function.arguments.get()) { for (size_t i = 0; i < child->children.size(); ++i) { if (is_dict_get && i == 0) { if (auto * identifier = child->children[i]->as()) { /// Identifier already qualified if (identifier->compound()) continue; auto qualified_dictionary_name = context->getExternalDictionariesLoader().qualifyDictionaryNameWithDatabase(identifier->name(), context); child->children[i] = std::make_shared(qualified_dictionary_name.getParts()); } else if (auto * literal = child->children[i]->as()) { auto & literal_value = literal->value; if (literal_value.getType() != Field::Types::String) continue; auto dictionary_name = literal_value.safeGet(); auto qualified_dictionary_name = context->getExternalDictionariesLoader().qualifyDictionaryNameWithDatabase(dictionary_name, context); literal_value = qualified_dictionary_name.getFullName(); } } else if (is_operator_in && i == 1) { /// XXX: for some unknown reason this place assumes that argument can't be an alias, /// like in the similar code in `MarkTableIdentifierVisitor`. if (auto * identifier = child->children[i]->as()) { /// If identifier is broken then we can do nothing and get an exception auto maybe_table_identifier = identifier->createTable(); if (maybe_table_identifier) child->children[i] = maybe_table_identifier; } /// Second argument of the "in" function (or similar) may be a table name or a subselect. /// Rewrite the table name or descend into subselect. if (!tryVisit(child->children[i])) visit(child->children[i]); } else { visit(child->children[i]); } } } else { visit(child); } } } void visit(ASTRefreshStrategy & refresh, ASTPtr &) const { if (refresh.dependencies) for (auto & table : refresh.dependencies->children) tryVisit(table); } void visitChildren(IAST & ast) const { for (auto & child : ast.children) visit(child); } template bool tryVisit(ASTPtr & ast) const { if (T * t = typeid_cast(ast.get())) { visit(*t, ast); return true; } return false; } void visitDDL(ASTQueryWithTableAndOutput & node, ASTPtr &) const { if (only_replace_current_database_function) return; if (!node.database) node.setDatabase(database_name); } void visitDDL(ASTRenameQuery & node, ASTPtr &) const { if (only_replace_current_database_function) return; node.setDatabaseIfNotExists(database_name); } void visitDDL(ASTAlterQuery & node, ASTPtr &) const { if (only_replace_current_database_function) return; if (!node.database) node.setDatabase(database_name); for (const auto & child : node.command_list->children) { auto * command_ast = child->as(); if (command_ast->from_database.empty()) command_ast->from_database = database_name; if (command_ast->to_database.empty()) command_ast->to_database = database_name; } } void visitDDL(ASTFunction & function, ASTPtr & node) const { if (function.name == "currentDatabase") { node = std::make_shared(database_name); return; } } void visitDDLChildren(ASTPtr & ast) const { for (auto & child : ast->children) visitDDL(child); } template bool tryVisitDynamicCast(ASTPtr & ast) const { if (T * t = dynamic_cast(ast.get())) { visitDDL(*t, ast); return true; } return false; } }; }