expire_cache 0.1.15

Efficient double-buffered expiration cache / 高效双缓冲过期缓存
Documentation
use std::{borrow::Borrow, hash::Hash, ops::Deref};

use dashmap::DashMap;
use sendptr::SendPtr;

use crate::Map;

pub struct RefVal<'a, K, V> {
  _mapref: dashmap::mapref::one::Ref<'a, K, V>,
  val: SendPtr<V>,
}

impl<'a, K, V> RefVal<'a, K, V>
where
  K: Eq + Hash,
{
  pub fn new(mapref: dashmap::mapref::one::Ref<'a, K, V>) -> Self {
    let val = SendPtr::new(mapref.value() as *const V);
    Self {
      _mapref: mapref,
      val,
    }
  }
}

impl<'a, K, V> Borrow<V> for RefVal<'a, K, V> {
  fn borrow(&self) -> &V {
    &self.val
  }
}

impl<'a, K, V> AsRef<V> for RefVal<'a, K, V> {
  fn as_ref(&self) -> &V {
    &self.val
  }
}

impl<'a, K, V> Deref for RefVal<'a, K, V> {
  type Target = V;

  fn deref(&self) -> &Self::Target {
    &self.val
  }
}

#[cfg(feature = "get_or_init_async")]
impl<K, V> crate::GetOrInitAsync for DashMap<K, V>
where
  K: Clone + Eq + Hash + Send + Sync + 'static,
  V: Send + Sync + 'static,
{
  async fn get_or_init_async<'a, I: crate::AsyncInit<Key = K, Val = V>>(
    &'a self,
    key: &Self::Key,
  ) -> Result<Self::RefVal<'a>, I::Error> {
    if let Some(r) = self.get(key) {
      return Ok(RefVal::new(r));
    }
    let val = I::init(key).await?;
    Ok(RefVal::new(
      self.entry(key.clone()).or_insert(val).downgrade(),
    ))
  }
}

#[cfg(feature = "get_or_init")]
impl<K, V> crate::GetOrInit for DashMap<K, V>
where
  K: Clone + Eq + Hash + Send + Sync + 'static,
  V: Send + Sync + 'static,
{
  fn get_or_init<'a, E>(
    &'a self,
    key: &Self::Key,
    func: impl Fn(&Self::Key) -> Result<Self::Val, E>,
  ) -> Result<Self::RefVal<'a>, E> {
    if let Some(r) = self.get(key) {
      return Ok(RefVal::new(r));
    }
    let val = func(key)?;
    Ok(RefVal::new(
      self.entry(key.clone()).or_insert(val).downgrade(),
    ))
  }
}

impl<K, V> Map for DashMap<K, V>
where
  K: Eq + Hash + Send + Sync + 'static,
  V: Send + Sync + 'static,
{
  type Key = K;
  type Val = V;
  type RefVal<'a> = RefVal<'a, K, V>;

  fn clear(&self) {
    self.clear();
  }

  fn insert(&self, key: Self::Key, val: Self::Val) {
    self.insert(key, val);
  }

  fn get<'a>(&'a self, key: &Self::Key) -> Option<Self::RefVal<'a>> {
    let mapref = self.get(key.borrow())?;
    Some(RefVal::new(mapref))
  }
}