mirror of
https://github.com/meilisearch/MeiliSearch
synced 2024-11-12 07:58:54 +01:00
9026867d17
+ cargo fmt, oops, sorry for the bad history :(
137 lines
3.6 KiB
Rust
137 lines
3.6 KiB
Rust
mod field_doc_id_facet_f64_codec;
|
|
mod field_doc_id_facet_string_codec;
|
|
mod ordered_f64_codec;
|
|
mod str_ref;
|
|
|
|
use std::borrow::Cow;
|
|
use std::convert::TryFrom;
|
|
use std::marker::PhantomData;
|
|
|
|
use heed::types::OwnedType;
|
|
use heed::{BytesDecode, BytesEncode};
|
|
use roaring::RoaringBitmap;
|
|
|
|
pub use self::field_doc_id_facet_f64_codec::FieldDocIdFacetF64Codec;
|
|
pub use self::field_doc_id_facet_string_codec::FieldDocIdFacetStringCodec;
|
|
pub use self::ordered_f64_codec::OrderedF64Codec;
|
|
pub use self::str_ref::StrRefCodec;
|
|
use crate::{CboRoaringBitmapCodec, BEU16};
|
|
|
|
pub type FieldIdCodec = OwnedType<BEU16>;
|
|
|
|
/// Tries to split a slice in half at the given middle point,
|
|
/// `None` if the slice is too short.
|
|
pub fn try_split_at(slice: &[u8], mid: usize) -> Option<(&[u8], &[u8])> {
|
|
if slice.len() >= mid {
|
|
Some(slice.split_at(mid))
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
|
|
pub struct FacetGroupKey<T> {
|
|
pub field_id: u16,
|
|
pub level: u8,
|
|
pub left_bound: T,
|
|
}
|
|
impl<'a> FacetGroupKey<&'a [u8]> {
|
|
pub fn into_owned(self) -> FacetGroupKey<Vec<u8>> {
|
|
FacetGroupKey {
|
|
field_id: self.field_id,
|
|
level: self.level,
|
|
left_bound: self.left_bound.to_vec(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> FacetGroupKey<Vec<u8>> {
|
|
pub fn as_ref(&self) -> FacetGroupKey<&[u8]> {
|
|
FacetGroupKey {
|
|
field_id: self.field_id,
|
|
level: self.level,
|
|
left_bound: self.left_bound.as_slice(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub struct FacetGroupValue {
|
|
pub size: u8,
|
|
pub bitmap: RoaringBitmap,
|
|
}
|
|
|
|
pub struct FacetGroupKeyCodec<T> {
|
|
_phantom: PhantomData<T>,
|
|
}
|
|
|
|
impl<'a, T> heed::BytesEncode<'a> for FacetGroupKeyCodec<T>
|
|
where
|
|
T: BytesEncode<'a>,
|
|
T::EItem: Sized,
|
|
{
|
|
type EItem = FacetGroupKey<T::EItem>;
|
|
|
|
fn bytes_encode(value: &'a Self::EItem) -> Option<Cow<'a, [u8]>> {
|
|
let mut v = vec![];
|
|
v.extend_from_slice(&value.field_id.to_be_bytes());
|
|
v.extend_from_slice(&[value.level]);
|
|
|
|
let bound = T::bytes_encode(&value.left_bound)?;
|
|
v.extend_from_slice(&bound);
|
|
|
|
Some(Cow::Owned(v))
|
|
}
|
|
}
|
|
impl<'a, T> heed::BytesDecode<'a> for FacetGroupKeyCodec<T>
|
|
where
|
|
T: BytesDecode<'a>,
|
|
{
|
|
type DItem = FacetGroupKey<T::DItem>;
|
|
|
|
fn bytes_decode(bytes: &'a [u8]) -> Option<Self::DItem> {
|
|
let fid = u16::from_be_bytes(<[u8; 2]>::try_from(&bytes[0..=1]).ok()?);
|
|
let level = bytes[2];
|
|
let bound = T::bytes_decode(&bytes[3..])?;
|
|
Some(FacetGroupKey { field_id: fid, level, left_bound: bound })
|
|
}
|
|
}
|
|
|
|
pub struct FacetGroupValueCodec;
|
|
impl<'a> heed::BytesEncode<'a> for FacetGroupValueCodec {
|
|
type EItem = FacetGroupValue;
|
|
|
|
fn bytes_encode(value: &'a Self::EItem) -> Option<Cow<'a, [u8]>> {
|
|
let mut v = vec![];
|
|
v.push(value.size);
|
|
CboRoaringBitmapCodec::serialize_into(&value.bitmap, &mut v);
|
|
Some(Cow::Owned(v))
|
|
}
|
|
}
|
|
impl<'a> heed::BytesDecode<'a> for FacetGroupValueCodec {
|
|
type DItem = FacetGroupValue;
|
|
fn bytes_decode(bytes: &'a [u8]) -> Option<Self::DItem> {
|
|
let size = bytes[0];
|
|
let bitmap = CboRoaringBitmapCodec::deserialize_from(&bytes[1..]).ok()?;
|
|
Some(FacetGroupValue { size, bitmap })
|
|
}
|
|
}
|
|
|
|
pub struct ByteSliceRef;
|
|
|
|
impl<'a> BytesEncode<'a> for ByteSliceRef {
|
|
type EItem = &'a [u8];
|
|
|
|
fn bytes_encode(item: &'a Self::EItem) -> Option<Cow<'a, [u8]>> {
|
|
Some(Cow::Borrowed(item))
|
|
}
|
|
}
|
|
|
|
impl<'a> BytesDecode<'a> for ByteSliceRef {
|
|
type DItem = &'a [u8];
|
|
|
|
fn bytes_decode(bytes: &'a [u8]) -> Option<Self::DItem> {
|
|
Some(bytes)
|
|
}
|
|
}
|