use anyhow::{anyhow, Result};
use bincode::{Decode, Encode};
use serde::{Deserialize, Serialize};
use rucksack_lib::util;
use crate::crypto::{decrypt, encrypt};
use super::shared;
use super::v050;
pub use super::v050::Creds;
pub const VERSION: &str = "0.6.0";
pub type HashMap = dashmap::DashMap<String, EncryptedRecord>;
pub fn migrate_hashmap_from_v050(hm_v050: v050::HashMap) -> HashMap {
let hm: HashMap = dashmap::DashMap::new();
for i in hm_v050.iter() {
let r = i.value();
let _ = hm.insert(
i.key().to_string(),
migrate_encrypted_record_from_v050(r.clone()),
);
}
hm
}
pub fn decode_hashmap(bytes: Vec<u8>, mut version: versions::SemVer) -> Result<HashMap> {
log::debug!(version = version.to_string().as_str(), operation = "decode"; "Decoding hashmap from stored bytes");
version = shared::trim_version(version);
let hm: HashMap = dashmap::DashMap::new();
log::trace!(operation = "decode"; "Created hashmap");
let sorted_vec: Vec<(String, EncryptedRecord)>;
log::trace!(operation = "decode"; "Created vec for sorted data");
let current_version = shared::version(VERSION).map_err(|e| anyhow!("{}", e))?;
if version < current_version {
log::info!(version = "0.5.0", operation = "migrate"; "Attempting to decode hashmap from previous version");
let hm = v050::decode_hashmap(bytes, version)?;
return Ok(migrate_hashmap_from_v050(hm));
}
match bincode::decode_from_slice(bytes.as_ref(), util::bincode_cfg()) {
Ok((result, _len)) => {
sorted_vec = result;
for (key, val) in sorted_vec {
if hm.insert(key.clone(), val).is_some() {}
}
Ok(hm)
}
Err(e) => {
log::info!(error = e.to_string().as_str(), operation = "decode"; "couldn't deserialise bincoded hashmap bytes");
Err(anyhow!(e))
}
}
}
#[derive(Clone, Debug, Default, Serialize, Deserialize, Eq, PartialEq, Encode, Decode)]
pub enum Kind {
#[default]
Account,
Credential,
Password,
}
pub fn migrate_kind_from_v050(k: v050::Kind) -> Kind {
match k {
v050::Kind::Password => Kind::Password,
}
}
#[derive(Clone, Debug, Eq, PartialEq, Encode, Decode)]
pub struct DecryptedRecord {
pub creds: Creds,
pub metadata: Metadata,
}
impl DecryptedRecord {
pub fn key(&self) -> String {
format!("{}:{}", self.creds.user, self.metadata.url)
}
pub fn metadata(&self) -> Metadata {
self.metadata.clone()
}
pub fn password(&self) -> String {
self.creds.password.clone()
}
pub fn user(&self) -> String {
self.creds.user.clone()
}
pub fn encrypt(&self, store_pwd: String, salt: String) -> Result<EncryptedRecord> {
let encoded = bincode::encode_to_vec(&self.creds, util::bincode_cfg()).unwrap();
let encrypted = encrypt(encoded, store_pwd, salt)?;
Ok(EncryptedRecord {
key: self.key(),
value: encrypted,
metadata: self.metadata(),
})
}
}
#[derive(Clone, Debug, Serialize, Deserialize, Eq, PartialEq, Encode, Decode)]
pub struct EncryptedRecord {
pub key: String,
pub value: Vec<u8>,
pub metadata: Metadata,
}
pub fn migrate_encrypted_record_from_v050(er: v050::EncryptedRecord) -> EncryptedRecord {
EncryptedRecord {
key: er.key(),
value: er.value(),
metadata: migrate_metadata_from_v050(er.metadata()),
}
}
impl EncryptedRecord {
pub fn key(&self) -> String {
self.key.clone()
}
pub fn value(&self) -> Vec<u8> {
self.value.clone()
}
pub fn metadata(&self) -> Metadata {
self.metadata.clone()
}
pub fn decrypt(&self, store_pwd: String, salt: String) -> Result<DecryptedRecord> {
let decrypted = decrypt(self.value.clone(), store_pwd, salt)?;
let (decoded, _len) =
bincode::decode_from_slice(&decrypted[..], util::bincode_cfg()).unwrap();
Ok(DecryptedRecord {
creds: decoded,
metadata: self.metadata(),
})
}
}
#[derive(Clone, Debug, Serialize, Deserialize, Eq, PartialEq, Encode, Decode)]
pub struct Metadata {
pub kind: Kind,
pub url: String,
pub created: String,
pub imported: String,
pub updated: String,
pub password_changed: String,
pub last_used: String,
pub access_count: u64,
}
pub fn migrate_metadata_from_v050(md: v050::Metadata) -> Metadata {
Metadata {
kind: migrate_kind_from_v050(md.kind),
url: md.url,
created: md.created,
imported: md.imported,
updated: md.updated,
password_changed: md.password_changed,
last_used: md.last_used,
access_count: md.access_count,
}
}
#[cfg(test)]
mod tests {
use rucksack_lib::time;
use crate::testing;
#[test]
fn password_records() {
let pwd = testing::data::store_pwd();
let salt = time::now();
let dpr = testing::data::plaintext_record_v060();
assert_eq!(
format!("{}", dpr.creds),
"Creds{user: alice@site.com, password: *****}"
);
assert_eq!(
format!("{:?}", dpr.creds),
"Creds{user: alice@site.com, password: *****}"
);
let epr = dpr.encrypt(pwd.clone(), salt.clone()).unwrap();
assert_eq!(54, epr.value.len());
let re_dpr = epr.decrypt(pwd, salt).unwrap();
assert_eq!(re_dpr.creds.password, "4 s3kr1t");
}
}