#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int LOGICAL_ERROR; extern const int BAD_ARGUMENTS; extern const int DICTIONARY_IS_EMPTY; extern const int UNSUPPORTED_METHOD; extern const int TYPE_MISMATCH; } enum class RangeHashedDictionaryLookupStrategy : uint8_t { min, max }; struct RangeHashedDictionaryConfiguration { bool convert_null_range_bound_to_open; RangeHashedDictionaryLookupStrategy lookup_strategy; bool require_nonempty; bool use_async_executor = false; }; template class RangeHashedDictionary final : public IDictionary { public: using KeyType = std::conditional_t; RangeHashedDictionary( const StorageID & dict_id_, const DictionaryStructure & dict_struct_, DictionarySourcePtr source_ptr_, DictionaryLifetime dict_lifetime_, RangeHashedDictionaryConfiguration configuration_, BlockPtr update_field_loaded_block_ = nullptr); std::string getTypeName() const override { if constexpr (dictionary_key_type == DictionaryKeyType::Simple) return "RangeHashed"; else return "ComplexKeyRangeHashed"; } size_t getBytesAllocated() const override { return bytes_allocated; } size_t getQueryCount() const override { return query_count.load(); } double getFoundRate() const override { size_t queries = query_count.load(); if (!queries) return 0; return std::min(1.0, static_cast(found_count.load()) / queries); } double getHitRate() const override { return 1.0; } size_t getElementCount() const override { return element_count; } double getLoadFactor() const override { return static_cast(element_count) / bucket_count; } std::shared_ptr clone() const override { auto result = std::make_shared( getDictionaryID(), dict_struct, source_ptr->clone(), dict_lifetime, configuration, update_field_loaded_block); return result; } DictionarySourcePtr getSource() const override { return source_ptr; } const DictionaryLifetime & getLifetime() const override { return dict_lifetime; } const DictionaryStructure & getStructure() const override { return dict_struct; } bool isInjective(const std::string & attribute_name) const override { return dict_struct.getAttribute(attribute_name).injective; } DictionaryKeyType getKeyType() const override { return dictionary_key_type; } DictionarySpecialKeyType getSpecialKeyType() const override { return DictionarySpecialKeyType::Range;} ColumnPtr getColumn( const std::string & attribute_name, const DataTypePtr & attribute_type, const Columns & key_columns, const DataTypes & key_types, DefaultOrFilter default_or_filter) const override; ColumnUInt8::Ptr hasKeys(const Columns & key_columns, const DataTypes & key_types) const override; Pipe read(const Names & column_names, size_t max_block_size, size_t num_streams) const override; private: template using IntervalMap = IntervalMap, size_t>; template using KeyAttributeContainerType = std::conditional_t< dictionary_key_type == DictionaryKeyType::Simple, HashMap, DefaultHash>, HashMapWithSavedHash, DefaultHash>>; template using AttributeContainerType = std::conditional_t, std::vector, PaddedPODArray>; struct Attribute final { AttributeUnderlyingType type; std::variant< AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType, AttributeContainerType> container; std::optional> is_value_nullable; }; template struct InvalidIntervalWithKey { KeyType key; Interval interval; size_t attribute_value_index; }; template using InvalidIntervalsContainerType = PaddedPODArray>; template typename ContainerType> using RangeStorageTypeContainer = std::variant< ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType>; struct KeyAttribute final { RangeStorageTypeContainer container; RangeStorageTypeContainer invalid_intervals_container; }; void createAttributes(); void loadData(); void calculateBytesAllocated(); static Attribute createAttribute(const DictionaryAttribute & dictionary_attribute); template using ValueSetterFunc = std::function; template void getItemsImpl( const Attribute & attribute, const Columns & key_columns, ValueSetterFunc && set_value, DefaultValueExtractor & default_value_extractor) const; template void getItemsShortCircuitImpl( const Attribute & attribute, const Columns & key_columns, ValueSetterFunc && set_value, IColumn::Filter & default_mask) const; ColumnPtr getColumnInternal( const std::string & attribute_name, const DataTypePtr & result_type, const PaddedPODArray & key_to_index) const; template void getItemsInternalImpl( const Attribute & attribute, const PaddedPODArray & key_to_index, ValueSetter && set_value) const; void updateData(); void blockToAttributes(const Block & block); void setAttributeValue(Attribute & attribute, const Field & value); const DictionaryStructure dict_struct; const DictionarySourcePtr source_ptr; const DictionaryLifetime dict_lifetime; const RangeHashedDictionaryConfiguration configuration; BlockPtr update_field_loaded_block; std::vector attributes; KeyAttribute key_attribute; size_t bytes_allocated = 0; size_t element_count = 0; size_t bucket_count = 0; mutable std::atomic query_count{0}; mutable std::atomic found_count{0}; Arena string_arena; }; extern template class RangeHashedDictionary; extern template class RangeHashedDictionary; namespace impl { template void callOnRangeType(const DataTypePtr & range_type, F && func) { auto call = [&](const auto & types) { using Types = std::decay_t; using DataType = typename Types::LeftType; if constexpr (IsDataTypeDecimalOrNumber || IsDataTypeDateOrDateTime || IsDataTypeEnum) { using ColumnType = typename DataType::ColumnType; func(TypePair()); return true; } return false; }; auto type_index = range_type->getTypeId(); if (!callOnIndexAndDataType(type_index, call)) { throw Exception(ErrorCodes::BAD_ARGUMENTS, "Dictionary structure type of 'range_min' and 'range_max' should " "be an Integer, Float, Decimal, Date, Date32, DateTime DateTime64, or Enum." " Actual 'range_min' and 'range_max' type is {}", range_type->getName()); } } } template RangeHashedDictionary::RangeHashedDictionary( const StorageID & dict_id_, const DictionaryStructure & dict_struct_, DictionarySourcePtr source_ptr_, DictionaryLifetime dict_lifetime_, RangeHashedDictionaryConfiguration configuration_, BlockPtr update_field_loaded_block_) : IDictionary(dict_id_) , dict_struct(dict_struct_) , source_ptr(std::move(source_ptr_)) , dict_lifetime(dict_lifetime_) , configuration(configuration_) , update_field_loaded_block(std::move(update_field_loaded_block_)) { createAttributes(); loadData(); calculateBytesAllocated(); } template ColumnPtr RangeHashedDictionary::getColumnInternal( const std::string & attribute_name, const DataTypePtr & result_type, const PaddedPODArray & key_to_index) const { ColumnPtr result; const auto & dictionary_attribute = dict_struct.getAttribute(attribute_name, result_type); const size_t attribute_index = dict_struct.attribute_name_to_index.find(attribute_name)->second; const auto & attribute = attributes[attribute_index]; size_t keys_size = key_to_index.size(); bool is_attribute_nullable = attribute.is_value_nullable.has_value(); ColumnUInt8::MutablePtr col_null_map_to; ColumnUInt8::Container * vec_null_map_to = nullptr; if (is_attribute_nullable) { col_null_map_to = ColumnUInt8::create(keys_size, false); vec_null_map_to = &col_null_map_to->getData(); } auto type_call = [&](const auto & dictionary_attribute_type) { using Type = std::decay_t; using AttributeType = typename Type::AttributeType; using ValueType = DictionaryValueType; using ColumnProvider = DictionaryAttributeColumnProvider; auto column = ColumnProvider::getColumn(dictionary_attribute, keys_size); if constexpr (std::is_same_v) { auto * out = column.get(); getItemsInternalImpl( attribute, key_to_index, [&](size_t, const Array & value, bool) { out->insert(value); }); } else if constexpr (std::is_same_v) { auto * out = column.get(); if (is_attribute_nullable) getItemsInternalImpl( attribute, key_to_index, [&](size_t row, StringRef value, bool is_null) { (*vec_null_map_to)[row] = is_null; out->insertData(value.data, value.size); }); else getItemsInternalImpl( attribute, key_to_index, [&](size_t, StringRef value, bool) { out->insertData(value.data, value.size); }); } else { auto & out = column->getData(); if (is_attribute_nullable) getItemsInternalImpl( attribute, key_to_index, [&](size_t row, const auto value, bool is_null) { (*vec_null_map_to)[row] = is_null; out[row] = value; }); else getItemsInternalImpl( attribute, key_to_index, [&](size_t row, const auto value, bool) { out[row] = value; }); } result = std::move(column); }; callOnDictionaryAttributeType(attribute.type, type_call); if (is_attribute_nullable) result = ColumnNullable::create(result, std::move(col_null_map_to)); return result; } template ColumnUInt8::Ptr RangeHashedDictionary::hasKeys(const Columns & key_columns, const DataTypes & key_types) const { if (dictionary_key_type == DictionaryKeyType::Complex) { auto key_types_copy = key_types; key_types_copy.pop_back(); dict_struct.validateKeyTypes(key_types_copy); } /// Cast range column to storage type const ColumnPtr & range_storage_column = key_columns.back(); ColumnWithTypeAndName column_to_cast = {range_storage_column->convertToFullColumnIfConst(), key_types.back(), ""}; auto range_column_updated = castColumnAccurate(column_to_cast, dict_struct.range_min->type); auto key_columns_copy = key_columns; key_columns_copy.pop_back(); DictionaryKeysArenaHolder arena_holder; DictionaryKeysExtractor keys_extractor(key_columns_copy, arena_holder.getComplexKeyArena()); const size_t keys_size = keys_extractor.getKeysSize(); auto result = ColumnUInt8::create(keys_size); auto & out = result->getData(); size_t keys_found = 0; impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; const auto * range_column_typed = typeid_cast(range_column_updated.get()); if (!range_column_typed) throw Exception(ErrorCodes::TYPE_MISMATCH, "Dictionary {} range column type should be equal to {}", getFullName(), dict_struct.range_min->type->getName()); const auto & range_column_data = range_column_typed->getData(); const auto & key_attribute_container = std::get>(key_attribute.container); for (size_t key_index = 0; key_index < keys_size; ++key_index) { const auto key = keys_extractor.extractCurrentKey(); const auto it = key_attribute_container.find(key); if (it) { const auto date = range_column_data[key_index]; const auto & interval_tree = it->getMapped(); out[key_index] = interval_tree.has(date); keys_found += out[key_index]; } else { out[key_index] = false; } keys_extractor.rollbackCurrentKey(); } }); query_count.fetch_add(keys_size, std::memory_order_relaxed); found_count.fetch_add(keys_found, std::memory_order_relaxed); return result; } template void RangeHashedDictionary::createAttributes() { const auto size = dict_struct.attributes.size(); attributes.reserve(size); for (const auto & attribute : dict_struct.attributes) { attributes.push_back(createAttribute(attribute)); if (attribute.hierarchical) throw Exception(ErrorCodes::BAD_ARGUMENTS, "Hierarchical attributes not supported by {} dictionary.", getDictionaryID().getNameForLogs()); } impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; key_attribute.container = KeyAttributeContainerType(); key_attribute.invalid_intervals_container = InvalidIntervalsContainerType(); }); } template void RangeHashedDictionary::loadData() { if (!source_ptr->hasUpdateField()) { QueryPipeline pipeline(source_ptr->loadAll()); DictionaryPipelineExecutor executor(pipeline, configuration.use_async_executor); pipeline.setConcurrencyControl(false); Block block; while (executor.pull(block)) { blockToAttributes(block); } } else { updateData(); } impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; auto & key_attribute_container = std::get>(key_attribute.container); for (auto & [_, intervals] : key_attribute_container) intervals.build(); }); if (configuration.require_nonempty && 0 == element_count) throw Exception(ErrorCodes::DICTIONARY_IS_EMPTY, "{}: dictionary source is empty and 'require_nonempty' property is set.", getFullName()); } template void RangeHashedDictionary::calculateBytesAllocated() { impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; auto & key_attribute_container = std::get>(key_attribute.container); bucket_count = key_attribute_container.getBufferSizeInCells(); bytes_allocated += key_attribute_container.getBufferSizeInBytes(); for (auto & [_, intervals] : key_attribute_container) bytes_allocated += intervals.getSizeInBytes(); }); bytes_allocated += attributes.size() * sizeof(attributes.front()); for (const auto & attribute : attributes) { auto type_call = [&](const auto & dictionary_attribute_type) { using Type = std::decay_t; using AttributeType = typename Type::AttributeType; using ValueType = DictionaryValueType; const auto & container = std::get>(attribute.container); bytes_allocated += container.size() * sizeof(ValueType); if (attribute.is_value_nullable) bytes_allocated += (*attribute.is_value_nullable).size() * sizeof(bool); }; callOnDictionaryAttributeType(attribute.type, type_call); } if (update_field_loaded_block) bytes_allocated += update_field_loaded_block->allocatedBytes(); bytes_allocated += string_arena.allocatedBytes(); } template typename RangeHashedDictionary::Attribute RangeHashedDictionary::createAttribute(const DictionaryAttribute & dictionary_attribute) { std::optional> is_value_nullable; if (dictionary_attribute.is_nullable) is_value_nullable.emplace(std::vector()); Attribute attribute{dictionary_attribute.underlying_type, {}, std::move(is_value_nullable)}; auto type_call = [&](const auto & dictionary_attribute_type) { using Type = std::decay_t; using AttributeType = typename Type::AttributeType; using ValueType = DictionaryValueType; attribute.container = AttributeContainerType(); }; callOnDictionaryAttributeType(dictionary_attribute.underlying_type, type_call); return attribute; } template template void RangeHashedDictionary::getItemsInternalImpl( const Attribute & attribute, const PaddedPODArray & key_to_index, ValueSetter && set_value) const { size_t keys_size = key_to_index.size(); const auto & container = std::get>(attribute.container); size_t container_size = container.size(); for (size_t key_index = 0; key_index < keys_size; ++key_index) { UInt64 container_index = key_to_index[key_index]; if (unlikely(container_index >= container_size)) { throw Exception(ErrorCodes::LOGICAL_ERROR, "Dictionary {} expected attribute container index {} must be less than attribute container size {}", getFullName(), container_index, container_size ); } AttributeType value = container[container_index]; if constexpr (is_nullable) { bool is_null = (*attribute.is_value_nullable)[container_index]; if (!is_null) set_value(key_index, value, false); else set_value(key_index, value, true); } else { set_value(key_index, value, false); } } query_count.fetch_add(keys_size, std::memory_order_relaxed); found_count.fetch_add(keys_size, std::memory_order_relaxed); } template void RangeHashedDictionary::updateData() { if (!update_field_loaded_block || update_field_loaded_block->rows() == 0) { QueryPipeline pipeline(source_ptr->loadUpdatedAll()); DictionaryPipelineExecutor executor(pipeline, configuration.use_async_executor); pipeline.setConcurrencyControl(false); update_field_loaded_block.reset(); Block block; while (executor.pull(block)) { if (!block.rows()) continue; convertToFullIfSparse(block); /// We are using this to keep saved data if input stream consists of multiple blocks if (!update_field_loaded_block) update_field_loaded_block = std::make_shared(block.cloneEmpty()); for (size_t attribute_index = 0; attribute_index < block.columns(); ++attribute_index) { const IColumn & update_column = *block.getByPosition(attribute_index).column.get(); MutableColumnPtr saved_column = update_field_loaded_block->getByPosition(attribute_index).column->assumeMutable(); saved_column->insertRangeFrom(update_column, 0, update_column.size()); } } } else { static constexpr size_t range_columns_size = 2; auto pipe = source_ptr->loadUpdatedAll(); /// Use complex dictionary key type to count range columns as part of complex primary key during update mergeBlockWithPipe( dict_struct.getKeysSize() + range_columns_size, *update_field_loaded_block, std::move(pipe)); } if (update_field_loaded_block) { blockToAttributes(*update_field_loaded_block.get()); } } template void RangeHashedDictionary::blockToAttributes(const Block & block) { size_t attributes_size = attributes.size(); size_t dictionary_keys_size = dict_struct.getKeysSize(); static constexpr size_t ranges_size = 2; size_t block_columns = block.columns(); size_t range_dictionary_attributes_size = attributes_size + dictionary_keys_size + ranges_size; if (range_dictionary_attributes_size != block.columns()) { throw Exception(ErrorCodes::UNSUPPORTED_METHOD, "Block size mismatch. Actual {}. Expected {}", block_columns, range_dictionary_attributes_size); } Columns key_columns; key_columns.reserve(dictionary_keys_size); /// Split into keys columns and attribute columns for (size_t i = 0; i < dictionary_keys_size; ++i) key_columns.emplace_back(block.getByPosition(i).column); DictionaryKeysArenaHolder arena_holder; DictionaryKeysExtractor keys_extractor(key_columns, arena_holder.getComplexKeyArena()); const size_t keys_size = keys_extractor.getKeysSize(); size_t block_attributes_skip_offset = dictionary_keys_size; const auto * min_range_column = block.getByPosition(block_attributes_skip_offset).column.get(); const auto * max_range_column = block.getByPosition(block_attributes_skip_offset + 1).column.get(); const NullMap * min_range_null_map = nullptr; const NullMap * max_range_null_map = nullptr; if (const auto * min_range_column_nullable = checkAndGetColumn(min_range_column)) { min_range_column = &min_range_column_nullable->getNestedColumn(); min_range_null_map = &min_range_column_nullable->getNullMapColumn().getData(); } if (const auto * max_range_column_nullable = checkAndGetColumn(max_range_column)) { max_range_column = &max_range_column_nullable->getNestedColumn(); max_range_null_map = &max_range_column_nullable->getNullMapColumn().getData(); } impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; const auto * min_range_column_typed = typeid_cast(min_range_column); if (!min_range_column_typed) throw Exception(ErrorCodes::TYPE_MISMATCH, "Dictionary {} range min column type should be equal to {}", getFullName(), dict_struct.range_min->type->getName()); const auto * max_range_column_typed = typeid_cast(max_range_column); if (!max_range_column_typed) throw Exception(ErrorCodes::TYPE_MISMATCH, "Dictionary {} range max column type should be equal to {}", getFullName(), dict_struct.range_max->type->getName()); const auto & min_range_column_data = min_range_column_typed->getData(); const auto & max_range_column_data = max_range_column_typed->getData(); auto & key_attribute_container = std::get>(key_attribute.container); auto & invalid_intervals_container = std::get>(key_attribute.invalid_intervals_container); block_attributes_skip_offset += 2; Field column_value; for (size_t key_index = 0; key_index < keys_size; ++key_index) { auto key = keys_extractor.extractCurrentKey(); RangeStorageType lower_bound = min_range_column_data[key_index]; RangeStorageType upper_bound = max_range_column_data[key_index]; bool invalid_range = false; if (unlikely(min_range_null_map && (*min_range_null_map)[key_index])) { lower_bound = std::numeric_limits::min(); invalid_range = true; } if (unlikely(max_range_null_map && (*max_range_null_map)[key_index])) { upper_bound = std::numeric_limits::max(); invalid_range = true; } if (unlikely(!configuration.convert_null_range_bound_to_open && invalid_range)) { keys_extractor.rollbackCurrentKey(); continue; } if constexpr (std::is_same_v) key = copyStringInArena(string_arena, key); for (size_t attribute_index = 0; attribute_index < attributes.size(); ++attribute_index) { const auto & attribute_column = *block.getByPosition(attribute_index + block_attributes_skip_offset).column; auto & attribute = attributes[attribute_index]; attribute_column.get(key_index, column_value); setAttributeValue(attribute, column_value); } auto interval = Interval(lower_bound, upper_bound); auto it = key_attribute_container.find(key); bool emplaced_in_interval_tree = false; if (it) { auto & intervals = it->getMapped(); emplaced_in_interval_tree = intervals.emplace(interval, element_count); } else { IntervalMap intervals; emplaced_in_interval_tree = intervals.emplace(interval, element_count); key_attribute_container.insert({key, std::move(intervals)}); } if (unlikely(!emplaced_in_interval_tree)) { InvalidIntervalWithKey invalid_interval{key, interval, element_count}; invalid_intervals_container.emplace_back(invalid_interval); } ++element_count; keys_extractor.rollbackCurrentKey(); } }); } template void RangeHashedDictionary::setAttributeValue(Attribute & attribute, const Field & value) { auto type_call = [&](const auto & dictionary_attribute_type) { using Type = std::decay_t; using AttributeType = typename Type::AttributeType; using ValueType = DictionaryValueType; auto & container = std::get>(attribute.container); container.emplace_back(); if (unlikely(attribute.is_value_nullable.has_value())) { bool value_is_null = value.isNull(); attribute.is_value_nullable->emplace_back(value_is_null); if (unlikely(value_is_null)) return; } ValueType value_to_insert; if constexpr (std::is_same_v) { const auto & string = value.safeGet(); StringRef string_ref = copyStringInArena(string_arena, string); value_to_insert = string_ref; } else { value_to_insert = static_cast(value.safeGet()); } container.back() = value_to_insert; }; callOnDictionaryAttributeType(attribute.type, type_call); } template Pipe RangeHashedDictionary::read(const Names & column_names, size_t max_block_size, size_t num_streams) const { auto key_to_index_column = ColumnUInt64::create(); auto range_min_column = dict_struct.range_min->type->createColumn(); auto range_max_column = dict_struct.range_max->type->createColumn(); PaddedPODArray keys; impl::callOnRangeType(dict_struct.range_min->type, [&](const auto & types) { using Types = std::decay_t; using RangeColumnType = typename Types::LeftType; using RangeStorageType = typename RangeColumnType::ValueType; auto * range_min_column_typed = typeid_cast(range_min_column.get()); if (!range_min_column_typed) throw Exception(ErrorCodes::TYPE_MISMATCH, "Dictionary {} range min column type should be equal to {}", getFullName(), dict_struct.range_min->type->getName()); auto * range_max_column_typed = typeid_cast(range_max_column.get()); if (!range_max_column_typed) throw Exception(ErrorCodes::TYPE_MISMATCH, "Dictionary {} range max column type should be equal to {}", getFullName(), dict_struct.range_max->type->getName()); auto & key_to_index_column_data = key_to_index_column->getData(); auto & range_min_column_data = range_min_column_typed->getData(); auto & range_max_column_data = range_max_column_typed->getData(); const auto & container = std::get>(key_attribute.container); const auto & invalid_intervals_container = std::get>(key_attribute.invalid_intervals_container); keys.reserve(element_count); key_to_index_column_data.reserve(element_count); range_min_column_data.reserve(element_count); range_max_column_data.reserve(element_count); for (const auto & key : container) { for (const auto & [interval, index] : key.getMapped()) { keys.emplace_back(key.getKey()); key_to_index_column_data.emplace_back(index); range_min_column_data.push_back(interval.left); range_max_column_data.push_back(interval.right); } } for (const auto & invalid_interval_with_key : invalid_intervals_container) { keys.emplace_back(invalid_interval_with_key.key); key_to_index_column_data.emplace_back(invalid_interval_with_key.attribute_value_index); range_min_column_data.push_back(invalid_interval_with_key.interval.left); range_max_column_data.push_back(invalid_interval_with_key.interval.right); } }); auto range_min_column_with_type = ColumnWithTypeAndName{std::move(range_min_column), dict_struct.range_min->type, dict_struct.range_min->name}; auto range_max_column_with_type = ColumnWithTypeAndName{std::move(range_max_column), dict_struct.range_max->type, dict_struct.range_max->name}; ColumnsWithTypeAndName key_columns; if constexpr (dictionary_key_type == DictionaryKeyType::Simple) { auto keys_column = getColumnFromPODArray(std::move(keys)); key_columns = {ColumnWithTypeAndName(std::move(keys_column), std::make_shared(), dict_struct.id->name)}; } else { key_columns = deserializeColumnsWithTypeAndNameFromKeys(dict_struct, keys, 0, keys.size()); } key_columns.emplace_back(ColumnWithTypeAndName{std::move(key_to_index_column), std::make_shared(), ""}); ColumnsWithTypeAndName data_columns = {std::move(range_min_column_with_type), std::move(range_max_column_with_type)}; std::shared_ptr dictionary = shared_from_this(); DictionarySourceCoordinator::ReadColumnsFunc read_keys_func = [dictionary_copy = dictionary]( const Strings & attribute_names, const DataTypes & result_types, const Columns & key_columns_, const DataTypes, const Columns &) { auto range_dictionary_ptr = std::static_pointer_cast>(dictionary_copy); size_t attribute_names_size = attribute_names.size(); Columns result; result.reserve(attribute_names_size); const ColumnPtr & key_column = key_columns_.back(); const auto * key_to_index_column_ = typeid_cast(key_column.get()); if (!key_to_index_column_) throw Exception(ErrorCodes::LOGICAL_ERROR, "Dictionary {} read expect indexes column with type UInt64", range_dictionary_ptr->getFullName()); const auto & data = key_to_index_column_->getData(); for (size_t i = 0; i < attribute_names_size; ++i) { const auto & attribute_name = attribute_names[i]; const auto & result_type = result_types[i]; result.emplace_back(range_dictionary_ptr->getColumnInternal(attribute_name, result_type, data)); } return result; }; auto coordinator = std::make_shared( dictionary, column_names, std::move(key_columns), std::move(data_columns), max_block_size, std::move(read_keys_func)); auto result = coordinator->read(num_streams); return result; } }