2019-09-16 22:11:31 +00:00
|
|
|
/*
|
|
|
|
* This file is part of espanso.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// This functions are used to register/unregister espanso from the system daemon manager.
|
|
|
|
|
|
|
|
use crate::config::ConfigSet;
|
2020-05-10 16:01:04 +00:00
|
|
|
use crate::sysdaemon::VerifyResult::{EnabledAndValid, EnabledButInvalidPath, NotEnabled};
|
2019-09-16 22:11:31 +00:00
|
|
|
|
|
|
|
// INSTALLATION
|
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2020-05-10 16:01:04 +00:00
|
|
|
const MAC_PLIST_CONTENT: &str = include_str!("res/mac/com.federicoterzi.espanso.plist");
|
2019-09-16 22:11:31 +00:00
|
|
|
#[cfg(target_os = "macos")]
|
2020-05-10 16:01:04 +00:00
|
|
|
const MAC_PLIST_FILENAME: &str = "com.federicoterzi.espanso.plist";
|
2019-09-16 22:11:31 +00:00
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2019-09-20 20:33:14 +00:00
|
|
|
pub fn register(_config_set: ConfigSet) {
|
2019-09-18 08:28:34 +00:00
|
|
|
use std::fs::create_dir_all;
|
|
|
|
use std::process::{Command, ExitStatus};
|
|
|
|
|
2019-09-16 22:11:31 +00:00
|
|
|
let home_dir = dirs::home_dir().expect("Could not get user home directory");
|
|
|
|
let library_dir = home_dir.join("Library");
|
|
|
|
let agents_dir = library_dir.join("LaunchAgents");
|
|
|
|
|
|
|
|
// Make sure agents directory exists
|
|
|
|
if !agents_dir.exists() {
|
|
|
|
create_dir_all(agents_dir.clone()).expect("Could not create LaunchAgents directory");
|
|
|
|
}
|
|
|
|
|
|
|
|
let plist_file = agents_dir.join(MAC_PLIST_FILENAME);
|
|
|
|
if !plist_file.exists() {
|
2020-05-10 16:01:04 +00:00
|
|
|
println!(
|
|
|
|
"Creating LaunchAgents entry: {}",
|
|
|
|
plist_file.to_str().unwrap_or_default()
|
|
|
|
);
|
2019-09-16 22:11:31 +00:00
|
|
|
|
|
|
|
let espanso_path = std::env::current_exe().expect("Could not get espanso executable path");
|
2020-05-10 16:01:04 +00:00
|
|
|
println!(
|
|
|
|
"Entry will point to: {}",
|
|
|
|
espanso_path.to_str().unwrap_or_default()
|
|
|
|
);
|
2019-09-16 22:11:31 +00:00
|
|
|
|
2020-05-10 16:01:04 +00:00
|
|
|
let plist_content = String::from(MAC_PLIST_CONTENT).replace(
|
|
|
|
"{{{espanso_path}}}",
|
|
|
|
espanso_path.to_str().unwrap_or_default(),
|
|
|
|
);
|
2019-09-16 22:11:31 +00:00
|
|
|
|
2020-06-10 18:21:18 +00:00
|
|
|
// Copy the user PATH variable and inject it in the Plist file so that
|
|
|
|
// it gets loaded by Launchd.
|
|
|
|
// To see why this is necessary: https://github.com/federico-terzi/espanso/issues/233
|
|
|
|
let user_path = std::env::var("PATH").unwrap_or("".to_owned());
|
|
|
|
let plist_content = plist_content.replace("{{{PATH}}}", &user_path);
|
|
|
|
|
2019-09-16 22:11:31 +00:00
|
|
|
std::fs::write(plist_file.clone(), plist_content).expect("Unable to write plist file");
|
|
|
|
|
|
|
|
println!("Entry created correctly!")
|
|
|
|
}
|
|
|
|
|
|
|
|
println!("Reloading entry...");
|
|
|
|
|
|
|
|
let res = Command::new("launchctl")
|
2020-05-10 16:01:04 +00:00
|
|
|
.args(&["unload", "-w", plist_file.to_str().unwrap_or_default()])
|
|
|
|
.output();
|
2019-09-16 22:11:31 +00:00
|
|
|
|
|
|
|
let res = Command::new("launchctl")
|
|
|
|
.args(&["load", "-w", plist_file.to_str().unwrap_or_default()])
|
|
|
|
.status();
|
|
|
|
|
|
|
|
if let Ok(status) = res {
|
|
|
|
if status.success() {
|
|
|
|
println!("Entry loaded correctly!")
|
|
|
|
}
|
2020-05-10 16:01:04 +00:00
|
|
|
} else {
|
2019-09-16 22:11:31 +00:00
|
|
|
println!("Error loading new entry");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2019-09-20 20:33:14 +00:00
|
|
|
pub fn unregister(_config_set: ConfigSet) {
|
2019-09-18 08:28:34 +00:00
|
|
|
use std::fs::create_dir_all;
|
|
|
|
use std::process::{Command, ExitStatus};
|
|
|
|
|
2019-09-16 22:11:31 +00:00
|
|
|
let home_dir = dirs::home_dir().expect("Could not get user home directory");
|
|
|
|
let library_dir = home_dir.join("Library");
|
|
|
|
let agents_dir = library_dir.join("LaunchAgents");
|
|
|
|
|
|
|
|
let plist_file = agents_dir.join(MAC_PLIST_FILENAME);
|
|
|
|
if plist_file.exists() {
|
|
|
|
let _res = Command::new("launchctl")
|
|
|
|
.args(&["unload", "-w", plist_file.to_str().unwrap_or_default()])
|
|
|
|
.output();
|
|
|
|
|
|
|
|
std::fs::remove_file(&plist_file).expect("Could not remove espanso entry");
|
|
|
|
|
|
|
|
println!("Entry removed correctly!")
|
2020-05-10 16:01:04 +00:00
|
|
|
} else {
|
2019-09-16 22:11:31 +00:00
|
|
|
println!("espanso is not installed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-20 20:33:14 +00:00
|
|
|
// LINUX
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
2020-05-10 16:01:04 +00:00
|
|
|
const LINUX_SERVICE_CONTENT: &str = include_str!("res/linux/systemd.service");
|
2019-10-11 21:35:17 +00:00
|
|
|
#[cfg(target_os = "linux")]
|
2020-05-10 16:01:04 +00:00
|
|
|
const LINUX_SERVICE_FILENAME: &str = "espanso.service";
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
2020-04-03 18:10:06 +00:00
|
|
|
pub fn register(_: ConfigSet) {
|
2019-10-11 21:35:17 +00:00
|
|
|
use std::fs::create_dir_all;
|
2020-05-10 16:01:04 +00:00
|
|
|
use std::process::Command;
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
// Check if espanso service is already registered
|
|
|
|
let res = Command::new("systemctl")
|
|
|
|
.args(&["--user", "is-enabled", "espanso"])
|
|
|
|
.output();
|
|
|
|
if let Ok(res) = res {
|
|
|
|
let output = String::from_utf8_lossy(res.stdout.as_slice());
|
|
|
|
let output = output.trim();
|
2019-10-12 14:24:08 +00:00
|
|
|
if res.status.success() {
|
|
|
|
if output == "enabled" {
|
|
|
|
eprintln!("espanso service is already registered to systemd");
|
|
|
|
eprintln!("If you want to register it again, please uninstall it first with:");
|
|
|
|
eprintln!(" espanso unregister");
|
|
|
|
std::process::exit(5);
|
|
|
|
}
|
2020-05-10 16:01:04 +00:00
|
|
|
} else {
|
2019-10-12 14:24:08 +00:00
|
|
|
if output == "disabled" {
|
|
|
|
use dialoguer::Confirmation;
|
|
|
|
if !Confirmation::new()
|
|
|
|
.with_text("espanso is already registered but currently disabled. Do you want to override it?")
|
|
|
|
.default(false)
|
|
|
|
.show_default(true)
|
|
|
|
.interact().expect("Unable to read user answer") {
|
|
|
|
|
|
|
|
std::process::exit(6);
|
|
|
|
}
|
|
|
|
}
|
2019-10-11 21:35:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// User level systemd services should be placed in this directory:
|
|
|
|
// $XDG_CONFIG_HOME/systemd/user/, usually: ~/.config/systemd/user/
|
|
|
|
let config_dir = dirs::config_dir().expect("Could not get configuration directory");
|
|
|
|
let systemd_dir = config_dir.join("systemd");
|
|
|
|
let user_dir = systemd_dir.join("user");
|
|
|
|
|
|
|
|
// Make sure the directory exists
|
|
|
|
if !user_dir.exists() {
|
|
|
|
create_dir_all(user_dir.clone()).expect("Could not create systemd user directory");
|
|
|
|
}
|
|
|
|
|
|
|
|
let service_file = user_dir.join(LINUX_SERVICE_FILENAME);
|
|
|
|
if !service_file.exists() {
|
2020-05-10 16:01:04 +00:00
|
|
|
println!(
|
|
|
|
"Creating service entry: {}",
|
|
|
|
service_file.to_str().unwrap_or_default()
|
|
|
|
);
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
let espanso_path = std::env::current_exe().expect("Could not get espanso executable path");
|
2020-05-10 16:01:04 +00:00
|
|
|
println!(
|
|
|
|
"Entry will point to: {}",
|
|
|
|
espanso_path.to_str().unwrap_or_default()
|
|
|
|
);
|
2019-10-11 21:35:17 +00:00
|
|
|
|
2020-05-10 16:01:04 +00:00
|
|
|
let service_content = String::from(LINUX_SERVICE_CONTENT).replace(
|
|
|
|
"{{{espanso_path}}}",
|
|
|
|
espanso_path.to_str().unwrap_or_default(),
|
|
|
|
);
|
2019-10-11 21:35:17 +00:00
|
|
|
|
2020-05-10 16:01:04 +00:00
|
|
|
std::fs::write(service_file.clone(), service_content)
|
|
|
|
.expect("Unable to write service file");
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
println!("Service file created correctly!")
|
|
|
|
}
|
|
|
|
|
|
|
|
println!("Enabling espanso for systemd...");
|
|
|
|
|
|
|
|
let res = Command::new("systemctl")
|
|
|
|
.args(&["--user", "enable", "espanso"])
|
|
|
|
.status();
|
|
|
|
|
|
|
|
if let Ok(status) = res {
|
|
|
|
if status.success() {
|
|
|
|
println!("Service registered correctly!")
|
|
|
|
}
|
2020-05-10 16:01:04 +00:00
|
|
|
} else {
|
2019-10-11 21:35:17 +00:00
|
|
|
println!("Error loading espanso service");
|
|
|
|
}
|
2019-09-20 20:33:14 +00:00
|
|
|
}
|
|
|
|
|
2020-03-16 21:16:12 +00:00
|
|
|
pub enum VerifyResult {
|
|
|
|
EnabledAndValid,
|
|
|
|
EnabledButInvalidPath,
|
|
|
|
NotEnabled,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
pub fn verify() -> VerifyResult {
|
|
|
|
use regex::Regex;
|
2020-05-10 16:01:04 +00:00
|
|
|
use std::process::Command;
|
2020-03-16 21:16:12 +00:00
|
|
|
|
|
|
|
// Check if espanso service is already registered
|
|
|
|
let res = Command::new("systemctl")
|
|
|
|
.args(&["--user", "is-enabled", "espanso"])
|
|
|
|
.output();
|
|
|
|
if let Ok(res) = res {
|
|
|
|
let output = String::from_utf8_lossy(res.stdout.as_slice());
|
|
|
|
let output = output.trim();
|
|
|
|
if !res.status.success() || output != "enabled" {
|
2020-05-10 16:01:04 +00:00
|
|
|
return NotEnabled;
|
2020-03-16 21:16:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lazy_static! {
|
2020-04-03 18:10:06 +00:00
|
|
|
static ref EXEC_PATH_REGEX: Regex = Regex::new("ExecStart=(?P<path>.*?)\\s").unwrap();
|
2020-03-16 21:16:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the currently registered path is valid
|
|
|
|
let res = Command::new("systemctl")
|
|
|
|
.args(&["--user", "cat", "espanso"])
|
|
|
|
.output();
|
|
|
|
if let Ok(res) = res {
|
|
|
|
let output = String::from_utf8_lossy(res.stdout.as_slice());
|
|
|
|
let output = output.trim();
|
|
|
|
if res.status.success() {
|
2020-04-03 18:10:06 +00:00
|
|
|
let caps = EXEC_PATH_REGEX.captures(output).unwrap();
|
2020-03-16 21:16:12 +00:00
|
|
|
let path = caps.get(1).map_or("", |m| m.as_str());
|
2020-05-10 16:01:04 +00:00
|
|
|
let espanso_path =
|
|
|
|
std::env::current_exe().expect("Could not get espanso executable path");
|
2020-03-16 21:16:12 +00:00
|
|
|
|
|
|
|
if espanso_path.to_string_lossy() != path {
|
2020-05-10 16:01:04 +00:00
|
|
|
return EnabledButInvalidPath;
|
2020-03-16 21:16:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EnabledAndValid
|
|
|
|
}
|
|
|
|
|
2019-09-20 20:33:14 +00:00
|
|
|
#[cfg(target_os = "linux")]
|
2020-04-03 18:10:06 +00:00
|
|
|
pub fn unregister(_: ConfigSet) {
|
2020-05-10 16:01:04 +00:00
|
|
|
use std::process::Command;
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
// Disable the service first
|
2020-04-03 18:10:06 +00:00
|
|
|
Command::new("systemctl")
|
2019-10-11 21:35:17 +00:00
|
|
|
.args(&["--user", "disable", "espanso"])
|
2020-05-10 16:01:04 +00:00
|
|
|
.status()
|
|
|
|
.expect("Unable to invoke systemctl");
|
2019-10-11 21:35:17 +00:00
|
|
|
|
|
|
|
// Then delete the espanso.service entry
|
|
|
|
let config_dir = dirs::config_dir().expect("Could not get configuration directory");
|
|
|
|
let systemd_dir = config_dir.join("systemd");
|
|
|
|
let user_dir = systemd_dir.join("user");
|
|
|
|
let service_file = user_dir.join(LINUX_SERVICE_FILENAME);
|
|
|
|
|
|
|
|
if service_file.exists() {
|
|
|
|
let res = std::fs::remove_file(&service_file);
|
|
|
|
match res {
|
|
|
|
Ok(_) => {
|
|
|
|
println!("Deleted entry at {}", service_file.to_string_lossy());
|
|
|
|
println!("Service unregistered successfully!");
|
2020-05-10 16:01:04 +00:00
|
|
|
}
|
2019-10-11 21:35:17 +00:00
|
|
|
Err(e) => {
|
2020-05-10 16:01:04 +00:00
|
|
|
println!(
|
|
|
|
"Error, could not delete service entry at {} with error {}",
|
|
|
|
service_file.to_string_lossy(),
|
|
|
|
e
|
|
|
|
);
|
|
|
|
}
|
2019-10-11 21:35:17 +00:00
|
|
|
}
|
2020-05-10 16:01:04 +00:00
|
|
|
} else {
|
2019-10-11 21:35:17 +00:00
|
|
|
eprintln!("Error, could not find espanso service file");
|
|
|
|
}
|
2019-09-20 20:33:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// WINDOWS
|
|
|
|
|
2019-09-16 22:11:31 +00:00
|
|
|
#[cfg(target_os = "windows")]
|
2019-09-20 20:33:14 +00:00
|
|
|
pub fn register(_config_set: ConfigSet) {
|
|
|
|
println!("Windows does not support automatic system daemon integration.")
|
2019-09-18 08:28:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "windows")]
|
2019-09-20 20:33:14 +00:00
|
|
|
pub fn unregister(_config_set: ConfigSet) {
|
|
|
|
println!("Windows does not support automatic system daemon integration.")
|
2020-05-10 16:01:04 +00:00
|
|
|
}
|