use std::time::Duration;
use derive_more::Display;
use tokio::time::sleep;
use tracing::instrument;
#[derive(Copy, Clone, Debug, Display)]
pub enum Halted {
Normal,
}
#[instrument(skip())]
pub async fn global_shutdown_signal() {
let ctrl_c = async {
tokio::signal::ctrl_c().await.expect("failed to install Ctrl+C handler");
};
#[cfg(unix)]
let terminate = async {
tokio::signal::unix::signal(tokio::signal::unix::SignalKind::terminate())
.expect("failed to install signal handler")
.recv()
.await;
};
#[cfg(not(unix))]
let terminate = std::future::pending::<()>();
tokio::select! {
() = ctrl_c => {tracing::warn!("caught interrupt signal (ctrl-c), halting...");},
() = terminate => {tracing::warn!("caught interrupt signal (terminate), halting...");}
}
}
#[instrument(skip(rx_halt))]
pub async fn shutdown_signal(rx_halt: tokio::sync::oneshot::Receiver<Halted>) {
let halt = async {
match rx_halt.await {
Ok(signal) => signal,
Err(err) => panic!("Failed to install stop signal: {err}"),
}
};
tokio::select! {
signal = halt => { tracing::debug!("Halt signal processed: {}", signal) },
() = global_shutdown_signal() => { tracing::debug!("Global shutdown signal processed") }
}
}
#[instrument(skip(rx_halt))]
pub async fn shutdown_signal_with_message(rx_halt: tokio::sync::oneshot::Receiver<Halted>, message: String) {
shutdown_signal(rx_halt).await;
tracing::info!("{message}");
}
#[instrument(skip(handle, rx_halt, message))]
pub async fn graceful_shutdown(handle: axum_server::Handle, rx_halt: tokio::sync::oneshot::Receiver<Halted>, message: String) {
shutdown_signal_with_message(rx_halt, message).await;
tracing::debug!("Sending graceful shutdown signal");
handle.graceful_shutdown(Some(Duration::from_secs(90)));
println!("!! shuting down in 90 seconds !!");
loop {
sleep(Duration::from_secs(1)).await;
tracing::info!("remaining alive connections: {}", handle.connection_count());
}
}