use std::cell::RefCell;
use std::time::{Duration, Instant};
#[derive(Default)]
pub struct ThreadLocalTimedCache<T> {
inner: RefCell<Option<(T, Instant)>>,
}
#[derive(Default)]
pub struct ThreadLocalKeyedCache<T, K> {
inner: RefCell<Option<(T, Instant, K)>>,
}
impl<T> ThreadLocalTimedCache<T> {
pub fn new() -> Self {
Self {
inner: RefCell::new(None),
}
}
}
impl<T: Clone> ThreadLocalTimedCache<T> {
pub fn get_or_refresh<F>(&self, max_age: Duration, compute: F) -> T
where
F: FnOnce() -> T,
{
let mut guard = self.inner.borrow_mut();
let now = Instant::now();
if let Some((ref v, ref at)) = *guard {
if at.elapsed() < max_age {
return v.clone();
}
}
let v = compute();
*guard = Some((v.clone(), now));
v
}
}
impl<T, K> ThreadLocalKeyedCache<T, K> {
pub fn new() -> Self {
Self {
inner: RefCell::new(None),
}
}
}
impl<T: Clone, K: Clone + PartialEq> ThreadLocalKeyedCache<T, K> {
pub fn get_or_refresh<F>(&self, max_age: Duration, key: &K, compute: F) -> T
where
F: FnOnce() -> T,
{
let mut guard = self.inner.borrow_mut();
let now = Instant::now();
if let Some((ref v, ref at, ref k)) = *guard {
if k == key && at.elapsed() < max_age {
return v.clone();
}
}
let v = compute();
*guard = Some((v.clone(), now, key.clone()));
v
}
}