feat(core): port 'edit' subcommand implementation from previous version
This commit is contained in:
parent
8f291f4717
commit
5a75a04d5a
230
espanso/src/cli/edit.rs
Normal file
230
espanso/src/cli/edit.rs
Normal file
|
@ -0,0 +1,230 @@
|
||||||
|
/*
|
||||||
|
* This file is part of espanso.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2019-2021 Federico Terzi
|
||||||
|
*
|
||||||
|
* espanso is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* espanso is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use std::process::Command;
|
||||||
|
|
||||||
|
use super::{CliModule, CliModuleArgs};
|
||||||
|
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
fn default_editor() -> String {
|
||||||
|
"/bin/nano".to_owned()
|
||||||
|
}
|
||||||
|
#[cfg(target_os = "macos")]
|
||||||
|
fn default_editor() -> String {
|
||||||
|
"/usr/bin/nano".to_owned()
|
||||||
|
}
|
||||||
|
#[cfg(target_os = "windows")]
|
||||||
|
fn default_editor() -> String {
|
||||||
|
"C:\\Windows\\System32\\notepad.exe".to_owned()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new() -> CliModule {
|
||||||
|
CliModule {
|
||||||
|
requires_paths: true,
|
||||||
|
subcommand: "edit".to_string(),
|
||||||
|
entry: edit_main,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn edit_main(args: CliModuleArgs) -> i32 {
|
||||||
|
let paths = args.paths.expect("missing paths argument");
|
||||||
|
let cli_args = args.cli_args.expect("missing cli_args");
|
||||||
|
|
||||||
|
if !paths.config.is_dir() {
|
||||||
|
panic!(
|
||||||
|
"config directory does not exist in path: {:?}",
|
||||||
|
paths.config
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine which is the file to edit
|
||||||
|
let target_file = cli_args.value_of("target_file");
|
||||||
|
let target_path = determine_target_path(&paths.config, target_file);
|
||||||
|
|
||||||
|
println!(
|
||||||
|
"Editing file: {}",
|
||||||
|
&target_path.to_string_lossy().to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
open_editor(&target_path);
|
||||||
|
|
||||||
|
// The previous version automatically reloaded the config after saving.
|
||||||
|
// Given that the new version automatically reloads config changes, this could be avoided.
|
||||||
|
// Nevertheless, this assumption might be wrong, so I'm keeping the necessary code.
|
||||||
|
// TODO: evaluate if reload is needed after v2 becomes stable
|
||||||
|
|
||||||
|
// // Based on the fact that the file already exists or not, we should detect in different
|
||||||
|
// // ways if a reload is needed
|
||||||
|
// let should_reload = if target_path.exists() {
|
||||||
|
// // Get the last modified date, so that we can detect if the user actually edits the file
|
||||||
|
// // before reloading
|
||||||
|
// let metadata = std::fs::metadata(&target_path).expect("cannot gather file metadata");
|
||||||
|
// let last_modified = metadata
|
||||||
|
// .modified()
|
||||||
|
// .expect("cannot read file last modified date");
|
||||||
|
|
||||||
|
// let result = open_editor(&target_path);
|
||||||
|
// if result {
|
||||||
|
// let new_metadata = std::fs::metadata(&target_path).expect("cannot gather file metadata");
|
||||||
|
// let new_last_modified = new_metadata
|
||||||
|
// .modified()
|
||||||
|
// .expect("cannot read file last modified date");
|
||||||
|
|
||||||
|
// if last_modified != new_last_modified {
|
||||||
|
// println!("File has been modified, reloading configuration");
|
||||||
|
// true
|
||||||
|
// } else {
|
||||||
|
// println!("File has not been modified, avoiding reload");
|
||||||
|
// false
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
// false
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
// let result = open_editor(&target_path);
|
||||||
|
// if result {
|
||||||
|
// // If the file has been created, we should reload the espanso config
|
||||||
|
// if target_path.exists() {
|
||||||
|
// println!("A new file has been created, reloading configuration");
|
||||||
|
// true
|
||||||
|
// } else {
|
||||||
|
// println!("No file has been created, avoiding reload");
|
||||||
|
// false
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
// false
|
||||||
|
// }
|
||||||
|
// };
|
||||||
|
|
||||||
|
// let no_restart: bool = if cli_args.is_present("norestart") {
|
||||||
|
// println!("Skipping automatic restart");
|
||||||
|
// true
|
||||||
|
// } else {
|
||||||
|
// false
|
||||||
|
// };
|
||||||
|
|
||||||
|
// if should_reload && !no_restart {
|
||||||
|
// // Check if the new configuration is valid
|
||||||
|
|
||||||
|
// if let Err(err) = crate::config::load_config(&paths.config, &paths.packages) {
|
||||||
|
// eprintln!("Unable to reload espanso due to a configuration error:");
|
||||||
|
// eprintln!("{:?}", err);
|
||||||
|
// return 1;
|
||||||
|
// };
|
||||||
|
|
||||||
|
// restart_espanso(&paths_overrides).expect("unable to restart espanso");
|
||||||
|
// }
|
||||||
|
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
|
fn determine_target_path(config_path: &Path, target_file: Option<&str>) -> PathBuf {
|
||||||
|
if let Some(target_file) = target_file {
|
||||||
|
match target_file {
|
||||||
|
"default" => {
|
||||||
|
if espanso_config::is_legacy_config(config_path) {
|
||||||
|
config_path.join("default.yml")
|
||||||
|
} else {
|
||||||
|
config_path.join("config").join("default.yml")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"base" => {
|
||||||
|
if espanso_config::is_legacy_config(config_path) {
|
||||||
|
panic!("'base' alias cannot be used in compatibility mode, please migrate your configuration by running 'espanso migrate'")
|
||||||
|
} else {
|
||||||
|
config_path.join("match").join("base.yml")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
custom => {
|
||||||
|
if !custom.ends_with(".yml") && !custom.ends_with(".yaml") {
|
||||||
|
if espanso_config::is_legacy_config(config_path) {
|
||||||
|
config_path.join("user").join(format!("{}.yml", custom))
|
||||||
|
} else {
|
||||||
|
config_path.join("match").join(format!("{}.yml", custom))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
config_path.join(custom)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if espanso_config::is_legacy_config(config_path) {
|
||||||
|
config_path.join("default.yml")
|
||||||
|
} else {
|
||||||
|
config_path.join("match").join("base.yml")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn open_editor(file_path: &Path) -> bool {
|
||||||
|
// Check if another editor is defined in the environment variables
|
||||||
|
let editor_var = std::env::var_os("EDITOR");
|
||||||
|
let visual_var = std::env::var_os("VISUAL");
|
||||||
|
|
||||||
|
// Prioritize the editors specified by the environment variable, use the default one
|
||||||
|
let editor: String = if let Some(editor_var) = editor_var {
|
||||||
|
editor_var.to_string_lossy().to_string()
|
||||||
|
} else if let Some(visual_var) = visual_var {
|
||||||
|
visual_var.to_string_lossy().to_string()
|
||||||
|
} else {
|
||||||
|
default_editor()
|
||||||
|
};
|
||||||
|
|
||||||
|
// Start the editor and wait for its termination
|
||||||
|
let status = if cfg!(target_os = "windows") {
|
||||||
|
Command::new(&editor).arg(file_path).spawn()
|
||||||
|
} else {
|
||||||
|
// On Unix, spawn the editor using the shell so that it can
|
||||||
|
// accept parameters. See issue #245
|
||||||
|
Command::new("/bin/bash")
|
||||||
|
.arg("-c")
|
||||||
|
.arg(format!("{} '{}'", editor, file_path.to_string_lossy()))
|
||||||
|
.spawn()
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Ok(mut child) = status {
|
||||||
|
// Wait for the user to edit the configuration
|
||||||
|
let result = child.wait();
|
||||||
|
|
||||||
|
if let Ok(exit_status) = result {
|
||||||
|
exit_status.success()
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
println!("Error: could not start editor at: {}", &editor);
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// fn restart_espanso(paths_overrides: &PathsOverrides) -> Result<()> {
|
||||||
|
// let espanso_exe_path = std::env::current_exe()?;
|
||||||
|
|
||||||
|
// let mut command = Command::new(&espanso_exe_path.to_string_lossy().to_string());
|
||||||
|
// command.args(&["restart"]);
|
||||||
|
// command.with_paths_overrides(paths_overrides);
|
||||||
|
|
||||||
|
// let output = command.output()?;
|
||||||
|
|
||||||
|
// if output.status.success() {
|
||||||
|
// Ok(())
|
||||||
|
// } else {
|
||||||
|
// bail!("restart command returned a non-zero exit code");
|
||||||
|
// }
|
||||||
|
// }
|
|
@ -24,6 +24,7 @@ use espanso_config::{config::ConfigStore, error::NonFatalErrorSet, matches::stor
|
||||||
use espanso_path::Paths;
|
use espanso_path::Paths;
|
||||||
|
|
||||||
pub mod daemon;
|
pub mod daemon;
|
||||||
|
pub mod edit;
|
||||||
pub mod env_path;
|
pub mod env_path;
|
||||||
pub mod launcher;
|
pub mod launcher;
|
||||||
pub mod log;
|
pub mod log;
|
||||||
|
|
|
@ -60,6 +60,7 @@ const LOG_FILE_NAME: &str = "espanso.log";
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
static ref CLI_HANDLERS: Vec<CliModule> = vec![
|
static ref CLI_HANDLERS: Vec<CliModule> = vec![
|
||||||
cli::path::new(),
|
cli::path::new(),
|
||||||
|
cli::edit::new(),
|
||||||
cli::launcher::new(),
|
cli::launcher::new(),
|
||||||
cli::log::new(),
|
cli::log::new(),
|
||||||
cli::worker::new(),
|
cli::worker::new(),
|
||||||
|
@ -215,17 +216,20 @@ fn main() {
|
||||||
// .subcommand(SubCommand::with_name("toggle")
|
// .subcommand(SubCommand::with_name("toggle")
|
||||||
// .about("Toggle the status of the espanso replacement engine."))
|
// .about("Toggle the status of the espanso replacement engine."))
|
||||||
// )
|
// )
|
||||||
// .subcommand(SubCommand::with_name("edit")
|
.subcommand(SubCommand::with_name("edit")
|
||||||
// .about("Open the default text editor to edit config files and reload them automatically when exiting")
|
.about("Shortcut to open the default text editor to edit config files")
|
||||||
// .arg(Arg::with_name("config")
|
.arg(Arg::with_name("target_file")
|
||||||
// .help("Defaults to \"default\". The configuration file name to edit (without the .yml extension)."))
|
.help(r#"Defaults to "match/base.yml", it contains the relative path of the file you want to edit,
|
||||||
// .arg(Arg::with_name("norestart")
|
such as 'config/default.yml' or 'match/base.yml'.
|
||||||
// .short("n")
|
For convenience, you can also specify the name directly and Espanso will figure out the path.
|
||||||
// .long("norestart")
|
For example, specifying 'email' is equivalent to 'match/email.yml'."#))
|
||||||
// .required(false)
|
// .arg(Arg::with_name("norestart")
|
||||||
// .takes_value(false)
|
// .short("n")
|
||||||
// .help("Avoid restarting espanso after editing the file"))
|
// .long("norestart")
|
||||||
// )
|
// .required(false)
|
||||||
|
// .takes_value(false)
|
||||||
|
// .help("Avoid restarting espanso after editing the file"))
|
||||||
|
)
|
||||||
// .subcommand(SubCommand::with_name("detect")
|
// .subcommand(SubCommand::with_name("detect")
|
||||||
// .about("Tool to detect current window properties, to simplify filters creation."))
|
// .about("Tool to detect current window properties, to simplify filters creation."))
|
||||||
.subcommand(
|
.subcommand(
|
||||||
|
|
Loading…
Reference in New Issue
Block a user