tape: add chunk archive reader/writer

This commit is contained in:
Dietmar Maurer 2020-12-16 12:08:34 +01:00
parent c1c2c8f635
commit 0b7432ae09
3 changed files with 213 additions and 0 deletions

206
src/tape/chunk_archive.rs Normal file
View File

@ -0,0 +1,206 @@
use std::io::Read;
use anyhow::{bail, Error};
use endian_trait::Endian;
use proxmox::tools::{
Uuid,
io::ReadExt,
};
use crate::backup::DataBlob;
use crate::tape::{
TapeWrite,
file_formats::{
PROXMOX_TAPE_BLOCK_SIZE,
PROXMOX_BACKUP_CHUNK_ARCHIVE_MAGIC_1_0,
PROXMOX_BACKUP_CHUNK_ARCHIVE_ENTRY_MAGIC_1_0,
MediaContentHeader,
ChunkArchiveEntryHeader,
},
};
/// Writes chunk lists to tape.
///
/// A chunk archive consists of a 'MediaContentHeader' followed by a
/// list of chunks entries. Each chunk entry consists of a
/// 'ChunkArchiveEntryHeader' folowed by thew chunk data ('DataBlob').
///
/// | MediaContentHeader | ( ChunkArchiveEntryHeader | DataBlob )* |
pub struct ChunkArchiveWriter<'a> {
writer: Option<Box<dyn TapeWrite + 'a>>,
bytes_written: usize, // does not include bytes from current writer
close_on_leom: bool,
}
impl <'a> ChunkArchiveWriter<'a> {
pub const MAGIC: [u8; 8] = PROXMOX_BACKUP_CHUNK_ARCHIVE_MAGIC_1_0;
/// Creates a new instance
pub fn new(mut writer: Box<dyn TapeWrite + 'a>, close_on_leom: bool) -> Result<(Self,Uuid), Error> {
let header = MediaContentHeader::new(Self::MAGIC, 0);
writer.write_header(&header, &[])?;
let me = Self {
writer: Some(writer),
bytes_written: 0,
close_on_leom,
};
Ok((me, header.uuid.into()))
}
/// Returns the number of bytes written so far.
pub fn bytes_written(&self) -> usize {
match self.writer {
Some(ref writer) => writer.bytes_written(),
None => self.bytes_written, // finalize sets this
}
}
fn write_all(&mut self, data: &[u8]) -> Result<bool, std::io::Error> {
let result = match self.writer {
Some(ref mut writer) => {
let leom = writer.write_all(data)?;
Ok(leom)
}
None => proxmox::io_bail!(
"detected write after archive finished - internal error"),
};
result
}
/// Write chunk into archive.
///
/// This may return false when LEOM is detected (when close_on_leom is set).
/// In that case the archive only contains parts of the last chunk.
pub fn try_write_chunk(
&mut self,
digest: &[u8;32],
blob: &DataBlob,
) -> Result<bool, std::io::Error> {
if self.writer.is_none() {
return Ok(false);
}
let head = ChunkArchiveEntryHeader {
magic: PROXMOX_BACKUP_CHUNK_ARCHIVE_ENTRY_MAGIC_1_0,
digest: *digest,
size: blob.raw_size(),
};
let head = head.to_le();
let data = unsafe { std::slice::from_raw_parts(
&head as *const ChunkArchiveEntryHeader as *const u8,
std::mem::size_of::<ChunkArchiveEntryHeader>())
};
self.write_all(data)?;
let mut start = 0;
let blob_data = blob.raw_data();
loop {
if start >= blob_data.len() {
break;
}
let end = start + PROXMOX_TAPE_BLOCK_SIZE;
let mut chunk_is_complete = false;
let leom = if end > blob_data.len() {
chunk_is_complete = true;
self.write_all(&blob_data[start..])?
} else {
self.write_all(&blob_data[start..end])?
};
if leom {
println!("WRITE DATA LEOM at pos {}", self.bytes_written());
if self.close_on_leom {
let mut writer = self.writer.take().unwrap();
writer.finish(false)?;
self.bytes_written = writer.bytes_written();
return Ok(chunk_is_complete);
}
}
start = end;
}
Ok(true)
}
/// This must be called at the end to add padding and EOF
///
/// Returns true on LEOM or when we hit max archive size
pub fn finish(&mut self) -> Result<bool, std::io::Error> {
match self.writer.take() {
Some(mut writer) => {
self.bytes_written = writer.bytes_written();
writer.finish(false)
}
None => Ok(true),
}
}
}
/// Read chunk archives.
pub struct ChunkArchiveDecoder<R> {
reader: R,
}
impl <R: Read> ChunkArchiveDecoder<R> {
/// Creates a new instance
pub fn new(reader: R) -> Self {
Self { reader }
}
/// Allow access to the underyling reader
pub fn reader(&self) -> &R {
&self.reader
}
/// Returns the next chunk (if any).
pub fn next_chunk(&mut self) -> Result<Option<([u8;32], DataBlob)>, Error> {
let mut header = ChunkArchiveEntryHeader {
magic: [0u8; 8],
digest: [0u8; 32],
size: 0,
};
let data = unsafe {
std::slice::from_raw_parts_mut(
(&mut header as *mut ChunkArchiveEntryHeader) as *mut u8,
std::mem::size_of::<ChunkArchiveEntryHeader>())
};
match self.reader.read_exact_or_eof(data) {
Ok(true) => {},
Ok(false) => {
// last chunk is allowed to be incomplete - simply report EOD
return Ok(None);
}
Err(err) => return Err(err.into()),
};
if header.magic != PROXMOX_BACKUP_CHUNK_ARCHIVE_ENTRY_MAGIC_1_0 {
bail!("wrong magic number");
}
let raw_data = match self.reader.read_exact_allocated(header.size as usize) {
Ok(data) => data,
Err(err) if err.kind() == std::io::ErrorKind::UnexpectedEof => {
// last chunk is allowed to be incomplete - simply report EOD
return Ok(None);
}
Err(err) => return Err(err.into()),
};
let blob = DataBlob::from_raw(raw_data)?;
blob.verify_crc()?;
Ok(Some((header.digest, blob)))
}
}

View File

@ -145,9 +145,13 @@ impl MediaContentHeader {
#[derive(Endian)] #[derive(Endian)]
#[repr(C,packed)] #[repr(C,packed)]
/// Header for data blobs inside a chunk archive
pub struct ChunkArchiveEntryHeader { pub struct ChunkArchiveEntryHeader {
/// Magic number ('PROXMOX_BACKUP_CHUNK_ARCHIVE_ENTRY_MAGIC_1_0')
pub magic: [u8; 8], pub magic: [u8; 8],
/// Chunk digest
pub digest: [u8; 32], pub digest: [u8; 32],
/// Chunk size
pub size: u64, pub size: u64,
} }

View File

@ -39,6 +39,9 @@ pub use media_pool::*;
mod media_catalog; mod media_catalog;
pub use media_catalog::*; pub use media_catalog::*;
mod chunk_archive;
pub use chunk_archive::*;
/// Directory path where we store all tape status information /// Directory path where we store all tape status information
pub const TAPE_STATUS_DIR: &str = "/var/lib/proxmox-backup/tape"; pub const TAPE_STATUS_DIR: &str = "/var/lib/proxmox-backup/tape";