2022-05-31 09:33:23 +00:00
|
|
|
#include <Backups/RestorerFromBackup.h>
|
|
|
|
#include <Backups/IRestoreCoordination.h>
|
2022-06-23 18:49:44 +00:00
|
|
|
#include <Backups/BackupCoordinationHelpers.h>
|
2022-05-31 09:33:23 +00:00
|
|
|
#include <Backups/BackupSettings.h>
|
|
|
|
#include <Backups/IBackup.h>
|
|
|
|
#include <Backups/IBackupEntry.h>
|
2022-06-08 02:11:41 +00:00
|
|
|
#include <Backups/BackupUtils.h>
|
2022-06-24 19:29:38 +00:00
|
|
|
#include <Backups/DDLAdjustingForBackupVisitor.h>
|
2022-06-15 18:25:13 +00:00
|
|
|
#include <Access/AccessBackup.h>
|
2022-06-18 22:01:08 +00:00
|
|
|
#include <Access/AccessRights.h>
|
2022-05-31 09:33:23 +00:00
|
|
|
#include <Parsers/ParserCreateQuery.h>
|
|
|
|
#include <Parsers/parseQuery.h>
|
|
|
|
#include <Parsers/formatAST.h>
|
|
|
|
#include <Parsers/ASTCreateQuery.h>
|
2022-06-09 16:19:54 +00:00
|
|
|
#include <Parsers/ASTFunction.h>
|
2022-05-31 09:33:23 +00:00
|
|
|
#include <Interpreters/DatabaseCatalog.h>
|
|
|
|
#include <Interpreters/Context.h>
|
|
|
|
#include <Interpreters/InterpreterCreateQuery.h>
|
|
|
|
#include <Databases/IDatabase.h>
|
2022-06-06 19:15:17 +00:00
|
|
|
#include <Databases/DDLDependencyVisitor.h>
|
2022-05-31 09:33:23 +00:00
|
|
|
#include <Storages/IStorage.h>
|
|
|
|
#include <Common/escapeForFileName.h>
|
|
|
|
#include <Common/quoteString.h>
|
|
|
|
#include <base/insertAtEnd.h>
|
2022-06-06 09:50:20 +00:00
|
|
|
#include <boost/algorithm/string/join.hpp>
|
|
|
|
#include <filesystem>
|
|
|
|
|
|
|
|
namespace fs = std::filesystem;
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int BACKUP_ENTRY_NOT_FOUND;
|
|
|
|
extern const int CANNOT_RESTORE_TABLE;
|
|
|
|
extern const int CANNOT_RESTORE_DATABASE;
|
2022-06-19 13:48:52 +00:00
|
|
|
extern const int LOGICAL_ERROR;
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
namespace
|
|
|
|
{
|
2022-06-23 17:45:36 +00:00
|
|
|
/// Finding databases and tables in the backup which we're going to restore.
|
2022-06-23 18:49:44 +00:00
|
|
|
constexpr const char * kFindingTablesInBackupStatus = "finding tables in backup";
|
2022-06-23 17:45:36 +00:00
|
|
|
|
|
|
|
/// Creating databases or finding them and checking their definitions.
|
2022-06-23 18:49:44 +00:00
|
|
|
constexpr const char * kCreatingDatabasesStatus = "creating databases";
|
2022-06-23 17:45:36 +00:00
|
|
|
|
|
|
|
/// Creating tables or finding them and checking their definition.
|
2022-06-23 18:49:44 +00:00
|
|
|
constexpr const char * kCreatingTablesStatus = "creating tables";
|
2022-06-23 17:45:36 +00:00
|
|
|
|
|
|
|
/// Inserting restored data to tables.
|
2022-06-23 18:49:44 +00:00
|
|
|
constexpr const char * kInsertingDataToTablesStatus = "inserting data to tables";
|
2022-06-23 17:45:36 +00:00
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
/// Error status.
|
|
|
|
constexpr const char * kErrorStatus = BackupCoordinationStatusSync::kErrorStatus;
|
2022-06-23 17:45:36 +00:00
|
|
|
|
|
|
|
/// Uppercases the first character of a passed string.
|
|
|
|
String toUpperFirst(const String & str)
|
|
|
|
{
|
|
|
|
String res = str;
|
|
|
|
res[0] = std::toupper(res[0]);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Outputs "table <name>" or "temporary table <name>"
|
|
|
|
String tableNameWithTypeToString(const String & database_name, const String & table_name, bool first_upper)
|
2022-06-23 12:24:45 +00:00
|
|
|
{
|
2022-06-23 17:45:36 +00:00
|
|
|
String str;
|
2022-06-23 12:24:45 +00:00
|
|
|
if (database_name == DatabaseCatalog::TEMPORARY_DATABASE)
|
2022-06-23 17:45:36 +00:00
|
|
|
str = fmt::format("temporary table {}", backQuoteIfNeed(table_name));
|
2022-06-23 12:24:45 +00:00
|
|
|
else
|
2022-06-23 17:45:36 +00:00
|
|
|
str = fmt::format("table {}.{}", backQuoteIfNeed(database_name), backQuoteIfNeed(table_name));
|
|
|
|
if (first_upper)
|
|
|
|
str[0] = std::toupper(str[0]);
|
|
|
|
return str;
|
2022-06-23 12:24:45 +00:00
|
|
|
}
|
2022-06-18 11:56:04 +00:00
|
|
|
|
2022-06-28 07:59:02 +00:00
|
|
|
/// Whether a specified name corresponds one of the tables backuping ACL.
|
|
|
|
bool isSystemAccessTableName(const QualifiedTableName & table_name)
|
2022-06-09 16:19:54 +00:00
|
|
|
{
|
2022-06-28 07:59:02 +00:00
|
|
|
if (table_name.database != DatabaseCatalog::SYSTEM_DATABASE)
|
|
|
|
return false;
|
2022-06-18 22:01:08 +00:00
|
|
|
|
2022-06-28 07:59:02 +00:00
|
|
|
return (table_name.table == "users") || (table_name.table == "roles") || (table_name.table == "settings_profiles")
|
|
|
|
|| (table_name.table == "row_policies") || (table_name.table == "quotas");
|
2022-06-09 16:19:54 +00:00
|
|
|
}
|
2022-06-27 13:39:12 +00:00
|
|
|
|
|
|
|
/// Whether a specified name corresponds one of the tables backuping ACL.
|
|
|
|
bool isSystemFunctionsTableName(const QualifiedTableName & table_name)
|
|
|
|
{
|
|
|
|
return (table_name.database == DatabaseCatalog::SYSTEM_DATABASE) && (table_name.table == "functions");
|
|
|
|
}
|
|
|
|
}
|
2022-06-09 16:19:54 +00:00
|
|
|
|
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
RestorerFromBackup::RestorerFromBackup(
|
|
|
|
const ASTBackupQuery::Elements & restore_query_elements_,
|
|
|
|
const RestoreSettings & restore_settings_,
|
|
|
|
std::shared_ptr<IRestoreCoordination> restore_coordination_,
|
|
|
|
const BackupPtr & backup_,
|
2022-06-23 18:49:44 +00:00
|
|
|
const ContextMutablePtr & context_)
|
2022-05-31 09:33:23 +00:00
|
|
|
: restore_query_elements(restore_query_elements_)
|
|
|
|
, restore_settings(restore_settings_)
|
|
|
|
, restore_coordination(restore_coordination_)
|
|
|
|
, backup(backup_)
|
|
|
|
, context(context_)
|
2022-06-23 18:49:44 +00:00
|
|
|
, create_table_timeout(context->getConfigRef().getUInt64("backups.create_table_timeout", 300000))
|
2022-05-31 09:33:23 +00:00
|
|
|
, log(&Poco::Logger::get("RestorerFromBackup"))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
RestorerFromBackup::~RestorerFromBackup() = default;
|
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
RestorerFromBackup::DataRestoreTasks RestorerFromBackup::run(Mode mode)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2022-06-23 18:49:44 +00:00
|
|
|
/// run() can be called onle once.
|
|
|
|
if (!current_status.empty())
|
2022-05-31 09:33:23 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Already restoring");
|
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
/// Find other hosts working along with us to execute this ON CLUSTER query.
|
|
|
|
all_hosts = BackupSettings::Util::filterHostIDs(
|
|
|
|
restore_settings.cluster_host_ids, restore_settings.shard_num, restore_settings.replica_num);
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
/// Do renaming in the create queries according to the renaming config.
|
2022-06-08 02:11:41 +00:00
|
|
|
renaming_map = makeRenamingMapFromBackupQuery(restore_query_elements);
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
/// Calculate the root path in the backup for restoring, it's either empty or has the format "shards/<shard_num>/replicas/<replica_num>/".
|
|
|
|
findRootPathsInBackup();
|
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
/// Find all the databases and tables which we will read from the backup.
|
2022-06-23 17:45:36 +00:00
|
|
|
setStatus(kFindingTablesInBackupStatus);
|
2022-06-23 12:24:45 +00:00
|
|
|
findDatabasesAndTablesInBackup();
|
2022-06-19 13:48:52 +00:00
|
|
|
|
2022-06-18 22:01:08 +00:00
|
|
|
/// Check access rights.
|
2022-06-23 12:24:45 +00:00
|
|
|
checkAccessForObjectsFoundInBackup();
|
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
if (mode == Mode::CHECK_ACCESS_ONLY)
|
|
|
|
return {};
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
/// Create databases using the create queries read from the backup.
|
2022-06-23 17:45:36 +00:00
|
|
|
setStatus(kCreatingDatabasesStatus);
|
2022-05-31 09:33:23 +00:00
|
|
|
createDatabases();
|
|
|
|
|
|
|
|
/// Create tables using the create queries read from the backup.
|
2022-06-23 17:45:36 +00:00
|
|
|
setStatus(kCreatingTablesStatus);
|
2022-05-31 09:33:23 +00:00
|
|
|
createTables();
|
|
|
|
|
|
|
|
/// All what's left is to insert data to tables.
|
|
|
|
/// No more data restoring tasks are allowed after this point.
|
2022-06-23 17:45:36 +00:00
|
|
|
setStatus(kInsertingDataToTablesStatus);
|
2022-06-23 18:49:44 +00:00
|
|
|
return getDataRestoreTasks();
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
/// Other hosts should know that we've encountered an error.
|
2022-06-23 18:49:44 +00:00
|
|
|
setStatus(kErrorStatus, getCurrentExceptionMessage(false));
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
void RestorerFromBackup::setStatus(const String & new_status, const String & message)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 18:49:44 +00:00
|
|
|
if (new_status == kErrorStatus)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 18:49:44 +00:00
|
|
|
LOG_ERROR(log, "{} failed with {}", toUpperFirst(current_status), message);
|
2022-06-23 17:45:36 +00:00
|
|
|
if (restore_coordination)
|
2022-06-23 18:49:44 +00:00
|
|
|
restore_coordination->setStatus(restore_settings.host_id, new_status, message);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-06-23 17:45:36 +00:00
|
|
|
LOG_TRACE(log, "{}", toUpperFirst(new_status));
|
|
|
|
current_status = new_status;
|
|
|
|
if (restore_coordination)
|
2022-06-23 18:49:44 +00:00
|
|
|
restore_coordination->setStatusAndWait(restore_settings.host_id, new_status, message, all_hosts);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RestorerFromBackup::findRootPathsInBackup()
|
|
|
|
{
|
|
|
|
size_t shard_num = 1;
|
|
|
|
size_t replica_num = 1;
|
|
|
|
if (!restore_settings.host_id.empty())
|
|
|
|
{
|
|
|
|
std::tie(shard_num, replica_num)
|
|
|
|
= BackupSettings::Util::findShardNumAndReplicaNum(restore_settings.cluster_host_ids, restore_settings.host_id);
|
|
|
|
}
|
2022-06-19 13:48:52 +00:00
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
root_paths_in_backup.clear();
|
|
|
|
|
|
|
|
/// Start with "" as the root path and then we will add shard- and replica-related part to it.
|
2022-06-06 09:50:20 +00:00
|
|
|
fs::path root_path = "/";
|
2022-05-31 09:33:23 +00:00
|
|
|
root_paths_in_backup.push_back(root_path);
|
|
|
|
|
|
|
|
/// Add shard-related part to the root path.
|
2022-06-06 09:50:20 +00:00
|
|
|
Strings shards_in_backup = backup->listFiles(root_path / "shards");
|
2022-05-31 09:33:23 +00:00
|
|
|
if (shards_in_backup.empty())
|
|
|
|
{
|
|
|
|
if (restore_settings.shard_num_in_backup > 1)
|
|
|
|
throw Exception(ErrorCodes::BACKUP_ENTRY_NOT_FOUND, "No shard #{} in backup", restore_settings.shard_num_in_backup);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
String shard_name;
|
|
|
|
if (restore_settings.shard_num_in_backup)
|
|
|
|
shard_name = std::to_string(restore_settings.shard_num_in_backup);
|
|
|
|
else if (shards_in_backup.size() == 1)
|
|
|
|
shard_name = shards_in_backup.front();
|
|
|
|
else
|
|
|
|
shard_name = std::to_string(shard_num);
|
|
|
|
if (std::find(shards_in_backup.begin(), shards_in_backup.end(), shard_name) == shards_in_backup.end())
|
|
|
|
throw Exception(ErrorCodes::BACKUP_ENTRY_NOT_FOUND, "No shard #{} in backup", shard_name);
|
2022-06-06 09:50:20 +00:00
|
|
|
root_path = root_path / "shards" / shard_name;
|
2022-05-31 09:33:23 +00:00
|
|
|
root_paths_in_backup.push_back(root_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add replica-related part to the root path.
|
2022-06-06 09:50:20 +00:00
|
|
|
Strings replicas_in_backup = backup->listFiles(root_path / "replicas");
|
2022-05-31 09:33:23 +00:00
|
|
|
if (replicas_in_backup.empty())
|
|
|
|
{
|
|
|
|
if (restore_settings.replica_num_in_backup > 1)
|
|
|
|
throw Exception(ErrorCodes::BACKUP_ENTRY_NOT_FOUND, "No replica #{} in backup", restore_settings.replica_num_in_backup);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
String replica_name;
|
|
|
|
if (restore_settings.replica_num_in_backup)
|
|
|
|
{
|
|
|
|
replica_name = std::to_string(restore_settings.replica_num_in_backup);
|
|
|
|
if (std::find(replicas_in_backup.begin(), replicas_in_backup.end(), replica_name) == replicas_in_backup.end())
|
|
|
|
throw Exception(ErrorCodes::BACKUP_ENTRY_NOT_FOUND, "No replica #{} in backup", replica_name);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
replica_name = std::to_string(replica_num);
|
|
|
|
if (std::find(replicas_in_backup.begin(), replicas_in_backup.end(), replica_name) == replicas_in_backup.end())
|
|
|
|
replica_name = replicas_in_backup.front();
|
|
|
|
}
|
2022-06-06 09:50:20 +00:00
|
|
|
root_path = root_path / "replicas" / replica_name;
|
2022-05-31 09:33:23 +00:00
|
|
|
root_paths_in_backup.push_back(root_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Revert the list of root paths, because we need it in the following order:
|
2022-06-06 09:50:20 +00:00
|
|
|
/// "/shards/<shard_num>/replicas/<replica_num>/" (first we search tables here)
|
|
|
|
/// "/shards/<shard_num>/" (then here)
|
|
|
|
/// "/" (and finally here)
|
2022-05-31 09:33:23 +00:00
|
|
|
std::reverse(root_paths_in_backup.begin(), root_paths_in_backup.end());
|
|
|
|
|
2022-06-06 09:50:20 +00:00
|
|
|
LOG_TRACE(
|
|
|
|
log,
|
|
|
|
"Will use paths in backup: {}",
|
|
|
|
boost::algorithm::join(
|
|
|
|
root_paths_in_backup
|
|
|
|
| boost::adaptors::transformed([](const fs::path & path) -> String { return doubleQuoteString(String{path}); }),
|
|
|
|
", "));
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
void RestorerFromBackup::findDatabasesAndTablesInBackup()
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
database_infos.clear();
|
|
|
|
table_infos.clear();
|
|
|
|
for (const auto & element : restore_query_elements)
|
|
|
|
{
|
|
|
|
switch (element.type)
|
|
|
|
{
|
|
|
|
case ASTBackupQuery::ElementType::TABLE:
|
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
findTableInBackup({element.database_name, element.table_name}, element.partitions);
|
2022-06-11 11:36:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ASTBackupQuery::ElementType::TEMPORARY_TABLE:
|
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
findTableInBackup({DatabaseCatalog::TEMPORARY_DATABASE, element.table_name}, element.partitions);
|
2022-05-31 09:33:23 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ASTBackupQuery::ElementType::DATABASE:
|
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
findDatabaseInBackup(element.database_name, element.except_tables);
|
2022-05-31 09:33:23 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-06-11 11:36:02 +00:00
|
|
|
case ASTBackupQuery::ElementType::ALL:
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
findEverythingInBackup(element.except_databases, element.except_tables);
|
2022-05-31 09:33:23 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_INFO(log, "Will restore {} databases and {} tables", database_infos.size(), table_infos.size());
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
void RestorerFromBackup::findTableInBackup(const QualifiedTableName & table_name_in_backup, const std::optional<ASTs> & partitions)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
bool is_temporary_table = (table_name_in_backup.database == DatabaseCatalog::TEMPORARY_DATABASE);
|
2022-06-09 16:19:54 +00:00
|
|
|
|
2022-06-06 09:50:20 +00:00
|
|
|
std::optional<fs::path> metadata_path;
|
|
|
|
std::optional<fs::path> root_path_in_use;
|
|
|
|
for (const auto & root_path_in_backup : root_paths_in_backup)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-09 16:19:54 +00:00
|
|
|
fs::path try_metadata_path;
|
|
|
|
if (is_temporary_table)
|
|
|
|
{
|
|
|
|
try_metadata_path
|
|
|
|
= root_path_in_backup / "temporary_tables" / "metadata" / (escapeForFileName(table_name_in_backup.table) + ".sql");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
try_metadata_path = root_path_in_backup / "metadata" / escapeForFileName(table_name_in_backup.database)
|
|
|
|
/ (escapeForFileName(table_name_in_backup.table) + ".sql");
|
|
|
|
}
|
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
if (backup->fileExists(try_metadata_path))
|
|
|
|
{
|
|
|
|
metadata_path = try_metadata_path;
|
2022-06-06 09:50:20 +00:00
|
|
|
root_path_in_use = root_path_in_backup;
|
2022-05-31 09:33:23 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!metadata_path)
|
2022-06-23 12:24:45 +00:00
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::BACKUP_ENTRY_NOT_FOUND,
|
|
|
|
"{} not found in backup",
|
|
|
|
tableNameWithTypeToString(table_name_in_backup.database, table_name_in_backup.table, true));
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
fs::path data_path_in_backup;
|
|
|
|
if (is_temporary_table)
|
|
|
|
{
|
|
|
|
data_path_in_backup = *root_path_in_use / "temporary_tables" / "data" / escapeForFileName(table_name_in_backup.table);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
data_path_in_backup
|
|
|
|
= *root_path_in_use / "data" / escapeForFileName(table_name_in_backup.database) / escapeForFileName(table_name_in_backup.table);
|
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
auto read_buffer = backup->readFile(*metadata_path)->getReadBuffer();
|
|
|
|
String create_query_str;
|
|
|
|
readStringUntilEOF(create_query_str, *read_buffer);
|
|
|
|
read_buffer.reset();
|
|
|
|
ParserCreateQuery create_parser;
|
|
|
|
ASTPtr create_table_query = parseQuery(create_parser, create_query_str, 0, DBMS_DEFAULT_MAX_PARSER_DEPTH);
|
2022-06-24 19:29:38 +00:00
|
|
|
renameDatabaseAndTableNameInCreateQuery(create_table_query, renaming_map, context->getGlobalContext());
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
QualifiedTableName table_name = renaming_map.getNewTableName(table_name_in_backup);
|
|
|
|
|
|
|
|
if (auto it = table_infos.find(table_name); it != table_infos.end())
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
const TableInfo & table_info = it->second;
|
|
|
|
if (table_info.create_table_query && (serializeAST(*table_info.create_table_query) != serializeAST(*create_table_query)))
|
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_TABLE,
|
2022-06-23 12:24:45 +00:00
|
|
|
"Extracted two different create queries for the same {}: {} and {}",
|
|
|
|
tableNameWithTypeToString(table_name.database, table_name.table, false),
|
2022-05-31 09:33:23 +00:00
|
|
|
serializeAST(*table_info.create_table_query),
|
|
|
|
serializeAST(*create_table_query));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
TableInfo & res_table_info = table_infos[table_name];
|
2022-05-31 09:33:23 +00:00
|
|
|
res_table_info.create_table_query = create_table_query;
|
2022-06-28 07:59:02 +00:00
|
|
|
res_table_info.is_predefined_table = DatabaseCatalog::instance().isPredefinedTable(StorageID{table_name.database, table_name.table});
|
2022-06-23 12:24:45 +00:00
|
|
|
res_table_info.dependencies = getDependenciesSetFromCreateQuery(context->getGlobalContext(), table_name, create_table_query);
|
2022-06-29 12:42:23 +00:00
|
|
|
res_table_info.has_data = backup->hasFiles(data_path_in_backup);
|
|
|
|
res_table_info.data_path_in_backup = data_path_in_backup;
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
if (partitions)
|
|
|
|
{
|
|
|
|
if (!res_table_info.partitions)
|
|
|
|
res_table_info.partitions.emplace();
|
|
|
|
insertAtEnd(*res_table_info.partitions, *partitions);
|
|
|
|
}
|
2022-06-18 22:01:08 +00:00
|
|
|
|
2022-06-27 13:39:12 +00:00
|
|
|
if (!restore_settings.structure_only && isSystemAccessTableName(table_name))
|
2022-06-18 22:01:08 +00:00
|
|
|
{
|
2022-06-29 20:44:05 +00:00
|
|
|
if (!access_restorer)
|
|
|
|
access_restorer = std::make_unique<AccessRestorerFromBackup>(backup, restore_settings);
|
|
|
|
access_restorer->addDataPath(data_path_in_backup, table_name);
|
2022-06-18 22:01:08 +00:00
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
void RestorerFromBackup::findDatabaseInBackup(const String & database_name_in_backup, const std::set<DatabaseAndTableName> & except_table_names)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-06 09:50:20 +00:00
|
|
|
std::optional<fs::path> metadata_path;
|
2022-05-31 09:33:23 +00:00
|
|
|
std::unordered_set<String> table_names_in_backup;
|
2022-06-06 09:50:20 +00:00
|
|
|
for (const auto & root_path_in_backup : root_paths_in_backup)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
fs::path try_metadata_path, try_tables_metadata_path;
|
|
|
|
if (database_name_in_backup == DatabaseCatalog::TEMPORARY_DATABASE)
|
|
|
|
{
|
|
|
|
try_tables_metadata_path = root_path_in_backup / "temporary_tables" / "metadata";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
try_metadata_path = root_path_in_backup / "metadata" / (escapeForFileName(database_name_in_backup) + ".sql");
|
|
|
|
try_tables_metadata_path = root_path_in_backup / "metadata" / escapeForFileName(database_name_in_backup);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!metadata_path && !try_metadata_path.empty() && backup->fileExists(try_metadata_path))
|
2022-05-31 09:33:23 +00:00
|
|
|
metadata_path = try_metadata_path;
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
Strings file_names = backup->listFiles(try_tables_metadata_path);
|
2022-05-31 09:33:23 +00:00
|
|
|
for (const String & file_name : file_names)
|
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
if (!file_name.ends_with(".sql"))
|
2022-05-31 09:33:23 +00:00
|
|
|
continue;
|
2022-06-23 12:24:45 +00:00
|
|
|
String file_name_without_ext = file_name.substr(0, file_name.length() - strlen(".sql"));
|
2022-05-31 09:33:23 +00:00
|
|
|
table_names_in_backup.insert(unescapeForFileName(file_name_without_ext));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
if (!metadata_path && table_names_in_backup.empty())
|
2022-05-31 09:33:23 +00:00
|
|
|
throw Exception(ErrorCodes::BACKUP_ENTRY_NOT_FOUND, "Database {} not found in backup", backQuoteIfNeed(database_name_in_backup));
|
|
|
|
|
|
|
|
if (metadata_path)
|
|
|
|
{
|
|
|
|
auto read_buffer = backup->readFile(*metadata_path)->getReadBuffer();
|
|
|
|
String create_query_str;
|
|
|
|
readStringUntilEOF(create_query_str, *read_buffer);
|
|
|
|
read_buffer.reset();
|
|
|
|
ParserCreateQuery create_parser;
|
2022-06-18 11:56:04 +00:00
|
|
|
ASTPtr create_database_query = parseQuery(create_parser, create_query_str, 0, DBMS_DEFAULT_MAX_PARSER_DEPTH);
|
2022-06-24 19:29:38 +00:00
|
|
|
renameDatabaseAndTableNameInCreateQuery(create_database_query, renaming_map, context->getGlobalContext());
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-18 11:56:04 +00:00
|
|
|
String database_name = renaming_map.getNewDatabaseName(database_name_in_backup);
|
|
|
|
DatabaseInfo & database_info = database_infos[database_name];
|
2022-06-19 13:48:52 +00:00
|
|
|
|
2022-06-18 11:56:04 +00:00
|
|
|
if (database_info.create_database_query && (serializeAST(*database_info.create_database_query) != serializeAST(*create_database_query)))
|
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_DATABASE,
|
|
|
|
"Extracted two different create queries for the same database {}: {} and {}",
|
|
|
|
backQuoteIfNeed(database_name),
|
|
|
|
serializeAST(*database_info.create_database_query),
|
|
|
|
serializeAST(*create_database_query));
|
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-18 11:56:04 +00:00
|
|
|
database_info.create_database_query = create_database_query;
|
2022-06-28 07:59:02 +00:00
|
|
|
database_info.is_predefined_database = DatabaseCatalog::isPredefinedDatabase(database_name);
|
2022-06-18 11:56:04 +00:00
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
|
|
|
|
for (const String & table_name_in_backup : table_names_in_backup)
|
|
|
|
{
|
2022-06-11 11:36:02 +00:00
|
|
|
if (except_table_names.contains({database_name_in_backup, table_name_in_backup}))
|
2022-05-31 09:33:23 +00:00
|
|
|
continue;
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
findTableInBackup({database_name_in_backup, table_name_in_backup}, /* partitions= */ {});
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
void RestorerFromBackup::findEverythingInBackup(const std::set<String> & except_database_names, const std::set<DatabaseAndTableName> & except_table_names)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
std::unordered_set<String> database_names_in_backup;
|
2022-06-18 11:56:04 +00:00
|
|
|
|
2022-06-06 09:50:20 +00:00
|
|
|
for (const auto & root_path_in_backup : root_paths_in_backup)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-06 09:50:20 +00:00
|
|
|
Strings file_names = backup->listFiles(root_path_in_backup / "metadata");
|
2022-05-31 09:33:23 +00:00
|
|
|
for (String & file_name : file_names)
|
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
if (file_name.ends_with(".sql"))
|
|
|
|
file_name.resize(file_name.length() - strlen(".sql"));
|
2022-05-31 09:33:23 +00:00
|
|
|
database_names_in_backup.emplace(unescapeForFileName(file_name));
|
|
|
|
}
|
2022-06-18 11:56:04 +00:00
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
if (backup->hasFiles(root_path_in_backup / "temporary_tables" / "metadata"))
|
|
|
|
database_names_in_backup.emplace(DatabaseCatalog::TEMPORARY_DATABASE);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (const String & database_name_in_backup : database_names_in_backup)
|
|
|
|
{
|
|
|
|
if (except_database_names.contains(database_name_in_backup))
|
|
|
|
continue;
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
findDatabaseInBackup(database_name_in_backup, except_table_names);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
void RestorerFromBackup::checkAccessForObjectsFoundInBackup() const
|
2022-06-18 22:01:08 +00:00
|
|
|
{
|
|
|
|
AccessRightsElements required_access;
|
2022-06-28 07:59:02 +00:00
|
|
|
for (const auto & [database_name, database_info] : database_infos)
|
2022-06-18 22:01:08 +00:00
|
|
|
{
|
2022-06-28 07:59:02 +00:00
|
|
|
if (database_info.is_predefined_database)
|
2022-06-18 22:01:08 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
AccessFlags flags;
|
|
|
|
|
|
|
|
if (restore_settings.create_database != RestoreDatabaseCreationMode::kMustExist)
|
|
|
|
flags |= AccessType::CREATE_DATABASE;
|
|
|
|
|
|
|
|
if (!flags)
|
|
|
|
flags = AccessType::SHOW_DATABASES;
|
|
|
|
|
|
|
|
required_access.emplace_back(flags, database_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto & [table_name, table_info] : table_infos)
|
|
|
|
{
|
2022-06-28 07:59:02 +00:00
|
|
|
if (table_info.is_predefined_table)
|
2022-06-27 13:39:12 +00:00
|
|
|
{
|
|
|
|
if (isSystemFunctionsTableName(table_name))
|
|
|
|
{
|
|
|
|
/// CREATE_FUNCTION privilege is required to restore the "system.functions" table.
|
2022-06-29 12:42:23 +00:00
|
|
|
if (!restore_settings.structure_only && table_info.has_data)
|
2022-06-27 13:39:12 +00:00
|
|
|
required_access.emplace_back(AccessType::CREATE_FUNCTION);
|
|
|
|
}
|
|
|
|
/// Privileges required to restore ACL system tables are checked separately
|
|
|
|
/// (see access_restore_task->getRequiredAccess() below).
|
2022-06-18 22:01:08 +00:00
|
|
|
continue;
|
2022-06-27 13:39:12 +00:00
|
|
|
}
|
2022-06-18 22:01:08 +00:00
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
if (table_name.database == DatabaseCatalog::TEMPORARY_DATABASE)
|
2022-06-18 22:01:08 +00:00
|
|
|
{
|
|
|
|
if (restore_settings.create_table != RestoreTableCreationMode::kMustExist)
|
|
|
|
required_access.emplace_back(AccessType::CREATE_TEMPORARY_TABLE);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
AccessFlags flags;
|
|
|
|
const ASTCreateQuery & create = table_info.create_table_query->as<const ASTCreateQuery &>();
|
|
|
|
|
|
|
|
if (restore_settings.create_table != RestoreTableCreationMode::kMustExist)
|
|
|
|
{
|
|
|
|
if (create.is_dictionary)
|
|
|
|
flags |= AccessType::CREATE_DICTIONARY;
|
|
|
|
else if (create.is_ordinary_view || create.is_materialized_view || create.is_live_view)
|
|
|
|
flags |= AccessType::CREATE_VIEW;
|
|
|
|
else
|
|
|
|
flags |= AccessType::CREATE_TABLE;
|
|
|
|
}
|
|
|
|
|
2022-06-29 12:42:23 +00:00
|
|
|
if (!restore_settings.structure_only && table_info.has_data)
|
2022-06-18 22:01:08 +00:00
|
|
|
{
|
|
|
|
flags |= AccessType::INSERT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!flags)
|
|
|
|
{
|
|
|
|
if (create.is_dictionary)
|
|
|
|
flags = AccessType::SHOW_DICTIONARIES;
|
|
|
|
else
|
|
|
|
flags = AccessType::SHOW_TABLES;
|
|
|
|
}
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
required_access.emplace_back(flags, table_name.database, table_name.table);
|
2022-06-18 22:01:08 +00:00
|
|
|
}
|
|
|
|
|
2022-06-29 20:44:05 +00:00
|
|
|
if (access_restorer)
|
|
|
|
insertAtEnd(required_access, access_restorer->getRequiredAccess());
|
2022-06-18 22:01:08 +00:00
|
|
|
|
|
|
|
/// We convert to AccessRights and back to check access rights in a predictable way
|
|
|
|
/// (some elements could be duplicated or not sorted).
|
|
|
|
required_access = AccessRights{required_access}.getElements();
|
|
|
|
|
|
|
|
context->checkAccess(required_access);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
void RestorerFromBackup::createDatabases()
|
|
|
|
{
|
|
|
|
for (const auto & [database_name, database_info] : database_infos)
|
|
|
|
{
|
2022-06-09 16:19:54 +00:00
|
|
|
bool need_create_database = (restore_settings.create_database != RestoreDatabaseCreationMode::kMustExist);
|
2022-06-28 07:59:02 +00:00
|
|
|
if (database_info.is_predefined_database)
|
2022-06-09 16:19:54 +00:00
|
|
|
need_create_database = false; /// Predefined databases always exist.
|
|
|
|
|
|
|
|
if (need_create_database)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
/// Execute CREATE DATABASE query.
|
|
|
|
auto create_database_query = database_info.create_database_query;
|
|
|
|
if (restore_settings.create_table == RestoreTableCreationMode::kCreateIfNotExists)
|
|
|
|
{
|
|
|
|
create_database_query = create_database_query->clone();
|
|
|
|
create_database_query->as<ASTCreateQuery &>().if_not_exists = true;
|
|
|
|
}
|
2022-06-08 02:11:41 +00:00
|
|
|
LOG_TRACE(log, "Creating database {}: {}", backQuoteIfNeed(database_name), serializeAST(*create_database_query));
|
2022-06-23 12:24:45 +00:00
|
|
|
InterpreterCreateQuery interpreter{create_database_query, context};
|
|
|
|
interpreter.setInternal(true);
|
|
|
|
interpreter.execute();
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DatabasePtr database = DatabaseCatalog::instance().getDatabase(database_name);
|
|
|
|
|
2022-06-28 07:59:02 +00:00
|
|
|
if (!restore_settings.allow_different_database_def && !database_info.is_predefined_database)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
/// Check that the database's definition is the same as expected.
|
2022-06-24 19:29:38 +00:00
|
|
|
ASTPtr create_database_query = database->getCreateDatabaseQuery();
|
|
|
|
adjustCreateQueryForBackup(create_database_query, context->getGlobalContext(), nullptr);
|
2022-05-31 09:33:23 +00:00
|
|
|
ASTPtr expected_create_query = database_info.create_database_query;
|
2022-06-08 02:11:41 +00:00
|
|
|
if (serializeAST(*create_database_query) != serializeAST(*expected_create_query))
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_DATABASE,
|
|
|
|
"The database {} has a different definition: {} "
|
|
|
|
"comparing to its definition in the backup: {}",
|
|
|
|
backQuoteIfNeed(database_name),
|
2022-06-08 02:11:41 +00:00
|
|
|
serializeAST(*create_database_query),
|
2022-05-31 09:33:23 +00:00
|
|
|
serializeAST(*expected_create_query));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RestorerFromBackup::createTables()
|
|
|
|
{
|
2022-06-06 19:15:17 +00:00
|
|
|
while (true)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-06 19:15:17 +00:00
|
|
|
/// We need to create tables considering their dependencies.
|
|
|
|
auto tables_to_create = findTablesWithoutDependencies();
|
|
|
|
if (tables_to_create.empty())
|
|
|
|
break; /// We've already created all the tables.
|
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
for (const auto & table_name : tables_to_create)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
auto & table_info = table_infos.at(table_name);
|
2022-06-09 16:19:54 +00:00
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
DatabasePtr database = DatabaseCatalog::instance().getDatabase(table_name.database);
|
2022-06-09 16:19:54 +00:00
|
|
|
|
|
|
|
bool need_create_table = (restore_settings.create_table != RestoreTableCreationMode::kMustExist);
|
2022-06-28 07:59:02 +00:00
|
|
|
if (table_info.is_predefined_table)
|
|
|
|
need_create_table = false; /// Predefined tables always exist.
|
2022-06-09 16:19:54 +00:00
|
|
|
|
|
|
|
if (need_create_table)
|
2022-06-06 19:15:17 +00:00
|
|
|
{
|
|
|
|
auto create_table_query = table_info.create_table_query;
|
|
|
|
if (restore_settings.create_table == RestoreTableCreationMode::kCreateIfNotExists)
|
|
|
|
{
|
|
|
|
create_table_query = create_table_query->clone();
|
|
|
|
create_table_query->as<ASTCreateQuery &>().if_not_exists = true;
|
|
|
|
}
|
2022-06-30 08:10:12 +00:00
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
LOG_TRACE(
|
|
|
|
log,
|
2022-06-23 12:24:45 +00:00
|
|
|
"Creating {}: {}",
|
|
|
|
tableNameWithTypeToString(table_name.database, table_name.table, false),
|
2022-06-09 16:19:54 +00:00
|
|
|
serializeAST(*create_table_query));
|
|
|
|
|
2022-06-28 07:59:02 +00:00
|
|
|
/// Execute CREATE TABLE query (we call IDatabase::createTableRestoredFromBackup() to allow the database to do some
|
|
|
|
/// database-specific things).
|
2022-06-23 18:49:44 +00:00
|
|
|
database->createTableRestoredFromBackup(
|
|
|
|
create_table_query,
|
|
|
|
context,
|
|
|
|
restore_coordination,
|
|
|
|
std::chrono::duration_cast<std::chrono::milliseconds>(create_table_timeout).count());
|
2022-06-06 19:15:17 +00:00
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-06 19:15:17 +00:00
|
|
|
table_info.created = true;
|
2022-06-09 16:19:54 +00:00
|
|
|
|
2022-06-23 12:24:45 +00:00
|
|
|
auto resolved_id = (table_name.database == DatabaseCatalog::TEMPORARY_DATABASE)
|
|
|
|
? context->resolveStorageID(StorageID{"", table_name.table}, Context::ResolveExternal)
|
|
|
|
: context->resolveStorageID(StorageID{table_name.database, table_name.table}, Context::ResolveGlobal);
|
2022-06-09 16:19:54 +00:00
|
|
|
|
|
|
|
auto storage = database->getTable(resolved_id.table_name, context);
|
|
|
|
table_info.storage = storage;
|
|
|
|
table_info.table_lock = storage->lockForShare(context->getInitialQueryId(), context->getSettingsRef().lock_acquire_timeout);
|
2022-06-06 19:15:17 +00:00
|
|
|
|
2022-06-28 07:59:02 +00:00
|
|
|
if (!restore_settings.allow_different_table_def && !table_info.is_predefined_table)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-22 22:56:41 +00:00
|
|
|
ASTPtr create_table_query = database->getCreateTableQuery(resolved_id.table_name, context);
|
2022-06-24 19:29:38 +00:00
|
|
|
adjustCreateQueryForBackup(create_table_query, context->getGlobalContext(), nullptr);
|
2022-06-06 19:15:17 +00:00
|
|
|
ASTPtr expected_create_query = table_info.create_table_query;
|
|
|
|
if (serializeAST(*create_table_query) != serializeAST(*expected_create_query))
|
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_TABLE,
|
2022-06-23 12:24:45 +00:00
|
|
|
"{} has a different definition: {} "
|
2022-06-06 19:15:17 +00:00
|
|
|
"comparing to its definition in the backup: {}",
|
2022-06-23 12:24:45 +00:00
|
|
|
tableNameWithTypeToString(table_name.database, table_name.table, true),
|
2022-06-06 19:15:17 +00:00
|
|
|
serializeAST(*create_table_query),
|
|
|
|
serializeAST(*expected_create_query));
|
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-06 19:15:17 +00:00
|
|
|
if (!restore_settings.structure_only)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-06 19:15:17 +00:00
|
|
|
const auto & data_path_in_backup = table_info.data_path_in_backup;
|
|
|
|
const auto & partitions = table_info.partitions;
|
2022-06-29 12:42:23 +00:00
|
|
|
if (partitions && !storage->supportsBackupPartition())
|
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_TABLE,
|
|
|
|
"Table engine {} doesn't support partitions, cannot restore {}",
|
|
|
|
storage->getName(),
|
|
|
|
tableNameWithTypeToString(table_name.database, table_name.table, false));
|
|
|
|
}
|
|
|
|
|
2022-06-06 19:15:17 +00:00
|
|
|
storage->restoreDataFromBackup(*this, data_path_in_backup, partitions);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
}
|
2022-06-06 19:15:17 +00:00
|
|
|
}
|
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
|
2022-06-06 19:15:17 +00:00
|
|
|
/// Returns the list of tables without dependencies or those which dependencies have been created before.
|
2022-06-23 12:24:45 +00:00
|
|
|
std::vector<QualifiedTableName> RestorerFromBackup::findTablesWithoutDependencies() const
|
2022-06-06 19:15:17 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
std::vector<QualifiedTableName> tables_without_dependencies;
|
2022-06-06 19:15:17 +00:00
|
|
|
bool all_tables_created = true;
|
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
for (const auto & [key, table_info] : table_infos)
|
2022-06-06 19:15:17 +00:00
|
|
|
{
|
|
|
|
if (table_info.created)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/// Found a table which is not created yet.
|
|
|
|
all_tables_created = false;
|
|
|
|
|
|
|
|
/// Check if all dependencies have been created before.
|
|
|
|
bool all_dependencies_met = true;
|
|
|
|
for (const auto & dependency : table_info.dependencies)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 12:24:45 +00:00
|
|
|
auto it = table_infos.find(dependency);
|
2022-06-06 19:15:17 +00:00
|
|
|
if ((it != table_infos.end()) && !it->second.created)
|
|
|
|
{
|
|
|
|
all_dependencies_met = false;
|
|
|
|
break;
|
|
|
|
}
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
2022-06-06 19:15:17 +00:00
|
|
|
|
|
|
|
if (all_dependencies_met)
|
2022-06-09 16:19:54 +00:00
|
|
|
tables_without_dependencies.push_back(key);
|
2022-06-06 19:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!tables_without_dependencies.empty())
|
|
|
|
return tables_without_dependencies;
|
|
|
|
|
|
|
|
if (all_tables_created)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
/// Cyclic dependency? We'll try to create those tables anyway but probably it's going to fail.
|
2022-06-23 12:24:45 +00:00
|
|
|
std::vector<QualifiedTableName> tables_with_cyclic_dependencies;
|
2022-06-09 16:19:54 +00:00
|
|
|
for (const auto & [key, table_info] : table_infos)
|
2022-06-06 19:15:17 +00:00
|
|
|
{
|
|
|
|
if (!table_info.created)
|
2022-06-09 16:19:54 +00:00
|
|
|
tables_with_cyclic_dependencies.push_back(key);
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
2022-06-06 19:15:17 +00:00
|
|
|
|
|
|
|
/// Only show a warning here, proper exception will be thrown later on creating those tables.
|
|
|
|
LOG_WARNING(
|
|
|
|
log,
|
|
|
|
"Some tables have cyclic dependency from each other: {}",
|
|
|
|
boost::algorithm::join(
|
|
|
|
tables_with_cyclic_dependencies
|
2022-06-23 12:24:45 +00:00
|
|
|
| boost::adaptors::transformed([](const QualifiedTableName & table_name) -> String { return table_name.getFullName(); }),
|
2022-06-06 19:15:17 +00:00
|
|
|
", "));
|
|
|
|
|
|
|
|
return tables_with_cyclic_dependencies;
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
void RestorerFromBackup::addDataRestoreTask(DataRestoreTask && new_task)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 17:45:36 +00:00
|
|
|
if (current_status == kInsertingDataToTablesStatus)
|
2022-05-31 09:33:23 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Adding data-restoring tasks is not allowed");
|
2022-06-09 16:19:54 +00:00
|
|
|
data_restore_tasks.push_back(std::move(new_task));
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 16:19:54 +00:00
|
|
|
void RestorerFromBackup::addDataRestoreTasks(DataRestoreTasks && new_tasks)
|
2022-05-31 09:33:23 +00:00
|
|
|
{
|
2022-06-23 17:45:36 +00:00
|
|
|
if (current_status == kInsertingDataToTablesStatus)
|
2022-05-31 09:33:23 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Adding data-restoring tasks is not allowed");
|
2022-06-09 16:19:54 +00:00
|
|
|
insertAtEnd(data_restore_tasks, std::move(new_tasks));
|
2022-05-31 09:33:23 +00:00
|
|
|
}
|
|
|
|
|
2022-06-23 18:49:44 +00:00
|
|
|
RestorerFromBackup::DataRestoreTasks RestorerFromBackup::getDataRestoreTasks()
|
|
|
|
{
|
2022-06-29 20:44:05 +00:00
|
|
|
if (data_restore_tasks.empty())
|
2022-06-23 18:49:44 +00:00
|
|
|
return {};
|
|
|
|
|
|
|
|
LOG_TRACE(log, "Will insert data to tables");
|
|
|
|
|
|
|
|
/// Storages and table locks must exist while we're executing data restoring tasks.
|
|
|
|
auto storages = std::make_shared<std::vector<StoragePtr>>();
|
|
|
|
auto table_locks = std::make_shared<std::vector<TableLockHolder>>();
|
|
|
|
storages->reserve(table_infos.size());
|
|
|
|
table_locks->reserve(table_infos.size());
|
|
|
|
for (const auto & table_info : table_infos | boost::adaptors::map_values)
|
|
|
|
{
|
|
|
|
storages->push_back(table_info.storage);
|
|
|
|
table_locks->push_back(table_info.table_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
DataRestoreTasks res_tasks;
|
|
|
|
for (const auto & task : data_restore_tasks)
|
|
|
|
res_tasks.push_back([task, storages, table_locks] { task(); });
|
|
|
|
|
|
|
|
return res_tasks;
|
|
|
|
}
|
|
|
|
|
2022-06-29 20:44:05 +00:00
|
|
|
std::vector<std::pair<UUID, AccessEntityPtr>> RestorerFromBackup::getAccessEntitiesToRestore()
|
|
|
|
{
|
|
|
|
if (!access_restorer || access_restored)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
/// getAccessEntitiesToRestore() will return entities only when called first time (we don't want to restore the same entities again).
|
|
|
|
access_restored = true;
|
|
|
|
|
|
|
|
return access_restorer->getAccessEntities(context->getAccessControl());
|
|
|
|
}
|
|
|
|
|
2022-05-31 09:33:23 +00:00
|
|
|
void RestorerFromBackup::throwTableIsNotEmpty(const StorageID & storage_id)
|
|
|
|
{
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::CANNOT_RESTORE_TABLE,
|
|
|
|
"Cannot restore the table {} because it already contains some data. You can set structure_only=true or "
|
|
|
|
"allow_non_empty_tables=true to overcome that in the way you want",
|
|
|
|
storage_id.getFullTableName());
|
|
|
|
}
|
|
|
|
}
|