1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
//! DI container functionality.
use crate::{traits::DITarget, types::DI};
use std::{any::Any, collections::HashMap, future::Future, sync::Mutex};
/// DI container holds component refs.
#[derive(Debug)]
pub struct DIContainer {
/// Hold components by its type name (FQTN).
components: Mutex<HashMap<String, DI<dyn Any + Send + Sync>>>,
}
impl DIContainer {
/// Create new instance.
pub fn new() -> DIContainer {
DIContainer {
components: Mutex::new(HashMap::new()),
}
}
/// Get a component by type.
pub fn get<T: DITarget>(&self) -> Option<DI<T>> {
self.components
.lock()
.unwrap()
.get(std::any::type_name::<T>())
.map(|c| c.clone().downcast::<T>().unwrap())
}
/// Get a component by type with a initialization.
/// If a target component does not exists, create and put into the container.
pub fn get_or_init<T, F>(&self, init: F) -> DI<T>
where
T: DITarget,
F: Fn() -> T,
{
if let Some(c) = self.get::<T>() {
c
} else {
let c = DI::new(init());
self.put(&c);
c
}
}
/// Get a component by type with a async initialization.
/// If a target component does not exists, create and put into the container.
pub async fn get_or_init_async<T, F, Fut>(&self, init: F) -> DI<T>
where
T: DITarget,
F: Fn() -> Fut,
Fut: Future<Output = T>,
{
if let Some(c) = self.get::<T>() {
c
} else {
let v = init().await;
let c = DI::new(v);
self.put(&c);
c
}
}
/// Put a component into the container.
pub fn put<T: DITarget>(&self, c: &DI<T>) {
self.components
.lock()
.unwrap()
.insert(std::any::type_name::<T>().into(), c.clone());
}
}