From 2d559b7fb08cef3dc2cf92b21043e4eb61573c65 Mon Sep 17 00:00:00 2001 From: Federico Terzi Date: Wed, 21 Jul 2021 20:23:37 +0200 Subject: [PATCH] feat(core): implement run-count config in worker --- espanso/src/cli/daemon/mod.rs | 62 +++++++++++++++++++--------- espanso/src/cli/worker/engine/mod.rs | 7 +++- espanso/src/cli/worker/mod.rs | 11 ++++- espanso/src/main.rs | 6 +++ 4 files changed, 65 insertions(+), 21 deletions(-) diff --git a/espanso/src/cli/daemon/mod.rs b/espanso/src/cli/daemon/mod.rs index 8cb2252..34dfc38 100644 --- a/espanso/src/cli/daemon/mod.rs +++ b/espanso/src/cli/daemon/mod.rs @@ -50,6 +50,7 @@ pub fn new() -> CliModule { #[allow(clippy::needless_update)] CliModule { requires_paths: true, + enable_logs: true, log_mode: super::LogMode::CleanAndAppend, subcommand: "daemon".to_string(), entry: daemon_main, @@ -102,7 +103,14 @@ fn daemon_main(args: CliModuleArgs) -> i32 { // TODO: register signals to terminate the worker if the daemon terminates - spawn_worker(&paths, &paths_overrides, exit_notify.clone()); + let mut worker_run_count = 0; + + spawn_worker( + &paths, + &paths_overrides, + exit_notify.clone(), + &mut worker_run_count, + ); ipc::initialize_and_spawn(&paths.runtime, exit_notify.clone()) .expect("unable to initialize ipc server for daemon"); @@ -151,7 +159,7 @@ fn daemon_main(args: CliModuleArgs) -> i32 { } if !has_timed_out { - spawn_worker(&paths, &paths_overrides, exit_notify.clone()); + spawn_worker(&paths, &paths_overrides, exit_notify.clone(), &mut worker_run_count); } else { error!("could not restart worker, as the exit process has timed out"); } @@ -166,7 +174,7 @@ fn daemon_main(args: CliModuleArgs) -> i32 { } WORKER_RESTART => { info!("worker requested a restart, spawning a new one..."); - spawn_worker(&paths, &paths_overrides, exit_notify.clone()); + spawn_worker(&paths, &paths_overrides, exit_notify.clone(), &mut worker_run_count); } _ => { error!("received unexpected exit code from worker {}, exiting", code); @@ -223,36 +231,52 @@ fn terminate_worker_if_already_running(runtime_dir: &Path) { ) } -fn spawn_worker(paths: &Paths, paths_overrides: &PathsOverrides, exit_notify: Sender) { +fn spawn_worker( + paths: &Paths, + paths_overrides: &PathsOverrides, + exit_notify: Sender, + worker_run_count: &mut i32, +) { info!("spawning the worker process..."); let espanso_exe_path = std::env::current_exe().expect("unable to obtain espanso executable location"); // Before starting the worker, check if the configuration has any error, and if so, show the troubleshooting GUI - let troubleshoot_guard = match troubleshoot::load_config_or_troubleshoot(paths, paths_overrides, true) { - Ok((_, troubleshoot_guard)) => troubleshoot_guard, - Err(fatal_err) => { - error!("critical configuration error detected, unable to restart worker: {:?}", fatal_err); + let troubleshoot_guard = + match troubleshoot::load_config_or_troubleshoot(paths, paths_overrides, true) { + Ok((_, troubleshoot_guard)) => troubleshoot_guard, + Err(fatal_err) => { + error!( + "critical configuration error detected, unable to restart worker: {:?}", + fatal_err + ); - // TODO: we should show a "Reload & Retry" button in the troubleshooter to retry - // loading the configuration and: - // - if the configuration is good -> start the worker - // - if the configuration is bad -> show the window again - // Until we have this logic, we choose the safest option and kill the daemon - // (otherwise, we would have a dangling daemon running after closing the troubleshooting). + // TODO: we should show a "Reload & Retry" button in the troubleshooter to retry + // loading the configuration and: + // - if the configuration is good -> start the worker + // - if the configuration is bad -> show the window again + // Until we have this logic, we choose the safest option and kill the daemon + // (otherwise, we would have a dangling daemon running after closing the troubleshooting). - unimplemented!(); - return; - }, - }; + unimplemented!(); + return; + } + }; let mut command = Command::new(&espanso_exe_path.to_string_lossy().to_string()); - command.args(&["worker", "--monitor-daemon"]); + command.args(&[ + "worker", + "--monitor-daemon", + "--run-count", + &format!("{}", worker_run_count), + ]); command.with_paths_overrides(paths_overrides); let mut child = command.spawn().expect("unable to spawn worker process"); + *worker_run_count += 1; + // Create a monitor thread that will exit with the same non-zero code if // the worker thread exits std::thread::Builder::new() diff --git a/espanso/src/cli/worker/engine/mod.rs b/espanso/src/cli/worker/engine/mod.rs index aebc779..f6696e0 100644 --- a/espanso/src/cli/worker/engine/mod.rs +++ b/espanso/src/cli/worker/engine/mod.rs @@ -72,6 +72,7 @@ pub fn initialize_and_spawn( ui_event_receiver: Receiver, secure_input_receiver: Receiver, use_evdev_backend: bool, + run_count: i32, ) -> Result> { let handle = std::thread::Builder::new() .name("engine thread".to_string()) @@ -212,7 +213,11 @@ pub fn initialize_and_spawn( } // TODO: check config - ui_remote.show_notification("Espanso is running!"); + match run_count { + 0 => ui_remote.show_notification("Espanso is running!"), + 1 => ui_remote.show_notification("Configuration reloaded! Espanso automatically loads new changes as soon as you save them."), + _ => {}, + } let mut engine = crate::engine::Engine::new(&funnel, &mut processor, &dispatcher); let exit_mode = engine.run(); diff --git a/espanso/src/cli/worker/mod.rs b/espanso/src/cli/worker/mod.rs index b411638..2364445 100644 --- a/espanso/src/cli/worker/mod.rs +++ b/espanso/src/cli/worker/mod.rs @@ -18,7 +18,7 @@ */ use crossbeam::channel::unbounded; -use log::{error, info}; +use log::{debug, error, info}; use crate::{ engine::event::ExitMode, @@ -59,6 +59,14 @@ fn worker_main(args: CliModuleArgs) -> i32 { let paths = args.paths.expect("missing paths in worker main"); let cli_args = args.cli_args.expect("missing cli_args in worker main"); + // This number is passed by the daemon and incremented at each worker + // restart. + let run_count = cli_args + .value_of("run-count") + .map(|val| val.parse::().unwrap_or(0)) + .unwrap_or(0); + debug!("starting with run-count = {:?}", cli_args); + // Avoid running multiple worker instances let lock_file = acquire_worker_lock(&paths.runtime); if lock_file.is_none() { @@ -120,6 +128,7 @@ fn worker_main(args: CliModuleArgs) -> i32 { engine_ui_event_receiver, engine_secure_input_receiver, use_evdev_backend, + run_count, ) .expect("unable to initialize engine"); diff --git a/espanso/src/main.rs b/espanso/src/main.rs index a3bb207..0f94232 100644 --- a/espanso/src/main.rs +++ b/espanso/src/main.rs @@ -331,6 +331,12 @@ fn main() { .subcommand( SubCommand::with_name("worker") .setting(AppSettings::Hidden) + .arg( + Arg::with_name("run-count") + .long("run-count") + .required(false) + .takes_value(true), + ) .arg( Arg::with_name("monitor-daemon") .long("monitor-daemon")