2022-06-14 22:03:48 +08:00
|
|
|
use std::convert::TryInto;
|
2022-06-20 19:48:02 +08:00
|
|
|
use std::{error, fmt, io, str};
|
2021-08-31 17:44:15 +08:00
|
|
|
|
|
|
|
use obkv::KvReader;
|
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
use super::{DocumentsBatchIndex, Error, DOCUMENTS_BATCH_INDEX_KEY};
|
2021-08-31 17:44:15 +08:00
|
|
|
use crate::FieldId;
|
|
|
|
|
|
|
|
/// The `DocumentsBatchReader` provides a way to iterate over documents that have been created with
|
|
|
|
/// a `DocumentsBatchWriter`.
|
|
|
|
///
|
|
|
|
/// The documents are returned in the form of `obkv::Reader` where each field is identified with a
|
|
|
|
/// `FieldId`. The mapping between the field ids and the field names is done thanks to the index.
|
2022-06-14 22:03:48 +08:00
|
|
|
pub struct DocumentsBatchReader<R> {
|
|
|
|
cursor: grenad::ReaderCursor<R>,
|
|
|
|
fields_index: DocumentsBatchIndex,
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
impl<R: io::Read + io::Seek> DocumentsBatchReader<R> {
|
2021-08-31 17:44:15 +08:00
|
|
|
/// Construct a `DocumentsReader` from a reader.
|
|
|
|
///
|
2022-06-14 22:03:48 +08:00
|
|
|
/// It first retrieves the index, then moves to the first document. Use the `into_cursor`
|
|
|
|
/// method to iterator over the documents, from the first to the last.
|
|
|
|
pub fn from_reader(reader: R) -> Result<Self, Error> {
|
|
|
|
let reader = grenad::Reader::new(reader)?;
|
|
|
|
let mut cursor = reader.into_cursor()?;
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
let fields_index = match cursor.move_on_key_equal_to(DOCUMENTS_BATCH_INDEX_KEY)? {
|
|
|
|
Some((_, value)) => serde_json::from_slice(value).map_err(Error::Serialize)?,
|
|
|
|
None => return Err(Error::InvalidDocumentFormat),
|
|
|
|
};
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
Ok(DocumentsBatchReader { cursor, fields_index })
|
|
|
|
}
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
pub fn documents_count(&self) -> u32 {
|
|
|
|
self.cursor.len().saturating_sub(1).try_into().expect("Invalid number of documents")
|
|
|
|
}
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.cursor.len().saturating_sub(1) == 0
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
pub fn documents_batch_index(&self) -> &DocumentsBatchIndex {
|
|
|
|
&self.fields_index
|
|
|
|
}
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
/// This method returns a forward cursor over the documents.
|
|
|
|
pub fn into_cursor(self) -> DocumentsBatchCursor<R> {
|
|
|
|
let DocumentsBatchReader { cursor, fields_index } = self;
|
|
|
|
let mut cursor = DocumentsBatchCursor { cursor, fields_index };
|
|
|
|
cursor.reset();
|
|
|
|
cursor
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
2022-06-14 22:03:48 +08:00
|
|
|
}
|
2021-08-31 17:44:15 +08:00
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
/// A forward cursor over the documents in a `DocumentsBatchReader`.
|
|
|
|
pub struct DocumentsBatchCursor<R> {
|
|
|
|
cursor: grenad::ReaderCursor<R>,
|
|
|
|
fields_index: DocumentsBatchIndex,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<R> DocumentsBatchCursor<R> {
|
|
|
|
pub fn into_reader(self) -> DocumentsBatchReader<R> {
|
|
|
|
let DocumentsBatchCursor { cursor, fields_index, .. } = self;
|
|
|
|
DocumentsBatchReader { cursor, fields_index }
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
pub fn documents_batch_index(&self) -> &DocumentsBatchIndex {
|
|
|
|
&self.fields_index
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
|
|
|
|
2022-06-14 22:03:48 +08:00
|
|
|
/// Resets the cursor to be able to read from the start again.
|
|
|
|
pub fn reset(&mut self) {
|
|
|
|
self.cursor.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<R: io::Read + io::Seek> DocumentsBatchCursor<R> {
|
|
|
|
/// Returns the next document, starting from the first one. Subsequent calls to
|
|
|
|
/// `next_document` advance the document reader until all the documents have been read.
|
2022-06-16 18:03:43 +08:00
|
|
|
pub fn next_document(
|
|
|
|
&mut self,
|
|
|
|
) -> Result<Option<KvReader<FieldId>>, DocumentsBatchCursorError> {
|
2022-06-14 22:03:48 +08:00
|
|
|
match self.cursor.move_on_next()? {
|
|
|
|
Some((key, value)) if key != DOCUMENTS_BATCH_INDEX_KEY => {
|
|
|
|
Ok(Some(KvReader::new(value)))
|
|
|
|
}
|
|
|
|
_otherwise => Ok(None),
|
|
|
|
}
|
2021-08-31 17:44:15 +08:00
|
|
|
}
|
|
|
|
}
|
2022-06-16 18:03:43 +08:00
|
|
|
|
|
|
|
/// The possible error thrown by the `DocumentsBatchCursor` when iterating on the documents.
|
|
|
|
#[derive(Debug)]
|
2022-06-20 19:48:02 +08:00
|
|
|
pub enum DocumentsBatchCursorError {
|
|
|
|
Grenad(grenad::Error),
|
|
|
|
Utf8(str::Utf8Error),
|
2022-06-16 18:03:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl From<grenad::Error> for DocumentsBatchCursorError {
|
|
|
|
fn from(error: grenad::Error) -> DocumentsBatchCursorError {
|
2022-06-20 19:48:02 +08:00
|
|
|
DocumentsBatchCursorError::Grenad(error)
|
2022-06-16 18:03:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-20 19:48:02 +08:00
|
|
|
impl From<str::Utf8Error> for DocumentsBatchCursorError {
|
|
|
|
fn from(error: str::Utf8Error) -> DocumentsBatchCursorError {
|
|
|
|
DocumentsBatchCursorError::Utf8(error)
|
2022-06-16 18:03:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl error::Error for DocumentsBatchCursorError {}
|
|
|
|
|
|
|
|
impl fmt::Display for DocumentsBatchCursorError {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2022-06-20 19:48:02 +08:00
|
|
|
match self {
|
|
|
|
DocumentsBatchCursorError::Grenad(e) => e.fmt(f),
|
|
|
|
DocumentsBatchCursorError::Utf8(e) => e.fmt(f),
|
|
|
|
}
|
2022-06-16 18:03:43 +08:00
|
|
|
}
|
|
|
|
}
|