#pragma once #include #include namespace DB { class AccessRightsElements; class DDLGuard; /// To avoid deadlocks, we must acquire locks for tables in same order in any different RENAMEs. struct UniqueTableName { String database_name; String table_name; UniqueTableName(const String & database_name_, const String & table_name_) : database_name(database_name_), table_name(table_name_) {} bool operator< (const UniqueTableName & rhs) const { return std::tie(database_name, table_name) < std::tie(rhs.database_name, rhs.table_name); } }; struct RenameDescription { RenameDescription(const ASTRenameQuery::Element & elem, const String & current_database) : from_database_name(!elem.from.database ? current_database : elem.from.getDatabase()), from_table_name(elem.from.getTable()), to_database_name(!elem.to.database ? current_database : elem.to.getDatabase()), to_table_name(elem.to.getTable()), if_exists(elem.if_exists) {} String from_database_name; String from_table_name; String to_database_name; String to_table_name; bool if_exists; }; using RenameDescriptions = std::vector; using TableGuards = std::map>; /** Rename one table * or rename many tables at once. */ class InterpreterRenameQuery : public IInterpreter, WithContext { public: InterpreterRenameQuery(const ASTPtr & query_ptr_, ContextPtr context_); BlockIO execute() override; void extendQueryLogElemImpl(QueryLogElement & elem, const ASTPtr & ast, ContextPtr) const override; bool renamedInsteadOfExchange() const { return renamed_instead_of_exchange; } private: BlockIO executeToTables(const ASTRenameQuery & rename, const RenameDescriptions & descriptions, TableGuards & ddl_guards); BlockIO executeToDatabase(const ASTRenameQuery & rename, const RenameDescriptions & descriptions); enum class RenameType : uint8_t { RenameTable, RenameDatabase }; AccessRightsElements getRequiredAccess(RenameType type) const; ASTPtr query_ptr; bool renamed_instead_of_exchange{false}; }; }