mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-11-18 13:42:02 +00:00
282 lines
11 KiB
C++
282 lines
11 KiB
C++
#include <Databases/DatabaseWithDictionaries.h>
|
|
#include <Common/StatusInfo.h>
|
|
#include <Common/ExternalLoaderStatus.h>
|
|
#include <Interpreters/ExternalDictionariesLoader.h>
|
|
#include <Interpreters/ExternalLoaderTempConfigRepository.h>
|
|
#include <Interpreters/ExternalLoaderDatabaseConfigRepository.h>
|
|
#include <Dictionaries/getDictionaryConfigurationFromAST.h>
|
|
#include <Interpreters/Context.h>
|
|
#include <Storages/StorageDictionary.h>
|
|
#include <IO/WriteBufferFromFile.h>
|
|
#include <Poco/File.h>
|
|
#include <ext/scope_guard.h>
|
|
|
|
|
|
namespace CurrentStatusInfo
|
|
{
|
|
extern const Status DictionaryStatus;
|
|
}
|
|
|
|
namespace DB
|
|
{
|
|
|
|
namespace ErrorCodes
|
|
{
|
|
extern const int CANNOT_GET_CREATE_DICTIONARY_QUERY;
|
|
extern const int TABLE_ALREADY_EXISTS;
|
|
extern const int UNKNOWN_TABLE;
|
|
extern const int DICTIONARY_ALREADY_EXISTS;
|
|
}
|
|
|
|
|
|
void DatabaseWithDictionaries::attachDictionary(const String & dictionary_name, const Context & context)
|
|
{
|
|
String full_name = getDatabaseName() + "." + dictionary_name;
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
if (!dictionaries.emplace(dictionary_name).second)
|
|
throw Exception("Dictionary " + full_name + " already exists.", ErrorCodes::DICTIONARY_ALREADY_EXISTS);
|
|
}
|
|
|
|
CurrentStatusInfo::set(CurrentStatusInfo::DictionaryStatus, full_name, static_cast<Int8>(ExternalLoaderStatus::NOT_LOADED));
|
|
/// ExternalLoader::reloadConfig() will find out that the dictionary's config has been added
|
|
/// and in case `dictionaries_lazy_load == false` it will load the dictionary.
|
|
const auto & external_loader = context.getExternalDictionariesLoader();
|
|
external_loader.reloadConfig(getDatabaseName(), full_name);
|
|
}
|
|
|
|
void DatabaseWithDictionaries::detachDictionary(const String & dictionary_name, const Context & context)
|
|
{
|
|
String full_name = getDatabaseName() + "." + dictionary_name;
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
auto it = dictionaries.find(dictionary_name);
|
|
if (it == dictionaries.end())
|
|
throw Exception("Dictionary " + full_name + " doesn't exist.", ErrorCodes::UNKNOWN_TABLE);
|
|
dictionaries.erase(it);
|
|
}
|
|
|
|
CurrentStatusInfo::unset(CurrentStatusInfo::DictionaryStatus, full_name);
|
|
/// ExternalLoader::reloadConfig() will find out that the dictionary's config has been removed
|
|
/// and therefore it will unload the dictionary.
|
|
const auto & external_loader = context.getExternalDictionariesLoader();
|
|
external_loader.reloadConfig(getDatabaseName(), full_name);
|
|
|
|
}
|
|
|
|
void DatabaseWithDictionaries::createDictionary(const Context & context, const String & dictionary_name, const ASTPtr & query)
|
|
{
|
|
const auto & settings = context.getSettingsRef();
|
|
|
|
/** The code is based on the assumption that all threads share the same order of operations:
|
|
* - create the .sql.tmp file;
|
|
* - add the dictionary to ExternalDictionariesLoader;
|
|
* - load the dictionary in case dictionaries_lazy_load == false;
|
|
* - attach the dictionary;
|
|
* - rename .sql.tmp to .sql.
|
|
*/
|
|
|
|
/// A race condition would be possible if a dictionary with the same name is simultaneously created using CREATE and using ATTACH.
|
|
/// But there is protection from it - see using DDLGuard in InterpreterCreateQuery.
|
|
if (isDictionaryExist(context, dictionary_name))
|
|
throw Exception("Dictionary " + backQuote(getDatabaseName()) + "." + backQuote(dictionary_name) + " already exists.", ErrorCodes::DICTIONARY_ALREADY_EXISTS);
|
|
|
|
/// A dictionary with the same full name could be defined in *.xml config files.
|
|
String full_name = getDatabaseName() + "." + dictionary_name;
|
|
const auto & external_loader = context.getExternalDictionariesLoader();
|
|
if (external_loader.getCurrentStatus(full_name) != ExternalLoader::Status::NOT_EXIST)
|
|
throw Exception(
|
|
"Dictionary " + backQuote(getDatabaseName()) + "." + backQuote(dictionary_name) + " already exists.",
|
|
ErrorCodes::DICTIONARY_ALREADY_EXISTS);
|
|
|
|
if (isTableExist(context, dictionary_name))
|
|
throw Exception("Table " + backQuote(getDatabaseName()) + "." + backQuote(dictionary_name) + " already exists.", ErrorCodes::TABLE_ALREADY_EXISTS);
|
|
|
|
|
|
String dictionary_metadata_path = getObjectMetadataPath(dictionary_name);
|
|
String dictionary_metadata_tmp_path = dictionary_metadata_path + ".tmp";
|
|
String statement = getObjectDefinitionFromCreateQuery(query);
|
|
|
|
{
|
|
/// Exclusive flags guarantees, that table is not created right now in another thread. Otherwise, exception will be thrown.
|
|
WriteBufferFromFile out(dictionary_metadata_tmp_path, statement.size(), O_WRONLY | O_CREAT | O_EXCL);
|
|
writeString(statement, out);
|
|
out.next();
|
|
if (settings.fsync_metadata)
|
|
out.sync();
|
|
out.close();
|
|
}
|
|
|
|
bool succeeded = false;
|
|
SCOPE_EXIT({
|
|
if (!succeeded)
|
|
Poco::File(dictionary_metadata_tmp_path).remove();
|
|
});
|
|
|
|
/// Add a temporary repository containing the dictionary.
|
|
/// We need this temp repository to try loading the dictionary before actually attaching it to the database.
|
|
auto temp_repository
|
|
= const_cast<ExternalDictionariesLoader &>(external_loader) /// the change of ExternalDictionariesLoader is temporary
|
|
.addConfigRepository(std::make_unique<ExternalLoaderTempConfigRepository>(
|
|
getDatabaseName(), dictionary_metadata_tmp_path, getDictionaryConfigurationFromAST(query->as<const ASTCreateQuery &>())));
|
|
|
|
bool lazy_load = context.getConfigRef().getBool("dictionaries_lazy_load", true);
|
|
if (!lazy_load)
|
|
{
|
|
/// load() is called here to force loading the dictionary, wait until the loading is finished,
|
|
/// and throw an exception if the loading is failed.
|
|
external_loader.load(full_name);
|
|
}
|
|
|
|
attachDictionary(dictionary_name, context);
|
|
SCOPE_EXIT({
|
|
if (!succeeded)
|
|
detachDictionary(dictionary_name, context);
|
|
});
|
|
|
|
/// If it was ATTACH query and file with dictionary metadata already exist
|
|
/// (so, ATTACH is done after DETACH), then rename atomically replaces old file with new one.
|
|
Poco::File(dictionary_metadata_tmp_path).renameTo(dictionary_metadata_path);
|
|
|
|
/// ExternalDictionariesLoader doesn't know we renamed the metadata path.
|
|
/// So we have to manually call reloadConfig() here.
|
|
external_loader.reloadConfig(getDatabaseName(), full_name);
|
|
|
|
/// Everything's ok.
|
|
succeeded = true;
|
|
}
|
|
|
|
void DatabaseWithDictionaries::removeDictionary(const Context & context, const String & dictionary_name)
|
|
{
|
|
detachDictionary(dictionary_name, context);
|
|
|
|
String dictionary_metadata_path = getObjectMetadataPath(dictionary_name);
|
|
|
|
try
|
|
{
|
|
Poco::File(dictionary_metadata_path).remove();
|
|
CurrentStatusInfo::unset(CurrentStatusInfo::DictionaryStatus, getDatabaseName() + "." + dictionary_name);
|
|
}
|
|
catch (...)
|
|
{
|
|
/// If remove was not possible for some reason
|
|
attachDictionary(dictionary_name, context);
|
|
throw;
|
|
}
|
|
}
|
|
|
|
StoragePtr DatabaseWithDictionaries::tryGetTable(const Context & context, const String & table_name) const
|
|
{
|
|
if (auto table_ptr = DatabaseWithOwnTablesBase::tryGetTable(context, table_name))
|
|
return table_ptr;
|
|
|
|
if (isDictionaryExist(context, table_name))
|
|
/// We don't need lock database here, because database doesn't store dictionary itself
|
|
/// just metadata
|
|
return getDictionaryStorage(table_name);
|
|
|
|
return {};
|
|
}
|
|
|
|
DatabaseTablesIteratorPtr DatabaseWithDictionaries::getTablesWithDictionaryTablesIterator(
|
|
const FilterByNameFunction & filter_by_dictionary_name)
|
|
{
|
|
/// NOTE: it's not atomic
|
|
auto tables_it = getTablesIterator(filter_by_dictionary_name);
|
|
auto dictionaries_it = getDictionariesIterator(filter_by_dictionary_name);
|
|
|
|
Tables result;
|
|
while (tables_it && tables_it->isValid())
|
|
{
|
|
result.emplace(tables_it->name(), tables_it->table());
|
|
tables_it->next();
|
|
}
|
|
|
|
while (dictionaries_it && dictionaries_it->isValid())
|
|
{
|
|
auto table_name = dictionaries_it->name();
|
|
auto table_ptr = getDictionaryStorage(table_name);
|
|
if (table_ptr)
|
|
result.emplace(table_name, table_ptr);
|
|
dictionaries_it->next();
|
|
}
|
|
|
|
return std::make_unique<DatabaseTablesSnapshotIterator>(result);
|
|
}
|
|
|
|
DatabaseDictionariesIteratorPtr DatabaseWithDictionaries::getDictionariesIterator(const FilterByNameFunction & filter_by_dictionary_name)
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
if (!filter_by_dictionary_name)
|
|
return std::make_unique<DatabaseDictionariesSnapshotIterator>(dictionaries);
|
|
|
|
Dictionaries filtered_dictionaries;
|
|
for (const auto & dictionary_name : dictionaries)
|
|
if (filter_by_dictionary_name(dictionary_name))
|
|
filtered_dictionaries.emplace(dictionary_name);
|
|
return std::make_unique<DatabaseDictionariesSnapshotIterator>(std::move(filtered_dictionaries));
|
|
}
|
|
|
|
bool DatabaseWithDictionaries::isDictionaryExist(const Context & /*context*/, const String & dictionary_name) const
|
|
{
|
|
std::lock_guard lock(mutex);
|
|
return dictionaries.find(dictionary_name) != dictionaries.end();
|
|
}
|
|
|
|
StoragePtr DatabaseWithDictionaries::getDictionaryStorage(const String & table_name) const
|
|
{
|
|
auto dict_name = database_name + "." + table_name;
|
|
const auto & external_loader = global_context.getExternalDictionariesLoader();
|
|
auto dict_ptr = external_loader.tryGetDictionary(dict_name);
|
|
if (dict_ptr)
|
|
{
|
|
const DictionaryStructure & dictionary_structure = dict_ptr->getStructure();
|
|
auto columns = StorageDictionary::getNamesAndTypes(dictionary_structure);
|
|
return StorageDictionary::create(StorageID(database_name, table_name), ColumnsDescription{columns}, global_context, true, dict_name);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
ASTPtr DatabaseWithDictionaries::getCreateDictionaryQueryImpl(
|
|
const Context & context,
|
|
const String & dictionary_name,
|
|
bool throw_on_error) const
|
|
{
|
|
ASTPtr ast;
|
|
|
|
auto dictionary_metadata_path = getObjectMetadataPath(dictionary_name);
|
|
ast = getCreateQueryFromMetadata(context, dictionary_metadata_path, throw_on_error);
|
|
if (!ast && throw_on_error)
|
|
{
|
|
/// Handle system.* tables for which there are no table.sql files.
|
|
bool has_dictionary = isDictionaryExist(context, dictionary_name);
|
|
|
|
auto msg = has_dictionary ? "There is no CREATE DICTIONARY query for table " : "There is no metadata file for dictionary ";
|
|
|
|
throw Exception(msg + backQuote(dictionary_name), ErrorCodes::CANNOT_GET_CREATE_DICTIONARY_QUERY);
|
|
}
|
|
|
|
return ast;
|
|
}
|
|
|
|
void DatabaseWithDictionaries::shutdown()
|
|
{
|
|
detachFromExternalDictionariesLoader();
|
|
DatabaseOnDisk::shutdown();
|
|
}
|
|
|
|
DatabaseWithDictionaries::~DatabaseWithDictionaries() = default;
|
|
|
|
void DatabaseWithDictionaries::attachToExternalDictionariesLoader(Context & context)
|
|
{
|
|
database_as_config_repo_for_external_loader = context.getExternalDictionariesLoader().addConfigRepository(
|
|
std::make_unique<ExternalLoaderDatabaseConfigRepository>(*this, context));
|
|
}
|
|
|
|
void DatabaseWithDictionaries::detachFromExternalDictionariesLoader()
|
|
{
|
|
database_as_config_repo_for_external_loader = {};
|
|
}
|
|
|
|
}
|