/*
 * 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 .
 */
use anyhow::Result;
use serde::{Deserialize, Serialize};
use std::convert::TryFrom;
use crate::util::is_yaml_empty;
use super::ParsedConfig;
#[derive(Debug, Serialize, Deserialize, Clone)]
pub(crate) struct YAMLConfig {
  #[serde(default)]
  pub label: Option,
  #[serde(default)]
  pub backend: Option,
  #[serde(default)]
  pub clipboard_threshold: Option,
  #[serde(default)]
  pub pre_paste_delay: Option,
  #[serde(default)]
  pub toggle_key: Option,
  #[serde(default)]
  pub auto_restart: Option,
  #[serde(default)]
  pub preserve_clipboard: Option,
  #[serde(default)]
  pub restore_clipboard_delay: Option,
  #[serde(default)]
  pub paste_shortcut_event_delay: Option,
  
  #[serde(default)]
  pub paste_shortcut: Option,
  #[serde(default)]
  pub disable_x11_fast_inject: Option,
  #[serde(default)]
  pub inject_delay: Option,
  #[serde(default)]
  pub key_delay: Option,
  #[serde(default)]
  pub backspace_delay: Option,
  #[serde(default)]
  pub word_separators: Option>,
  #[serde(default)]
  pub backspace_limit: Option,
  // Include/Exclude
  #[serde(default)]
  pub includes: Option>,
  #[serde(default)]
  pub excludes: Option>,
  #[serde(default)]
  pub extra_includes: Option>,
  #[serde(default)]
  pub extra_excludes: Option>,
  #[serde(default)]
  pub use_standard_includes: Option,
  // Filters
  #[serde(default)]
  pub filter_title: Option,
  #[serde(default)]
  pub filter_class: Option,
  #[serde(default)]
  pub filter_exec: Option,
  #[serde(default)]
  pub filter_os: Option,
}
impl YAMLConfig {
  pub fn parse_from_str(yaml: &str) -> Result {
    // Because an empty string is not valid YAML but we want to support it anyway
    if is_yaml_empty(yaml) {
      return Ok(serde_yaml::from_str(
        "arbitrary_field_that_will_not_block_the_parser: true",
      )?);
    }
    Ok(serde_yaml::from_str(yaml)?)
  }
}
impl TryFrom for ParsedConfig {
  type Error = anyhow::Error;
  fn try_from(yaml_config: YAMLConfig) -> Result {
    Ok(Self {
      label: yaml_config.label,
      backend: yaml_config.backend,
      clipboard_threshold: yaml_config.clipboard_threshold,
      auto_restart: yaml_config.auto_restart,
      toggle_key: yaml_config.toggle_key,
      preserve_clipboard: yaml_config.preserve_clipboard,
      paste_shortcut: yaml_config.paste_shortcut,
      disable_x11_fast_inject: yaml_config.disable_x11_fast_inject,
      inject_delay: yaml_config.inject_delay,
      key_delay: yaml_config.key_delay.or(yaml_config.backspace_delay),
      word_separators: yaml_config.word_separators,
      backspace_limit: yaml_config.backspace_limit,
      pre_paste_delay: yaml_config.pre_paste_delay,
      restore_clipboard_delay: yaml_config.restore_clipboard_delay,
      paste_shortcut_event_delay: yaml_config.paste_shortcut_event_delay,
      use_standard_includes: yaml_config.use_standard_includes,
      includes: yaml_config.includes,
      extra_includes: yaml_config.extra_includes,
      excludes: yaml_config.excludes,
      extra_excludes: yaml_config.extra_excludes,
      filter_class: yaml_config.filter_class,
      filter_exec: yaml_config.filter_exec,
      filter_os: yaml_config.filter_os,
      filter_title: yaml_config.filter_title,
    })
  }
}
#[cfg(test)]
mod tests {
  use super::*;
  use std::convert::TryInto;
  #[test]
  fn conversion_to_parsed_config_works_correctly() {
    let config = YAMLConfig::parse_from_str(
      r#"
    label: "test"
    backend: clipboard
    clipboard_threshold: 200
    pre_paste_delay: 300
    toggle_key: CTRL
    auto_restart: false
    preserve_clipboard: false
    restore_clipboard_delay: 400
    paste_shortcut: CTRL+ALT+V
    paste_shortcut_event_delay: 10
    disable_x11_fast_inject: true
    inject_delay: 10
    key_delay: 20
    backspace_delay: 30
    word_separators: ["'", "."]
    backspace_limit: 10
      
    use_standard_includes: true
    includes: ["test1"]
    extra_includes: ["test2"]
    excludes: ["test3"]
    extra_excludes: ["test4"]
    
    filter_class: "test5"
    filter_exec: "test6"
    filter_os: "test7"
    filter_title: "test8"
    "#,
    )
    .unwrap();
    let parsed_config: ParsedConfig = config.try_into().unwrap();
    assert_eq!(
      parsed_config,
      ParsedConfig {
        label: Some("test".to_string()),
        backend: Some("clipboard".to_string()),
        clipboard_threshold: Some(200),
        auto_restart: Some(false),
        preserve_clipboard: Some(false),
        restore_clipboard_delay: Some(400),
        paste_shortcut: Some("CTRL+ALT+V".to_string()),
        paste_shortcut_event_delay: Some(10),
        disable_x11_fast_inject: Some(true),
        inject_delay: Some(10),
        key_delay: Some(20),
        backspace_limit: Some(10),
        pre_paste_delay: Some(300),
        
        toggle_key: Some("CTRL".to_string()),
        word_separators: Some(vec!["'".to_owned(), ".".to_owned()]),
        use_standard_includes: Some(true),
        includes: Some(vec!["test1".to_string()]),
        extra_includes: Some(vec!["test2".to_string()]),
        excludes: Some(vec!["test3".to_string()]),
        extra_excludes: Some(vec!["test4".to_string()]),
        filter_class: Some("test5".to_string()),
        filter_exec: Some("test6".to_string()),
        filter_os: Some("test7".to_string()),
        filter_title: Some("test8".to_string()),
      }
    )
  }
}