use crate::{
CompressionType,
checksum::ChecksummedWriter,
encryption::EncryptionProvider,
table::{
Block, IndexBlock, block::Header as BlockHeader, index_block::KeyedBlockHandle,
writer::index::BlockIndexWriter,
},
};
use std::sync::Arc;
pub struct FullIndexWriter {
compression: CompressionType,
restart_interval: u8,
block_handles: Vec<KeyedBlockHandle>,
encryption: Option<Arc<dyn EncryptionProvider>>,
}
impl FullIndexWriter {
pub fn new() -> Self {
Self {
compression: CompressionType::None,
restart_interval: 1,
block_handles: Vec::new(),
encryption: None,
}
}
}
impl<W: std::io::Write + std::io::Seek> BlockIndexWriter<W> for FullIndexWriter {
fn use_encryption(
mut self: Box<Self>,
encryption: Option<Arc<dyn EncryptionProvider>>,
) -> Box<dyn BlockIndexWriter<W>> {
self.encryption = encryption;
self
}
fn use_partition_size(self: Box<Self>, _: u32) -> Box<dyn BlockIndexWriter<W>> {
self
}
fn use_restart_interval(mut self: Box<Self>, interval: u8) -> Box<dyn BlockIndexWriter<W>> {
self.restart_interval = interval;
self
}
fn use_compression(
mut self: Box<Self>,
compression: CompressionType,
) -> Box<dyn BlockIndexWriter<W>> {
self.compression = compression;
self
}
fn register_data_block(&mut self, block_handle: KeyedBlockHandle) -> crate::Result<()> {
log::trace!(
"Registering block at {:?} with size {} [end_key={:?}]",
block_handle.offset(),
block_handle.size(),
block_handle.end_key(),
);
self.block_handles.push(block_handle);
Ok(())
}
fn finish(
self: Box<Self>,
file_writer: &mut sfa::Writer<ChecksummedWriter<W>>,
) -> crate::Result<usize> {
file_writer.start("tli")?;
let mut bytes = vec![];
IndexBlock::encode_into_with_restart_interval(
&mut bytes,
&self.block_handles,
self.restart_interval,
)?;
let header = Block::write_into(
file_writer,
&bytes,
crate::table::block::BlockType::Index,
self.compression,
self.encryption.as_deref(),
#[cfg(zstd_any)]
None, )?;
#[expect(
clippy::cast_possible_truncation,
reason = "blocks never even approach u32 size"
)]
let bytes_written = BlockHeader::serialized_len() as u32 + header.data_length;
debug_assert!(bytes_written > 0, "Block index should never be empty");
log::trace!(
"Written top level index, with {} pointers ({bytes_written}B)",
self.block_handles.len(),
);
Ok(1)
}
}