feat(config): add id to config and decouple variable params from serde_yaml
This commit is contained in:
parent
e643609d57
commit
f847d0cd81
|
@ -15,6 +15,8 @@ regex = "1.4.3"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
dunce = "1.0.1"
|
dunce = "1.0.1"
|
||||||
walkdir = "2.3.1"
|
walkdir = "2.3.1"
|
||||||
|
enum-as-inner = "0.3.3"
|
||||||
|
ordered-float = "2.0"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tempdir = "0.3.7"
|
tempdir = "0.3.7"
|
||||||
|
|
|
@ -28,6 +28,7 @@ mod util;
|
||||||
pub(crate) mod store;
|
pub(crate) mod store;
|
||||||
|
|
||||||
pub trait Config {
|
pub trait Config {
|
||||||
|
fn id(&self) -> i32;
|
||||||
fn label(&self) -> &str;
|
fn label(&self) -> &str;
|
||||||
fn match_paths(&self) -> &[String];
|
fn match_paths(&self) -> &[String];
|
||||||
fn backend(&self) -> Backend;
|
fn backend(&self) -> Backend;
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use super::{AppProperties, Backend, Config, parse::ParsedConfig, path::calculate_paths, util::os_matches};
|
use super::{AppProperties, Backend, Config, parse::ParsedConfig, path::calculate_paths, util::os_matches};
|
||||||
use crate::merge;
|
use crate::{counter::next_id, merge};
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use log::error;
|
use log::error;
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
|
@ -34,6 +34,7 @@ pub(crate) struct ResolvedConfig {
|
||||||
parsed: ParsedConfig,
|
parsed: ParsedConfig,
|
||||||
|
|
||||||
// Generated properties
|
// Generated properties
|
||||||
|
id: i32,
|
||||||
match_paths: Vec<String>,
|
match_paths: Vec<String>,
|
||||||
|
|
||||||
filter_title: Option<Regex>,
|
filter_title: Option<Regex>,
|
||||||
|
@ -45,6 +46,7 @@ impl Default for ResolvedConfig {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self {
|
Self {
|
||||||
parsed: Default::default(),
|
parsed: Default::default(),
|
||||||
|
id: 0,
|
||||||
match_paths: Vec::new(),
|
match_paths: Vec::new(),
|
||||||
filter_title: None,
|
filter_title: None,
|
||||||
filter_class: None,
|
filter_class: None,
|
||||||
|
@ -54,6 +56,10 @@ impl Default for ResolvedConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config for ResolvedConfig {
|
impl Config for ResolvedConfig {
|
||||||
|
fn id(&self) -> i32 {
|
||||||
|
self.id
|
||||||
|
}
|
||||||
|
|
||||||
fn label(&self) -> &str {
|
fn label(&self) -> &str {
|
||||||
self.parsed.label.as_deref().unwrap_or("none")
|
self.parsed.label.as_deref().unwrap_or("none")
|
||||||
}
|
}
|
||||||
|
@ -162,6 +168,7 @@ impl ResolvedConfig {
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
parsed: config,
|
parsed: config,
|
||||||
|
id: next_id(),
|
||||||
match_paths,
|
match_paths,
|
||||||
filter_title,
|
filter_title,
|
||||||
filter_class,
|
filter_class,
|
||||||
|
|
|
@ -134,6 +134,10 @@ mod tests {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config for MockConfig {
|
impl Config for MockConfig {
|
||||||
|
fn id(&self) -> i32 {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
fn label(&self) -> &str {
|
fn label(&self) -> &str {
|
||||||
&self.label
|
&self.label
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
use std::sync::atomic::{AtomicI32, Ordering};
|
use std::sync::atomic::{AtomicI32, Ordering};
|
||||||
|
|
||||||
thread_local! {
|
thread_local! {
|
||||||
|
// TODO: if thread local, we probably don't need an atomic
|
||||||
static STRUCT_COUNTER: AtomicI32 = AtomicI32::new(0);
|
static STRUCT_COUNTER: AtomicI32 = AtomicI32::new(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,6 +31,7 @@ use crate::{
|
||||||
store::{MatchSet, MatchStore},
|
store::{MatchSet, MatchStore},
|
||||||
Match, Variable,
|
Match, Variable,
|
||||||
},
|
},
|
||||||
|
counter::next_id,
|
||||||
};
|
};
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
|
|
||||||
|
@ -137,6 +138,8 @@ struct LegacyInteropConfig {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
match_paths: Vec<String>,
|
match_paths: Vec<String>,
|
||||||
|
|
||||||
|
id: i32,
|
||||||
|
|
||||||
config: LegacyConfig,
|
config: LegacyConfig,
|
||||||
|
|
||||||
filter_title: Option<Regex>,
|
filter_title: Option<Regex>,
|
||||||
|
@ -147,6 +150,7 @@ struct LegacyInteropConfig {
|
||||||
impl From<config::LegacyConfig> for LegacyInteropConfig {
|
impl From<config::LegacyConfig> for LegacyInteropConfig {
|
||||||
fn from(config: config::LegacyConfig) -> Self {
|
fn from(config: config::LegacyConfig) -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
id: next_id(),
|
||||||
config: config.clone(),
|
config: config.clone(),
|
||||||
name: config.name.clone(),
|
name: config.name.clone(),
|
||||||
match_paths: vec![config.name],
|
match_paths: vec![config.name],
|
||||||
|
@ -170,6 +174,10 @@ impl From<config::LegacyConfig> for LegacyInteropConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config for LegacyInteropConfig {
|
impl Config for LegacyInteropConfig {
|
||||||
|
fn id(&self) -> i32 {
|
||||||
|
self.id
|
||||||
|
}
|
||||||
|
|
||||||
fn label(&self) -> &str {
|
fn label(&self) -> &str {
|
||||||
&self.config.name
|
&self.config.name
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,21 +17,25 @@
|
||||||
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
|
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use crate::{counter::next_id, matches::{
|
use crate::{
|
||||||
|
counter::next_id,
|
||||||
|
matches::{
|
||||||
group::{path::resolve_imports, MatchGroup},
|
group::{path::resolve_imports, MatchGroup},
|
||||||
Match, Variable,
|
Match, Variable,
|
||||||
}};
|
},
|
||||||
|
};
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use log::warn;
|
use log::warn;
|
||||||
use parse::YAMLMatchGroup;
|
use parse::YAMLMatchGroup;
|
||||||
use std::convert::{TryFrom, TryInto};
|
use std::convert::{TryFrom, TryInto};
|
||||||
|
|
||||||
use self::parse::{YAMLMatch, YAMLVariable};
|
use self::{parse::{YAMLMatch, YAMLVariable}, util::convert_params};
|
||||||
use crate::matches::{MatchCause, MatchEffect, TextEffect, TriggerCause};
|
use crate::matches::{MatchCause, MatchEffect, TextEffect, TriggerCause};
|
||||||
|
|
||||||
use super::Importer;
|
use super::Importer;
|
||||||
|
|
||||||
pub(crate) mod parse;
|
pub(crate) mod parse;
|
||||||
|
mod util;
|
||||||
|
|
||||||
pub(crate) struct YAMLImporter {}
|
pub(crate) struct YAMLImporter {}
|
||||||
|
|
||||||
|
@ -150,7 +154,7 @@ impl TryFrom<YAMLVariable> for Variable {
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
name: yaml_var.name,
|
name: yaml_var.name,
|
||||||
var_type: yaml_var.var_type,
|
var_type: yaml_var.var_type,
|
||||||
params: yaml_var.params,
|
params: convert_params(yaml_var.params)?,
|
||||||
id: next_id(),
|
id: next_id(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -159,8 +163,7 @@ impl TryFrom<YAMLVariable> for Variable {
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{matches::Match, util::tests::use_test_directory};
|
use crate::{matches::{Match, Params, Value}, util::tests::use_test_directory};
|
||||||
use serde_yaml::{Mapping, Value};
|
|
||||||
use std::fs::create_dir_all;
|
use std::fs::create_dir_all;
|
||||||
|
|
||||||
fn create_match(yaml: &str) -> Result<Match> {
|
fn create_match(yaml: &str) -> Result<Match> {
|
||||||
|
@ -331,8 +334,8 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn vars_maps_correctly() {
|
fn vars_maps_correctly() {
|
||||||
let mut params = Mapping::new();
|
let mut params = Params::new();
|
||||||
params.insert(Value::String("param1".to_string()), Value::Bool(true));
|
params.insert("param1".to_string(), Value::Bool(true));
|
||||||
let vars = vec![Variable {
|
let vars = vec![Variable {
|
||||||
name: "var1".to_string(),
|
name: "var1".to_string(),
|
||||||
var_type: "test".to_string(),
|
var_type: "test".to_string(),
|
||||||
|
@ -371,7 +374,7 @@ mod tests {
|
||||||
let vars = vec![Variable {
|
let vars = vec![Variable {
|
||||||
name: "var1".to_string(),
|
name: "var1".to_string(),
|
||||||
var_type: "test".to_string(),
|
var_type: "test".to_string(),
|
||||||
params: Mapping::new(),
|
params: Params::new(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
}];
|
}];
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
@ -444,7 +447,7 @@ mod tests {
|
||||||
let vars = vec![Variable {
|
let vars = vec![Variable {
|
||||||
name: "var1".to_string(),
|
name: "var1".to_string(),
|
||||||
var_type: "test".to_string(),
|
var_type: "test".to_string(),
|
||||||
params: Mapping::new(),
|
params: Params::new(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
}];
|
}];
|
||||||
|
|
||||||
|
|
167
espanso-config/src/matches/group/loader/yaml/util.rs
Normal file
167
espanso-config/src/matches/group/loader/yaml/util.rs
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
/*
|
||||||
|
* 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::convert::TryInto;
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
use serde_yaml::Mapping;
|
||||||
|
use thiserror::Error;
|
||||||
|
|
||||||
|
use crate::matches::{Number, Params, Value};
|
||||||
|
|
||||||
|
pub(crate) fn convert_params(m: Mapping) -> Result<Params> {
|
||||||
|
let mut params = Params::new();
|
||||||
|
|
||||||
|
for (key, value) in m {
|
||||||
|
let key = key.as_str().ok_or(ConversionError::InvalidKeyFormat)?;
|
||||||
|
let value = convert_value(value)?;
|
||||||
|
params.insert(key.to_owned(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(params)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn convert_value(value: serde_yaml::Value) -> Result<Value> {
|
||||||
|
Ok(match value {
|
||||||
|
serde_yaml::Value::Null => Value::Null,
|
||||||
|
serde_yaml::Value::Bool(val) => Value::Bool(val),
|
||||||
|
serde_yaml::Value::Number(n) => {
|
||||||
|
if n.is_i64() {
|
||||||
|
Value::Number(Number::Integer(
|
||||||
|
n.as_i64().ok_or(ConversionError::InvalidNumberFormat)?,
|
||||||
|
))
|
||||||
|
} else if n.is_u64() {
|
||||||
|
Value::Number(Number::Integer(
|
||||||
|
n.as_u64()
|
||||||
|
.ok_or(ConversionError::InvalidNumberFormat)?
|
||||||
|
.try_into()?,
|
||||||
|
))
|
||||||
|
} else if n.is_f64() {
|
||||||
|
Value::Number(Number::Float(
|
||||||
|
n.as_f64()
|
||||||
|
.ok_or(ConversionError::InvalidNumberFormat)?
|
||||||
|
.into(),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
return Err(ConversionError::InvalidNumberFormat.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
serde_yaml::Value::String(s) => Value::String(s),
|
||||||
|
serde_yaml::Value::Sequence(arr) => Value::Array(
|
||||||
|
arr
|
||||||
|
.into_iter()
|
||||||
|
.map(convert_value)
|
||||||
|
.collect::<Result<Vec<Value>>>()?,
|
||||||
|
),
|
||||||
|
serde_yaml::Value::Mapping(m) => Value::Object(convert_params(m)?),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Error, Debug)]
|
||||||
|
pub enum ConversionError {
|
||||||
|
#[error("invalid key format")]
|
||||||
|
InvalidKeyFormat,
|
||||||
|
|
||||||
|
#[error("invalid number format")]
|
||||||
|
InvalidNumberFormat,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_value_null() {
|
||||||
|
assert_eq!(convert_value(serde_yaml::Value::Null).unwrap(), Value::Null);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_value_bool() {
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Bool(true)).unwrap(),
|
||||||
|
Value::Bool(true)
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Bool(false)).unwrap(),
|
||||||
|
Value::Bool(false)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_value_number() {
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Number(0.into())).unwrap(),
|
||||||
|
Value::Number(Number::Integer(0))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Number((-100).into())).unwrap(),
|
||||||
|
Value::Number(Number::Integer(-100))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Number(1.5.into())).unwrap(),
|
||||||
|
Value::Number(Number::Float(1.5.into()))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
#[test]
|
||||||
|
fn convert_value_string() {
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::String("hello".to_string())).unwrap(),
|
||||||
|
Value::String("hello".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
#[test]
|
||||||
|
fn convert_value_array() {
|
||||||
|
assert_eq!(
|
||||||
|
convert_value(serde_yaml::Value::Sequence(vec![
|
||||||
|
serde_yaml::Value::Bool(true),
|
||||||
|
serde_yaml::Value::Null,
|
||||||
|
]))
|
||||||
|
.unwrap(),
|
||||||
|
Value::Array(vec![Value::Bool(true), Value::Null,])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_value_params() {
|
||||||
|
let mut mapping = serde_yaml::Mapping::new();
|
||||||
|
mapping.insert(serde_yaml::Value::String("test".to_string()), serde_yaml::Value::Null);
|
||||||
|
|
||||||
|
let mut expected = Params::new();
|
||||||
|
expected.insert("test".to_string(), Value::Null);
|
||||||
|
assert_eq!(convert_value(serde_yaml::Value::Mapping(mapping)).unwrap(), Value::Object(expected));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_params_works_correctly() {
|
||||||
|
let mut mapping = serde_yaml::Mapping::new();
|
||||||
|
mapping.insert(serde_yaml::Value::String("test".to_string()), serde_yaml::Value::Null);
|
||||||
|
|
||||||
|
let mut expected = Params::new();
|
||||||
|
expected.insert("test".to_string(), Value::Null);
|
||||||
|
assert_eq!(convert_params(mapping).unwrap(), expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn convert_params_invalid_key_type() {
|
||||||
|
let mut mapping = serde_yaml::Mapping::new();
|
||||||
|
mapping.insert(serde_yaml::Value::Null, serde_yaml::Value::Null);
|
||||||
|
|
||||||
|
assert!(convert_params(mapping).is_err());
|
||||||
|
}
|
||||||
|
}
|
|
@ -17,7 +17,9 @@
|
||||||
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
|
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use serde_yaml::Mapping;
|
use std::collections::{BTreeMap};
|
||||||
|
use enum_as_inner::EnumAsInner;
|
||||||
|
use ordered_float::OrderedFloat;
|
||||||
|
|
||||||
use crate::counter::{StructId};
|
use crate::counter::{StructId};
|
||||||
|
|
||||||
|
@ -107,7 +109,7 @@ pub struct Variable {
|
||||||
pub id: StructId,
|
pub id: StructId,
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub var_type: String,
|
pub var_type: String,
|
||||||
pub params: Mapping,
|
pub params: Params,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Variable {
|
impl Default for Variable {
|
||||||
|
@ -116,7 +118,25 @@ impl Default for Variable {
|
||||||
id: 0,
|
id: 0,
|
||||||
name: String::new(),
|
name: String::new(),
|
||||||
var_type: String::new(),
|
var_type: String::new(),
|
||||||
params: Mapping::new(),
|
params: Params::new(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub type Params = BTreeMap<String, Value>;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Hash, EnumAsInner)]
|
||||||
|
pub enum Value {
|
||||||
|
Null,
|
||||||
|
Bool(bool),
|
||||||
|
Number(Number),
|
||||||
|
String(String),
|
||||||
|
Array(Vec<Value>),
|
||||||
|
Object(Params),
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Eq, Hash, PartialEq)]
|
||||||
|
pub enum Number {
|
||||||
|
Integer(i64),
|
||||||
|
Float(OrderedFloat<f64>),
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user