#pragma once #include #include #include #include #include #include #include #include #include #include #include #include /// GinIndexStore manages the generalized inverted index ("gin") (full-text index )for a data part, and it is made up of one or more /// immutable index segments. /// /// There are 4 types of index files in a store: /// 1. Segment ID file(.gin_sid): it contains one byte for version followed by the next available segment ID. /// 2. Segment Metadata file(.gin_seg): it contains index segment metadata. /// - Its file format is an array of GinIndexSegment as defined in this file. /// - postings_start_offset points to the file(.gin_post) starting position for the segment's postings list. /// - dict_start_offset points to the file(.gin_dict) starting position for the segment's dictionaries. /// 3. Dictionary file(.gin_dict): it contains dictionaries. /// - It contains an array of (FST_size, FST_blob) which has size and actual data of FST. /// 4. Postings Lists(.gin_post): it contains postings lists data. /// - It contains an array of serialized postings lists. /// /// During the searching in the segment, the segment's meta data can be found in .gin_seg file. From the meta data, /// the starting position of its dictionary is used to locate its FST. Then FST is read into memory. /// By using the term and FST, the offset("output" in FST) of the postings list for the term /// in FST is found. The offset plus the postings_start_offset is the file location in .gin_post file /// for its postings list. namespace DB { /// GinIndexPostingsList which uses 32-bit Roaring using GinIndexPostingsList = roaring::Roaring; using GinIndexPostingsListPtr = std::shared_ptr; /// Build a postings list for a term class GinIndexPostingsBuilder { public: explicit GinIndexPostingsBuilder(UInt64 limit); /// Check whether a row_id is already added bool contains(UInt32 row_id) const; /// Add a row_id into the builder void add(UInt32 row_id); /// Serialize the content of builder to given WriteBuffer, returns the bytes of serialized data UInt64 serialize(WriteBuffer & buffer); /// Deserialize the postings list data from given ReadBuffer, return a pointer to the GinIndexPostingsList created by deserialization static GinIndexPostingsListPtr deserialize(ReadBuffer & buffer); private: constexpr static int MIN_SIZE_FOR_ROARING_ENCODING = 16; static constexpr auto GIN_COMPRESSION_CODEC = "ZSTD"; static constexpr auto GIN_COMPRESSION_LEVEL = 1; /// When the list length is no greater than MIN_SIZE_FOR_ROARING_ENCODING, array 'rowid_lst' is used /// As a special case, rowid_lst[0] == CONTAINS_ALL encodes that all rowids are set. std::array rowid_lst; /// When the list length is greater than MIN_SIZE_FOR_ROARING_ENCODING, roaring bitmap 'rowid_bitmap' is used roaring::Roaring rowid_bitmap; /// rowid_lst_length stores the number of row IDs in 'rowid_lst' array, can also be a flag(0xFF) indicating that roaring bitmap is used UInt8 rowid_lst_length = 0; /// Indicates that all rowids are contained, see 'rowid_lst' static constexpr UInt32 CONTAINS_ALL = std::numeric_limits::max(); /// Indicates that roaring bitmap is used, see 'rowid_lst_length'. static constexpr UInt8 USES_BIT_MAP = 0xFF; /// Clear the postings list and reset it with MATCHALL flags when the size of the postings list is beyond the limit UInt64 size_limit; /// Check whether the builder is using roaring bitmap bool useRoaring() const { return rowid_lst_length == USES_BIT_MAP; } /// Check whether the postings list has been flagged to contain all row ids bool containsAllRows() const { return rowid_lst[0] == CONTAINS_ALL; } }; using GinIndexPostingsBuilderPtr = std::shared_ptr; /// Gin index segment descriptor, which contains: struct GinIndexSegment { /// Segment ID retrieved from next available ID from file .gin_sid UInt32 segment_id = 0; /// Start row ID for this segment UInt32 next_row_id = 1; /// .gin_post file offset of this segment's postings lists UInt64 postings_start_offset = 0; /// .gin_dict file offset of this segment's dictionaries UInt64 dict_start_offset = 0; }; struct GinSegmentDictionary { /// .gin_post file offset of this segment's postings lists UInt64 postings_start_offset; /// .gin_dict file offset of this segment's dictionaries UInt64 dict_start_offset; /// (Minimized) Finite State Transducer, which can be viewed as a map of , where offset is the /// offset to the term's posting list in postings list file FST::FiniteStateTransducer offsets; }; using GinSegmentDictionaryPtr = std::shared_ptr; /// Gin index store which has gin index meta data for the corresponding column data part class GinIndexStore { public: /// Container for all term's Gin Index Postings List Builder using GinIndexPostingsBuilderContainer = absl::flat_hash_map; GinIndexStore(const String & name_, DataPartStoragePtr storage_); GinIndexStore(const String & name_, DataPartStoragePtr storage_, MutableDataPartStoragePtr data_part_storage_builder_, UInt64 max_digestion_size_); /// Check existence by checking the existence of file .gin_sid bool exists() const; /// Get a range of next 'numIDs'-many available row IDs UInt32 getNextRowIDRange(size_t numIDs); /// Get next available segment ID by updating file .gin_sid UInt32 getNextSegmentID(); /// Get total number of segments in the store UInt32 getNumOfSegments(); /// Get current postings list builder const GinIndexPostingsBuilderContainer & getPostingsListBuilder() const { return current_postings; } /// Set postings list builder for given term void setPostingsBuilder(const String & term, GinIndexPostingsBuilderPtr builder) { current_postings[term] = builder; } /// Check if we need to write segment to Gin index files bool needToWrite() const; /// Accumulate the size of text data which has been digested void incrementCurrentSizeBy(UInt64 sz) { current_size += sz; } UInt32 getCurrentSegmentID() const { return current_segment.segment_id; } /// Do last segment writing void finalize(); /// Method for writing segment data to Gin index files void writeSegment(); const String & getName() const { return name; } private: friend class GinIndexStoreDeserializer; /// Initialize all indexing files for this store void initFileStreams(); /// Get a range of next available segment IDs by updating file .gin_sid UInt32 getNextSegmentIDRange(const String & file_name, size_t n); String name; DataPartStoragePtr storage; MutableDataPartStoragePtr data_part_storage_builder; UInt32 cached_segment_num = 0; std::mutex mutex; /// Dictionaries indexed by segment ID using GinSegmentDictionaries = std::unordered_map; /// Term's dictionaries which are loaded from .gin_dict files GinSegmentDictionaries segment_dictionaries; /// Container for building postings lists during index construction GinIndexPostingsBuilderContainer current_postings; /// For the segmentation of Gin indexes GinIndexSegment current_segment; UInt64 current_size = 0; const UInt64 max_digestion_size = 0; /// File streams for segment, dictionaries and postings lists std::unique_ptr metadata_file_stream; std::unique_ptr dict_file_stream; std::unique_ptr postings_file_stream; static constexpr auto GIN_SEGMENT_ID_FILE_TYPE = ".gin_sid"; static constexpr auto GIN_SEGMENT_METADATA_FILE_TYPE = ".gin_seg"; static constexpr auto GIN_DICTIONARY_FILE_TYPE = ".gin_dict"; static constexpr auto GIN_POSTINGS_FILE_TYPE = ".gin_post"; enum class Format : uint8_t { v0 = 0, v1 = 1, /// Initial version }; static constexpr auto CURRENT_GIN_FILE_FORMAT_VERSION = Format::v1; }; using GinIndexStorePtr = std::shared_ptr; /// Container for postings lists for each segment using GinSegmentedPostingsListContainer = std::unordered_map; /// Postings lists and terms built from query string using GinPostingsCache = std::unordered_map; using GinPostingsCachePtr = std::shared_ptr; /// Gin index store reader which helps to read segments, dictionaries and postings list class GinIndexStoreDeserializer : private boost::noncopyable { public: explicit GinIndexStoreDeserializer(const GinIndexStorePtr & store_); /// Read segment information from .gin_seg files void readSegments(); /// Read all dictionaries from .gin_dict files void readSegmentDictionaries(); /// Read dictionary for given segment id void readSegmentDictionary(UInt32 segment_id); /// Read postings lists for the term GinSegmentedPostingsListContainer readSegmentedPostingsLists(const String & term); /// Read postings lists for terms (which are created by tokenzing query string) GinPostingsCachePtr createPostingsCacheFromTerms(const std::vector & terms); private: /// Initialize gin index files void initFileStreams(); /// The store for the reader GinIndexStorePtr store; /// File streams for reading Gin Index std::unique_ptr metadata_file_stream; std::unique_ptr dict_file_stream; std::unique_ptr postings_file_stream; /// Current segment, used in building index GinIndexSegment current_segment; }; /// PostingsCacheForStore contains postings lists from 'store' which are retrieved from Gin index files for the terms in query strings /// GinPostingsCache is per query string (one query can have multiple query strings): when skipping index (row ID ranges) is used for the part during the /// query, the postings cache is created and associated with the store where postings lists are read /// for the tokenized query string. The postings caches are released automatically when the query is done. struct PostingsCacheForStore { /// Which store to retrieve postings lists GinIndexStorePtr store; /// map of std::unordered_map cache; /// Get postings lists for query string, return nullptr if not found GinPostingsCachePtr getPostings(const String & query_string) const; }; /// A singleton for storing GinIndexStores class GinIndexStoreFactory : private boost::noncopyable { public: /// Get singleton of GinIndexStoreFactory static GinIndexStoreFactory & instance(); /// Get GinIndexStore by using index name, disk and part_path (which are combined to create key in stores) GinIndexStorePtr get(const String & name, DataPartStoragePtr storage); /// Remove all Gin index files which are under the same part_path void remove(const String & part_path); /// GinIndexStores indexed by part file path using GinIndexStores = std::unordered_map; private: GinIndexStores stores; std::mutex mutex; }; inline bool isGinFile(const String &file_name) { return (file_name.ends_with(".gin_dict") || file_name.ends_with(".gin_post") || file_name.ends_with(".gin_seg") || file_name.ends_with(".gin_sid")); } }