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; | ||||
| 
 | ||||
| pub mod daemon; | ||||
| pub mod edit; | ||||
| pub mod env_path; | ||||
| pub mod launcher; | ||||
| pub mod log; | ||||
|  |  | |||
|  | @ -60,6 +60,7 @@ const LOG_FILE_NAME: &str = "espanso.log"; | |||
| lazy_static! { | ||||
|   static ref CLI_HANDLERS: Vec<CliModule> = vec![ | ||||
|     cli::path::new(), | ||||
|     cli::edit::new(), | ||||
|     cli::launcher::new(), | ||||
|     cli::log::new(), | ||||
|     cli::worker::new(), | ||||
|  | @ -215,17 +216,20 @@ fn main() { | |||
|     //     .subcommand(SubCommand::with_name("toggle")
 | ||||
|     //         .about("Toggle the status of the espanso replacement engine."))
 | ||||
|     // )
 | ||||
|     // .subcommand(SubCommand::with_name("edit")
 | ||||
|     //     .about("Open the default text editor to edit config files and reload them automatically when exiting")
 | ||||
|     //     .arg(Arg::with_name("config")
 | ||||
|     //         .help("Defaults to \"default\". The configuration file name to edit (without the .yml extension)."))
 | ||||
|     //     .arg(Arg::with_name("norestart")
 | ||||
|     //         .short("n")
 | ||||
|     //         .long("norestart")
 | ||||
|     //         .required(false)
 | ||||
|     //         .takes_value(false)
 | ||||
|     //         .help("Avoid restarting espanso after editing the file"))
 | ||||
|     // )
 | ||||
|     .subcommand(SubCommand::with_name("edit") | ||||
|         .about("Shortcut to open the default text editor to edit config files") | ||||
|         .arg(Arg::with_name("target_file") | ||||
|             .help(r#"Defaults to "match/base.yml", it contains the relative path of the file you want to edit, 
 | ||||
| such as 'config/default.yml' or 'match/base.yml'. 
 | ||||
| For convenience, you can also specify the name directly and Espanso will figure out the path. 
 | ||||
| For example, specifying 'email' is equivalent to 'match/email.yml'."#))
 | ||||
|         // .arg(Arg::with_name("norestart")
 | ||||
|         //     .short("n")
 | ||||
|         //     .long("norestart")
 | ||||
|         //     .required(false)
 | ||||
|         //     .takes_value(false)
 | ||||
|         //     .help("Avoid restarting espanso after editing the file"))
 | ||||
|     ) | ||||
|     // .subcommand(SubCommand::with_name("detect")
 | ||||
|     //     .about("Tool to detect current window properties, to simplify filters creation."))
 | ||||
|     .subcommand( | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	Block a user