use crate::client::Client;
use crate::crypto::shared_secretbox;
use crate::errors::CoreError;
use crate::nfs::{data_map, File, NfsError};
use crate::self_encryption_storage::SelfEncryptionStorage;
use chrono::Utc;
use log::trace;
use safe_nd::Error as SndError;
use self_encryption::SequentialEncryptor;
#[derive(Clone, Copy, Debug)]
pub enum Mode {
Overwrite,
Append,
}
pub struct Writer<C: Client + 'static> {
client: C,
file: File,
self_encryptor: SequentialEncryptor<SelfEncryptionStorage<C>>,
encryption_key: Option<shared_secretbox::Key>,
}
impl<C: Sync + Client + 'static> Writer<C> {
pub async fn new(
client: &C,
storage: SelfEncryptionStorage<C>,
file: File,
mode: Mode,
encryption_key: Option<shared_secretbox::Key>,
) -> Result<Writer<C>, NfsError> {
let data_map = match mode {
Mode::Append => {
match data_map::get(client, file.data_address(), encryption_key.clone()).await {
Ok(map) => Some(map),
Err(NfsError::CoreError(CoreError::DataError(SndError::NoSuchData))) => None,
Err(err) => return Err(err),
}
}
Mode::Overwrite => None,
};
let self_encryptor = SequentialEncryptor::new(storage, data_map).await?;
Ok(Self {
client: client.clone(),
file,
self_encryptor,
encryption_key,
})
}
pub async fn write(&self, data: &[u8]) -> Result<(), NfsError> {
trace!(
"Writer writing file data of size {} into self-encryptor.",
data.len()
);
self.self_encryptor.write(data).await.map_err(From::from)
}
pub async fn close(self) -> Result<File, NfsError> {
trace!("Writer induced self-encryptor close.");
let mut file = self.file;
let size = self.self_encryptor.len().await;
let client = self.client;
let encryption_key = self.encryption_key;
let published = file.published();
let (data_map, _storage) = self.self_encryptor.close().await?;
let data_map_name = data_map::put(&client, &data_map, published, encryption_key).await?;
file.set_data_map_name(data_map_name);
file.set_modified_time(Utc::now());
file.set_size(size);
Ok(file)
}
}