use std::collections::hash_map::Entry; use std::hash::Hash; use fxhash::FxHashMap; use heed::types::ByteSlice; use heed::{BytesEncode, Database, RoTxn}; use super::interner::{DedupInterner, Interned}; use crate::{Index, Result}; /// A cache storing pointers to values in the LMDB databases. /// /// Used for performance reasons only. By using this cache, we avoid performing a /// database lookup and instead get a direct reference to the value using a fast /// local HashMap lookup. #[derive(Default)] pub struct DatabaseCache<'ctx> { pub word_pair_proximity_docids: FxHashMap<(u8, Interned, Interned), Option<&'ctx [u8]>>, pub word_prefix_pair_proximity_docids: FxHashMap<(u8, Interned, Interned), Option<&'ctx [u8]>>, pub prefix_word_pair_proximity_docids: FxHashMap<(u8, Interned, Interned), Option<&'ctx [u8]>>, pub word_docids: FxHashMap, Option<&'ctx [u8]>>, pub exact_word_docids: FxHashMap, Option<&'ctx [u8]>>, pub word_prefix_docids: FxHashMap, Option<&'ctx [u8]>>, } impl<'ctx> DatabaseCache<'ctx> { fn get_value<'v, K1, KC>( txn: &'ctx RoTxn, cache_key: K1, db_key: &'v KC::EItem, cache: &mut FxHashMap>, db: Database, ) -> Result> where K1: Copy + Eq + Hash, KC: BytesEncode<'v>, { let bitmap_ptr = match cache.entry(cache_key) { Entry::Occupied(bitmap_ptr) => *bitmap_ptr.get(), Entry::Vacant(entry) => { let bitmap_ptr = db.get(txn, db_key)?; entry.insert(bitmap_ptr); bitmap_ptr } }; Ok(bitmap_ptr) } /// Retrieve or insert the given value in the `word_docids` database. pub fn get_word_docids( &mut self, index: &Index, txn: &'ctx RoTxn, word_interner: &DedupInterner, word: Interned, ) -> Result> { Self::get_value( txn, word, word_interner.get(word).as_str(), &mut self.word_docids, index.word_docids.remap_data_type::(), ) } /// Retrieve or insert the given value in the `word_prefix_docids` database. pub fn get_word_prefix_docids( &mut self, index: &Index, txn: &'ctx RoTxn, word_interner: &DedupInterner, prefix: Interned, ) -> Result> { Self::get_value( txn, prefix, word_interner.get(prefix).as_str(), &mut self.word_prefix_docids, index.word_prefix_docids.remap_data_type::(), ) } pub fn get_word_pair_proximity_docids( &mut self, index: &Index, txn: &'ctx RoTxn, word_interner: &DedupInterner, word1: Interned, word2: Interned, proximity: u8, ) -> Result> { Self::get_value( txn, (proximity, word1, word2), &(proximity, word_interner.get(word1).as_str(), word_interner.get(word2).as_str()), &mut self.word_pair_proximity_docids, index.word_pair_proximity_docids.remap_data_type::(), ) } pub fn get_word_prefix_pair_proximity_docids( &mut self, index: &Index, txn: &'ctx RoTxn, word_interner: &DedupInterner, word1: Interned, prefix2: Interned, proximity: u8, ) -> Result> { Self::get_value( txn, (proximity, word1, prefix2), &(proximity, word_interner.get(word1).as_str(), word_interner.get(prefix2).as_str()), &mut self.word_prefix_pair_proximity_docids, index.word_prefix_pair_proximity_docids.remap_data_type::(), ) } pub fn get_prefix_word_pair_proximity_docids( &mut self, index: &Index, txn: &'ctx RoTxn, word_interner: &DedupInterner, left_prefix: Interned, right: Interned, proximity: u8, ) -> Result> { Self::get_value( txn, (proximity, left_prefix, right), &( proximity, word_interner.get(left_prefix).as_str(), word_interner.get(right).as_str(), ), &mut self.prefix_word_pair_proximity_docids, index.prefix_word_pair_proximity_docids.remap_data_type::(), ) } }