dprint 0.8.0

Binary for dprint.
use std::collections::HashMap;
use dprint_core::types::ErrBox;
use dprint_core::configuration::{ConfigKeyMap};

use crate::plugins::Plugin;
use super::{ConfigMap, ConfigMapValue};

pub fn get_plugin_config_map(
    plugin: &Box<dyn Plugin>,
    config_map: &mut ConfigMap,
) -> Result<ConfigKeyMap, ErrBox> {
    match get_plugin_config_map_inner(plugin, config_map) {
        Ok(result) => Ok(result),
        Err(err) => err!("Error initializing from configuration file. {}", err.to_string()),
    }
}

fn get_plugin_config_map_inner(
    plugin: &Box<dyn Plugin>,
    config_map: &mut ConfigMap,
) -> Result<ConfigKeyMap, ErrBox> {
    let config_key = plugin.config_key();

    if let Some(plugin_config_map) = config_map.remove(config_key) {
        if let ConfigMapValue::HashMap(plugin_config_map) = plugin_config_map {
            let mut plugin_config_map = plugin_config_map;
            plugin_config_map.remove("$schema");
            Ok(plugin_config_map)
        } else {
            return err!("Expected the configuration property '{}' to be an object.", config_key);
        }
    } else {
        Ok(HashMap::new())
    }
}

#[cfg(test)]
mod tests {
    use std::collections::HashMap;
    use crate::plugins::{TestPlugin, Plugin};
    use dprint_core::configuration::{ConfigKeyValue};

    use super::super::{ConfigMapValue, ConfigMap};
    use super::*;

    #[test]
    fn it_should_get_config_for_plugin() {
        let mut config_map = HashMap::new();
        let mut ts_config_map = HashMap::new();
        ts_config_map.insert(String::from("lineWidth"), ConfigKeyValue::from_i32(40));
        ts_config_map.insert(String::from("$schema"), ConfigKeyValue::from_str("test"));

        config_map.insert(String::from("lineWidth"), ConfigMapValue::from_i32(80));
        config_map.insert(String::from("typescript"), ConfigMapValue::HashMap(ts_config_map.clone()));
        let plugin = create_plugin();
        let result = get_plugin_config_map(&(Box::new(plugin) as Box<dyn Plugin>), &mut config_map).unwrap();
        ts_config_map.remove("$schema"); // should not be in result
        assert_eq!(result, ts_config_map.clone());
        assert_eq!(config_map.contains_key("typescript"), false);
    }

    #[test]
    fn it_should_error_plugin_key_is_not_object() {
        let mut config_map = HashMap::new();
        config_map.insert(String::from("lineWidth"), ConfigMapValue::from_i32(80));
        config_map.insert(String::from("typescript"), ConfigMapValue::from_str(""));
        assert_errors(
            &mut config_map,
            "Expected the configuration property 'typescript' to be an object.",
        );
    }

    fn assert_errors(config_map: &mut ConfigMap, message: &str) {
        let test_plugin = Box::new(create_plugin()) as Box<dyn Plugin>;
        let result = get_plugin_config_map(&test_plugin, config_map);
        assert_eq!(result.err().unwrap().to_string(), format!("Error initializing from configuration file. {}", message));
    }

    fn create_plugin() -> TestPlugin {
        TestPlugin::new(
            "dprint-plugin-typescript",
            "typescript",
            vec![".ts"]
        )
    }
}