2021-05-24 18:58:18 +00:00
|
|
|
/*
|
|
|
|
* 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::{cmp::Ordering, collections::HashMap, path::PathBuf};
|
2021-05-29 11:08:32 +00:00
|
|
|
use yaml_rust::{yaml::Hash, Yaml};
|
2021-05-24 18:58:18 +00:00
|
|
|
|
2021-05-29 11:08:32 +00:00
|
|
|
pub struct ConvertedFile {
|
|
|
|
pub origin: String,
|
|
|
|
pub content: Hash,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn convert(input_files: HashMap<String, Hash>) -> HashMap<String, ConvertedFile> {
|
2021-05-24 18:58:18 +00:00
|
|
|
let mut output_files = HashMap::new();
|
|
|
|
|
|
|
|
let sorted_input_files = sort_input_files(&input_files);
|
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
let mut config_names_to_path = HashMap::new();
|
|
|
|
|
2021-05-24 18:58:18 +00:00
|
|
|
for input_path in sorted_input_files {
|
|
|
|
let yaml = input_files
|
|
|
|
.get(&input_path)
|
|
|
|
.expect("received unexpected file in input function");
|
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
let yaml_matches = yaml_get_vec(yaml, "matches");
|
|
|
|
let yaml_global_vars = yaml_get_vec(yaml, "global_vars");
|
2021-05-24 18:58:18 +00:00
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
let yaml_parent = yaml_get_string(yaml, "parent");
|
|
|
|
let yaml_name = yaml_get_string(yaml, "name");
|
2021-05-24 18:58:18 +00:00
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
let should_generate_match = yaml_matches.is_some() || yaml_global_vars.is_some();
|
2021-05-27 20:06:39 +00:00
|
|
|
let match_file_path_if_unlisted = if should_generate_match {
|
2021-05-25 20:07:26 +00:00
|
|
|
let should_underscore = !input_path.starts_with("default") && yaml_parent != Some("default");
|
|
|
|
let match_output_path = calculate_output_match_path(&input_path, should_underscore);
|
|
|
|
if match_output_path.is_none() {
|
|
|
|
eprintln!(
|
|
|
|
"unable to determine output path for {}, skipping...",
|
|
|
|
input_path
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let match_output_path = match_output_path.unwrap();
|
|
|
|
|
|
|
|
if let Some(name) = yaml_name {
|
|
|
|
config_names_to_path.insert(name.to_string(), match_output_path.clone());
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:14:42 +00:00
|
|
|
let output_yaml = output_files
|
|
|
|
.entry(match_output_path.clone())
|
2021-05-29 11:08:32 +00:00
|
|
|
.or_insert(ConvertedFile {
|
|
|
|
origin: input_path.to_string(),
|
|
|
|
content: Hash::new(),
|
|
|
|
});
|
2021-05-25 20:07:26 +00:00
|
|
|
|
|
|
|
if let Some(global_vars) = yaml_global_vars {
|
2021-05-29 11:08:32 +00:00
|
|
|
let output_global_vars = output_yaml.content
|
2021-05-25 20:07:26 +00:00
|
|
|
.entry(Yaml::String("global_vars".to_string()))
|
|
|
|
.or_insert(Yaml::Array(Vec::new()));
|
|
|
|
if let Yaml::Array(out_global_vars) = output_global_vars {
|
|
|
|
out_global_vars.extend(global_vars.clone());
|
|
|
|
} else {
|
|
|
|
eprintln!("unable to transform global_vars for file: {}", input_path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(matches) = yaml_matches {
|
2021-05-29 11:08:32 +00:00
|
|
|
let output_matches = output_yaml.content
|
2021-05-25 20:07:26 +00:00
|
|
|
.entry(Yaml::String("matches".to_string()))
|
|
|
|
.or_insert(Yaml::Array(Vec::new()));
|
|
|
|
if let Yaml::Array(out_matches) = output_matches {
|
|
|
|
out_matches.extend(matches.clone());
|
|
|
|
} else {
|
|
|
|
eprintln!("unable to transform matches for file: {}", input_path);
|
|
|
|
}
|
|
|
|
}
|
2021-05-29 08:14:42 +00:00
|
|
|
|
2021-05-27 20:06:39 +00:00
|
|
|
if should_underscore {
|
|
|
|
Some(match_output_path)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
2021-05-25 20:07:26 +00:00
|
|
|
|
|
|
|
let yaml_filter_class = yaml_get_string(yaml, "filter_class");
|
|
|
|
let yaml_filter_title = yaml_get_string(yaml, "filter_title");
|
|
|
|
let yaml_filter_exec = yaml_get_string(yaml, "filter_exec");
|
|
|
|
|
|
|
|
let should_generate_config = input_path.starts_with("default")
|
|
|
|
|| yaml_filter_class.is_some()
|
|
|
|
|| yaml_filter_exec.is_some()
|
|
|
|
|| yaml_filter_title.is_some();
|
|
|
|
|
|
|
|
if should_generate_config {
|
|
|
|
let config_output_path = calculate_output_config_path(&input_path);
|
|
|
|
|
|
|
|
let mut output_yaml = Hash::new();
|
|
|
|
|
|
|
|
copy_field_if_present(yaml, "filter_title", &mut output_yaml, "filter_title");
|
|
|
|
copy_field_if_present(yaml, "filter_class", &mut output_yaml, "filter_class");
|
|
|
|
copy_field_if_present(yaml, "filter_exec", &mut output_yaml, "filter_exec");
|
2021-05-27 20:06:39 +00:00
|
|
|
copy_field_if_present(yaml, "enable_active", &mut output_yaml, "enable");
|
2021-05-29 08:14:42 +00:00
|
|
|
copy_field_if_present(yaml, "backend", &mut output_yaml, "backend");
|
|
|
|
map_field_if_present(
|
|
|
|
yaml,
|
|
|
|
"paste_shortcut",
|
|
|
|
&mut output_yaml,
|
|
|
|
"paste_shortcut",
|
|
|
|
|val| match val {
|
|
|
|
Yaml::String(shortcut) if shortcut == "CtrlV" => Some(Yaml::String("CTRL+V".to_string())),
|
|
|
|
Yaml::String(shortcut) if shortcut == "CtrlShiftV" => Some(Yaml::String("CTRL+SHIFT+V".to_string())),
|
|
|
|
Yaml::String(shortcut) if shortcut == "ShiftInsert" => Some(Yaml::String("SHIFT+INSERT".to_string())),
|
|
|
|
Yaml::String(shortcut) if shortcut == "CtrlAltV" => Some(Yaml::String("CTRL+ALT+V".to_string())),
|
|
|
|
Yaml::String(shortcut) if shortcut == "MetaV" => Some(Yaml::String("META+V".to_string())),
|
|
|
|
Yaml::String(_) => None,
|
|
|
|
_ => None,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
copy_field_if_present(yaml, "secure_input_watcher_enabled", &mut output_yaml, "secure_input_watcher_enabled");
|
|
|
|
copy_field_if_present(yaml, "secure_input_watcher_interval", &mut output_yaml, "secure_input_watcher_interval");
|
|
|
|
copy_field_if_present(yaml, "secure_input_notification", &mut output_yaml, "secure_input_notification");
|
|
|
|
copy_field_if_present(yaml, "config_caching_interval", &mut output_yaml, "config_caching_interval");
|
|
|
|
copy_field_if_present(yaml, "toggle_interval", &mut output_yaml, "toggle_interval");
|
|
|
|
copy_field_if_present(yaml, "toggle_key", &mut output_yaml, "toggle_key");
|
|
|
|
copy_field_if_present(yaml, "preserve_clipboard", &mut output_yaml, "preserve_clipboard");
|
|
|
|
copy_field_if_present(yaml, "backspace_limit", &mut output_yaml, "backspace_limit");
|
|
|
|
copy_field_if_present(yaml, "fast_inject", &mut output_yaml, "x11_fast_inject");
|
|
|
|
copy_field_if_present(yaml, "auto_restart", &mut output_yaml, "auto_restart");
|
|
|
|
copy_field_if_present(yaml, "undo_backspace", &mut output_yaml, "undo_backspace");
|
|
|
|
copy_field_if_present(yaml, "show_icon", &mut output_yaml, "show_icon");
|
|
|
|
copy_field_if_present(yaml, "show_notifications", &mut output_yaml, "show_notifications");
|
|
|
|
copy_field_if_present(yaml, "inject_delay", &mut output_yaml, "inject_delay");
|
|
|
|
copy_field_if_present(yaml, "restore_clipboard_delay", &mut output_yaml, "restore_clipboard_delay");
|
|
|
|
copy_field_if_present(yaml, "use_system_agent", &mut output_yaml, "use_system_agent");
|
|
|
|
copy_field_if_present(yaml, "word_separators", &mut output_yaml, "word_separators");
|
2021-05-27 20:06:39 +00:00
|
|
|
|
|
|
|
// TODO: warn if passive mode parameters are used
|
2021-05-25 20:07:26 +00:00
|
|
|
|
|
|
|
// TODO: copy other config fields: https://github.com/federico-terzi/espanso/blob/master/src/config/mod.rs#L169
|
|
|
|
|
2021-05-29 08:14:42 +00:00
|
|
|
// Link any unlisted match file (the ones starting with the _ underscore, which are excluded by the
|
2021-05-27 20:06:39 +00:00
|
|
|
// default.yml config) explicitly, if present.
|
|
|
|
if let Some(match_file_path) = match_file_path_if_unlisted {
|
2021-05-29 08:14:42 +00:00
|
|
|
let yaml_exclude_default_entries =
|
|
|
|
yaml_get_bool(yaml, "exclude_default_entries").unwrap_or(false);
|
|
|
|
let key_name = if yaml_exclude_default_entries {
|
|
|
|
"includes"
|
|
|
|
} else {
|
|
|
|
"extra_includes"
|
|
|
|
};
|
|
|
|
|
2021-05-27 20:06:39 +00:00
|
|
|
let includes = vec![Yaml::String(format!("../{}", match_file_path))];
|
|
|
|
|
|
|
|
output_yaml.insert(Yaml::String(key_name.to_string()), Yaml::Array(includes));
|
|
|
|
}
|
2021-05-25 20:07:26 +00:00
|
|
|
|
2021-05-29 11:08:32 +00:00
|
|
|
output_files.insert(config_output_path, ConvertedFile {
|
|
|
|
origin: input_path,
|
|
|
|
content: output_yaml,
|
|
|
|
});
|
2021-05-25 20:07:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: create config file
|
|
|
|
|
|
|
|
// TODO: execute the actual conversion
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: here resolve parent: name imports
|
|
|
|
|
|
|
|
// TODO: remove this prints
|
2021-05-27 20:06:39 +00:00
|
|
|
// for (file, content) in output_files {
|
|
|
|
// let mut out_str = String::new();
|
|
|
|
// {
|
|
|
|
// let mut emitter = YamlEmitter::new(&mut out_str);
|
|
|
|
// emitter.dump(&Yaml::Hash(content)).unwrap(); // dump the YAML object to a String
|
|
|
|
// }
|
|
|
|
// println!("\n------- {} ------------\n{}", file, out_str);
|
|
|
|
// }
|
2021-05-24 18:58:18 +00:00
|
|
|
|
|
|
|
output_files
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sort_input_files(input_files: &HashMap<String, Hash>) -> Vec<String> {
|
|
|
|
let mut files: Vec<String> = input_files.iter().map(|(key, _)| key.clone()).collect();
|
|
|
|
files.sort_by(|f1, f2| {
|
|
|
|
let f1_slashes = f1.matches("/").count();
|
|
|
|
let f2_slashes = f2.matches("/").count();
|
|
|
|
if f1_slashes > f2_slashes {
|
|
|
|
Ordering::Greater
|
|
|
|
} else if f1_slashes < f2_slashes {
|
|
|
|
Ordering::Less
|
|
|
|
} else {
|
|
|
|
f1.cmp(f2)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
files
|
|
|
|
}
|
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
// TODO: test
|
|
|
|
fn calculate_output_match_path(path: &str, is_underscored: bool) -> Option<String> {
|
2021-05-24 18:58:18 +00:00
|
|
|
let path_buf = PathBuf::from(path);
|
|
|
|
let file_name = path_buf.file_name()?.to_string_lossy().to_string();
|
2021-05-25 20:07:26 +00:00
|
|
|
|
|
|
|
let path = if is_underscored {
|
|
|
|
path.replace(&file_name, &format!("_{}", file_name))
|
|
|
|
} else {
|
|
|
|
path.to_string()
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(if path.starts_with("user/") {
|
|
|
|
format!("match/{}", path.trim_start_matches("user/"))
|
|
|
|
} else if path.starts_with("packages/") {
|
|
|
|
format!("match/packages/{}", path.trim_start_matches("packages/"))
|
|
|
|
} else if path == "default.yml" {
|
|
|
|
"match/base.yml".to_string()
|
|
|
|
} else {
|
|
|
|
format!("match/{}", path)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: test
|
|
|
|
fn calculate_output_config_path(path: &str) -> String {
|
|
|
|
if path.starts_with("user/") {
|
|
|
|
format!("config/{}", path.trim_start_matches("user/"))
|
|
|
|
} else if path.starts_with("packages/") {
|
|
|
|
format!("config/packages/{}", path.trim_start_matches("packages/"))
|
|
|
|
} else {
|
|
|
|
format!("config/{}", path)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn yaml_get_vec<'a>(yaml: &'a Hash, name: &str) -> Option<&'a Vec<Yaml>> {
|
|
|
|
yaml
|
|
|
|
.get(&Yaml::String(name.to_string()))
|
|
|
|
.and_then(|v| v.as_vec())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn yaml_get_string<'a>(yaml: &'a Hash, name: &str) -> Option<&'a str> {
|
|
|
|
yaml
|
|
|
|
.get(&Yaml::String(name.to_string()))
|
|
|
|
.and_then(|v| v.as_str())
|
|
|
|
}
|
|
|
|
|
2021-05-27 20:06:39 +00:00
|
|
|
fn yaml_get_bool<'a>(yaml: &'a Hash, name: &str) -> Option<bool> {
|
|
|
|
yaml
|
|
|
|
.get(&Yaml::String(name.to_string()))
|
|
|
|
.and_then(|v| v.as_bool())
|
|
|
|
}
|
|
|
|
|
2021-05-25 20:07:26 +00:00
|
|
|
fn copy_field_if_present(
|
|
|
|
input_yaml: &Hash,
|
|
|
|
input_field_name: &str,
|
|
|
|
output_yaml: &mut Hash,
|
|
|
|
output_field_name: &str,
|
|
|
|
) {
|
|
|
|
if let Some(value) = input_yaml.get(&Yaml::String(input_field_name.to_string())) {
|
|
|
|
output_yaml.insert(Yaml::String(output_field_name.to_string()), value.clone());
|
|
|
|
}
|
|
|
|
}
|
2021-05-29 08:14:42 +00:00
|
|
|
|
|
|
|
fn map_field_if_present(
|
|
|
|
input_yaml: &Hash,
|
|
|
|
input_field_name: &str,
|
|
|
|
output_yaml: &mut Hash,
|
|
|
|
output_field_name: &str,
|
|
|
|
transform: impl FnOnce(&Yaml) -> Option<Yaml>,
|
|
|
|
) {
|
|
|
|
if let Some(value) = input_yaml.get(&Yaml::String(input_field_name.to_string())) {
|
|
|
|
let transformed = transform(value);
|
|
|
|
if let Some(transformed) = transformed {
|
|
|
|
output_yaml.insert(Yaml::String(output_field_name.to_string()), transformed);
|
|
|
|
} else {
|
|
|
|
eprintln!("could not convert value for field: {}", input_field_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|