#include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int UNKNOWN_FUNCTION; extern const int LOGICAL_ERROR; } const String & getFunctionCanonicalNameIfAny(const String & name) { return FunctionFactory::instance().getCanonicalNameIfAny(name); } void FunctionFactory::registerFunction(const std::string & name, Value creator, CaseSensitiveness case_sensitiveness) { if (!functions.emplace(name, creator).second) throw Exception("FunctionFactory: the function name '" + name + "' is not unique", ErrorCodes::LOGICAL_ERROR); String function_name_lowercase = Poco::toLower(name); if (isAlias(name) || isAlias(function_name_lowercase)) throw Exception("FunctionFactory: the function name '" + name + "' is already registered as alias", ErrorCodes::LOGICAL_ERROR); if (case_sensitiveness == CaseInsensitive) { if (!case_insensitive_functions.emplace(function_name_lowercase, creator).second) throw Exception("FunctionFactory: the case insensitive function name '" + name + "' is not unique", ErrorCodes::LOGICAL_ERROR); case_insensitive_name_mapping[function_name_lowercase] = name; } } FunctionOverloadResolverPtr FunctionFactory::getImpl( const std::string & name, ContextPtr context) const { auto res = tryGetImpl(name, context); if (!res) { String extra_info; if (AggregateFunctionFactory::instance().hasNameOrAlias(name)) extra_info = ". There is an aggregate function with the same name, but ordinary function is expected here"; auto hints = this->getHints(name); if (!hints.empty()) throw Exception(ErrorCodes::UNKNOWN_FUNCTION, "Unknown function {}{}. Maybe you meant: {}", name, extra_info, toString(hints)); else throw Exception(ErrorCodes::UNKNOWN_FUNCTION, "Unknown function {}{}", name, extra_info); } return res; } std::vector FunctionFactory::getAllNames() const { std::vector res; res.reserve(functions.size()); for (const auto & func : functions) res.emplace_back(func.first); return res; } FunctionOverloadResolverPtr FunctionFactory::get( const std::string & name, ContextPtr context) const { return getImpl(name, context); } FunctionOverloadResolverPtr FunctionFactory::tryGetImpl( const std::string & name_param, ContextPtr context) const { String name = getAliasToOrName(name_param); FunctionOverloadResolverPtr res; auto it = functions.find(name); if (functions.end() != it) res = it->second(context); else { name = Poco::toLower(name); it = case_insensitive_functions.find(name); if (case_insensitive_functions.end() != it) res = it->second(context); } if (!res) return nullptr; if (CurrentThread::isInitialized()) { auto query_context = CurrentThread::get().getQueryContext(); if (query_context && query_context->getSettingsRef().log_queries) query_context->addQueryFactoriesInfo(Context::QueryLogFactories::Function, name); } return res; } FunctionOverloadResolverPtr FunctionFactory::tryGet( const std::string & name, ContextPtr context) const { auto impl = tryGetImpl(name, context); return impl ? std::move(impl) : nullptr; } FunctionFactory & FunctionFactory::instance() { static FunctionFactory ret; return ret; } }