#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace DB; InterpreterAlterQuery::InterpreterAlterQuery(ASTPtr query_ptr_, Context & context_) : query_ptr(query_ptr_), context(context_) { } static bool namesEqual(const String &name, const ASTPtr & name_type_) { const ASTNameTypePair & name_type = dynamic_cast(*name_type_); return name_type.name == name; } /// одинаковыми считаются имена, если они совпадают целиком или name_without_dot совпадает с частью имени до точки static bool namesEqualIgnoreAfterDot(const String & name_without_dot, const ASTPtr & name_type_) { const ASTNameTypePair & name_type = dynamic_cast(*name_type_); String name_with_dot = name_without_dot + "."; return (name_without_dot == name_type.name || name_with_dot == name_type.name.substr(0, name_with_dot.length())); } void InterpreterAlterQuery::execute() { ASTAlterQuery & alter = dynamic_cast(*query_ptr); String & table_name = alter.table; String database_name = alter.database.empty() ? context.getCurrentDatabase() : alter.database; StoragePtr table = context.getTable(database_name, table_name); auto table_soft_lock = table->lockDataForAlter(); const DataTypeFactory & data_type_factory = context.getDataTypeFactory(); String path = context.getPath(); String database_name_escaped = escapeForFileName(database_name); String table_name_escaped = escapeForFileName(table_name); String metadata_path = path + "metadata/" + database_name_escaped + "/" + table_name_escaped + ".sql"; ASTPtr attach_ptr = context.getCreateQuery(database_name, table_name); ASTCreateQuery & attach = dynamic_cast< ASTCreateQuery &>(*attach_ptr); attach.attach = true; ASTs & columns = dynamic_cast(*attach.columns).children; /// Различные проверки, на возможность выполнения запроса ASTs columns_copy = columns; IdentifierNameSet identifier_names; attach.storage->collectIdentifierNames(identifier_names); for (ASTAlterQuery::ParameterContainer::const_iterator alter_it = alter.parameters.begin(); alter_it != alter.parameters.end(); ++alter_it) { const ASTAlterQuery::Parameters & params = *alter_it; if (params.type == ASTAlterQuery::ADD) { const ASTNameTypePair & name_type = dynamic_cast(*params.name_type); StringRange type_range = name_type.type->range; /// проверяем корректность типа. В случае некоректного типа будет исключение data_type_factory.get(String(type_range.first, type_range.second - type_range.first)); /// Проверяем, что колонка еще не существует if (std::find_if(columns_copy.begin(), columns_copy.end(), boost::bind(namesEqual, name_type.name, _1)) != columns_copy.end()) throw Exception("Wrong column name. Column " + name_type.name + " already exists", DB::ErrorCodes::ILLEGAL_COLUMN); /// Проверяем опциональный аргумент AFTER ASTs::iterator insert_it = columns_copy.end(); if (params.column) { const ASTIdentifier & col_after = dynamic_cast(*params.column); insert_it = std::find_if(columns_copy.begin(), columns_copy.end(), boost::bind(namesEqualIgnoreAfterDot, col_after.name, _1)) ; if (insert_it == columns_copy.end()) throw Exception("Wrong column name. Cannot find column " + col_after.name + " to insert after", DB::ErrorCodes::ILLEGAL_COLUMN); } columns_copy.insert(insert_it, params.name_type); } else if (params.type == ASTAlterQuery::DROP) { const ASTIdentifier & drop_column = dynamic_cast (*params.column); /// Проверяем, что поле не является ключевым if (identifier_names.find(drop_column.name) != identifier_names.end()) throw Exception("Cannot drop key column", DB::ErrorCodes::ILLEGAL_COLUMN); ASTs::iterator drop_it = std::find_if(columns_copy.begin(), columns_copy.end(), boost::bind(namesEqual, drop_column.name, _1)); if (drop_it == columns_copy.end()) throw Exception("Wrong column name. Cannot find column " + drop_column.name +" to drop", DB::ErrorCodes::ILLEGAL_COLUMN); else columns_copy.erase(drop_it); } else if (params.type == ASTAlterQuery::MODIFY) { const ASTNameTypePair & name_type = dynamic_cast(*params.name_type); StringRange type_range = name_type.type->range; /// проверяем корректность типа. В случае некоректного типа будет исключение String type(type_range.first, type_range.second - type_range.first); data_type_factory.get(type); /// проверяем, что колонка существует auto modified_column = std::find_if(columns_copy.begin(), columns_copy.end(), boost::bind(namesEqual, name_type.name, _1)); if ( modified_column == columns_copy.end()) throw Exception("Wrong column name. Column " + name_type.name + " not exists", DB::ErrorCodes::ILLEGAL_COLUMN); /// Проверяем, что поле не является ключевым if (identifier_names.find(name_type.name) != identifier_names.end()) throw Exception("Modification of primary column not supported", DB::ErrorCodes::ILLEGAL_COLUMN); /// к сожалению, проверить на возможно ли это приведение типов можно только во время выполнения } } /// Пока разрешим читать из таблицы. Запретим при первой попытке изменить структуру таблицы. /// Это позволит сделать большую часть первого MODIFY, не останавливая чтение из таблицы. IStorage::TableStructureWriteLockPtr table_hard_lock; /// todo cycle over sub tables and tables /// Применяем изменения for (ASTAlterQuery::ParameterContainer::const_iterator alter_it = alter.parameters.begin(); alter_it != alter.parameters.end(); ++alter_it) { const ASTAlterQuery::Parameters & params = *alter_it; if (params.type == ASTAlterQuery::MODIFY) { table->prepareAlterModify(params); if (!table_hard_lock) table_hard_lock = table->lockStructureForAlter(); table->commitAlterModify(params); } else { if (!table_hard_lock) table_hard_lock = table->lockStructureForAlter(); table->alter(params); } if (params.type == ASTAlterQuery::ADD) { /// Применяем опциональный аргумент AFTER ASTs::iterator insert_it = columns.end(); if (params.column) { const ASTIdentifier & col_after = dynamic_cast(*params.column); insert_it = std::find_if(columns.begin(), columns.end(), boost::bind(namesEqualIgnoreAfterDot, col_after.name, _1)) ; ++insert_it; /// increase iterator because we want to insert after found element not before } columns.insert(insert_it, params.name_type); } else if (params.type == ASTAlterQuery::DROP) { const ASTIdentifier & drop_column = dynamic_cast (*params.column); ASTs::iterator drop_it = std::find_if(columns.begin(), columns.end(), boost::bind(namesEqual, drop_column.name, _1)); columns.erase(drop_it); } else if (params.type == ASTAlterQuery::MODIFY) { const ASTNameTypePair & name_type = dynamic_cast(*params.name_type); ASTs::iterator modify_it = std::find_if(columns.begin(), columns.end(), boost::bind(namesEqual, name_type.name, _1)); ASTNameTypePair & modified_column = dynamic_cast(**modify_it); modified_column.type = name_type.type; } /// Перезаписываем файл метадата каждую итерацию Poco::FileOutputStream ostr(metadata_path); formatAST(attach, ostr, 0, false); } }