121 lines
2.6 KiB
Rust
121 lines
2.6 KiB
Rust
use std::io::{Read, Write};
|
|
use std::path::{Path, PathBuf};
|
|
use std::sync::{Arc, RwLock};
|
|
|
|
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
|
|
|
use crate::error::Error;
|
|
use crate::migrate::Migrate;
|
|
|
|
pub struct Persister<T: Migrate> {
|
|
path: PathBuf,
|
|
|
|
_marker: std::marker::PhantomData<T>,
|
|
}
|
|
|
|
impl<T: Migrate> Persister<T> {
|
|
pub fn new(base_dir: &Path, file_name: &str) -> Self {
|
|
let mut path = base_dir.to_path_buf();
|
|
path.push(file_name);
|
|
Self {
|
|
path,
|
|
_marker: Default::default(),
|
|
}
|
|
}
|
|
|
|
fn decode(&self, bytes: &[u8]) -> Result<T, Error> {
|
|
match T::decode(bytes) {
|
|
Some(v) => Ok(v),
|
|
None => {
|
|
error!(
|
|
"Unable to decode persisted data file {}",
|
|
self.path.display()
|
|
);
|
|
for line in hexdump::hexdump_iter(bytes) {
|
|
debug!("{}", line);
|
|
}
|
|
Err(Error::Message(format!(
|
|
"Unable to decode persisted data file {}",
|
|
self.path.display()
|
|
)))
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn load(&self) -> Result<T, Error> {
|
|
let mut file = std::fs::OpenOptions::new().read(true).open(&self.path)?;
|
|
|
|
let mut bytes = vec![];
|
|
file.read_to_end(&mut bytes)?;
|
|
|
|
let value = self.decode(&bytes[..])?;
|
|
Ok(value)
|
|
}
|
|
|
|
pub fn save(&self, t: &T) -> Result<(), Error> {
|
|
let bytes = t.encode()?;
|
|
|
|
let mut file = std::fs::OpenOptions::new()
|
|
.write(true)
|
|
.create(true)
|
|
.truncate(true)
|
|
.open(&self.path)?;
|
|
|
|
file.write_all(&bytes[..])?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub async fn load_async(&self) -> Result<T, Error> {
|
|
let mut file = tokio::fs::File::open(&self.path).await?;
|
|
|
|
let mut bytes = vec![];
|
|
file.read_to_end(&mut bytes).await?;
|
|
|
|
let value = self.decode(&bytes[..])?;
|
|
Ok(value)
|
|
}
|
|
|
|
pub async fn save_async(&self, t: &T) -> Result<(), Error> {
|
|
let bytes = t.encode()?;
|
|
|
|
let mut file = tokio::fs::File::create(&self.path).await?;
|
|
file.write_all(&bytes[..]).await?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub struct PersisterShared<V: Migrate + Default>(Arc<(Persister<V>, RwLock<V>)>);
|
|
|
|
impl<V: Migrate + Default> Clone for PersisterShared<V> {
|
|
fn clone(&self) -> PersisterShared<V> {
|
|
PersisterShared(self.0.clone())
|
|
}
|
|
}
|
|
|
|
impl<V: Migrate + Default> PersisterShared<V> {
|
|
pub fn new(base_dir: &Path, file_name: &str) -> Self {
|
|
let persister = Persister::new(base_dir, file_name);
|
|
let value = persister.load().unwrap_or_default();
|
|
Self(Arc::new((persister, RwLock::new(value))))
|
|
}
|
|
|
|
pub fn get_with<F, R>(&self, f: F) -> R
|
|
where
|
|
F: FnOnce(&V) -> R,
|
|
{
|
|
let value = self.0 .1.read().unwrap();
|
|
f(&value)
|
|
}
|
|
|
|
pub fn set_with<F>(&self, f: F) -> Result<(), Error>
|
|
where
|
|
F: FnOnce(&mut V),
|
|
{
|
|
let mut value = self.0 .1.write().unwrap();
|
|
f(&mut value);
|
|
self.0 .0.save(&value)
|
|
}
|
|
}
|