mirror of
https://github.com/meilisearch/MeiliSearch
synced 2025-01-13 14:57:30 +01:00
109 lines
3.4 KiB
Rust
109 lines
3.4 KiB
Rust
|
use std::collections::HashSet;
|
||
|
|
||
|
use heed::Database;
|
||
|
use heed::{types::Bytes, RwTxn};
|
||
|
use roaring::RoaringBitmap;
|
||
|
|
||
|
use crate::{CboRoaringBitmapCodec, Index, Prefix, Result};
|
||
|
|
||
|
struct WordPrefixDocids {
|
||
|
database: Database<Bytes, CboRoaringBitmapCodec>,
|
||
|
prefix_database: Database<Bytes, CboRoaringBitmapCodec>,
|
||
|
}
|
||
|
|
||
|
impl WordPrefixDocids {
|
||
|
fn new(
|
||
|
database: Database<Bytes, CboRoaringBitmapCodec>,
|
||
|
prefix_database: Database<Bytes, CboRoaringBitmapCodec>,
|
||
|
) -> WordPrefixDocids {
|
||
|
WordPrefixDocids { database, prefix_database }
|
||
|
}
|
||
|
|
||
|
fn execute(
|
||
|
self,
|
||
|
wtxn: &mut heed::RwTxn,
|
||
|
prefix_to_compute: &HashSet<Prefix>,
|
||
|
prefix_to_delete: &HashSet<Prefix>,
|
||
|
) -> Result<()> {
|
||
|
self.delete_prefixes(wtxn, prefix_to_delete)?;
|
||
|
self.recompute_modified_prefixes(wtxn, prefix_to_compute)
|
||
|
}
|
||
|
|
||
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::prefix")]
|
||
|
fn delete_prefixes(&self, wtxn: &mut heed::RwTxn, prefixes: &HashSet<Prefix>) -> Result<()> {
|
||
|
// We remove all the entries that are no more required in this word prefix docids database.
|
||
|
for prefix in prefixes {
|
||
|
let prefix = prefix.as_bytes();
|
||
|
if !self.prefix_database.delete(wtxn, prefix)? {
|
||
|
unreachable!("We tried to delete an unknown key")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::prefix")]
|
||
|
fn recompute_modified_prefixes(
|
||
|
&self,
|
||
|
wtxn: &mut RwTxn,
|
||
|
prefixes: &HashSet<Prefix>,
|
||
|
) -> Result<()> {
|
||
|
// We fetch the docids associated to the newly added word prefix fst only.
|
||
|
let mut docids = RoaringBitmap::new();
|
||
|
for prefix in prefixes {
|
||
|
docids.clear();
|
||
|
let prefix = prefix.as_bytes();
|
||
|
for result in self.database.prefix_iter(wtxn, prefix)? {
|
||
|
let (_word, data) = result?;
|
||
|
docids |= &data;
|
||
|
}
|
||
|
|
||
|
self.prefix_database.put(wtxn, prefix, &docids)?;
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::prefix")]
|
||
|
pub fn compute_word_prefix_docids(
|
||
|
wtxn: &mut RwTxn,
|
||
|
index: &Index,
|
||
|
prefix_to_compute: &HashSet<Prefix>,
|
||
|
prefix_to_delete: &HashSet<Prefix>,
|
||
|
) -> Result<()> {
|
||
|
WordPrefixDocids::new(
|
||
|
index.word_docids.remap_key_type(),
|
||
|
index.word_prefix_docids.remap_key_type(),
|
||
|
)
|
||
|
.execute(wtxn, prefix_to_compute, prefix_to_delete)
|
||
|
}
|
||
|
|
||
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::prefix")]
|
||
|
pub fn compute_word_prefix_fid_docids(
|
||
|
wtxn: &mut RwTxn,
|
||
|
index: &Index,
|
||
|
prefix_to_compute: &HashSet<Prefix>,
|
||
|
prefix_to_delete: &HashSet<Prefix>,
|
||
|
) -> Result<()> {
|
||
|
WordPrefixDocids::new(
|
||
|
index.word_fid_docids.remap_key_type(),
|
||
|
index.word_prefix_fid_docids.remap_key_type(),
|
||
|
)
|
||
|
.execute(wtxn, prefix_to_compute, prefix_to_delete)
|
||
|
}
|
||
|
|
||
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::prefix")]
|
||
|
pub fn compute_word_prefix_position_docids(
|
||
|
wtxn: &mut RwTxn,
|
||
|
index: &Index,
|
||
|
prefix_to_compute: &HashSet<Prefix>,
|
||
|
prefix_to_delete: &HashSet<Prefix>,
|
||
|
) -> Result<()> {
|
||
|
WordPrefixDocids::new(
|
||
|
index.word_position_docids.remap_key_type(),
|
||
|
index.word_prefix_position_docids.remap_key_type(),
|
||
|
)
|
||
|
.execute(wtxn, prefix_to_compute, prefix_to_delete)
|
||
|
}
|