diff --git a/monolake-core/src/server/mod.rs b/monolake-core/src/server/mod.rs index f1ae38c..98ea3d7 100644 --- a/monolake-core/src/server/mod.rs +++ b/monolake-core/src/server/mod.rs @@ -157,6 +157,7 @@ impl Default for WorkerController { pub struct SiteHandler { handler_slot: HandlerSlot, + two_stage_handler_slot: UnsafeCell>, _stop: OReceiver<()>, } @@ -166,6 +167,7 @@ impl SiteHandler { ( Self { handler_slot, + two_stage_handler_slot: UnsafeCell::new(None), _stop: rx, }, tx, @@ -210,6 +212,9 @@ pub enum Command { Add(Arc, F, LF), Update(Arc, F), Remove(Arc), + TwoStageCreate(Arc, F), + TwoStageApply(Arc), + TwoStageAbort(Arc), } pub struct Update { @@ -254,6 +259,38 @@ where None => bail_into!("site {name} not exist"), } } + Command::TwoStageCreate(name, factory) => { + let sites = unsafe { &mut *controller.sites.get() }; + let Some(sh) = sites.get(&name) else { + bail_into!("site {name} not exist"); + }; + + let current_svc = sh.handler_slot.clone(); + let svc = factory + .make_via_ref(Some(¤t_svc.get_svc())) + .map_err(|e| anyhow!("build service fail {e:?}"))?; + unsafe { *sh.two_stage_handler_slot.get() = Some(svc) }; + Ok(()) + } + Command::TwoStageApply(name) => { + let sites = unsafe { &mut *controller.sites.get() }; + let Some(sh) = sites.get(&name) else { + bail_into!("site {name} not exist"); + }; + + let svc = unsafe { (*sh.two_stage_handler_slot.get()).take() } + .ok_or_else(|| anyhow!("two stage service not exist"))?; + sh.handler_slot.update_svc(Rc::new(svc)); + Ok(()) + } + Command::TwoStageAbort(name) => { + let sites = unsafe { &mut *controller.sites.get() }; + let Some(sh) = sites.get(&name) else { + bail_into!("site {name} not exist"); + }; + unsafe { (*sh.two_stage_handler_slot.get()) = None }; + Ok(()) + } Command::Add(name, factory, listener_factory) => { // TODO: make sure the named service has not been started let listener = match listener_factory.make() {