use crate::client::Client;
use crate::crypto::shared_secretbox;
use crate::errors::CoreError;
use crate::nfs::{data_map, File, NfsError, NfsFuture};
use crate::self_encryption_storage::SelfEncryptionStorage;
use crate::utils::FutureExt;
use chrono::Utc;
use futures::Future;
use safe_nd::Error as SndError;
use self_encryption::{DataMap, SequentialEncryptor};
#[derive(Clone, Copy, Debug)]
pub enum Mode {
Overwrite,
Append,
}
pub struct Writer<C: Client> {
client: C,
file: File,
self_encryptor: SequentialEncryptor<SelfEncryptionStorage<C>>,
encryption_key: Option<shared_secretbox::Key>,
}
impl<C: Client> Writer<C> {
pub fn new(
client: &C,
storage: SelfEncryptionStorage<C>,
file: File,
mode: Mode,
encryption_key: Option<shared_secretbox::Key>,
) -> Box<NfsFuture<Writer<C>>> {
let fut = match mode {
Mode::Append => data_map::get(client, file.data_address(), encryption_key.clone())
.map(Some)
.into_box(),
Mode::Overwrite => ok!(None),
};
let client = client.clone();
fut.or_else(|err| -> Box<NfsFuture<Option<DataMap>>> {
match err {
NfsError::CoreError(CoreError::DataError(SndError::NoSuchData)) => ok!(None),
_ => err!(err),
}
})
.and_then(move |data_map| SequentialEncryptor::new(storage, data_map).map_err(From::from))
.map(move |self_encryptor| Writer {
client,
file,
self_encryptor,
encryption_key,
})
.map_err(From::from)
.into_box()
}
pub fn write(&self, data: &[u8]) -> Box<NfsFuture<()>> {
trace!(
"Writer writing file data of size {} into self-encryptor.",
data.len()
);
self.self_encryptor
.write(data)
.map_err(From::from)
.into_box()
}
pub fn close(self) -> Box<NfsFuture<File>> {
trace!("Writer induced self-encryptor close.");
let mut file = self.file;
let size = self.self_encryptor.len();
let client = self.client;
let encryption_key = self.encryption_key;
let published = file.published();
self.self_encryptor
.close()
.map_err(From::from)
.and_then(move |(data_map, _)| {
data_map::put(&client, &data_map, published, encryption_key)
})
.map(move |data_map_name| {
file.set_data_map_name(data_map_name);
file.set_modified_time(Utc::now());
file.set_size(size);
file
})
.into_box()
}
}