/* * meli - configuration module. * * Copyright 2019 Manos Pitsidianakis * * This file is part of meli. * * meli 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. * * meli 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 meli. If not, see . */ //! E-mail tag configuration and {de,}serializing. use super::DotAddressable; use crate::terminal::Color; use melib::{MeliError, Result}; use serde::{Deserialize, Deserializer}; use std::collections::{hash_map::DefaultHasher, HashMap, HashSet}; use std::hash::Hasher; #[derive(Debug, Deserialize, Clone, Serialize)] pub struct TagsSettings { #[serde(default, deserialize_with = "tag_color_de")] pub colors: HashMap, #[serde(default, deserialize_with = "tag_set_de", alias = "ignore-tags")] pub ignore_tags: HashSet, } impl Default for TagsSettings { fn default() -> Self { TagsSettings { colors: Default::default(), ignore_tags: Default::default(), } } } pub fn tag_set_de<'de, D, T: std::convert::From>>( deserializer: D, ) -> std::result::Result where D: Deserializer<'de>, { Ok(>::deserialize(deserializer)? .into_iter() .map(|tag| { let mut hasher = DefaultHasher::new(); hasher.write(tag.as_bytes()); hasher.finish() }) .collect::>() .into()) } pub fn tag_color_de<'de, D, T: std::convert::From>>( deserializer: D, ) -> std::result::Result where D: Deserializer<'de>, { #[derive(Deserialize)] #[serde(untagged)] enum _Color { B(u8), C(Color), } Ok(>::deserialize(deserializer)? .into_iter() .map(|(tag, color)| { let mut hasher = DefaultHasher::new(); hasher.write(tag.as_bytes()); ( hasher.finish(), match color { _Color::B(b) => Color::Byte(b), _Color::C(c) => c, }, ) }) .collect::>() .into()) } impl DotAddressable for TagsSettings { fn lookup(&self, parent_field: &str, path: &[&str]) -> Result { match path.first() { Some(field) => { let tail = &path[1..]; match *field { "colors" => self.colors.lookup(field, tail), "ignore_tags" => self.ignore_tags.lookup(field, tail), other => Err(MeliError::new(format!( "{} has no field named {}", parent_field, other ))), } } None => Ok(toml::to_string(self).map_err(|err| err.to_string())?), } } }