2017-12-28 21:36:27 +00:00
|
|
|
#include <Storages/StorageFactory.h>
|
|
|
|
#include <Interpreters/Context.h>
|
|
|
|
#include <Parsers/ASTFunction.h>
|
2017-12-30 00:36:06 +00:00
|
|
|
#include <Parsers/ASTCreateQuery.h>
|
2017-12-28 21:36:27 +00:00
|
|
|
#include <Common/Exception.h>
|
2018-06-05 19:46:49 +00:00
|
|
|
#include <Common/StringUtils/StringUtils.h>
|
2019-04-03 11:13:22 +00:00
|
|
|
#include <IO/WriteHelpers.h>
|
2020-03-13 10:30:55 +00:00
|
|
|
#include <Interpreters/StorageID.h>
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int UNKNOWN_STORAGE;
|
|
|
|
extern const int LOGICAL_ERROR;
|
|
|
|
extern const int INCORRECT_QUERY;
|
|
|
|
extern const int ENGINE_REQUIRED;
|
|
|
|
extern const int FUNCTION_CANNOT_HAVE_PARAMETERS;
|
|
|
|
extern const int BAD_ARGUMENTS;
|
2017-12-30 00:36:06 +00:00
|
|
|
extern const int DATA_TYPE_CANNOT_BE_USED_IN_TABLES;
|
2017-12-28 21:36:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Some types are only for intermediate values of expressions and cannot be used in tables.
|
|
|
|
static void checkAllTypesAreAllowedInTable(const NamesAndTypesList & names_and_types)
|
|
|
|
{
|
|
|
|
for (const auto & elem : names_and_types)
|
|
|
|
if (elem.type->cannotBeStoredInTables())
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::DATA_TYPE_CANNOT_BE_USED_IN_TABLES, "Data type {} cannot be used in tables", elem.type->getName());
|
2017-12-28 21:36:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr StorageFactory::Arguments::getContext() const
|
2021-04-10 23:33:54 +00:00
|
|
|
{
|
|
|
|
auto ptr = context.lock();
|
|
|
|
if (!ptr)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Context has expired");
|
2021-04-10 23:33:54 +00:00
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr StorageFactory::Arguments::getLocalContext() const
|
2021-04-10 23:33:54 +00:00
|
|
|
{
|
|
|
|
auto ptr = local_context.lock();
|
|
|
|
if (!ptr)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "Context has expired");
|
2021-04-10 23:33:54 +00:00
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-01-25 19:14:33 +00:00
|
|
|
void StorageFactory::registerStorage(const std::string & name, CreatorFn creator_fn, StorageFeatures features)
|
2017-12-28 21:36:27 +00:00
|
|
|
{
|
2020-01-25 19:14:33 +00:00
|
|
|
if (!storages.emplace(name, Creator{std::move(creator_fn), features}).second)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::LOGICAL_ERROR, "TableFunctionFactory: the table function name '{}' is not unique", name);
|
2017-12-28 21:36:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
StoragePtr StorageFactory::get(
|
2019-12-16 18:10:35 +00:00
|
|
|
const ASTCreateQuery & query,
|
2019-10-25 19:07:47 +00:00
|
|
|
const String & relative_data_path,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr local_context,
|
|
|
|
ContextMutablePtr context,
|
2018-03-06 20:18:34 +00:00
|
|
|
const ColumnsDescription & columns,
|
2019-08-24 21:20:20 +00:00
|
|
|
const ConstraintsDescription & constraints,
|
2017-12-28 21:36:27 +00:00
|
|
|
bool has_force_restore_data_flag) const
|
|
|
|
{
|
2021-05-15 07:05:31 +00:00
|
|
|
String name, comment;
|
2022-01-31 17:10:59 +00:00
|
|
|
|
2017-12-30 00:36:06 +00:00
|
|
|
ASTStorage * storage_def = query.storage;
|
2017-12-28 21:36:27 +00:00
|
|
|
|
2020-02-19 18:58:29 +00:00
|
|
|
bool has_engine_args = false;
|
|
|
|
|
2020-12-16 03:19:38 +00:00
|
|
|
if (query.is_ordinary_view)
|
2017-12-28 21:36:27 +00:00
|
|
|
{
|
|
|
|
if (query.storage)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY, "Specifying ENGINE is not allowed for a View");
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
name = "View";
|
|
|
|
}
|
2019-05-28 21:17:48 +00:00
|
|
|
else if (query.is_live_view)
|
|
|
|
{
|
|
|
|
if (query.storage)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY, "Specifying ENGINE is not allowed for a LiveView");
|
2019-05-28 21:17:48 +00:00
|
|
|
|
|
|
|
name = "LiveView";
|
|
|
|
}
|
2021-04-21 13:45:13 +00:00
|
|
|
else if (query.is_dictionary)
|
|
|
|
{
|
|
|
|
if (query.storage)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY, "Specifying ENGINE is not allowed for a Dictionary");
|
2021-04-21 13:45:13 +00:00
|
|
|
|
|
|
|
name = "Dictionary";
|
|
|
|
}
|
2017-12-28 21:36:27 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/// Check for some special types, that are not allowed to be stored in tables. Example: NULL data type.
|
|
|
|
/// Exception: any type is allowed in View, because plain (non-materialized) View does not store anything itself.
|
2018-03-10 17:03:57 +00:00
|
|
|
checkAllTypesAreAllowedInTable(columns.getAll());
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
if (query.is_materialized_view)
|
|
|
|
{
|
|
|
|
name = "MaterializedView";
|
|
|
|
}
|
2021-05-28 07:36:19 +00:00
|
|
|
else if (query.is_window_view)
|
|
|
|
{
|
|
|
|
name = "WindowView";
|
|
|
|
}
|
2017-12-28 21:36:27 +00:00
|
|
|
else
|
|
|
|
{
|
2022-02-01 16:59:41 +00:00
|
|
|
if (!query.storage)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY, "Incorrect CREATE query: storage required");
|
2022-02-01 16:59:41 +00:00
|
|
|
|
2022-01-31 17:10:59 +00:00
|
|
|
if (!storage_def->engine)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::ENGINE_REQUIRED, "Incorrect CREATE query: ENGINE required");
|
2017-12-28 21:36:27 +00:00
|
|
|
|
2017-12-30 00:36:06 +00:00
|
|
|
const ASTFunction & engine_def = *storage_def->engine;
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
if (engine_def.parameters)
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::FUNCTION_CANNOT_HAVE_PARAMETERS, "Engine definition cannot take the form of a parametric function");
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
if (engine_def.arguments)
|
2020-02-19 18:58:29 +00:00
|
|
|
has_engine_args = true;
|
2017-12-28 21:36:27 +00:00
|
|
|
|
|
|
|
name = engine_def.name;
|
|
|
|
|
|
|
|
if (name == "View")
|
|
|
|
{
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY, "Direct creation of tables with ENGINE View is not supported, use CREATE VIEW statement");
|
2017-12-28 21:36:27 +00:00
|
|
|
}
|
|
|
|
else if (name == "MaterializedView")
|
|
|
|
{
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY,
|
|
|
|
"Direct creation of tables with ENGINE MaterializedView "
|
|
|
|
"is not supported, use CREATE MATERIALIZED VIEW statement");
|
2017-12-28 21:36:27 +00:00
|
|
|
}
|
2019-05-28 21:17:48 +00:00
|
|
|
else if (name == "LiveView")
|
|
|
|
{
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY,
|
|
|
|
"Direct creation of tables with ENGINE LiveView "
|
|
|
|
"is not supported, use CREATE LIVE VIEW statement");
|
2019-05-28 21:17:48 +00:00
|
|
|
}
|
2020-01-14 03:07:31 +00:00
|
|
|
else if (name == "WindowView")
|
|
|
|
{
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::INCORRECT_QUERY,
|
|
|
|
"Direct creation of tables with ENGINE WindowView "
|
|
|
|
"is not supported, use CREATE WINDOW VIEW statement");
|
2020-01-14 03:07:31 +00:00
|
|
|
}
|
2017-12-28 21:36:27 +00:00
|
|
|
|
2020-01-27 18:28:27 +00:00
|
|
|
auto it = storages.find(name);
|
|
|
|
if (it == storages.end())
|
2020-01-25 19:14:33 +00:00
|
|
|
{
|
2020-01-27 18:28:27 +00:00
|
|
|
auto hints = getHints(name);
|
|
|
|
if (!hints.empty())
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::UNKNOWN_STORAGE, "Unknown table engine {}. Maybe you meant: {}", name, toString(hints));
|
2020-01-27 18:28:27 +00:00
|
|
|
else
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::UNKNOWN_STORAGE, "Unknown table engine {}", name);
|
2020-01-25 19:14:33 +00:00
|
|
|
}
|
2021-05-15 07:56:46 +00:00
|
|
|
|
2020-03-23 02:12:31 +00:00
|
|
|
auto check_feature = [&](String feature_description, FeatureMatcherFn feature_matcher_fn)
|
2020-01-27 18:28:27 +00:00
|
|
|
{
|
|
|
|
if (!feature_matcher_fn(it->second.features))
|
|
|
|
{
|
2023-01-23 21:13:58 +00:00
|
|
|
String msg;
|
2020-01-27 18:28:27 +00:00
|
|
|
auto supporting_engines = getAllRegisteredNamesByFeatureMatcherFn(feature_matcher_fn);
|
|
|
|
for (size_t index = 0; index < supporting_engines.size(); ++index)
|
|
|
|
{
|
|
|
|
if (index)
|
|
|
|
msg += ", ";
|
|
|
|
msg += supporting_engines[index];
|
|
|
|
}
|
2023-01-23 21:13:58 +00:00
|
|
|
throw Exception(ErrorCodes::BAD_ARGUMENTS, "Engine {} doesn't support {}. "
|
|
|
|
"Currently only the following engines have support for the feature: [{}]",
|
|
|
|
name, feature_description, msg);
|
2020-01-27 18:28:27 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (storage_def->settings)
|
2020-03-23 02:12:31 +00:00
|
|
|
check_feature(
|
2020-01-27 18:28:27 +00:00
|
|
|
"SETTINGS clause",
|
|
|
|
[](StorageFeatures features) { return features.supports_settings; });
|
|
|
|
|
|
|
|
if (storage_def->partition_by || storage_def->primary_key || storage_def->order_by || storage_def->sample_by)
|
2020-03-23 02:12:31 +00:00
|
|
|
check_feature(
|
2020-01-27 18:28:27 +00:00
|
|
|
"PARTITION_BY, PRIMARY_KEY, ORDER_BY or SAMPLE_BY clauses",
|
|
|
|
[](StorageFeatures features) { return features.supports_sort_order; });
|
|
|
|
|
|
|
|
if (storage_def->ttl_table || !columns.getColumnTTLs().empty())
|
2020-03-23 02:12:31 +00:00
|
|
|
check_feature(
|
2020-01-27 18:28:27 +00:00
|
|
|
"TTL clause",
|
|
|
|
[](StorageFeatures features) { return features.supports_ttl; });
|
|
|
|
|
|
|
|
if (query.columns_list && query.columns_list->indices && !query.columns_list->indices->children.empty())
|
2020-03-23 02:12:31 +00:00
|
|
|
check_feature(
|
2020-01-27 18:28:27 +00:00
|
|
|
"skipping indices",
|
|
|
|
[](StorageFeatures features) { return features.supports_skipping_indices; });
|
2021-02-10 14:12:49 +00:00
|
|
|
|
|
|
|
if (query.columns_list && query.columns_list->projections && !query.columns_list->projections->children.empty())
|
|
|
|
check_feature(
|
|
|
|
"projections",
|
|
|
|
[](StorageFeatures features) { return features.supports_projections; });
|
2020-01-27 18:28:27 +00:00
|
|
|
}
|
2020-01-27 13:04:01 +00:00
|
|
|
}
|
2020-01-25 19:14:33 +00:00
|
|
|
|
2021-11-03 20:32:29 +00:00
|
|
|
if (query.comment)
|
|
|
|
comment = query.comment->as<ASTLiteral &>().value.get<String>();
|
|
|
|
|
2020-02-19 18:58:29 +00:00
|
|
|
ASTs empty_engine_args;
|
2021-05-13 06:22:05 +00:00
|
|
|
Arguments arguments{
|
2017-12-30 00:36:06 +00:00
|
|
|
.engine_name = name,
|
2020-02-19 18:58:29 +00:00
|
|
|
.engine_args = has_engine_args ? storage_def->engine->arguments->children : empty_engine_args,
|
2017-12-30 00:36:06 +00:00
|
|
|
.storage_def = storage_def,
|
|
|
|
.query = query,
|
2019-10-25 19:07:47 +00:00
|
|
|
.relative_data_path = relative_data_path,
|
2021-09-06 22:13:54 +00:00
|
|
|
.table_id = StorageID(query.getDatabase(), query.getTable(), query.uuid),
|
2017-12-30 00:36:06 +00:00
|
|
|
.local_context = local_context,
|
|
|
|
.context = context,
|
|
|
|
.columns = columns,
|
2019-08-24 21:20:20 +00:00
|
|
|
.constraints = constraints,
|
2019-12-04 16:06:55 +00:00
|
|
|
.attach = query.attach,
|
2021-04-23 12:18:23 +00:00
|
|
|
.has_force_restore_data_flag = has_force_restore_data_flag,
|
2021-05-13 06:22:05 +00:00
|
|
|
.comment = comment};
|
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
assert(arguments.getContext() == arguments.getContext()->getGlobalContext());
|
2017-09-18 14:18:29 +00:00
|
|
|
|
2020-09-26 19:18:28 +00:00
|
|
|
auto res = storages.at(name).creator_fn(arguments);
|
2023-02-19 22:15:09 +00:00
|
|
|
if (!empty_engine_args.empty())
|
2020-09-26 19:18:28 +00:00
|
|
|
{
|
|
|
|
/// Storage creator modified empty arguments list, so we should modify the query
|
|
|
|
assert(storage_def && storage_def->engine && !storage_def->engine->arguments);
|
|
|
|
storage_def->engine->arguments = std::make_shared<ASTExpressionList>();
|
|
|
|
storage_def->engine->children.push_back(storage_def->engine->arguments);
|
|
|
|
storage_def->engine->arguments->children = empty_engine_args;
|
|
|
|
}
|
2021-01-20 10:54:11 +00:00
|
|
|
|
2021-08-04 18:44:19 +00:00
|
|
|
if (local_context->hasQueryContext() && local_context->getSettingsRef().log_queries)
|
2021-04-10 23:33:54 +00:00
|
|
|
local_context->getQueryContext()->addQueryFactoriesInfo(Context::QueryLogFactories::Storage, name);
|
2021-01-20 10:54:11 +00:00
|
|
|
|
2020-09-26 19:18:28 +00:00
|
|
|
return res;
|
2011-10-31 17:30:44 +00:00
|
|
|
}
|
|
|
|
|
2019-08-22 03:24:05 +00:00
|
|
|
StorageFactory & StorageFactory::instance()
|
|
|
|
{
|
|
|
|
static StorageFactory ret;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-04-06 05:19:40 +00:00
|
|
|
|
|
|
|
AccessType StorageFactory::getSourceAccessType(const String & table_engine) const
|
|
|
|
{
|
|
|
|
auto it = storages.find(table_engine);
|
|
|
|
if (it == storages.end())
|
|
|
|
return AccessType::NONE;
|
|
|
|
return it->second.features.source_access_type;
|
|
|
|
}
|
|
|
|
|
2011-10-31 17:30:44 +00:00
|
|
|
}
|