1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#![doc = include_str!("../README.md")]

pub mod args;

use rand::RngCore;
use serde::{de::DeserializeOwned, Serialize};
use std::{error::Error, path::Path};
use streambed::{decrypt_buf, encrypt_struct, secret_store};
use tokio::{
    fs,
    io::{AsyncReadExt, AsyncWriteExt},
};

/// Loads and deserializes a structure described by T, if the file is present.
/// If there are IO issues outside of the file not being there, they will be returned
/// as an error. Beyond IO, state is attempted to be decrypted and deserialized when present.
/// Any issues there cause the default representation of the structure to be returned. The default
/// structure is also returned where there is no file present in the first place.
pub async fn load_struct<T, D, DE>(
    state_storage_path: &Path,
    ss: &impl secret_store::SecretStore,
    secret_path: &str,
    deserialize: D,
) -> Result<T, Box<dyn Error>>
where
    T: Default + DeserializeOwned,
    D: FnOnce(&[u8]) -> Result<T, DE>,
{
    if let Ok(mut f) = fs::File::open(state_storage_path).await {
        let mut buf = vec![];
        f.read_to_end(&mut buf).await?;
        Ok(decrypt_buf(ss, secret_path, &mut buf, deserialize)
            .await
            .unwrap_or_default())
    } else {
        Ok(T::default())
    }
}

/// Saves an encrypted structure described by T. Any IO errors are returned.
pub async fn save_struct<T, U, F, S, SE>(
    state_storage_path: &Path,
    ss: &impl secret_store::SecretStore,
    secret_path: &str,
    serialize: S,
    rng: F,
    state: &T,
) -> Result<(), Box<dyn Error>>
where
    T: Serialize,
    S: FnOnce(&T) -> Result<Vec<u8>, SE>,
    F: FnOnce() -> U,
    U: RngCore,
{
    if let Some(buf) = encrypt_struct(ss, secret_path, serialize, rng, state).await {
        let mut f = fs::File::create(state_storage_path).await?;
        f.write_all(&buf).await?;
    }
    Ok(())
}