#include #include #include namespace DB { namespace { /// Substitute return type for Date and DateTime class AggregateFunctionTopKDate : public AggregateFunctionTopK { DataTypePtr getReturnType() const override { return std::make_shared(std::make_shared()); } }; class AggregateFunctionTopKDateTime : public AggregateFunctionTopK { DataTypePtr getReturnType() const override { return std::make_shared(std::make_shared()); } }; static IAggregateFunction * createWithExtraTypes(const IDataType & argument_type) { if (typeid_cast(&argument_type)) return new AggregateFunctionTopKDate; else if (typeid_cast(&argument_type)) return new AggregateFunctionTopKDateTime; else { /// Check that we can use plain version of AggregateFunctionTopKGeneric if (typeid_cast(&argument_type) || typeid_cast(&argument_type)) return new AggregateFunctionTopKGeneric; auto * array_type = typeid_cast(&argument_type); if (array_type) { auto nested_type = array_type->getNestedType(); if (nested_type->isNumeric() || typeid_cast(nested_type.get())) return new AggregateFunctionTopKGeneric; } return new AggregateFunctionTopKGeneric; } } AggregateFunctionPtr createAggregateFunctionTopK(const std::string & name, const DataTypes & argument_types) { if (argument_types.size() != 1) throw Exception("Incorrect number of arguments for aggregate function " + name, ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); AggregateFunctionPtr res(createWithNumericType(*argument_types[0])); if (!res) res = AggregateFunctionPtr(createWithExtraTypes(*argument_types[0])); if (!res) throw Exception("Illegal type " + argument_types[0]->getName() + " of argument for aggregate function " + name, ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT); return res; } } void registerAggregateFunctionTopK(AggregateFunctionFactory & factory) { factory.registerFunction("topK", createAggregateFunctionTopK); } }