#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "config.h" namespace DB { namespace ErrorCodes { extern const int ILLEGAL_TYPE_OF_ARGUMENT; extern const int DICTIONARIES_WAS_NOT_LOADED; extern const int BAD_ARGUMENTS; extern const int ILLEGAL_COLUMN; extern const int NUMBER_OF_ARGUMENTS_DOESNT_MATCH; } /** Functions using deprecated dictionaries * - dictionaries of regions, operating systems, search engines. * * Climb up the tree to a certain level. * regionToCity, regionToArea, regionToCountry, ... * * Convert values of a column * regionToName * * Whether the first identifier is a descendant of the second. * regionIn * * Get an array of region identifiers, consisting of the source and the parents chain. Order implementation defined. * regionHierarchy */ struct RegionToCityImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toCity(x); } }; struct RegionToAreaImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toArea(x); } }; struct RegionToDistrictImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toDistrict(x); } }; struct RegionToCountryImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toCountry(x); } }; struct RegionToContinentImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toContinent(x); } }; struct RegionToTopContinentImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toTopContinent(x); } }; struct RegionToPopulationImpl { static UInt32 apply(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.getPopulation(x); } }; struct RegionInImpl { static bool apply(UInt32 x, UInt32 y, const RegionsHierarchy & hierarchy) { return hierarchy.in(x, y); } }; struct RegionHierarchyImpl { static UInt32 toParent(UInt32 x, const RegionsHierarchy & hierarchy) { return hierarchy.toParent(x); } }; /** Auxiliary thing, allowing to get from the dictionary a specific dictionary, corresponding to the point of view * (the dictionary key passed as function argument). * Example: when calling regionToCountry(x, 'ua'), a dictionary can be used, in which Crimea refers to Ukraine. */ struct RegionsHierarchyGetter { using Src = const RegionsHierarchies; using Dst = const RegionsHierarchy; static Dst & get(Src & src, const std::string & key) { return src.get(key); } }; /** For dictionaries without key support. Doing nothing. */ template struct IdentityDictionaryGetter { using Src = const Dict; using Dst = const Dict; static Dst & get(Src & src, const std::string & key) { if (key.empty()) return src; throw Exception(ErrorCodes::BAD_ARGUMENTS, "Dictionary doesn't support 'point of view' keys."); } }; /// Converts an identifier using a dictionary. template class FunctionTransformWithDictionary : public IFunction { public: static constexpr auto name = Name::name; using base_type = FunctionTransformWithDictionary; private: const std::shared_ptr owned_dict; public: explicit FunctionTransformWithDictionary(const std::shared_ptr & owned_dict_) : owned_dict(owned_dict_) { if (!owned_dict) throw Exception(ErrorCodes::DICTIONARIES_WAS_NOT_LOADED, "Embedded dictionaries were not loaded. You need to check configuration file."); } String getName() const override { return name; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } DataTypePtr getReturnTypeImpl(const DataTypes & arguments) const override { if (arguments.size() != 1 && arguments.size() != 2) throw Exception(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, "Number of arguments for function {} doesn't match: passed {}, should be 1 or 2.", getName(), arguments.size()); if (arguments[0]->getName() != TypeName) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of argument of function {} (must be {})", arguments[0]->getName(), getName(), TypeName); if (arguments.size() == 2 && arguments[1]->getName() != "String") throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of the second ('point of view') argument of function {} (must be String)", arguments[1]->getName(), getName()); return arguments[0]; } bool useDefaultImplementationForConstants() const override { return true; } ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {1}; } bool isDeterministic() const override { return false; } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr &, size_t input_rows_count) const override { /// The dictionary key that defines the "point of view". std::string dict_key; if (arguments.size() == 2) { const ColumnConst * key_col = checkAndGetColumnConst(arguments[1].column.get()); if (!key_col) throw Exception(ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of second ('point of view') argument of function {}. " "Must be constant string.", arguments[1].column->getName(), name); dict_key = key_col->getValue(); } const typename DictGetter::Dst & dict = DictGetter::get(*owned_dict, dict_key); if (const ColumnVector * col_from = checkAndGetColumn>(arguments[0].column.get())) { auto col_to = ColumnVector::create(); const typename ColumnVector::Container & vec_from = col_from->getData(); typename ColumnVector::Container & vec_to = col_to->getData(); vec_to.resize(input_rows_count); for (size_t i = 0; i < input_rows_count; ++i) vec_to[i] = Transform::apply(vec_from[i], dict); return col_to; } throw Exception( ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of first argument of function {}", arguments[0].column->getName(), name); } }; /// Checks belonging using a dictionary. template class FunctionIsInWithDictionary : public IFunction { public: static constexpr auto name = Name::name; using base_type = FunctionIsInWithDictionary; private: const std::shared_ptr owned_dict; public: explicit FunctionIsInWithDictionary(const std::shared_ptr & owned_dict_) : owned_dict(owned_dict_) { if (!owned_dict) throw Exception(ErrorCodes::DICTIONARIES_WAS_NOT_LOADED, "Embedded dictionaries were not loaded. You need to check configuration file."); } String getName() const override { return name; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } DataTypePtr getReturnTypeImpl(const DataTypes & arguments) const override { if (arguments.size() != 2 && arguments.size() != 3) throw Exception(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, "Number of arguments for function {} doesn't match: passed {}, should be 2 or 3.", getName(), arguments.size()); if (arguments[0]->getName() != TypeName) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of first argument of function {} (must be {})", arguments[0]->getName(), getName(), TypeName); if (arguments[1]->getName() != TypeName) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of second argument of function {} (must be {})", arguments[1]->getName(), getName(), TypeName); if (arguments.size() == 3 && arguments[2]->getName() != "String") throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of the third ('point of view') argument of function {} (must be String)", arguments[2]->getName(), getName()); return std::make_shared(); } bool isDeterministic() const override { return false; } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr &, size_t input_rows_count) const override { /// The dictionary key that defines the "point of view". std::string dict_key; if (arguments.size() == 3) { const ColumnConst * key_col = checkAndGetColumnConst(arguments[2].column.get()); if (!key_col) throw Exception(ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of third ('point of view') argument of function {}. " "Must be constant string.", arguments[2].column->getName(), name); dict_key = key_col->getValue(); } const typename DictGetter::Dst & dict = DictGetter::get(*owned_dict, dict_key); const ColumnVector * col_vec1 = checkAndGetColumn>(arguments[0].column.get()); const ColumnVector * col_vec2 = checkAndGetColumn>(arguments[1].column.get()); const ColumnConst * col_const1 = checkAndGetColumnConst>(arguments[0].column.get()); const ColumnConst * col_const2 = checkAndGetColumnConst>(arguments[1].column.get()); if (col_vec1 && col_vec2) { auto col_to = ColumnUInt8::create(); const typename ColumnVector::Container & vec_from1 = col_vec1->getData(); const typename ColumnVector::Container & vec_from2 = col_vec2->getData(); typename ColumnUInt8::Container & vec_to = col_to->getData(); vec_to.resize(input_rows_count); for (size_t i = 0; i < input_rows_count; ++i) vec_to[i] = Transform::apply(vec_from1[i], vec_from2[i], dict); return col_to; } if (col_vec1 && col_const2) { auto col_to = ColumnUInt8::create(); const typename ColumnVector::Container & vec_from1 = col_vec1->getData(); const T const_from2 = col_const2->template getValue(); typename ColumnUInt8::Container & vec_to = col_to->getData(); vec_to.resize(input_rows_count); for (size_t i = 0; i < input_rows_count; ++i) vec_to[i] = Transform::apply(vec_from1[i], const_from2, dict); return col_to; } if (col_const1 && col_vec2) { auto col_to = ColumnUInt8::create(); const T const_from1 = col_const1->template getValue(); const typename ColumnVector::Container & vec_from2 = col_vec2->getData(); typename ColumnUInt8::Container & vec_to = col_to->getData(); vec_to.resize(input_rows_count); for (size_t i = 0; i < input_rows_count; ++i) vec_to[i] = Transform::apply(const_from1, vec_from2[i], dict); return col_to; } if (col_const1 && col_const2) { return DataTypeUInt8().createColumnConst( col_const1->size(), toField(Transform::apply(col_const1->template getValue(), col_const2->template getValue(), dict))); } throw Exception( ErrorCodes::ILLEGAL_COLUMN, "Illegal columns {} and {} of arguments of function {}", arguments[0].column->getName(), arguments[1].column->getName(), name); } }; /// Gets an array of identifiers consisting of the source and the parents chain. template class FunctionHierarchyWithDictionary : public IFunction { public: static constexpr auto name = Name::name; using base_type = FunctionHierarchyWithDictionary; private: const std::shared_ptr owned_dict; public: explicit FunctionHierarchyWithDictionary(const std::shared_ptr & owned_dict_) : owned_dict(owned_dict_) { if (!owned_dict) throw Exception(ErrorCodes::DICTIONARIES_WAS_NOT_LOADED, "Embedded dictionaries were not loaded. You need to check configuration file."); } String getName() const override { return name; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } DataTypePtr getReturnTypeImpl(const DataTypes & arguments) const override { if (arguments.size() != 1 && arguments.size() != 2) throw Exception(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, "Number of arguments for function {} doesn't match: passed {}, should be 1 or 2.", getName(), arguments.size()); if (arguments[0]->getName() != TypeName) throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of argument of function {} (must be {})", arguments[0]->getName(), getName(), TypeName); if (arguments.size() == 2 && arguments[1]->getName() != "String") throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of the second ('point of view') argument of function {} (must be String)", arguments[1]->getName(), getName()); return std::make_shared(arguments[0]); } bool useDefaultImplementationForConstants() const override { return true; } ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {1}; } bool isDeterministic() const override { return false; } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr &, size_t input_rows_count) const override { /// The dictionary key that defines the "point of view". std::string dict_key; if (arguments.size() == 2) { const ColumnConst * key_col = checkAndGetColumnConst(arguments[1].column.get()); if (!key_col) throw Exception(ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of second ('point of view') argument of function {}. " "Must be constant string.", arguments[1].column->getName(), name); dict_key = key_col->getValue(); } const typename DictGetter::Dst & dict = DictGetter::get(*owned_dict, dict_key); if (const ColumnVector * col_from = checkAndGetColumn>(arguments[0].column.get())) { auto col_values = ColumnVector::create(); auto col_offsets = ColumnArray::ColumnOffsets::create(); auto & res_offsets = col_offsets->getData(); auto & res_values = col_values->getData(); const typename ColumnVector::Container & vec_from = col_from->getData(); res_offsets.resize(input_rows_count); res_values.reserve(input_rows_count * 4); for (size_t i = 0; i < input_rows_count; ++i) { T cur = vec_from[i]; for (size_t depth = 0; cur && depth < DBMS_HIERARCHICAL_DICTIONARY_MAX_DEPTH; ++depth) { res_values.push_back(cur); cur = Transform::toParent(cur, dict); } res_offsets[i] = res_values.size(); } return ColumnArray::create(std::move(col_values), std::move(col_offsets)); } throw Exception( ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of first argument of function {}", arguments[0].column->getName(), name); } }; struct NameRegionToCity { static constexpr auto name = "regionToCity"; }; struct NameRegionToArea { static constexpr auto name = "regionToArea"; }; struct NameRegionToDistrict { static constexpr auto name = "regionToDistrict"; }; struct NameRegionToCountry { static constexpr auto name = "regionToCountry"; }; struct NameRegionToContinent { static constexpr auto name = "regionToContinent"; }; struct NameRegionToTopContinent { static constexpr auto name = "regionToTopContinent"; }; struct NameRegionToPopulation { static constexpr auto name = "regionToPopulation"; }; struct NameRegionHierarchy { static constexpr auto name = "regionHierarchy"; }; struct NameRegionIn { static constexpr auto name = "regionIn"; }; struct FunctionRegionToCity : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToArea : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToDistrict : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToCountry : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToContinent : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToTopContinent : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionToPopulation : public FunctionTransformWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionIn : public FunctionIsInWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; struct FunctionRegionHierarchy : public FunctionHierarchyWithDictionary { static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsHierarchies()); } }; /// Converts a region's numeric identifier to a name in the specified language using a dictionary. class FunctionRegionToName : public IFunction { public: static constexpr auto name = "regionToName"; static FunctionPtr create(ContextPtr context) { return std::make_shared(context->getEmbeddedDictionaries().getRegionsNames()); } private: const MultiVersion::Version owned_dict; public: explicit FunctionRegionToName(const MultiVersion::Version & owned_dict_) : owned_dict(owned_dict_) { if (!owned_dict) throw Exception(ErrorCodes::DICTIONARIES_WAS_NOT_LOADED, "Embedded dictionaries were not loaded. You need to check configuration file."); } String getName() const override { return name; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } /// For the purpose of query optimization, we assume this function to be injective /// even in face of fact that there are many different cities named Paris. bool isInjective(const ColumnsWithTypeAndName &) const override { return true; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } DataTypePtr getReturnTypeImpl(const DataTypes & arguments) const override { if (arguments.size() != 1 && arguments.size() != 2) throw Exception(ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH, "Number of arguments for function {} doesn't match: passed {}, should be 1 or 2.", getName(), arguments.size()); if (arguments[0]->getName() != "UInt32") throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of the first argument of function {} (must be UInt32)", arguments[0]->getName(), getName()); if (arguments.size() == 2 && arguments[1]->getName() != "String") throw Exception(ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT, "Illegal type {} of the second argument of function {} (must be String)", arguments[0]->getName(), getName()); return std::make_shared(); } DataTypePtr getReturnTypeForDefaultImplementationForDynamic() const override { return std::make_shared(); } bool useDefaultImplementationForConstants() const override { return true; } ColumnNumbers getArgumentsThatAreAlwaysConstant() const override { return {1}; } bool isDeterministic() const override { return false; } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr &, size_t /*input_rows_count*/) const override { RegionsNames::Language language = RegionsNames::Language::ru; /// If the result language is specified if (arguments.size() == 2) { if (const ColumnConst * col_language = checkAndGetColumnConst(arguments[1].column.get())) language = RegionsNames::getLanguageEnum(col_language->getValue()); else throw Exception(ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of the second argument of function {}", arguments[1].column->getName(), getName()); } const RegionsNames & dict = *owned_dict; if (const ColumnUInt32 * col_from = typeid_cast(arguments[0].column.get())) { auto col_to = ColumnString::create(); const ColumnUInt32::Container & region_ids = col_from->getData(); for (unsigned int region_id : region_ids) { const StringRef & name_ref = dict.getRegionName(region_id, language); col_to->insertData(name_ref.data, name_ref.size); } return col_to; } throw Exception( ErrorCodes::ILLEGAL_COLUMN, "Illegal column {} of the first argument of function {}", arguments[0].column->getName(), getName()); } }; }