mirror of
https://github.com/meilisearch/meilisearch.git
synced 2025-01-23 19:43:10 +08:00
252 lines
8.4 KiB
Rust
252 lines
8.4 KiB
Rust
use std::cell::RefCell;
|
|
|
|
use hashbrown::HashSet;
|
|
use heed::types::Bytes;
|
|
use heed::{Database, RoTxn};
|
|
use memmap2::Mmap;
|
|
use rayon::iter::{IntoParallelIterator, ParallelIterator};
|
|
use roaring::RoaringBitmap;
|
|
|
|
use super::channel::*;
|
|
use super::extract::{
|
|
merge_caches, transpose_and_freeze_caches, BalancedCaches, DelAddRoaringBitmap, FacetKind,
|
|
GeoExtractorData,
|
|
};
|
|
use crate::{CboRoaringBitmapCodec, FieldId, GeoPoint, Index, InternalError, Result};
|
|
|
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::merge")]
|
|
pub fn merge_and_send_rtree<'extractor, MSP>(
|
|
datastore: impl IntoIterator<Item = RefCell<GeoExtractorData<'extractor>>>,
|
|
rtxn: &RoTxn,
|
|
index: &Index,
|
|
geo_sender: GeoSender<'_, '_>,
|
|
must_stop_processing: &MSP,
|
|
) -> Result<()>
|
|
where
|
|
MSP: Fn() -> bool + Sync,
|
|
{
|
|
let mut rtree = index.geo_rtree(rtxn)?.unwrap_or_default();
|
|
let mut faceted = index.geo_faceted_documents_ids(rtxn)?;
|
|
|
|
for data in datastore {
|
|
if must_stop_processing() {
|
|
return Err(InternalError::AbortedIndexation.into());
|
|
}
|
|
|
|
let mut frozen = data.into_inner().freeze()?;
|
|
for result in frozen.iter_and_clear_removed() {
|
|
let extracted_geo_point = result?;
|
|
let removed = rtree.remove(&GeoPoint::from(extracted_geo_point));
|
|
debug_assert!(removed.is_some());
|
|
let removed = faceted.remove(extracted_geo_point.docid);
|
|
debug_assert!(removed);
|
|
}
|
|
|
|
for result in frozen.iter_and_clear_inserted() {
|
|
let extracted_geo_point = result?;
|
|
rtree.insert(GeoPoint::from(extracted_geo_point));
|
|
let inserted = faceted.insert(extracted_geo_point.docid);
|
|
debug_assert!(inserted);
|
|
}
|
|
}
|
|
|
|
let mut file = tempfile::tempfile()?;
|
|
bincode::serialize_into(&mut file, &rtree).map_err(InternalError::BincodeError)?;
|
|
file.sync_all()?;
|
|
|
|
let rtree_mmap = unsafe { Mmap::map(&file)? };
|
|
geo_sender.set_rtree(rtree_mmap).unwrap();
|
|
geo_sender.set_geo_faceted(&faceted).unwrap();
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::merge")]
|
|
pub fn merge_and_send_docids<'extractor, MSP, D>(
|
|
mut caches: Vec<BalancedCaches<'extractor>>,
|
|
database: Database<Bytes, Bytes>,
|
|
index: &Index,
|
|
docids_sender: WordDocidsSender<D>,
|
|
must_stop_processing: &MSP,
|
|
) -> Result<()>
|
|
where
|
|
MSP: Fn() -> bool + Sync,
|
|
D: DatabaseType + Sync,
|
|
{
|
|
transpose_and_freeze_caches(&mut caches)?.into_par_iter().try_for_each(|frozen| {
|
|
let rtxn = index.read_txn()?;
|
|
if must_stop_processing() {
|
|
return Err(InternalError::AbortedIndexation.into());
|
|
}
|
|
merge_caches(frozen, |key, DelAddRoaringBitmap { del, add }| {
|
|
let current = database.get(&rtxn, key)?;
|
|
match merge_cbo_bitmaps(current, del, add)? {
|
|
Operation::Write(bitmap) => {
|
|
docids_sender.write(key, &bitmap).unwrap();
|
|
Ok(())
|
|
}
|
|
Operation::Delete => {
|
|
docids_sender.delete(key).unwrap();
|
|
Ok(())
|
|
}
|
|
Operation::Ignore => Ok(()),
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
#[tracing::instrument(level = "trace", skip_all, target = "indexing::merge")]
|
|
pub fn merge_and_send_facet_docids<'extractor>(
|
|
mut caches: Vec<BalancedCaches<'extractor>>,
|
|
database: FacetDatabases,
|
|
index: &Index,
|
|
docids_sender: FacetDocidsSender,
|
|
) -> Result<FacetFieldIdsDelta> {
|
|
transpose_and_freeze_caches(&mut caches)?
|
|
.into_par_iter()
|
|
.map(|frozen| {
|
|
let mut facet_field_ids_delta = FacetFieldIdsDelta::default();
|
|
let rtxn = index.read_txn()?;
|
|
merge_caches(frozen, |key, DelAddRoaringBitmap { del, add }| {
|
|
let current = database.get_cbo_roaring_bytes_value(&rtxn, key)?;
|
|
match merge_cbo_bitmaps(current, del, add)? {
|
|
Operation::Write(bitmap) => {
|
|
facet_field_ids_delta.register_from_key(key);
|
|
docids_sender.write(key, &bitmap).unwrap();
|
|
Ok(())
|
|
}
|
|
Operation::Delete => {
|
|
facet_field_ids_delta.register_from_key(key);
|
|
docids_sender.delete(key).unwrap();
|
|
Ok(())
|
|
}
|
|
Operation::Ignore => Ok(()),
|
|
}
|
|
})?;
|
|
|
|
Ok(facet_field_ids_delta)
|
|
})
|
|
.reduce(|| Ok(FacetFieldIdsDelta::default()), |lhs, rhs| Ok(lhs?.merge(rhs?)))
|
|
}
|
|
|
|
pub struct FacetDatabases<'a> {
|
|
index: &'a Index,
|
|
}
|
|
|
|
impl<'a> FacetDatabases<'a> {
|
|
pub fn new(index: &'a Index) -> Self {
|
|
Self { index }
|
|
}
|
|
|
|
fn get_cbo_roaring_bytes_value<'t>(
|
|
&self,
|
|
rtxn: &'t RoTxn<'_>,
|
|
key: &[u8],
|
|
) -> heed::Result<Option<&'t [u8]>> {
|
|
let (facet_kind, key) = FacetKind::extract_from_key(key);
|
|
|
|
let value =
|
|
super::channel::Database::from(facet_kind).database(self.index).get(rtxn, key)?;
|
|
match facet_kind {
|
|
// skip level group size
|
|
FacetKind::String | FacetKind::Number => Ok(value.map(|v| &v[1..])),
|
|
_ => Ok(value),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Default)]
|
|
pub struct FacetFieldIdsDelta {
|
|
/// The field ids that have been modified
|
|
modified_facet_string_ids: HashSet<FieldId>,
|
|
modified_facet_number_ids: HashSet<FieldId>,
|
|
}
|
|
|
|
impl FacetFieldIdsDelta {
|
|
fn register_facet_string_id(&mut self, field_id: FieldId) {
|
|
self.modified_facet_string_ids.insert(field_id);
|
|
}
|
|
|
|
fn register_facet_number_id(&mut self, field_id: FieldId) {
|
|
self.modified_facet_number_ids.insert(field_id);
|
|
}
|
|
|
|
fn register_from_key(&mut self, key: &[u8]) {
|
|
let (facet_kind, field_id) = self.extract_key_data(key);
|
|
match facet_kind {
|
|
FacetKind::Number => self.register_facet_number_id(field_id),
|
|
FacetKind::String => self.register_facet_string_id(field_id),
|
|
_ => (),
|
|
}
|
|
}
|
|
|
|
fn extract_key_data(&self, key: &[u8]) -> (FacetKind, FieldId) {
|
|
let facet_kind = FacetKind::from(key[0]);
|
|
let field_id = FieldId::from_be_bytes([key[1], key[2]]);
|
|
(facet_kind, field_id)
|
|
}
|
|
|
|
pub fn modified_facet_string_ids(&self) -> Option<Vec<FieldId>> {
|
|
if self.modified_facet_string_ids.is_empty() {
|
|
None
|
|
} else {
|
|
Some(self.modified_facet_string_ids.iter().copied().collect())
|
|
}
|
|
}
|
|
|
|
pub fn modified_facet_number_ids(&self) -> Option<Vec<FieldId>> {
|
|
if self.modified_facet_number_ids.is_empty() {
|
|
None
|
|
} else {
|
|
Some(self.modified_facet_number_ids.iter().copied().collect())
|
|
}
|
|
}
|
|
|
|
pub fn merge(mut self, rhs: Self) -> Self {
|
|
let Self { modified_facet_number_ids, modified_facet_string_ids } = rhs;
|
|
modified_facet_number_ids.into_iter().for_each(|fid| {
|
|
self.modified_facet_number_ids.insert(fid);
|
|
});
|
|
modified_facet_string_ids.into_iter().for_each(|fid| {
|
|
self.modified_facet_string_ids.insert(fid);
|
|
});
|
|
self
|
|
}
|
|
}
|
|
|
|
enum Operation {
|
|
Write(RoaringBitmap),
|
|
Delete,
|
|
Ignore,
|
|
}
|
|
|
|
/// A function that merges the DelAdd CboRoaringBitmaps with the current bitmap.
|
|
fn merge_cbo_bitmaps(
|
|
current: Option<&[u8]>,
|
|
del: Option<RoaringBitmap>,
|
|
add: Option<RoaringBitmap>,
|
|
) -> Result<Operation> {
|
|
let current = current.map(CboRoaringBitmapCodec::deserialize_from).transpose()?;
|
|
match (current, del, add) {
|
|
(None, None, None) => Ok(Operation::Ignore), // but it's strange
|
|
(None, None, Some(add)) => Ok(Operation::Write(add)),
|
|
(None, Some(_del), None) => Ok(Operation::Ignore), // but it's strange
|
|
(None, Some(_del), Some(add)) => Ok(Operation::Write(add)),
|
|
(Some(_current), None, None) => Ok(Operation::Ignore), // but it's strange
|
|
(Some(current), None, Some(add)) => Ok(Operation::Write(current | add)),
|
|
(Some(current), Some(del), add) => {
|
|
let output = match add {
|
|
Some(add) => (¤t - del) | add,
|
|
None => ¤t - del,
|
|
};
|
|
if output.is_empty() {
|
|
Ok(Operation::Delete)
|
|
} else if current == output {
|
|
Ok(Operation::Ignore)
|
|
} else {
|
|
Ok(Operation::Write(output))
|
|
}
|
|
}
|
|
}
|
|
}
|