move services ctor/dtor detail into service
Signed-off-by: Jason Volk <jason@zemos.net>
This commit is contained in:
parent
427aa4645c
commit
e4aa20ebeb
|
@ -6,14 +6,12 @@ use tracing::{debug, error, info};
|
||||||
|
|
||||||
extern crate conduit_admin as admin;
|
extern crate conduit_admin as admin;
|
||||||
extern crate conduit_core as conduit;
|
extern crate conduit_core as conduit;
|
||||||
extern crate conduit_database as database;
|
|
||||||
extern crate conduit_service as service;
|
extern crate conduit_service as service;
|
||||||
|
|
||||||
use std::sync::atomic::Ordering;
|
use std::sync::atomic::Ordering;
|
||||||
|
|
||||||
use conduit::{debug_info, trace, Error, Result, Server};
|
use conduit::{debug_info, trace, Error, Result, Server};
|
||||||
use database::KeyValueDatabase;
|
use service::services;
|
||||||
use service::{services, Services};
|
|
||||||
|
|
||||||
use crate::{layers, serve};
|
use crate::{layers, serve};
|
||||||
|
|
||||||
|
@ -58,15 +56,10 @@ pub(crate) async fn run(server: Arc<Server>) -> Result<(), Error> {
|
||||||
|
|
||||||
/// Async initializations
|
/// Async initializations
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
#[allow(clippy::let_underscore_must_use)]
|
|
||||||
pub(crate) async fn start(server: Arc<Server>) -> Result<(), Error> {
|
pub(crate) async fn start(server: Arc<Server>) -> Result<(), Error> {
|
||||||
debug!("Starting...");
|
debug!("Starting...");
|
||||||
let d = Arc::new(KeyValueDatabase::load_or_create(&server).await?);
|
|
||||||
let s = Box::new(Services::build(server, d.clone()).await?);
|
service::init(&server).await?;
|
||||||
_ = service::SERVICES
|
|
||||||
.write()
|
|
||||||
.expect("write locked")
|
|
||||||
.insert(Box::leak(s));
|
|
||||||
services().start().await?;
|
services().start().await?;
|
||||||
|
|
||||||
#[cfg(feature = "systemd")]
|
#[cfg(feature = "systemd")]
|
||||||
|
@ -84,23 +77,10 @@ pub(crate) async fn stop(_server: Arc<Server>) -> Result<(), Error> {
|
||||||
// Wait for all completions before dropping or we'll lose them to the module
|
// Wait for all completions before dropping or we'll lose them to the module
|
||||||
// unload and explode.
|
// unload and explode.
|
||||||
services().shutdown().await;
|
services().shutdown().await;
|
||||||
|
|
||||||
// Deactivate services(). Any further use will panic the caller.
|
// Deactivate services(). Any further use will panic the caller.
|
||||||
let s = service::SERVICES
|
service::fini();
|
||||||
.write()
|
|
||||||
.expect("write locked")
|
|
||||||
.take()
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
let s: *mut Services = std::ptr::from_ref(s).cast_mut();
|
|
||||||
//SAFETY: Services was instantiated in start() and leaked into the SERVICES
|
|
||||||
// global perusing as 'static for the duration of run_server(). Now we reclaim
|
|
||||||
// it to drop it before unloading the module. If this is not done there will be
|
|
||||||
// multiple instances after module reload.
|
|
||||||
let s = unsafe { Box::from_raw(s) };
|
|
||||||
debug!("Cleaning up...");
|
debug!("Cleaning up...");
|
||||||
// Drop it so we encounter any trouble before the infolog message
|
|
||||||
drop(s);
|
|
||||||
|
|
||||||
#[cfg(feature = "systemd")]
|
#[cfg(feature = "systemd")]
|
||||||
sd_notify::notify(true, &[sd_notify::NotifyState::Stopping]).expect("failed to notify systemd of stopping state");
|
sd_notify::notify(true, &[sd_notify::NotifyState::Stopping]).expect("failed to notify systemd of stopping state");
|
||||||
|
|
|
@ -17,9 +17,9 @@ pub mod users;
|
||||||
|
|
||||||
extern crate conduit_core as conduit;
|
extern crate conduit_core as conduit;
|
||||||
extern crate conduit_database as database;
|
extern crate conduit_database as database;
|
||||||
use std::sync::RwLock;
|
use std::sync::{Arc, RwLock};
|
||||||
|
|
||||||
pub(crate) use conduit::{config, debug_error, debug_info, debug_warn, utils, Config, Error, PduCount, Result};
|
pub(crate) use conduit::{config, debug_error, debug_info, debug_warn, utils, Config, Error, PduCount, Result, Server};
|
||||||
pub(crate) use database::KeyValueDatabase;
|
pub(crate) use database::KeyValueDatabase;
|
||||||
pub use globals::{server_is_ours, user_is_local};
|
pub use globals::{server_is_ours, user_is_local};
|
||||||
pub use pdu::PduEvent;
|
pub use pdu::PduEvent;
|
||||||
|
@ -30,7 +30,35 @@ pub(crate) use crate as service;
|
||||||
conduit::mod_ctor! {}
|
conduit::mod_ctor! {}
|
||||||
conduit::mod_dtor! {}
|
conduit::mod_dtor! {}
|
||||||
|
|
||||||
pub static SERVICES: RwLock<Option<&'static Services>> = RwLock::new(None);
|
static SERVICES: RwLock<Option<&Services>> = RwLock::new(None);
|
||||||
|
|
||||||
|
#[allow(clippy::let_underscore_must_use)]
|
||||||
|
pub async fn init(server: &Arc<Server>) -> Result<()> {
|
||||||
|
let d = Arc::new(KeyValueDatabase::load_or_create(server).await?);
|
||||||
|
let s = Box::new(Services::build(server.clone(), d.clone()).await?);
|
||||||
|
_ = SERVICES.write().expect("write locked").insert(Box::leak(s));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fini() {
|
||||||
|
// Deactivate services(). Any further use will panic the caller.
|
||||||
|
let s = SERVICES
|
||||||
|
.write()
|
||||||
|
.expect("write locked")
|
||||||
|
.take()
|
||||||
|
.expect("services initialized");
|
||||||
|
|
||||||
|
let s: *mut Services = std::ptr::from_ref(s).cast_mut();
|
||||||
|
//SAFETY: Services was instantiated in init() and leaked into the SERVICES
|
||||||
|
// global perusing as 'static for the duration of service. Now we reclaim
|
||||||
|
// it to drop it before unloading the module. If this is not done there wil
|
||||||
|
// be multiple instances after module reload.
|
||||||
|
let s = unsafe { Box::from_raw(s) };
|
||||||
|
|
||||||
|
// Drop it so we encounter any trouble before the infolog message
|
||||||
|
drop(s);
|
||||||
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn services() -> &'static Services {
|
pub fn services() -> &'static Services {
|
||||||
|
|
Loading…
Reference in New Issue