#pragma once #include #include #include #include #include #include #include namespace DB { class Context; class IDataType; using DataTypePtr = std::shared_ptr; using DataTypes = std::vector; /** Creates an aggregate function by name. */ class AggregateFunctionFactory final : public ext::singleton { friend class StorageSystemFunctions; public: using Creator = std::function; /// For compatibility with SQL, it's possible to specify that certain aggregate function name is case insensitive. enum CaseSensitiveness { CaseSensitive, CaseInsensitive }; /// Register a function by its name. /// No locking, you must register all functions before usage of get. void registerFunction( const String & name, Creator creator, CaseSensitiveness case_sensitiveness = CaseSensitive); /// Throws an exception if not found. AggregateFunctionPtr get( const String & name, const DataTypes & argument_types, const Array & parameters = {}, int recursion_level = 0) const; /// Returns nullptr if not found. AggregateFunctionPtr tryGet( const String & name, const DataTypes & argument_types, const Array & parameters = {}) const; bool isAggregateFunctionName(const String & name, int recursion_level = 0) const; private: AggregateFunctionPtr getImpl( const String & name, const DataTypes & argument_types, const Array & parameters, int recursion_level) const; private: using AggregateFunctions = std::unordered_map; AggregateFunctions aggregate_functions; /// Case insensitive aggregate functions will be additionally added here with lowercased name. AggregateFunctions case_insensitive_aggregate_functions; }; }