use std::sync::Arc;
use std::sync::atomic::{AtomicBool, Ordering};
use tokio::signal;
pub struct ShutdownSignal {
shutdown: Arc<AtomicBool>,
}
impl Default for ShutdownSignal {
fn default() -> Self {
Self::new()
}
}
impl ShutdownSignal {
pub fn new() -> Self {
Self {
shutdown: Arc::new(AtomicBool::new(false)),
}
}
pub fn is_shutdown(&self) -> bool {
self.shutdown.load(Ordering::Relaxed)
}
pub async fn wait_for_signal(&self) {
let shutdown_clone = self.shutdown.clone();
tokio::spawn(async move {
let ctrl_c = async {
signal::ctrl_c()
.await
.expect("failed to install Ctrl+C handler");
};
#[cfg(unix)]
let terminate = async {
signal::unix::signal(signal::unix::SignalKind::terminate())
.expect("failed to install signal handler")
.recv()
.await;
};
#[cfg(not(unix))]
let terminate = std::future::pending::<()>();
tokio::select! {
_ = ctrl_c => {},
_ = terminate => {},
}
shutdown_clone.store(true, Ordering::Relaxed);
});
}
}