maf 0.1.0-alpha.6

MAF is an authoritative realtime framework for writing simple, secure, and scalable apps.
Documentation
use std::{
    any::{Any, TypeId},
    collections::HashMap,
    sync::Arc,
};

use tokio::sync::{RwLock, RwLockMappedWriteGuard, RwLockReadGuard, RwLockWriteGuard};

use crate::callable::{CallableFetch, CallableParam, SupportsAsync};

use super::App;

#[derive(Debug, Default)]
pub struct LocalStateStore {
    states: HashMap<TypeId, AnyState>,
}

#[derive(Debug, Clone)]
pub struct AnyState {
    data: Arc<RwLock<Box<dyn Any + Send + Sync>>>,
}

/// [`Local`] is a wrapper around shared data that can be accessed throughout the app **without
/// being synchronized to clients**.
///
/// If client synchronization is needed, use [`crate::Store`] instead.
///
/// See [`crate::AppBuilder::local`] for usage.
#[derive(Debug, Clone)]
pub struct Local<T> {
    inner: AnyState,
    _phantom: std::marker::PhantomData<T>,
}

#[derive(Debug, thiserror::Error)]
pub enum LocalStateError {
    #[error("local state of type {0} not found")]
    NotFound(String),
}

impl LocalStateStore {
    pub fn insert<T: Send + Sync + 'static>(&mut self, data: T) {
        self.states.insert(
            TypeId::of::<T>(),
            AnyState {
                data: Arc::new(RwLock::new(Box::new(data))),
            },
        );
    }

    pub fn get<T: Send + Sync + 'static>(&self) -> Option<Local<T>> {
        self.states.get(&TypeId::of::<T>()).map(|state| Local {
            inner: state.clone(),
            _phantom: std::marker::PhantomData,
        })
    }
}

impl<T: 'static> Local<T> {
    pub async fn read(&self) -> RwLockReadGuard<'_, T> {
        RwLockReadGuard::map(self.inner.data.read().await, |inner| {
            inner
                .downcast_ref::<T>()
                .expect("failed to downcast state (is the state of the right type?)")
        })
    }

    pub async fn write(&self) -> RwLockMappedWriteGuard<'_, T> {
        RwLockWriteGuard::map(self.inner.data.write().await, |inner| {
            inner
                .downcast_mut::<T>()
                .expect("failed to downcast state (is the state of the right type?)")
        })
    }
}

impl<T: Send + Sync + 'static, Ctx: CallableFetch<App> + Send + Sync, Init: Send + Sync>
    CallableParam<Ctx, Init> for Local<T>
{
    type Error = LocalStateError;

    async fn extract(ctx: &mut Ctx, _init: &Init) -> Result<Self, Self::Error> {
        let state = ctx
            .fetch()
            .inner
            .states
            .get::<T>()
            .ok_or_else(|| LocalStateError::NotFound(std::any::type_name::<T>().to_string()))?;

        Ok(state)
    }
}

impl<T> SupportsAsync for Local<T> {}