2020-12-16 11:08:34 +00:00
|
|
|
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,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2020-12-16 11:43:51 +00:00
|
|
|
/// Writes chunk archives to tape.
|
2020-12-16 11:08:34 +00:00
|
|
|
///
|
2020-12-16 11:43:51 +00:00
|
|
|
/// A chunk archive consists of a `MediaContentHeader` followed by a
|
2020-12-16 11:08:34 +00:00
|
|
|
/// list of chunks entries. Each chunk entry consists of a
|
2020-12-16 11:43:51 +00:00
|
|
|
/// `ChunkArchiveEntryHeader` folowed by the chunk data (`DataBlob`).
|
2020-12-16 11:08:34 +00:00
|
|
|
///
|
2020-12-16 11:43:51 +00:00
|
|
|
/// `| MediaContentHeader | ( ChunkArchiveEntryHeader | DataBlob )* |`
|
2020-12-16 11:08:34 +00:00
|
|
|
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> {
|
2021-01-19 13:24:31 +00:00
|
|
|
match self.writer {
|
|
|
|
Some(ref mut writer) => writer.write_all(data),
|
2020-12-16 11:08:34 +00:00
|
|
|
None => proxmox::io_bail!(
|
|
|
|
"detected write after archive finished - internal error"),
|
2021-01-19 13:24:31 +00:00
|
|
|
}
|
2020-12-16 11:08:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Write chunk into archive.
|
|
|
|
///
|
2020-12-16 11:43:51 +00:00
|
|
|
/// This may return false when `LEOM` is detected (when close_on_leom is set).
|
2020-12-16 11:08:34 +00:00
|
|
|
/// 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)
|
|
|
|
}
|
|
|
|
|
2020-12-16 11:43:51 +00:00
|
|
|
/// This must be called at the end to add padding and `EOF`
|
2020-12-16 11:08:34 +00:00
|
|
|
///
|
2020-12-16 11:43:51 +00:00
|
|
|
/// Returns true on `LEOM` or when we hit max archive size
|
2020-12-16 11:08:34 +00:00
|
|
|
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)))
|
|
|
|
}
|
|
|
|
}
|