use std::collections::HashMap;
use std::fs;
use std::io;
use std::path::{Path, PathBuf};
use rns_crypto::identity::Identity;
use rns_crypto::OsRng;
#[derive(Debug, Clone)]
pub struct StoragePaths {
pub config_dir: PathBuf,
pub storage: PathBuf,
pub cache: PathBuf,
pub identities: PathBuf,
pub discovered_interfaces: PathBuf,
}
#[derive(Debug, Clone)]
pub struct KnownDestination {
pub identity_hash: [u8; 16],
pub public_key: [u8; 64],
pub app_data: Option<Vec<u8>>,
pub hops: u8,
pub received_at: f64,
pub receiving_interface: u64,
pub was_used: bool,
pub last_used_at: Option<f64>,
pub retained: bool,
}
pub fn ensure_storage_dirs(config_dir: &Path) -> io::Result<StoragePaths> {
let storage = config_dir.join("storage");
let cache = config_dir.join("cache");
let identities = storage.join("identities");
let announces = cache.join("announces");
let discovered_interfaces = storage.join("discovery").join("interfaces");
fs::create_dir_all(&storage)?;
fs::create_dir_all(&cache)?;
fs::create_dir_all(&identities)?;
fs::create_dir_all(&announces)?;
fs::create_dir_all(&discovered_interfaces)?;
Ok(StoragePaths {
config_dir: config_dir.to_path_buf(),
storage,
cache,
identities,
discovered_interfaces,
})
}
pub fn save_identity(identity: &Identity, path: &Path) -> io::Result<()> {
let private_key = identity
.get_private_key()
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "Identity has no private key"))?;
fs::write(path, &private_key)
}
pub fn load_identity(path: &Path) -> io::Result<Identity> {
let data = fs::read(path)?;
if data.len() != 64 {
return Err(io::Error::new(
io::ErrorKind::InvalidData,
format!("Identity file must be 64 bytes, got {}", data.len()),
));
}
let mut key = [0u8; 64];
key.copy_from_slice(&data);
Ok(Identity::from_private_key(&key))
}
pub fn save_known_destinations(
destinations: &HashMap<[u8; 16], KnownDestination>,
path: &Path,
) -> io::Result<()> {
use rns_core::msgpack::{self, Value};
let entries: Vec<(Value, Value)> = destinations
.iter()
.map(|(hash, dest)| {
let key = Value::Bin(hash.to_vec());
let app_data = match &dest.app_data {
Some(d) => Value::Bin(d.clone()),
None => Value::Nil,
};
let value = Value::Array(vec![
Value::UInt(dest.received_at as u64),
Value::Bin(dest.public_key.to_vec()),
app_data,
Value::Bin(dest.identity_hash.to_vec()),
Value::UInt(dest.hops as u64),
Value::UInt(dest.receiving_interface),
Value::Bool(dest.was_used),
match dest.last_used_at {
Some(last_used_at) => Value::UInt(last_used_at as u64),
None => Value::Nil,
},
Value::Bool(dest.retained),
]);
(key, value)
})
.collect();
let packed = msgpack::pack(&Value::Map(entries));
fs::write(path, packed)
}
pub fn load_known_destinations(path: &Path) -> io::Result<HashMap<[u8; 16], KnownDestination>> {
use rns_core::msgpack;
let data = fs::read(path)?;
if data.is_empty() {
return Ok(HashMap::new());
}
let (value, _) = msgpack::unpack(&data)
.map_err(|e| io::Error::new(io::ErrorKind::InvalidData, format!("msgpack error: {}", e)))?;
let map = value
.as_map()
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "Expected msgpack map"))?;
let mut result = HashMap::new();
for (k, v) in map {
let hash_bytes = k
.as_bin()
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "Expected bin key"))?;
if hash_bytes.len() != 16 {
continue; }
let mut dest_hash = [0u8; 16];
dest_hash.copy_from_slice(hash_bytes);
let arr = v
.as_array()
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "Expected array value"))?;
if arr.len() < 3 {
continue;
}
let received_at = arr[0].as_uint().unwrap_or(0) as f64;
let pub_key_bytes = arr[1]
.as_bin()
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "Expected bin public_key"))?;
if pub_key_bytes.len() != 64 {
continue;
}
let mut public_key = [0u8; 64];
public_key.copy_from_slice(pub_key_bytes);
let app_data = if arr.len() > 2 {
arr[2].as_bin().map(|b| b.to_vec())
} else {
None
};
let identity_hash = if arr.len() > 3 {
let hash_bytes = arr[3]
.as_bin()
.filter(|bytes| bytes.len() == 16)
.map(|bytes| {
let mut hash = [0u8; 16];
hash.copy_from_slice(bytes);
hash
});
hash_bytes.unwrap_or_else(|| {
let identity = Identity::from_public_key(&public_key);
*identity.hash()
})
} else {
let identity = Identity::from_public_key(&public_key);
*identity.hash()
};
let hops = arr.get(4).and_then(|value| value.as_uint()).unwrap_or(0) as u8;
let receiving_interface = arr.get(5).and_then(|value| value.as_uint()).unwrap_or(0);
let was_used = arr
.get(6)
.and_then(|value| value.as_bool())
.unwrap_or(false);
let last_used_at = arr
.get(7)
.and_then(|value| value.as_uint())
.map(|value| value as f64);
let retained = arr
.get(8)
.and_then(|value| value.as_bool())
.unwrap_or(false);
result.insert(
dest_hash,
KnownDestination {
identity_hash,
public_key,
app_data,
hops,
received_at,
receiving_interface,
was_used,
last_used_at,
retained,
},
);
}
Ok(result)
}
pub fn resolve_config_dir(explicit: Option<&Path>) -> PathBuf {
if let Some(p) = explicit {
p.to_path_buf()
} else {
let home = std::env::var("HOME").unwrap_or_else(|_| "/tmp".into());
PathBuf::from(home).join(".reticulum")
}
}
pub fn load_or_create_identity(identities_dir: &Path) -> io::Result<Identity> {
let id_path = identities_dir.join("identity");
if id_path.exists() {
load_identity(&id_path)
} else {
let identity = Identity::new(&mut OsRng);
save_identity(&identity, &id_path)?;
Ok(identity)
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::sync::atomic::{AtomicU64, Ordering};
static TEST_COUNTER: AtomicU64 = AtomicU64::new(0);
fn temp_dir() -> PathBuf {
let id = TEST_COUNTER.fetch_add(1, Ordering::Relaxed);
let dir = std::env::temp_dir().join(format!("rns-test-{}-{}", std::process::id(), id));
let _ = fs::remove_dir_all(&dir);
fs::create_dir_all(&dir).unwrap();
dir
}
#[test]
fn save_load_identity_roundtrip() {
let dir = temp_dir();
let path = dir.join("test_identity");
let identity = Identity::new(&mut OsRng);
let original_hash = *identity.hash();
save_identity(&identity, &path).unwrap();
let loaded = load_identity(&path).unwrap();
assert_eq!(*loaded.hash(), original_hash);
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn identity_file_format() {
let dir = temp_dir();
let path = dir.join("test_identity_fmt");
let identity = Identity::new(&mut OsRng);
save_identity(&identity, &path).unwrap();
let data = fs::read(&path).unwrap();
assert_eq!(data.len(), 64, "Identity file must be exactly 64 bytes");
let private_key = identity.get_private_key();
let private_key = private_key.unwrap();
assert_eq!(&data[..], &private_key[..]);
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn save_load_known_destinations_empty() {
let dir = temp_dir();
let path = dir.join("known_destinations");
let empty: HashMap<[u8; 16], KnownDestination> = HashMap::new();
save_known_destinations(&empty, &path).unwrap();
let loaded = load_known_destinations(&path).unwrap();
assert!(loaded.is_empty());
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn save_load_known_destinations_roundtrip() {
let dir = temp_dir();
let path = dir.join("known_destinations");
let mut dests = HashMap::new();
dests.insert(
[0x01u8; 16],
KnownDestination {
identity_hash: [0x11u8; 16],
public_key: [0xABu8; 64],
app_data: Some(vec![0x01, 0x02, 0x03]),
hops: 2,
received_at: 1700000000.0,
receiving_interface: 7,
was_used: true,
last_used_at: Some(1700000010.0),
retained: true,
},
);
dests.insert(
[0x02u8; 16],
KnownDestination {
identity_hash: [0x22u8; 16],
public_key: [0xCDu8; 64],
app_data: None,
hops: 1,
received_at: 1700000001.0,
receiving_interface: 0,
was_used: false,
last_used_at: None,
retained: false,
},
);
save_known_destinations(&dests, &path).unwrap();
let loaded = load_known_destinations(&path).unwrap();
assert_eq!(loaded.len(), 2);
let d1 = &loaded[&[0x01u8; 16]];
assert_eq!(d1.identity_hash, [0x11u8; 16]);
assert_eq!(d1.public_key, [0xABu8; 64]);
assert_eq!(d1.app_data, Some(vec![0x01, 0x02, 0x03]));
assert_eq!(d1.hops, 2);
assert_eq!(d1.received_at as u64, 1700000000);
assert_eq!(d1.receiving_interface, 7);
assert!(d1.was_used);
assert_eq!(d1.last_used_at, Some(1700000010.0));
assert!(d1.retained);
let d2 = &loaded[&[0x02u8; 16]];
assert_eq!(d2.app_data, None);
assert!(!d2.was_used);
assert_eq!(d2.last_used_at, None);
assert!(!d2.retained);
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn ensure_dirs_creates() {
let dir = temp_dir().join("new_config");
let _ = fs::remove_dir_all(&dir);
let paths = ensure_storage_dirs(&dir).unwrap();
assert!(paths.storage.exists());
assert!(paths.cache.exists());
assert!(paths.identities.exists());
assert!(paths.discovered_interfaces.exists());
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn ensure_dirs_existing() {
let dir = temp_dir().join("existing_config");
fs::create_dir_all(dir.join("storage")).unwrap();
fs::create_dir_all(dir.join("cache")).unwrap();
let paths = ensure_storage_dirs(&dir).unwrap();
assert!(paths.storage.exists());
assert!(paths.identities.exists());
let _ = fs::remove_dir_all(&dir);
}
#[test]
fn load_or_create_identity_new() {
let dir = temp_dir().join("load_or_create");
fs::create_dir_all(&dir).unwrap();
let identity = load_or_create_identity(&dir).unwrap();
let id_path = dir.join("identity");
assert!(id_path.exists());
let loaded = load_or_create_identity(&dir).unwrap();
assert_eq!(*identity.hash(), *loaded.hash());
let _ = fs::remove_dir_all(&dir);
}
}