2020-10-20 16:40:50 +02:00
|
|
|
use std::borrow::Cow;
|
|
|
|
|
2020-10-22 14:23:33 +02:00
|
|
|
use anyhow::{bail, ensure};
|
2020-10-20 14:20:17 +02:00
|
|
|
use bstr::ByteSlice as _;
|
|
|
|
use fst::IntoStreamer;
|
|
|
|
use roaring::RoaringBitmap;
|
|
|
|
|
|
|
|
use crate::heed_codec::CboRoaringBitmapCodec;
|
|
|
|
|
2020-10-21 15:55:48 +02:00
|
|
|
const WORDS_FST_KEY: &[u8] = crate::index::WORDS_FST_KEY.as_bytes();
|
2020-10-22 14:23:33 +02:00
|
|
|
const FIELDS_IDS_MAP_KEY: &[u8] = crate::index::FIELDS_IDS_MAP_KEY.as_bytes();
|
2020-10-21 15:55:48 +02:00
|
|
|
const DOCUMENTS_IDS_KEY: &[u8] = crate::index::DOCUMENTS_IDS_KEY.as_bytes();
|
2020-10-20 14:20:17 +02:00
|
|
|
|
2020-10-20 16:40:50 +02:00
|
|
|
pub fn main_merge(key: &[u8], values: &[Cow<[u8]>]) -> anyhow::Result<Vec<u8>> {
|
2020-10-20 14:20:17 +02:00
|
|
|
match key {
|
|
|
|
WORDS_FST_KEY => {
|
|
|
|
let fsts: Vec<_> = values.iter().map(|v| fst::Set::new(v).unwrap()).collect();
|
|
|
|
|
|
|
|
// Union of the FSTs
|
|
|
|
let mut op = fst::set::OpBuilder::new();
|
|
|
|
fsts.iter().for_each(|fst| op.push(fst.into_stream()));
|
|
|
|
let op = op.r#union();
|
|
|
|
|
|
|
|
let mut build = fst::SetBuilder::memory();
|
|
|
|
build.extend_stream(op.into_stream()).unwrap();
|
|
|
|
Ok(build.into_inner().unwrap())
|
|
|
|
},
|
2020-10-22 14:23:33 +02:00
|
|
|
FIELDS_IDS_MAP_KEY => {
|
|
|
|
ensure!(values.windows(2).all(|vs| vs[0] == vs[1]), "fields ids map doesn't match");
|
2020-10-20 14:20:17 +02:00
|
|
|
Ok(values[0].to_vec())
|
|
|
|
},
|
|
|
|
DOCUMENTS_IDS_KEY => word_docids_merge(&[], values),
|
2020-10-20 15:14:06 +02:00
|
|
|
otherwise => bail!("wut {:?}", otherwise),
|
2020-10-20 14:20:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-20 16:40:50 +02:00
|
|
|
pub fn word_docids_merge(_key: &[u8], values: &[Cow<[u8]>]) -> anyhow::Result<Vec<u8>> {
|
2020-10-20 14:20:17 +02:00
|
|
|
let (head, tail) = values.split_first().unwrap();
|
2020-10-20 16:40:50 +02:00
|
|
|
let mut head = RoaringBitmap::deserialize_from(&head[..])?;
|
2020-10-20 14:20:17 +02:00
|
|
|
|
|
|
|
for value in tail {
|
2020-10-20 16:40:50 +02:00
|
|
|
let bitmap = RoaringBitmap::deserialize_from(&value[..])?;
|
2020-10-20 14:20:17 +02:00
|
|
|
head.union_with(&bitmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut vec = Vec::with_capacity(head.serialized_size());
|
2020-10-20 15:14:06 +02:00
|
|
|
head.serialize_into(&mut vec)?;
|
2020-10-20 14:20:17 +02:00
|
|
|
Ok(vec)
|
|
|
|
}
|
|
|
|
|
2020-10-20 16:40:50 +02:00
|
|
|
pub fn docid_word_positions_merge(key: &[u8], _values: &[Cow<[u8]>]) -> anyhow::Result<Vec<u8>> {
|
2020-10-20 15:14:06 +02:00
|
|
|
bail!("merging docid word positions is an error ({:?})", key.as_bstr())
|
2020-10-20 14:20:17 +02:00
|
|
|
}
|
|
|
|
|
2020-10-20 16:40:50 +02:00
|
|
|
pub fn words_pairs_proximities_docids_merge(_key: &[u8], values: &[Cow<[u8]>]) -> anyhow::Result<Vec<u8>> {
|
2020-10-20 14:20:17 +02:00
|
|
|
let (head, tail) = values.split_first().unwrap();
|
2020-10-20 16:40:50 +02:00
|
|
|
let mut head = CboRoaringBitmapCodec::deserialize_from(&head[..])?;
|
2020-10-20 14:20:17 +02:00
|
|
|
|
|
|
|
for value in tail {
|
2020-10-20 16:40:50 +02:00
|
|
|
let bitmap = CboRoaringBitmapCodec::deserialize_from(&value[..])?;
|
2020-10-20 14:20:17 +02:00
|
|
|
head.union_with(&bitmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut vec = Vec::new();
|
2020-10-20 15:14:06 +02:00
|
|
|
CboRoaringBitmapCodec::serialize_into(&head, &mut vec)?;
|
2020-10-20 14:20:17 +02:00
|
|
|
Ok(vec)
|
|
|
|
}
|
|
|
|
|
2020-10-20 16:40:50 +02:00
|
|
|
pub fn documents_merge(key: &[u8], _values: &[Cow<[u8]>]) -> anyhow::Result<Vec<u8>> {
|
2020-10-20 15:14:06 +02:00
|
|
|
bail!("merging documents is an error ({:?})", key.as_bstr())
|
2020-10-20 14:20:17 +02:00
|
|
|
}
|
2020-10-26 20:18:10 +01:00
|
|
|
|
|
|
|
pub fn merge_two_obkv(base: obkv::KvReader, update: obkv::KvReader, buffer: &mut Vec<u8>) {
|
|
|
|
use itertools::merge_join_by;
|
|
|
|
use itertools::EitherOrBoth::{Both, Left, Right};
|
|
|
|
|
|
|
|
buffer.clear();
|
|
|
|
|
|
|
|
let mut writer = obkv::KvWriter::new(buffer);
|
|
|
|
for eob in merge_join_by(base.iter(), update.iter(), |(b, _), (u, _)| b.cmp(u)) {
|
|
|
|
match eob {
|
|
|
|
Both(_, (k, v)) | Left((k, v)) | Right((k, v)) => writer.insert(k, v).unwrap(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
writer.finish().unwrap();
|
|
|
|
}
|