use std::{
num::NonZeroUsize,
sync::{Arc, Mutex},
};
use iroh_base::PublicKey;
type SignatureError = <PublicKey as TryFrom<&'static [u8]>>::Error;
type PublicKeyBytes = [u8; PublicKey::LENGTH];
#[derive(Debug, Clone)]
pub enum KeyCache {
Disabled,
Shared(Arc<Mutex<lru::LruCache<PublicKey, ()>>>),
}
impl KeyCache {
#[cfg(all(test, feature = "server"))]
pub fn test() -> Self {
Self::Disabled
}
pub fn new(capacity: usize) -> Self {
let Some(capacity) = NonZeroUsize::new(capacity) else {
return Self::Disabled;
};
let cache = lru::LruCache::new(capacity);
Self::Shared(Arc::new(Mutex::new(cache)))
}
pub fn key_from_slice(&self, slice: &[u8]) -> Result<PublicKey, SignatureError> {
let Self::Shared(cache) = self else {
return PublicKey::try_from(slice);
};
let Ok(bytes) = PublicKeyBytes::try_from(slice) else {
return Err(PublicKey::try_from(slice).expect_err("invalid length"));
};
let mut cache = cache.lock().expect("not poisoned");
if let Some((key, _)) = cache.get_key_value(&bytes) {
return Ok(*key);
}
let key = PublicKey::from_bytes(&bytes)?;
cache.put(key, ());
Ok(key)
}
}