use actix::{Actor, Addr, ArbiterHandle, Context, Supervisor, System, SystemService};
use once_cell::sync::Lazy;
use parking_lot::Mutex;
use std::any::{Any, TypeId};
use std::collections::HashMap;
static SREG: Lazy<Mutex<HashMap<usize, PatchedSystemRegistry>>> =
Lazy::new(|| Mutex::new(HashMap::new()));
#[derive(Debug)]
struct PatchedSystemRegistry {
#[allow(dead_code)]
system: ArbiterHandle,
registry: HashMap<TypeId, Box<dyn Any + Send>>,
}
impl PatchedSystemRegistry {
pub(crate) fn new(system: ArbiterHandle) -> Self {
Self {
system,
registry: HashMap::default(),
}
}
}
pub trait CustomSystemService: Actor<Context = Context<Self>> + SystemService {
fn start_service_with(
f: impl Fn() -> Self + std::marker::Sync + 'static + std::marker::Send,
) -> Addr<Self> {
let sys = System::current();
let arbiter = sys.arbiter();
let addr = Supervisor::start_in_arbiter(arbiter, move |ctx| {
let mut act = f();
act.custom_service_started(ctx);
act
});
Self::add_to_registry(addr)
}
#[allow(dead_code, unused_variables)]
fn custom_service_started(&mut self, ctx: &mut Context<Self>) {}
fn add_to_registry(addr: Addr<Self>) -> Addr<Self> {
let sys = System::current();
let mut sreg = SREG.lock();
let reg = sreg
.entry(sys.id())
.or_insert_with(|| PatchedSystemRegistry::new(sys.arbiter().clone()));
reg.registry
.insert(TypeId::of::<Self>(), Box::new(addr.clone()));
addr
}
fn from_custom_registry() -> Addr<Self> {
let sys = System::current();
let mut sreg = SREG.lock();
let reg = sreg
.entry(sys.id())
.or_insert_with(|| PatchedSystemRegistry::new(sys.arbiter().clone()));
if let Some(addr) = reg.registry.get(&TypeId::of::<Self>())
&& let Some(addr) = addr.downcast_ref::<Addr<Self>>()
{
return addr.clone();
}
panic!("Please start Actor before asking for it in registry!");
}
}