use std::hash::Hash;
use std::num::NonZeroUsize;
use std::sync::{Arc, Mutex, MutexGuard};
use lru::LruCache as InnerCache;
use tracing::instrument;
#[derive(Clone)]
pub struct LruCache<K, V>(Arc<Mutex<InnerCache<K, V>>>);
impl<K, V> LruCache<K, V>
where
K: Hash + Eq,
V: Clone,
{
pub fn new(capacity: NonZeroUsize) -> Self {
Self(Arc::new(Mutex::new(InnerCache::new(capacity))))
}
pub fn get(&self, key: &K) -> Option<V> {
self.lock().get(key).cloned()
}
pub fn put(&self, key: K, value: V) {
self.lock().put(key, value);
}
pub fn get_many<'a>(&self, keys: impl IntoIterator<Item = &'a K>) -> Vec<Option<V>>
where
K: 'a,
{
let mut cache = self.lock();
keys.into_iter().map(|key| cache.get(key).cloned()).collect()
}
pub fn put_many(&self, entries: impl IntoIterator<Item = (K, V)>) {
let mut cache = self.lock();
for (key, value) in entries {
cache.put(key, value);
}
}
pub fn clear(&self) {
self.lock().clear();
}
#[instrument(name = "lru.lock", skip_all)]
fn lock(&self) -> MutexGuard<'_, InnerCache<K, V>> {
self.0.lock().expect("LRU cache mutex poisoned")
}
}