#pragma once #include #include #include #include #include #include #include namespace DB { class Context; /** Creates function by name. * Function could use for initialization (take ownership of shared_ptr, for example) * some dictionaries from Context. */ class FunctionFactory : public ext::singleton, public IFactoryWithAliases> { public: template void registerFunction(CaseSensitiveness case_sensitiveness = CaseSensitive) { registerFunction(Function::name, case_sensitiveness); } template void registerFunction(const std::string & name, CaseSensitiveness case_sensitiveness = CaseSensitive) { if constexpr (std::is_base_of::value) registerFunction(name, &createDefaultFunction, case_sensitiveness); else registerFunction(name, &Function::create, case_sensitiveness); } /// Throws an exception if not found. FunctionBuilderPtr get(const std::string & name, const Context & context) const; /// Returns nullptr if not found. FunctionBuilderPtr tryGet(const std::string & name, const Context & context) const; private: using Functions = std::unordered_map; Functions functions; Functions case_insensitive_functions; template static FunctionBuilderPtr createDefaultFunction(const Context & context) { return std::make_shared(Function::create(context)); } const Functions & getCreatorMap() const override { return functions; } const Functions & getCaseInsensitiveCreatorMap() const override { return case_insensitive_functions; } String getFactoryName() const override { return "FunctionFactory"; } /// Register a function by its name. /// No locking, you must register all functions before usage of get. void registerFunction( const std::string & name, Creator creator, CaseSensitiveness case_sensitiveness = CaseSensitive); }; }