dls: basic offline store ops

This commit is contained in:
Mikaël Cluseau
2026-05-11 12:00:44 +02:00
parent 2dfd67ec0d
commit 35a2609f29
4 changed files with 252 additions and 49 deletions
+24
View File
@@ -3,6 +3,7 @@ use clap::{CommandFactory, Parser, Subcommand};
use eyre::format_err;
use futures_util::Stream;
use futures_util::StreamExt;
use std::path::PathBuf;
use std::time::{Duration, SystemTime};
use tokio::fs;
use tokio::io::{AsyncWrite, AsyncWriteExt};
@@ -40,6 +41,11 @@ enum Command {
Hash {
salt: String,
},
Store {
store_path: PathBuf,
#[command(subcommand)]
op: StoreOp,
},
}
#[derive(Subcommand)]
@@ -96,6 +102,12 @@ enum ClusterCommand {
},
}
#[derive(Subcommand)]
enum StoreOp {
Get { data_path: PathBuf },
Set { data_path: PathBuf, value: String },
}
#[tokio::main(flavor = "current_thread")]
async fn main() -> eyre::Result<()> {
clap_complete::CompleteEnv::with_factory(Cli::command).complete();
@@ -231,6 +243,18 @@ async fn main() -> eyre::Result<()> {
println!("hash (hex): {}", hex::encode(&hash));
println!("hash (base64): {}", dkl::base64_encode(&hash));
}
C::Store { store_path, op } => {
let mut s = dls::store::Store::new(store_path);
s.unlock(&std::env::var("DLS_STORE_PW").unwrap()).await?;
match op {
StoreOp::Get { data_path } => {
let mut data = std::io::Cursor::new(s.read(data_path).await?);
tokio::io::copy(&mut data, &mut tokio::io::stdout()).await?;
}
StoreOp::Set { data_path, value } => s.write(data_path, value.as_bytes()).await?,
}
}
};
Ok(())
+192 -4
View File
@@ -1,4 +1,13 @@
pub fn hash_password(salt: &[u8], passphrase: &str) -> argon2::Result<[u8; 32]> {
use openssl::symm::Mode;
use std::borrow::Cow;
use std::path::{Path, PathBuf};
use tokio::{fs, io::AsyncWriteExt};
pub type Salt = [u8; 16];
pub type Key = [u8; 32];
pub type Hash = Vec<u8>;
pub fn hash_password(salt: &[u8], passphrase: &str) -> Result<Key> {
let hash = argon2::hash_raw(
passphrase.as_bytes(),
salt,
@@ -6,12 +15,191 @@ pub fn hash_password(salt: &[u8], passphrase: &str) -> argon2::Result<[u8; 32]>
variant: argon2::Variant::Argon2id,
hash_length: 32,
time_cost: 1,
mem_cost: 65536,
mem_cost: 64 << 10,
thread_mode: argon2::ThreadMode::Parallel,
lanes: 4,
..Default::default()
},
)?;
)
.map_err(Error::Hash)?;
unsafe { Ok(hash.try_into().unwrap_unchecked()) }
Ok(hash.try_into().unwrap())
}
pub struct Store {
path: PathBuf,
key: Option<Key>,
}
impl Store {
pub fn new(path: impl Into<PathBuf>) -> Self {
Self {
path: path.into(),
key: None,
}
}
pub async fn unlock(&mut self, passphrase: &str) -> Result<()> {
let keys = self.read_keys().await?;
let salt = keys.salt;
let user_key = hash_password(&salt, passphrase)?;
let user_hash = openssl::sha::sha512(&user_key);
for nk in keys.keys {
if nk.hash != user_hash {
continue;
}
// build iv+data from salt
let mut enc_data = Vec::with_capacity(salt.len() + nk.enc_key.len());
enc_data.extend_from_slice(&salt);
enc_data.extend_from_slice(&nk.enc_key);
let key = crypt(&enc_data, &user_key, Mode::Decrypt)?;
let key = key.try_into().map_err(|_| Error::InvalidKey)?;
self.key = Some(key);
return Ok(());
}
Err(Error::KeyNotFound)
}
pub async fn read(&self, path: impl AsRef<Path>) -> Result<Vec<u8>> {
let enc_data = self.read_file(path.as_ref().with_extension("data")).await?;
self.decrypt(&enc_data)
}
pub async fn read_to_string(&self, path: impl AsRef<Path>) -> Result<String> {
let path = path.as_ref();
String::from_utf8(self.read(path).await?).map_err(|_| Error::InvalidUtf8(path.into()))
}
pub async fn write(&self, path: impl AsRef<Path>, data: &[u8]) -> Result<()> {
let enc_data = self.encrypt(data)?;
safe_write(self.path.join(&path).with_extension("data"), &enc_data).await
}
async fn read_keys(&self) -> Result<Keys<'_>> {
let keys = self.read_file(".keys").await?;
let Some(keys) = keys.strip_prefix(b"{json}") else {
return Err(Error::InvalidKeys);
};
serde_json::from_slice(keys).map_err(Error::KeysParse)
}
async fn read_file(&self, subpath: impl AsRef<Path>) -> Result<Vec<u8>> {
let path = self.path.join(subpath);
fs::read(&path).await.map_err(|e| Error::Read(path, e))
}
fn encrypt(&self, src: &[u8]) -> Result<Vec<u8>> {
self.crypt(src, Mode::Encrypt)
}
fn decrypt(&self, src: &[u8]) -> Result<Vec<u8>> {
self.crypt(src, Mode::Decrypt)
}
fn crypt(&self, src: &[u8], mode: Mode) -> Result<Vec<u8>> {
let key = self.key.as_ref().ok_or(Error::NotUnlocked)?;
crypt(src, key, mode)
}
}
async fn safe_write(path: impl AsRef<Path>, contents: &[u8]) -> Result<()> {
let path = path.as_ref();
let tmp = path.with_added_extension("new");
let tmp_err = |e| Error::Write(tmp.clone(), e);
let mut file = fs::OpenOptions::new()
.create(true)
.truncate(true)
.write(true)
.open(&tmp)
.await
.map_err(tmp_err)?;
file.write_all(contents).await.map_err(tmp_err)?;
file.sync_all().await.map_err(tmp_err)?;
file.shutdown().await.map_err(tmp_err)?;
fs::rename(tmp, &path)
.await
.map_err(|e| Error::Write(path.into(), e))
}
fn crypt(mut src: &[u8], key: &Key, mode: Mode) -> Result<Vec<u8>> {
use openssl::symm::{Cipher, Crypter};
let mut iv = [0u8; 16];
let mut dst: Vec<u8>;
let crypt_dst: &mut [u8];
match mode {
Mode::Encrypt => {
getrandom::fill(&mut iv).unwrap();
dst = vec![0u8; iv.len() + src.len()];
dst[..iv.len()].copy_from_slice(&iv);
crypt_dst = &mut dst[iv.len()..];
}
Mode::Decrypt => {
iv = src[..iv.len()]
.try_into()
.map_err(|_| Error::DecryptInputToSmall)?;
src = &src[iv.len()..];
dst = vec![0u8; src.len()];
crypt_dst = &mut dst;
}
}
let cipher = Cipher::aes_256_cfb128();
Crypter::new(cipher, mode, key, Some(&iv))
.expect("Failed to init AES")
.update(src, crypt_dst)
.expect("AES CFB encrypt/decrypt failed");
Ok(dst)
}
pub type Result<T> = std::result::Result<T, Error>;
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("hash error: {0}")]
Hash(argon2::Error),
#[error("read {0} failed: {1}")]
Read(PathBuf, std::io::Error),
#[error("write {0} failed: {1}")]
Write(PathBuf, std::io::Error),
#[error("read {0} failed: invalid UTF-8")]
InvalidUtf8(PathBuf),
#[error("invalid keys data")]
InvalidKeys,
#[error("invalid key")]
InvalidKey,
#[error("keys parse error: {0}")]
KeysParse(serde_json::Error),
#[error("key not found")]
KeyNotFound,
#[error("store not unlocked")]
NotUnlocked,
#[error("decrypt input too small")]
DecryptInputToSmall,
}
#[derive(Clone, Debug, serde::Serialize, serde::Deserialize)]
#[serde(rename_all = "PascalCase")]
struct Keys<'t> {
salt: Salt,
keys: Vec<NamedKey<'t>>,
}
#[derive(Clone, Debug, serde::Serialize, serde::Deserialize)]
#[serde(rename_all = "PascalCase")]
struct NamedKey<'t> {
name: Cow<'t, str>,
hash: Hash,
enc_key: Key,
}