#include "config.h" #if USE_AWS_S3 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace fs = std::filesystem; namespace DB { static const std::unordered_set optional_configuration_keys = { "url", "access_key_id", "secret_access_key", "no_sign_request" }; namespace ErrorCodes { extern const int LOGICAL_ERROR; extern const int UNKNOWN_TABLE; extern const int BAD_ARGUMENTS; extern const int FILE_DOESNT_EXIST; extern const int UNACCEPTABLE_URL; extern const int S3_ERROR; extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH; } DatabaseS3::DatabaseS3(const String & name_, const Configuration& config_, ContextPtr context_) : IDatabase(name_) , WithContext(context_->getGlobalContext()) , config(config_) , log(&Poco::Logger::get("DatabaseS3(" + name_ + ")")) { } void DatabaseS3::addTable(const std::string & table_name, StoragePtr table_storage) const { std::lock_guard lock(mutex); auto [_, inserted] = loaded_tables.emplace(table_name, table_storage); if (!inserted) throw Exception( ErrorCodes::LOGICAL_ERROR, "Table with name `{}` already exists in database `{}` (engine {})", table_name, getDatabaseName(), getEngineName()); } std::string DatabaseS3::getFullUrl(const std::string & name) const { if (!config.url_prefix.empty()) return fs::path(config.url_prefix) / name; return name; } bool DatabaseS3::checkUrl(const std::string & url, ContextPtr context_, bool throw_on_error) const { try { S3::URI uri(url); context_->getGlobalContext()->getRemoteHostFilter().checkURL(uri.uri); } catch (...) { if (throw_on_error) throw; return false; } return true; } bool DatabaseS3::isTableExist(const String & name, ContextPtr context_) const { std::lock_guard lock(mutex); if (loaded_tables.find(name) != loaded_tables.end()) return true; return checkUrl(getFullUrl(name), context_, false); } StoragePtr DatabaseS3::getTableImpl(const String & name, ContextPtr context_) const { /// Check if the table exists in the loaded tables map. { std::lock_guard lock(mutex); auto it = loaded_tables.find(name); if (it != loaded_tables.end()) return it->second; } auto url = getFullUrl(name); checkUrl(url, context_, /* throw_on_error */true); auto function = std::make_shared(); function->name = "s3"; function->arguments = std::make_shared(); function->children.push_back(function->arguments); function->arguments->children.push_back(std::make_shared(url)); if (config.no_sign_request) { function->arguments->children.push_back(std::make_shared("NOSIGN")); } else if (config.access_key_id.has_value() && config.secret_access_key.has_value()) { function->arguments->children.push_back(std::make_shared(config.access_key_id.value())); function->arguments->children.push_back(std::make_shared(config.secret_access_key.value())); } auto table_function = TableFunctionFactory::instance().get(function, context_); if (!table_function) return nullptr; /// TableFunctionS3 throws exceptions, if table cannot be created. auto table_storage = table_function->execute(function, context_, name); if (table_storage) addTable(name, table_storage); return table_storage; } StoragePtr DatabaseS3::getTable(const String & name, ContextPtr context_) const { /// Rethrow all exceptions from TableFunctionS3 to show correct error to user. if (auto storage = getTableImpl(name, context_)) return storage; throw Exception(ErrorCodes::UNKNOWN_TABLE, "Table {}.{} doesn't exist", backQuoteIfNeed(getDatabaseName()), backQuoteIfNeed(name)); } StoragePtr DatabaseS3::tryGetTable(const String & name, ContextPtr context_) const { try { return getTableImpl(name, context_); } catch (const Exception & e) { /// Ignore exceptions thrown by TableFunctionS3, which indicate that there is no table. if (e.code() == ErrorCodes::BAD_ARGUMENTS || e.code() == ErrorCodes::S3_ERROR || e.code() == ErrorCodes::FILE_DOESNT_EXIST || e.code() == ErrorCodes::UNACCEPTABLE_URL) { return nullptr; } throw; } catch (const Poco::URISyntaxException &) { return nullptr; } } bool DatabaseS3::empty() const { std::lock_guard lock(mutex); return loaded_tables.empty(); } ASTPtr DatabaseS3::getCreateDatabaseQuery() const { const auto & settings = getContext()->getSettingsRef(); ParserCreateQuery parser; std::string creation_args; creation_args += fmt::format("'{}'", config.url_prefix); if (config.no_sign_request) creation_args += ", 'NOSIGN'"; else if (config.access_key_id.has_value() && config.secret_access_key.has_value()) creation_args += fmt::format(", '{}', '{}'", config.access_key_id.value(), config.secret_access_key.value()); const String query = fmt::format("CREATE DATABASE {} ENGINE = S3({})", backQuoteIfNeed(getDatabaseName()), creation_args); ASTPtr ast = parseQuery(parser, query.data(), query.data() + query.size(), "", 0, settings.max_parser_depth); if (const auto database_comment = getDatabaseComment(); !database_comment.empty()) { auto & ast_create_query = ast->as(); ast_create_query.set(ast_create_query.comment, std::make_shared(database_comment)); } return ast; } void DatabaseS3::shutdown() { Tables tables_snapshot; { std::lock_guard lock(mutex); tables_snapshot = loaded_tables; } for (const auto & kv : tables_snapshot) { auto table_id = kv.second->getStorageID(); kv.second->flushAndShutdown(); } std::lock_guard lock(mutex); loaded_tables.clear(); } DatabaseS3::Configuration DatabaseS3::parseArguments(ASTs engine_args, ContextPtr context_) { Configuration result; if (auto named_collection = tryGetNamedCollectionWithOverrides(engine_args, context_)) { auto & collection = *named_collection; validateNamedCollection(collection, {}, optional_configuration_keys); result.url_prefix = collection.getOrDefault("url", ""); result.no_sign_request = collection.getOrDefault("no_sign_request", false); auto key_id = collection.getOrDefault("access_key_id", ""); auto secret_key = collection.getOrDefault("secret_access_key", ""); if (!key_id.empty()) result.access_key_id = key_id; if (!secret_key.empty()) result.secret_access_key = secret_key; } else { const std::string supported_signature = " - S3()\n" " - S3('url')\n" " - S3('url', 'NOSIGN')\n" " - S3('url', 'access_key_id', 'secret_access_key')\n"; const auto error_message = fmt::format("Engine DatabaseS3 must have the following arguments signature\n{}", supported_signature); for (auto & arg : engine_args) arg = evaluateConstantExpressionOrIdentifierAsLiteral(arg, context_); if (engine_args.size() > 3) throw Exception::createRuntime(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, error_message.c_str()); if (engine_args.empty()) return result; result.url_prefix = checkAndGetLiteralArgument(engine_args[0], "url"); // url, NOSIGN if (engine_args.size() == 2) { auto second_arg = checkAndGetLiteralArgument(engine_args[1], "NOSIGN"); if (boost::iequals(second_arg, "NOSIGN")) result.no_sign_request = true; else throw Exception::createRuntime(ErrorCodes::BAD_ARGUMENTS, error_message.c_str()); } // url, access_key_id, secret_access_key if (engine_args.size() == 3) { auto key_id = checkAndGetLiteralArgument(engine_args[1], "access_key_id"); auto secret_key = checkAndGetLiteralArgument(engine_args[2], "secret_access_key"); if (key_id.empty() || secret_key.empty() || boost::iequals(key_id, "NOSIGN")) throw Exception::createRuntime(ErrorCodes::BAD_ARGUMENTS, error_message.c_str()); result.access_key_id = key_id; result.secret_access_key = secret_key; } } return result; } /** * Returns an empty vector because the database is read-only and no tables can be backed up */ std::vector> DatabaseS3::getTablesForBackup(const FilterByNameFunction &, const ContextPtr &) const { return {}; } /** * * Returns an empty iterator because the database does not have its own tables * But only caches them for quick access */ DatabaseTablesIteratorPtr DatabaseS3::getTablesIterator(ContextPtr, const FilterByNameFunction &) const { return std::make_unique(Tables{}, getDatabaseName()); } } #endif