#pragma once #include #include namespace DB { static inline constexpr auto FULL_TEXT_INDEX_NAME = "full_text"; static inline constexpr auto INVERTED_INDEX_NAME = "inverted"; static inline constexpr UInt64 UNLIMITED_ROWS_PER_POSTINGS_LIST = 0; static inline constexpr UInt64 MIN_ROWS_PER_POSTINGS_LIST = 8 * 1024; static inline constexpr UInt64 DEFAULT_MAX_ROWS_PER_POSTINGS_LIST = 64 * 1024; struct GinFilterParameters { GinFilterParameters(size_t ngrams_, UInt64 max_rows_per_postings_list_); size_t ngrams; UInt64 max_rows_per_postings_list; }; struct GinSegmentWithRowIdRange { /// Segment ID of the row ID range UInt32 segment_id; /// First row ID in the range UInt32 range_start; /// Last row ID in the range (inclusive) UInt32 range_end; }; using GinSegmentWithRowIdRangeVector = std::vector; /// GinFilter provides underlying functionalities for building full-text index and also /// it does filtering the unmatched rows according to its query string. /// It also builds and uses skipping index which stores (segmentID, RowIDStart, RowIDEnd) triples. class GinFilter { public: explicit GinFilter(const GinFilterParameters & params_); /// Add term (located at 'data' with length 'len') and its row ID to the postings list builder /// for building full-text index for the given store. void add(const char * data, size_t len, UInt32 rowID, GinIndexStorePtr & store) const; /// Accumulate (segmentID, RowIDStart, RowIDEnd) for building skipping index void addRowRangeToGinFilter(UInt32 segmentID, UInt32 rowIDStart, UInt32 rowIDEnd); /// Clear the content void clear(); /// Check if the filter (built from query string) contains any rows in given filter by using /// given postings list cache bool contains(const GinFilter & filter, PostingsCacheForStore & cache_store) const; /// Set the query string of the filter void setQueryString(const char * data, size_t len) { query_string = String(data, len); } /// Add term which are tokens generated from the query string void addTerm(const char * data, size_t len) { if (len > FST::MAX_TERM_LENGTH) return; terms.push_back(String(data, len)); } /// Getter const String & getQueryString() const { return query_string; } const std::vector & getTerms() const { return terms; } const GinSegmentWithRowIdRangeVector & getFilter() const { return rowid_ranges; } GinSegmentWithRowIdRangeVector & getFilter() { return rowid_ranges; } private: /// Filter parameters const GinFilterParameters & params; /// Query string of the filter String query_string; /// Tokenized terms from query string std::vector terms; /// Row ID ranges which are (segmentID, RowIDStart, RowIDEnd) GinSegmentWithRowIdRangeVector rowid_ranges; /// Check if the given postings list cache has matched rows by using the filter bool match(const GinPostingsCache & postings_cache) const; }; using GinFilters = std::vector; }