mirror of
https://github.com/meilisearch/meilisearch.git
synced 2025-01-24 12:03:14 +08:00
73ce67862d
Co-authored-by: ManyTheFish <many@meilisearch.com>
370 lines
11 KiB
Rust
370 lines
11 KiB
Rust
use std::fs::File;
|
|
use std::marker::PhantomData;
|
|
|
|
use crossbeam_channel::{IntoIter, Receiver, SendError, Sender};
|
|
use grenad::Merger;
|
|
use heed::types::Bytes;
|
|
use memmap2::Mmap;
|
|
|
|
use super::StdResult;
|
|
use crate::index::main_key::{DOCUMENTS_IDS_KEY, WORDS_FST_KEY};
|
|
use crate::update::new::KvReaderFieldId;
|
|
use crate::update::MergeDeladdCboRoaringBitmaps;
|
|
use crate::{DocumentId, Index};
|
|
|
|
/// The capacity of the channel is currently in number of messages.
|
|
pub fn merger_writer_channel(cap: usize) -> (MergerSender, WriterReceiver) {
|
|
let (sender, receiver) = crossbeam_channel::bounded(cap);
|
|
(MergerSender(sender), WriterReceiver(receiver))
|
|
}
|
|
|
|
/// The capacity of the channel is currently in number of messages.
|
|
pub fn extractors_merger_channels(cap: usize) -> (ExtractorSender, MergerReceiver) {
|
|
let (sender, receiver) = crossbeam_channel::bounded(cap);
|
|
(ExtractorSender(sender), MergerReceiver(receiver))
|
|
}
|
|
|
|
pub enum KeyValueEntry {
|
|
SmallInMemory { key_length: usize, data: Box<[u8]> },
|
|
LargeOnDisk { key: Box<[u8]>, value: Mmap },
|
|
}
|
|
|
|
impl KeyValueEntry {
|
|
pub fn from_small_key_value(key: &[u8], value: &[u8]) -> Self {
|
|
let mut data = Vec::with_capacity(key.len() + value.len());
|
|
data.extend_from_slice(key);
|
|
data.extend_from_slice(value);
|
|
KeyValueEntry::SmallInMemory { key_length: key.len(), data: data.into_boxed_slice() }
|
|
}
|
|
|
|
pub fn from_large_key_value(key: &[u8], value: Mmap) -> Self {
|
|
KeyValueEntry::LargeOnDisk { key: key.to_vec().into_boxed_slice(), value }
|
|
}
|
|
|
|
pub fn key(&self) -> &[u8] {
|
|
match self {
|
|
KeyValueEntry::SmallInMemory { key_length, data } => &data.as_ref()[..*key_length],
|
|
KeyValueEntry::LargeOnDisk { key, value: _ } => key.as_ref(),
|
|
}
|
|
}
|
|
|
|
pub fn value(&self) -> &[u8] {
|
|
match self {
|
|
KeyValueEntry::SmallInMemory { key_length, data } => &data.as_ref()[*key_length..],
|
|
KeyValueEntry::LargeOnDisk { key: _, value } => value.as_ref(),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct KeyEntry {
|
|
data: Box<[u8]>,
|
|
}
|
|
|
|
impl KeyEntry {
|
|
pub fn from_key(key: &[u8]) -> Self {
|
|
KeyEntry { data: key.to_vec().into_boxed_slice() }
|
|
}
|
|
|
|
pub fn entry(&self) -> &[u8] {
|
|
self.data.as_ref()
|
|
}
|
|
}
|
|
|
|
pub enum EntryOperation {
|
|
Delete(KeyEntry),
|
|
Write(KeyValueEntry),
|
|
}
|
|
|
|
pub struct DocumentEntry {
|
|
docid: DocumentId,
|
|
content: Box<[u8]>,
|
|
}
|
|
|
|
impl DocumentEntry {
|
|
pub fn new_uncompressed(docid: DocumentId, content: Box<KvReaderFieldId>) -> Self {
|
|
DocumentEntry { docid, content: content.into() }
|
|
}
|
|
|
|
pub fn new_compressed(docid: DocumentId, content: Box<[u8]>) -> Self {
|
|
DocumentEntry { docid, content }
|
|
}
|
|
|
|
pub fn key(&self) -> [u8; 4] {
|
|
self.docid.to_be_bytes()
|
|
}
|
|
|
|
pub fn content(&self) -> &[u8] {
|
|
&self.content
|
|
}
|
|
}
|
|
|
|
pub struct DocumentDeletionEntry(DocumentId);
|
|
|
|
impl DocumentDeletionEntry {
|
|
pub fn key(&self) -> [u8; 4] {
|
|
self.0.to_be_bytes()
|
|
}
|
|
}
|
|
|
|
pub struct WriterOperation {
|
|
database: Database,
|
|
entry: EntryOperation,
|
|
}
|
|
|
|
pub enum Database {
|
|
Documents,
|
|
ExactWordDocids,
|
|
FidWordCountDocids,
|
|
Main,
|
|
WordDocids,
|
|
WordFidDocids,
|
|
WordPairProximityDocids,
|
|
WordPositionDocids,
|
|
}
|
|
|
|
impl WriterOperation {
|
|
pub fn database(&self, index: &Index) -> heed::Database<Bytes, Bytes> {
|
|
match self.database {
|
|
Database::Documents => index.documents.remap_types(),
|
|
Database::ExactWordDocids => index.exact_word_docids.remap_types(),
|
|
Database::Main => index.main.remap_types(),
|
|
Database::WordDocids => index.word_docids.remap_types(),
|
|
Database::WordFidDocids => index.word_fid_docids.remap_types(),
|
|
Database::WordPositionDocids => index.word_position_docids.remap_types(),
|
|
Database::FidWordCountDocids => index.field_id_word_count_docids.remap_types(),
|
|
Database::WordPairProximityDocids => index.word_pair_proximity_docids.remap_types(),
|
|
}
|
|
}
|
|
|
|
pub fn entry(self) -> EntryOperation {
|
|
self.entry
|
|
}
|
|
}
|
|
|
|
pub struct WriterReceiver(Receiver<WriterOperation>);
|
|
|
|
impl IntoIterator for WriterReceiver {
|
|
type Item = WriterOperation;
|
|
type IntoIter = IntoIter<Self::Item>;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
self.0.into_iter()
|
|
}
|
|
}
|
|
|
|
pub struct MergerSender(Sender<WriterOperation>);
|
|
|
|
impl MergerSender {
|
|
pub fn main(&self) -> MainSender<'_> {
|
|
MainSender(&self.0)
|
|
}
|
|
|
|
pub fn docids<D: DatabaseType>(&self) -> DocidsSender<'_, D> {
|
|
DocidsSender { sender: &self.0, _marker: PhantomData }
|
|
}
|
|
|
|
pub fn documents(&self) -> DocumentsSender<'_> {
|
|
DocumentsSender(&self.0)
|
|
}
|
|
|
|
pub fn send_documents_ids(&self, bitmap: &[u8]) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Write(KeyValueEntry::from_small_key_value(
|
|
DOCUMENTS_IDS_KEY.as_bytes(),
|
|
bitmap,
|
|
));
|
|
match self.0.send(WriterOperation { database: Database::Main, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct MainSender<'a>(&'a Sender<WriterOperation>);
|
|
|
|
impl MainSender<'_> {
|
|
pub fn write_words_fst(&self, value: Mmap) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Write(KeyValueEntry::from_large_key_value(
|
|
WORDS_FST_KEY.as_bytes(),
|
|
value,
|
|
));
|
|
match self.0.send(WriterOperation { database: Database::Main, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
|
|
pub fn delete(&self, key: &[u8]) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Delete(KeyEntry::from_key(key));
|
|
match self.0.send(WriterOperation { database: Database::Main, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub enum ExactWordDocids {}
|
|
pub enum FidWordCountDocids {}
|
|
pub enum WordDocids {}
|
|
pub enum WordFidDocids {}
|
|
pub enum WordPairProximityDocids {}
|
|
pub enum WordPositionDocids {}
|
|
|
|
pub trait DatabaseType {
|
|
const DATABASE: Database;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation;
|
|
}
|
|
|
|
impl DatabaseType for ExactWordDocids {
|
|
const DATABASE: Database = Database::ExactWordDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::ExactWordDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
impl DatabaseType for FidWordCountDocids {
|
|
const DATABASE: Database = Database::FidWordCountDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::FidWordCountDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
impl DatabaseType for WordDocids {
|
|
const DATABASE: Database = Database::WordDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::WordDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
impl DatabaseType for WordFidDocids {
|
|
const DATABASE: Database = Database::WordFidDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::WordFidDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
impl DatabaseType for WordPairProximityDocids {
|
|
const DATABASE: Database = Database::WordPairProximityDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::WordPairProximityDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
impl DatabaseType for WordPositionDocids {
|
|
const DATABASE: Database = Database::WordPositionDocids;
|
|
|
|
fn new_merger_operation(merger: Merger<File, MergeDeladdCboRoaringBitmaps>) -> MergerOperation {
|
|
MergerOperation::WordPositionDocidsMerger(merger)
|
|
}
|
|
}
|
|
|
|
pub struct DocidsSender<'a, D> {
|
|
sender: &'a Sender<WriterOperation>,
|
|
_marker: PhantomData<D>,
|
|
}
|
|
|
|
impl<D: DatabaseType> DocidsSender<'_, D> {
|
|
pub fn write(&self, key: &[u8], value: &[u8]) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Write(KeyValueEntry::from_small_key_value(key, value));
|
|
match self.sender.send(WriterOperation { database: D::DATABASE, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
|
|
pub fn delete(&self, key: &[u8]) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Delete(KeyEntry::from_key(key));
|
|
match self.sender.send(WriterOperation { database: D::DATABASE, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct DocumentsSender<'a>(&'a Sender<WriterOperation>);
|
|
|
|
impl DocumentsSender<'_> {
|
|
/// TODO do that efficiently
|
|
pub fn uncompressed(
|
|
&self,
|
|
docid: DocumentId,
|
|
document: &KvReaderFieldId,
|
|
) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Write(KeyValueEntry::from_small_key_value(
|
|
&docid.to_be_bytes(),
|
|
document.as_bytes(),
|
|
));
|
|
match self.0.send(WriterOperation { database: Database::Documents, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
|
|
pub fn delete(&self, docid: DocumentId) -> StdResult<(), SendError<()>> {
|
|
let entry = EntryOperation::Delete(KeyEntry::from_key(&docid.to_be_bytes()));
|
|
match self.0.send(WriterOperation { database: Database::Documents, entry }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub enum MergerOperation {
|
|
ExactWordDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
FidWordCountDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
WordDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
WordFidDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
WordPairProximityDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
WordPositionDocidsMerger(Merger<File, MergeDeladdCboRoaringBitmaps>),
|
|
DeleteDocument { docid: DocumentId },
|
|
InsertDocument { docid: DocumentId, document: Box<KvReaderFieldId> },
|
|
}
|
|
|
|
pub struct MergerReceiver(Receiver<MergerOperation>);
|
|
|
|
impl IntoIterator for MergerReceiver {
|
|
type Item = MergerOperation;
|
|
type IntoIter = IntoIter<Self::Item>;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
self.0.into_iter()
|
|
}
|
|
}
|
|
|
|
pub struct ExtractorSender(Sender<MergerOperation>);
|
|
|
|
impl ExtractorSender {
|
|
pub fn document_insert(
|
|
&self,
|
|
docid: DocumentId,
|
|
document: Box<KvReaderFieldId>,
|
|
) -> StdResult<(), SendError<()>> {
|
|
match self.0.send(MergerOperation::InsertDocument { docid, document }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
|
|
pub fn document_delete(&self, docid: DocumentId) -> StdResult<(), SendError<()>> {
|
|
match self.0.send(MergerOperation::DeleteDocument { docid }) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
|
|
pub fn send_searchable<D: DatabaseType>(
|
|
&self,
|
|
merger: Merger<File, MergeDeladdCboRoaringBitmaps>,
|
|
) -> StdResult<(), SendError<()>> {
|
|
match self.0.send(D::new_merger_operation(merger)) {
|
|
Ok(()) => Ok(()),
|
|
Err(SendError(_)) => Err(SendError(())),
|
|
}
|
|
}
|
|
}
|