feat: solely handle single terminating signal for now
This commit is contained in:
parent
0faa6a8578
commit
50cdd3115f
6 changed files with 107 additions and 19 deletions
14
src/main.rs
14
src/main.rs
|
|
@ -82,7 +82,7 @@ fn backups_thread(counter: Arc<Mutex<server::ServerProcess>>, frequency: u64) {
|
|||
}
|
||||
|
||||
fn main() -> io::Result<()> {
|
||||
let (term, mut signals) = signals::install_signal_handlers()?;
|
||||
let (_, mut signals) = signals::install_signal_handlers()?;
|
||||
let cli = Cli::parse();
|
||||
|
||||
let cmd = server::ServerCommand::new(cli.type_, &cli.server_version)
|
||||
|
|
@ -94,17 +94,17 @@ fn main() -> io::Result<()> {
|
|||
.xms(cli.xms)
|
||||
.xmx(cli.xmx)
|
||||
.max_backups(cli.max_backups);
|
||||
let server = Arc::new(cmd.spawn()?);
|
||||
let counter = Arc::new(Mutex::new(cmd.spawn()?));
|
||||
|
||||
if cli.frequency > 0 {
|
||||
let clone = Arc::clone(&server);
|
||||
std::thread::spawn(move || backups_thread(server, cli.frequency));
|
||||
let clone = Arc::clone(&counter);
|
||||
std::thread::spawn(move || backups_thread(clone, cli.frequency));
|
||||
}
|
||||
|
||||
// Spawn thread that handles the main stdin loop
|
||||
let clone = Arc::clone(&server);
|
||||
std::thread::spawn(move || stdin::handle_stdin(server));
|
||||
let clone = Arc::clone(&counter);
|
||||
std::thread::spawn(move || stdin::handle_stdin(clone));
|
||||
|
||||
// Signal handler loop exits the process when necessary
|
||||
signals::handle_signals(&mut signals, term, server)
|
||||
signals::handle_signals(&mut signals, counter)
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
use std::io;
|
||||
use std::sync::atomic::AtomicBool;
|
||||
use std::sync::{Arc, Mutex};
|
||||
use std::sync::atomic::{AtomicBool, Ordering};
|
||||
|
||||
use signal_hook::consts::TERM_SIGNALS;
|
||||
use signal_hook::flag;
|
||||
|
|
@ -16,12 +16,15 @@ pub fn install_signal_handlers() -> io::Result<(Arc<AtomicBool>, SignalsInfo)> {
|
|||
// atomic bool. With this, the process will get killed immediately once it receives a second
|
||||
// termination signal (e.g. a double ctrl-c).
|
||||
// https://docs.rs/signal-hook/0.3.15/signal_hook/#a-complex-signal-handling-with-a-background-thread
|
||||
// for sig in TERM_SIGNALS {
|
||||
// // But this will "arm" the above for the second time, by setting it to true.
|
||||
// // The order of registering these is important, if you put this one first, it will
|
||||
// // first arm and then terminate ‒ all in the first round.
|
||||
// flag::register(*sig, Arc::clone(&term))?;
|
||||
// }
|
||||
for sig in TERM_SIGNALS {
|
||||
// When terminated by a second term signal, exit with exit code 1.
|
||||
// This will do nothing the first time (because term_now is false).
|
||||
flag::register_conditional_shutdown(*sig, 1, Arc::clone(&term))?;
|
||||
// But this will "arm" the above for the second time, by setting it to true.
|
||||
// The order of registering these is important, if you put this one first, it will
|
||||
// first arm and then terminate ‒ all in the first round.
|
||||
flag::register(*sig, Arc::clone(&term))?;
|
||||
}
|
||||
|
||||
let signals = TERM_SIGNALS;
|
||||
|
||||
|
|
@ -29,7 +32,10 @@ pub fn install_signal_handlers() -> io::Result<(Arc<AtomicBool>, SignalsInfo)> {
|
|||
}
|
||||
|
||||
/// Loop that handles terminating signals as they come in.
|
||||
pub fn handle_signals(signals: &mut SignalsInfo, term: Arc<AtomicBool>, server: Arc<server::ServerProcess>) -> io::Result<()> {
|
||||
pub fn handle_signals(
|
||||
signals: &mut SignalsInfo,
|
||||
counter: Arc<Mutex<server::ServerProcess>>,
|
||||
) -> io::Result<()> {
|
||||
let mut force = false;
|
||||
|
||||
// We only register terminating signals, so we don't need to differentiate between what kind of
|
||||
|
|
@ -37,15 +43,20 @@ pub fn handle_signals(signals: &mut SignalsInfo, term: Arc<AtomicBool>, server:
|
|||
for _ in signals {
|
||||
// If term is already true, this is the second signal, meaning we kill the process
|
||||
// immediately.
|
||||
// This will currently not work, as the initial stop command will block the kill from
|
||||
// happening.
|
||||
if force {
|
||||
return server.kill()
|
||||
let mut server = counter.lock().unwrap();
|
||||
return server.kill();
|
||||
}
|
||||
// The stop command runs in a separate thread to avoid blocking the signal handling loop.
|
||||
// After stopping the server, the thread terminates the process.
|
||||
else {
|
||||
let clone = Arc::clone(&server);
|
||||
let clone = Arc::clone(&counter);
|
||||
|
||||
std::thread::spawn(move || {
|
||||
let _ = clone.stop();
|
||||
let mut server = clone.lock().unwrap();
|
||||
let _ = server.stop();
|
||||
std::process::exit(0);
|
||||
});
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,5 +1,5 @@
|
|||
use std::sync::{Arc, Mutex};
|
||||
use std::io;
|
||||
use std::sync::{Arc, Mutex};
|
||||
|
||||
use crate::server;
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue