implement server state/signal handling, depend on tokio-signal

This commit is contained in:
Dietmar Maurer 2019-04-08 12:21:29 +02:00
parent 5a12c0e2fb
commit 7a630df709
5 changed files with 166 additions and 3 deletions

View File

@ -21,8 +21,9 @@ futures = "0.1"
tokio-threadpool = "0.1" tokio-threadpool = "0.1"
tokio = "0.1" tokio = "0.1"
tokio-fs = "0.1" tokio-fs = "0.1"
tokio-tls = "0.2.1" tokio-tls = "0.2"
native-tls = "0.2.2" tokio-signal = "0.2"
native-tls = "0.2"
http = "0.1" http = "0.1"
hyper = "0.12" hyper = "0.12"
hyper-tls = "0.3" hyper-tls = "0.3"

View File

@ -7,9 +7,14 @@
mod environment; mod environment;
pub use environment::*; pub use environment::*;
mod state;
pub use state::*;
mod worker_task; mod worker_task;
pub use worker_task::*; pub use worker_task::*;
pub mod formatter; pub mod formatter;
#[macro_use] #[macro_use]
pub mod rest; pub mod rest;

151
src/server/state.rs Normal file
View File

@ -0,0 +1,151 @@
use failure::*;
use lazy_static::lazy_static;
use std::sync::Mutex;
use futures::*;
use futures::stream::Stream;
use tokio::sync::oneshot;
use tokio_signal::unix::{Signal, SIGHUP, SIGINT};
use crate::tools;
#[derive(PartialEq, Copy, Clone, Debug)]
pub enum ServerMode {
Normal,
Shutdown,
}
pub struct ServerState {
pub mode: ServerMode,
pub shutdown_listeners: Vec<oneshot::Sender<()>>,
pub last_worker_listeners: Vec<oneshot::Sender<()>>,
pub worker_count: usize,
pub reload_request: bool,
}
lazy_static! {
static ref SERVER_STATE: Mutex<ServerState> = Mutex::new(ServerState {
mode: ServerMode::Normal,
shutdown_listeners: vec![],
last_worker_listeners: vec![],
worker_count: 0,
reload_request: false,
});
}
pub fn server_state_init() -> Result<(), Error> {
let stream = Signal::new(SIGINT).flatten_stream();
let future = stream.for_each(|_| {
println!("got shutdown request (SIGINT)");
SERVER_STATE.lock().unwrap().reload_request = false;
tools::request_shutdown();
Ok(())
}).map_err(|_| {});
let abort_future = last_worker_future().map_err(|_| {});
let task = future.select(abort_future);
tokio::spawn(task.map(|_| {}).map_err(|_| {}));
let stream = Signal::new(SIGHUP).flatten_stream();
let future = stream.for_each(|_| {
println!("got reload request (SIGHUP)");
SERVER_STATE.lock().unwrap().reload_request = true;
tools::request_shutdown();
Ok(())
}).map_err(|_| {});
let abort_future = last_worker_future().map_err(|_| {});
let task = future.select(abort_future);
tokio::spawn(task.map(|_| {}).map_err(|_| {}));
Ok(())
}
pub fn is_reload_request() -> bool {
let data = SERVER_STATE.lock().unwrap();
if data.mode == ServerMode::Shutdown && data.reload_request {
true
} else {
false
}
}
pub fn server_shutdown() {
let mut data = SERVER_STATE.lock().unwrap();
println!("SET SHUTDOWN MODE");
data.mode = ServerMode::Shutdown;
notify_listeners(&mut data.shutdown_listeners);
drop(data); // unlock
check_last_worker();
}
pub fn shutdown_future() -> oneshot::Receiver<()> {
let (tx, rx) = oneshot::channel::<()>();
let mut data = SERVER_STATE.lock().unwrap();
match data.mode {
ServerMode::Normal => { data.shutdown_listeners.push(tx); },
ServerMode::Shutdown => { let _ = tx.send(()); },
}
rx
}
pub fn last_worker_future() -> oneshot::Receiver<()> {
let (tx, rx) = oneshot::channel::<()>();
let mut data = SERVER_STATE.lock().unwrap();
if data.mode == ServerMode::Shutdown && data.worker_count == 0 {
let _ = tx.send(());
} else {
data.last_worker_listeners.push(tx);
}
rx
}
pub fn set_worker_count(count: usize) {
let mut data = SERVER_STATE.lock().unwrap();
data.worker_count = count;
if !(data.mode == ServerMode::Shutdown && data.worker_count == 0) { return; }
notify_listeners(&mut data.last_worker_listeners);
}
pub fn check_last_worker() {
let mut data = SERVER_STATE.lock().unwrap();
if !(data.mode == ServerMode::Shutdown && data.worker_count == 0) { return; }
notify_listeners(&mut data.last_worker_listeners);
}
fn notify_listeners(list: &mut Vec<oneshot::Sender<()>>) {
loop {
match list.pop() {
None => { break; },
Some(ch) => {
println!("SEND ABORT");
if let Err(_) = ch.send(()) {
eprintln!("SEND ABORT failed");
}
},
}
}
}

View File

@ -410,7 +410,10 @@ impl WorkerTask {
}), }),
}); });
WORKER_TASK_LIST.lock().unwrap().insert(task_id, worker.clone()); let mut hash = WORKER_TASK_LIST.lock().unwrap();
hash.insert(task_id, worker.clone());
super::set_worker_count(hash.len());
Ok(worker) Ok(worker)
} }
@ -434,6 +437,7 @@ impl WorkerTask {
WORKER_TASK_LIST.lock().unwrap().remove(&task_id); WORKER_TASK_LIST.lock().unwrap().remove(&task_id);
worker.log_result(result); worker.log_result(result);
let _ = update_active_workers(None); let _ = update_active_workers(None);
super::set_worker_count(WORKER_TASK_LIST.lock().unwrap().len());
Ok(()) Ok(())
})); }));
@ -464,6 +468,7 @@ impl WorkerTask {
worker.log_result(result); worker.log_result(result);
let _ = update_active_workers(None); let _ = update_active_workers(None);
p.send(()).unwrap(); p.send(()).unwrap();
super::set_worker_count(WORKER_TASK_LIST.lock().unwrap().len());
}); });
tokio::spawn(c.then(|_| Ok(()))); tokio::spawn(c.then(|_| Ok(())));

View File

@ -627,6 +627,7 @@ static mut SHUTDOWN_REQUESTED: bool = false;
pub fn request_shutdown() { pub fn request_shutdown() {
unsafe { SHUTDOWN_REQUESTED = true; } unsafe { SHUTDOWN_REQUESTED = true; }
crate::server::server_shutdown();
} }
#[inline(always)] #[inline(always)]