2019-11-14 18:38:24 +06:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
|
|
|
use std::task::{Context, Poll};
|
2018-12-10 21:06:54 -08:00
|
|
|
|
2021-11-01 23:36:51 +00:00
|
|
|
use crate::server::ServerHandle;
|
2018-12-10 21:06:54 -08:00
|
|
|
|
2021-10-11 05:14:34 +01:00
|
|
|
/// Types of process signals.
|
2019-12-06 14:06:14 +06:00
|
|
|
#[allow(dead_code)]
|
2018-12-10 21:06:54 -08:00
|
|
|
#[derive(PartialEq, Clone, Copy, Debug)]
|
|
|
|
pub(crate) enum Signal {
|
2021-10-11 05:14:34 +01:00
|
|
|
/// `SIGINT`
|
2018-12-10 21:06:54 -08:00
|
|
|
Int,
|
2021-10-11 05:14:34 +01:00
|
|
|
|
|
|
|
/// `SIGTERM`
|
2018-12-10 21:06:54 -08:00
|
|
|
Term,
|
2021-10-11 05:14:34 +01:00
|
|
|
|
|
|
|
/// `SIGQUIT`
|
2018-12-10 21:06:54 -08:00
|
|
|
Quit,
|
|
|
|
}
|
|
|
|
|
2021-10-11 05:14:34 +01:00
|
|
|
/// Process signal listener.
|
2018-12-10 21:06:54 -08:00
|
|
|
pub(crate) struct Signals {
|
2021-11-01 23:36:51 +00:00
|
|
|
srv: ServerHandle,
|
2021-10-11 05:14:34 +01:00
|
|
|
|
2018-12-10 21:06:54 -08:00
|
|
|
#[cfg(not(unix))]
|
2021-02-05 19:38:11 -08:00
|
|
|
signals: futures_core::future::LocalBoxFuture<'static, std::io::Result<()>>,
|
2021-10-11 05:14:34 +01:00
|
|
|
|
2018-12-10 21:06:54 -08:00
|
|
|
#[cfg(unix)]
|
2021-02-05 19:38:11 -08:00
|
|
|
signals: Vec<(Signal, actix_rt::signal::unix::Signal)>,
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Signals {
|
2021-10-11 05:14:34 +01:00
|
|
|
/// Spawns a signal listening future that is able to send commands to the `Server`.
|
2021-11-01 23:36:51 +00:00
|
|
|
pub(crate) fn start(srv: ServerHandle) {
|
2020-12-29 07:44:53 +08:00
|
|
|
#[cfg(not(unix))]
|
|
|
|
{
|
|
|
|
actix_rt::spawn(Signals {
|
|
|
|
srv,
|
|
|
|
signals: Box::pin(actix_rt::signal::ctrl_c()),
|
|
|
|
});
|
|
|
|
}
|
2021-10-11 05:14:34 +01:00
|
|
|
|
2020-12-29 07:44:53 +08:00
|
|
|
#[cfg(unix)]
|
|
|
|
{
|
|
|
|
use actix_rt::signal::unix;
|
2019-11-14 18:38:24 +06:00
|
|
|
|
2020-12-29 07:44:53 +08:00
|
|
|
let sig_map = [
|
|
|
|
(unix::SignalKind::interrupt(), Signal::Int),
|
|
|
|
(unix::SignalKind::terminate(), Signal::Term),
|
|
|
|
(unix::SignalKind::quit(), Signal::Quit),
|
|
|
|
];
|
2019-11-14 18:38:24 +06:00
|
|
|
|
2021-02-05 19:38:11 -08:00
|
|
|
let signals = sig_map
|
|
|
|
.iter()
|
|
|
|
.filter_map(|(kind, sig)| {
|
|
|
|
unix::signal(*kind)
|
|
|
|
.map(|tokio_sig| (*sig, tokio_sig))
|
|
|
|
.map_err(|e| {
|
|
|
|
log::error!(
|
|
|
|
"Can not initialize stream handler for {:?} err: {}",
|
|
|
|
sig,
|
|
|
|
e
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.ok()
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
2020-12-29 07:44:53 +08:00
|
|
|
|
|
|
|
actix_rt::spawn(Signals { srv, signals });
|
|
|
|
}
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Future for Signals {
|
2019-11-14 18:38:24 +06:00
|
|
|
type Output = ();
|
|
|
|
|
2019-11-26 17:03:52 +06:00
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
2018-12-10 21:06:54 -08:00
|
|
|
#[cfg(not(unix))]
|
2020-12-29 07:44:53 +08:00
|
|
|
match self.signals.as_mut().poll(cx) {
|
2019-12-06 14:06:14 +06:00
|
|
|
Poll::Ready(_) => {
|
|
|
|
self.srv.signal(Signal::Int);
|
|
|
|
Poll::Ready(())
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
2020-12-29 07:44:53 +08:00
|
|
|
Poll::Pending => Poll::Pending,
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
2021-10-11 05:14:34 +01:00
|
|
|
|
2018-12-10 21:06:54 -08:00
|
|
|
#[cfg(unix)]
|
|
|
|
{
|
2020-12-29 07:44:53 +08:00
|
|
|
for (sig, fut) in self.signals.iter_mut() {
|
2021-02-05 19:38:11 -08:00
|
|
|
if Pin::new(fut).poll_recv(cx).is_ready() {
|
2020-12-29 07:44:53 +08:00
|
|
|
let sig = *sig;
|
|
|
|
self.srv.signal(sig);
|
|
|
|
return Poll::Ready(());
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
|
|
|
}
|
2019-11-26 17:03:52 +06:00
|
|
|
Poll::Pending
|
2018-12-10 21:06:54 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|