use std::future::Future;
use std::time::Instant;
use tokio::sync::Mutex;
#[derive(Debug, Clone)]
pub struct TemporaryToken<T> {
pub token: T,
pub expiry: Option<Instant>,
}
#[derive(Debug)]
pub struct TokenCache<T> {
cache: Mutex<Option<TemporaryToken<T>>>,
}
impl<T> Default for TokenCache<T> {
fn default() -> Self {
Self {
cache: Default::default(),
}
}
}
impl<T: Clone + Send> TokenCache<T> {
pub async fn get_or_insert_with<F, Fut, E>(&self, f: F) -> Result<T, E>
where
F: FnOnce() -> Fut + Send,
Fut: Future<Output = Result<TemporaryToken<T>, E>> + Send,
{
let now = Instant::now();
let mut locked = self.cache.lock().await;
if let Some(cached) = locked.as_ref() {
match cached.expiry {
Some(ttl)
if ttl
.checked_duration_since(now)
.unwrap_or_default()
.as_secs()
> 300 =>
{
return Ok(cached.token.clone());
}
None => return Ok(cached.token.clone()),
_ => (),
}
}
let cached = f().await?;
let token = cached.token.clone();
*locked = Some(cached);
Ok(token)
}
}