MeiliSearch/src/data/doc_indexes.rs

203 lines
5.7 KiB
Rust
Raw Normal View History

2018-10-28 14:24:04 +01:00
use std::slice::from_raw_parts;
use std::io::{self, Write};
2018-12-01 18:37:21 +01:00
use std::mem::size_of;
use std::ops::Index;
2018-10-28 14:24:04 +01:00
use std::path::Path;
use std::sync::Arc;
2018-10-28 14:24:04 +01:00
use fst::raw::MmapReadOnly;
use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt};
2018-10-28 14:24:04 +01:00
use crate::DocIndex;
use crate::data::Data;
2018-10-28 14:24:04 +01:00
2018-12-01 11:35:16 +01:00
#[derive(Debug)]
2018-10-28 14:24:04 +01:00
#[repr(C)]
struct Range {
start: u64,
end: u64,
}
#[derive(Clone, Default)]
2018-10-28 14:24:04 +01:00
pub struct DocIndexes {
ranges: Data,
indexes: Data,
2018-10-28 14:24:04 +01:00
}
impl DocIndexes {
pub unsafe fn from_path<P: AsRef<Path>>(path: P) -> io::Result<Self> {
let mmap = MmapReadOnly::open_path(path)?;
DocIndexes::from_data(Data::Mmap(mmap))
2018-10-28 14:24:04 +01:00
}
pub fn from_bytes(vec: Vec<u8>) -> io::Result<Self> {
let len = vec.len();
DocIndexes::from_shared_bytes(Arc::new(vec), 0, len)
}
pub fn from_shared_bytes(bytes: Arc<Vec<u8>>, offset: usize, len: usize) -> io::Result<Self> {
let data = Data::Shared { bytes, offset, len };
DocIndexes::from_data(data)
}
2018-10-28 14:24:04 +01:00
fn from_data(data: Data) -> io::Result<Self> {
2018-12-01 18:37:21 +01:00
let ranges_len_offset = data.len() - size_of::<u64>();
let ranges_len = (&data[ranges_len_offset..]).read_u64::<LittleEndian>()?;
2018-12-01 11:35:16 +01:00
let ranges_len = ranges_len as usize;
2018-10-28 14:24:04 +01:00
let ranges_offset = ranges_len_offset - ranges_len;
let ranges = data.range(ranges_offset, ranges_len);
2018-10-28 14:24:04 +01:00
let indexes = data.range(0, ranges_offset);
2018-10-28 14:24:04 +01:00
Ok(DocIndexes { ranges, indexes })
}
2018-12-01 18:37:21 +01:00
pub fn to_vec(&self) -> Vec<u8> {
let capacity = self.indexes.len() + self.ranges.len() + size_of::<u64>();
let mut bytes = Vec::with_capacity(capacity);
bytes.extend_from_slice(&self.indexes);
bytes.extend_from_slice(&self.ranges);
bytes.write_u64::<LittleEndian>(self.ranges.len() as u64).unwrap();
bytes
}
pub fn get(&self, index: usize) -> Option<&[DocIndex]> {
2018-10-28 14:24:04 +01:00
self.ranges().get(index as usize).map(|Range { start, end }| {
let start = *start as usize;
let end = *end as usize;
&self.indexes()[start..end]
})
}
fn ranges(&self) -> &[Range] {
let slice = &self.ranges;
let ptr = slice.as_ptr() as *const Range;
2018-12-01 18:37:21 +01:00
let len = slice.len() / size_of::<Range>();
2018-10-28 14:24:04 +01:00
unsafe { from_raw_parts(ptr, len) }
}
fn indexes(&self) -> &[DocIndex] {
let slice = &self.indexes;
let ptr = slice.as_ptr() as *const DocIndex;
2018-12-01 18:37:21 +01:00
let len = slice.len() / size_of::<DocIndex>();
2018-10-28 14:24:04 +01:00
unsafe { from_raw_parts(ptr, len) }
}
}
2018-12-01 18:37:21 +01:00
impl Index<usize> for DocIndexes {
type Output = [DocIndex];
fn index(&self, index: usize) -> &Self::Output {
match self.get(index) {
Some(indexes) => indexes,
None => panic!("index {} out of range for a maximum of {} ranges", index, self.ranges().len()),
}
2018-11-23 18:00:24 +01:00
}
}
2018-12-01 11:35:16 +01:00
pub struct DocIndexesBuilder<W> {
ranges: Vec<Range>,
wtr: W,
}
2018-12-01 11:35:16 +01:00
impl DocIndexesBuilder<Vec<u8>> {
pub fn memory() -> Self {
2018-12-01 11:35:16 +01:00
DocIndexesBuilder::new(Vec::new())
}
}
2018-12-01 11:35:16 +01:00
impl<W: Write> DocIndexesBuilder<W> {
pub fn new(wtr: W) -> Self {
2018-12-01 11:35:16 +01:00
DocIndexesBuilder {
ranges: Vec::new(),
wtr: wtr,
}
}
pub fn insert(&mut self, indexes: &[DocIndex]) -> io::Result<()> {
let len = indexes.len() as u64;
2018-12-01 11:35:16 +01:00
let start = self.ranges.last().map(|r| r.end).unwrap_or(0);
let range = Range { start, end: start + len };
self.ranges.push(range);
// write the values
let indexes = unsafe { into_u8_slice(indexes) };
self.wtr.write_all(indexes)
}
pub fn finish(self) -> io::Result<()> {
self.into_inner().map(drop)
}
pub fn into_inner(mut self) -> io::Result<W> {
// write the ranges
let ranges = unsafe { into_u8_slice(self.ranges.as_slice()) };
self.wtr.write_all(ranges)?;
// write the length of the ranges
let len = ranges.len() as u64;
self.wtr.write_u64::<LittleEndian>(len)?;
Ok(self.wtr)
}
}
2018-12-01 11:35:16 +01:00
unsafe fn into_u8_slice<T>(slice: &[T]) -> &[u8] {
let ptr = slice.as_ptr() as *const u8;
2018-12-01 18:37:21 +01:00
let len = slice.len() * size_of::<T>();
2018-12-01 11:35:16 +01:00
from_raw_parts(ptr, len)
2018-10-28 14:24:04 +01:00
}
2018-12-01 11:35:16 +01:00
#[cfg(test)]
mod tests {
use super::*;
use std::error::Error;
2018-10-28 14:24:04 +01:00
2018-12-01 11:35:16 +01:00
#[test]
2018-12-01 18:37:21 +01:00
fn builder_serialize_deserialize() -> Result<(), Box<Error>> {
2018-12-01 11:35:16 +01:00
let a = DocIndex { document_id: 0, attribute: 3, attribute_index: 11 };
let b = DocIndex { document_id: 1, attribute: 4, attribute_index: 21 };
let c = DocIndex { document_id: 2, attribute: 8, attribute_index: 2 };
2018-12-01 11:35:16 +01:00
let mut builder = DocIndexesBuilder::memory();
2018-12-01 11:35:16 +01:00
builder.insert(&[a])?;
builder.insert(&[a, b, c])?;
builder.insert(&[a, c])?;
2018-10-28 14:24:04 +01:00
2018-12-01 11:35:16 +01:00
let bytes = builder.into_inner()?;
let docs = DocIndexes::from_bytes(bytes)?;
2018-10-28 14:24:04 +01:00
2018-12-01 18:37:21 +01:00
assert_eq!(docs.get(0), Some(&[a][..]));
assert_eq!(docs.get(1), Some(&[a, b, c][..]));
assert_eq!(docs.get(2), Some(&[a, c][..]));
assert_eq!(docs.get(3), None);
Ok(())
}
#[test]
fn serialize_deserialize() -> Result<(), Box<Error>> {
let a = DocIndex { document_id: 0, attribute: 3, attribute_index: 11 };
let b = DocIndex { document_id: 1, attribute: 4, attribute_index: 21 };
let c = DocIndex { document_id: 2, attribute: 8, attribute_index: 2 };
let mut builder = DocIndexesBuilder::memory();
builder.insert(&[a])?;
builder.insert(&[a, b, c])?;
builder.insert(&[a, c])?;
let builder_bytes = builder.into_inner()?;
let docs = DocIndexes::from_bytes(builder_bytes.clone())?;
let bytes = docs.to_vec();
assert_eq!(builder_bytes, bytes);
2018-10-28 14:24:04 +01:00
2018-12-01 11:35:16 +01:00
Ok(())
2018-10-28 14:24:04 +01:00
}
}