src/backup/data_blob.rs: impl. DataBlobWriter for authenticated blobs

This commit is contained in:
Dietmar Maurer 2019-08-12 07:34:41 +02:00
parent cb0eea29d9
commit f796351c54

View File

@ -294,17 +294,22 @@ impl DataBlob {
use std::io::{Read, BufRead, Write, Seek, SeekFrom}; use std::io::{Read, BufRead, Write, Seek, SeekFrom};
enum BlobWriterState<W: Write> { enum BlobWriterState<'a, W: Write> {
Uncompressed { writer: W, hasher: crc32fast::Hasher }, Uncompressed { writer: W, hasher: crc32fast::Hasher },
Compressed { compr: zstd::stream::write::Encoder<W>, hasher: crc32fast::Hasher }, Compressed { compr: zstd::stream::write::Encoder<W>, hasher: crc32fast::Hasher },
Signed {
writer: W,
hasher: crc32fast::Hasher,
signer: openssl::sign::Signer<'a>,
},
} }
/// Write compressed data blobs /// Write compressed data blobs
pub struct DataBlobWriter<W: Write> { pub struct DataBlobWriter<'a, W: Write> {
state: BlobWriterState<W>, state: BlobWriterState<'a, W>,
} }
impl <W: Write + Seek> DataBlobWriter<W> { impl <'a, W: Write + Seek> DataBlobWriter<'a, W> {
pub fn new_uncompressed(mut writer: W) -> Result<Self, Error> { pub fn new_uncompressed(mut writer: W) -> Result<Self, Error> {
let hasher = crc32fast::Hasher::new(); let hasher = crc32fast::Hasher::new();
@ -329,6 +334,22 @@ impl <W: Write + Seek> DataBlobWriter<W> {
Ok(Self { state }) Ok(Self { state })
} }
pub fn new_signed(mut writer: W, config: &'a CryptConfig) -> Result<Self, Error> {
let hasher = crc32fast::Hasher::new();
writer.seek(SeekFrom::Start(0))?;
let head = AuthenticatedDataBlobHeader {
head: DataBlobHeader { magic: AUTHENTICATED_BLOB_MAGIC_1_0, crc: [0; 4] },
tag: [0u8; 32],
};
unsafe {
writer.write_le_value(head)?;
}
let signer = config.data_signer();
let state = BlobWriterState::Signed { writer, hasher, signer };
Ok(Self { state })
}
pub fn finish(self) -> Result<W, Error> { pub fn finish(self) -> Result<W, Error> {
match self.state { match self.state {
BlobWriterState::Uncompressed { mut writer, hasher } => { BlobWriterState::Uncompressed { mut writer, hasher } => {
@ -355,13 +376,30 @@ impl <W: Write + Seek> DataBlobWriter<W> {
writer.write_le_value(head)?; writer.write_le_value(head)?;
} }
return Ok(writer)
}
BlobWriterState::Signed { mut writer, hasher, signer, .. } => {
// write CRC and hmac
let crc = hasher.finalize();
let mut head = AuthenticatedDataBlobHeader {
head: DataBlobHeader { magic: AUTHENTICATED_BLOB_MAGIC_1_0, crc: crc.to_le_bytes() },
tag: [0u8; 32],
};
signer.sign(&mut head.tag)?;
writer.seek(SeekFrom::Start(0))?;
unsafe {
writer.write_le_value(head)?;
}
return Ok(writer) return Ok(writer)
} }
} }
} }
} }
impl <W: Write + Seek> Write for DataBlobWriter<W> { impl <'a, W: Write + Seek> Write for DataBlobWriter<'a, W> {
fn write(&mut self, buf: &[u8]) -> Result<usize, std::io::Error> { fn write(&mut self, buf: &[u8]) -> Result<usize, std::io::Error> {
match self.state { match self.state {
@ -373,6 +411,16 @@ impl <W: Write + Seek> Write for DataBlobWriter<W> {
hasher.update(buf); hasher.update(buf);
compr.write(buf) compr.write(buf)
} }
BlobWriterState::Signed { ref mut writer, ref mut hasher, ref mut signer, .. } => {
hasher.update(buf);
signer.update(buf).
map_err(|err| {
std::io::Error::new(
std::io::ErrorKind::Other,
format!("hmac update failed - {}", err))
})?;
writer.write(buf)
}
} }
} }
@ -384,6 +432,9 @@ impl <W: Write + Seek> Write for DataBlobWriter<W> {
BlobWriterState::Compressed { ref mut compr, .. } => { BlobWriterState::Compressed { ref mut compr, .. } => {
compr.flush() compr.flush()
} }
BlobWriterState::Signed { ref mut writer, .. } => {
writer.flush()
}
} }
} }
} }