use std::net::SocketAddr;
use std::time::Duration;
use derive_more::Display;
use tokio::time::sleep;
use tracing::info;
#[derive(Copy, Clone, Debug, Display)]
pub struct Started {
pub socket_addr: SocketAddr,
}
#[derive(Copy, Clone, Debug, Display)]
pub enum Halted {
Normal,
}
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;
info!("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;
info!("remaining alive connections: {}", handle.connection_count());
}
}
pub async fn shutdown_signal_with_message(rx_halt: tokio::sync::oneshot::Receiver<Halted>, message: String) {
shutdown_signal(rx_halt).await;
info!("{message}");
}
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 => { info!("Halt signal processed: {}", signal) },
() = global_shutdown_signal() => { info!("Global shutdown signal processed") }
}
}
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 => {},
() = terminate => {}
}
}