use tokio::sync::OnceCell;
use crate::io::{FileRead, InputFile};
use crate::puffin::blob::Blob;
use crate::puffin::metadata::{BlobMetadata, FileMetadata};
use crate::Result;
pub(crate) struct PuffinReader {
input_file: InputFile,
file_metadata: OnceCell<FileMetadata>,
}
impl PuffinReader {
pub(crate) fn new(input_file: InputFile) -> Self {
Self {
input_file,
file_metadata: OnceCell::new(),
}
}
pub(crate) async fn file_metadata(&self) -> Result<&FileMetadata> {
self.file_metadata
.get_or_try_init(|| FileMetadata::read(&self.input_file))
.await
}
pub(crate) async fn blob(&self, blob_metadata: &BlobMetadata) -> Result<Blob> {
let file_read = self.input_file.reader().await?;
let start = blob_metadata.offset;
let end = start + blob_metadata.length;
let bytes = file_read.read(start..end).await?.to_vec();
let data = blob_metadata.compression_codec.decompress(bytes)?;
Ok(Blob {
r#type: blob_metadata.r#type.clone(),
fields: blob_metadata.fields.clone(),
snapshot_id: blob_metadata.snapshot_id,
sequence_number: blob_metadata.sequence_number,
data,
properties: blob_metadata.properties.clone(),
})
}
}
#[cfg(test)]
mod tests {
use crate::puffin::reader::PuffinReader;
use crate::puffin::test_utils::{
blob_0, blob_1, java_uncompressed_metric_input_file,
java_zstd_compressed_metric_input_file, uncompressed_metric_file_metadata,
zstd_compressed_metric_file_metadata,
};
#[tokio::test]
async fn test_puffin_reader_uncompressed_metric_data() {
let input_file = java_uncompressed_metric_input_file();
let puffin_reader = PuffinReader::new(input_file);
let file_metadata = puffin_reader.file_metadata().await.unwrap().clone();
assert_eq!(file_metadata, uncompressed_metric_file_metadata());
assert_eq!(
puffin_reader
.blob(file_metadata.blobs.first().unwrap())
.await
.unwrap(),
blob_0()
);
assert_eq!(
puffin_reader
.blob(file_metadata.blobs.get(1).unwrap())
.await
.unwrap(),
blob_1(),
)
}
#[tokio::test]
async fn test_puffin_reader_zstd_compressed_metric_data() {
let input_file = java_zstd_compressed_metric_input_file();
let puffin_reader = PuffinReader::new(input_file);
let file_metadata = puffin_reader.file_metadata().await.unwrap().clone();
assert_eq!(file_metadata, zstd_compressed_metric_file_metadata());
assert_eq!(
puffin_reader
.blob(file_metadata.blobs.first().unwrap())
.await
.unwrap(),
blob_0()
);
assert_eq!(
puffin_reader
.blob(file_metadata.blobs.get(1).unwrap())
.await
.unwrap(),
blob_1(),
)
}
}