ohkami 0.24.9

A performant, declarative, and runtime-flexible web framework for Rust
Documentation
use crate::fang::SendSyncOnThreaded;
use ohkami_lib::map::TupleMap;
use std::any::{Any, TypeId};

#[cfg(feature = "__rt_threaded__")]
type StoreItem = Box<dyn Any + Send + Sync>;
#[cfg(not(feature = "__rt_threaded__"))]
type StoreItem = Box<dyn Any>;

pub struct Context {
    store: Option<Box<TupleMap<TypeId, StoreItem>>>,

    #[cfg(feature = "rt_worker")]
    worker: std::mem::MaybeUninit<(::worker::Context, ::worker::Env)>,

    #[cfg(feature = "rt_lambda")]
    lambda: Option<Box<crate::x_lambda::LambdaHTTPRequestContext>>,
}

impl Context {
    #[cfg(feature = "__rt__")]
    pub(super) const fn init() -> Self {
        Self {
            store: None,

            #[cfg(feature = "rt_worker")]
            worker: std::mem::MaybeUninit::uninit(),

            #[cfg(feature = "rt_lambda")]
            lambda: None,
        }
    }

    #[cfg(feature = "rt_worker")]
    pub(super) fn load(&mut self, worker: (::worker::Context, ::worker::Env)) {
        self.worker.write(worker);
    }

    #[cfg(feature = "rt_lambda")]
    pub(super) fn load(&mut self, request_context: crate::x_lambda::LambdaHTTPRequestContext) {
        self.lambda = Some(Box::new(request_context));
    }

    #[cfg(feature = "__rt_native__")]
    pub(super) fn clear(&mut self) {
        if let Some(map) = &mut self.store {
            map.clear()
        }
    }
}

impl Context {
    #[inline]
    pub fn set<Data: SendSyncOnThreaded + 'static>(&mut self, value: Data) {
        if self.store.is_none() {
            self.store = Some(Box::new(TupleMap::new()));
        }
        (unsafe { self.store.as_mut().unwrap_unchecked() })
            .insert(TypeId::of::<Data>(), Box::new(value));
    }

    #[inline]
    pub fn get<Data: SendSyncOnThreaded + 'static>(&self) -> Option<&Data> {
        self.store.as_ref().and_then(|map| {
            map.get(&TypeId::of::<Data>()).map(|boxed| {
                let data: &dyn Any = &**boxed;
                #[cfg(debug_assertions)]
                {
                    assert!(data.is::<Data>(), "Request store is poisoned!!!");
                }
                unsafe { &*(data as *const dyn Any as *const Data) }
            })
        })
    }

    #[cfg(feature = "rt_worker")]
    #[inline(always)]
    pub fn worker(&self) -> &::worker::Context {
        // SAFETY: User can touch here **ONLY AFTER `Self::load`** called by `Request`
        unsafe { &self.worker.assume_init_ref().0 }
    }
    #[cfg(feature = "rt_worker")]
    #[inline(always)]
    pub fn env(&self) -> &::worker::Env {
        // SAFETY: User can touch here **ONLY AFTER `Self::load`** called by `Request`
        unsafe { &self.worker.assume_init_ref().1 }
    }

    #[cfg(feature = "rt_lambda")]
    #[inline(always)]
    pub fn lambda(&self) -> &crate::x_lambda::LambdaHTTPRequestContext {
        // SAFETY: User can touch here **ONLY AFTER `Self::load`** called by `Request`
        unsafe { self.lambda.as_ref().unwrap_unchecked() }
    }
}