meli/melib/src/conf.rs

382 lines
11 KiB
Rust

/*
* meli - configuration module.
*
* Copyright 2017 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 <http://www.gnu.org/licenses/>.
*/
//! Basic mail account configuration to use with
//! [`backends`](./backends/index.html)
use std::{collections::HashMap, path::Path};
use crate::{
backends::SpecialUsageMailbox,
email::Address,
error::{Error, ErrorKind, Result},
};
pub use crate::{SortField, SortOrder};
#[derive(Clone, Debug, Default, Serialize)]
pub struct AccountSettings {
pub name: String,
/// Name of mailbox that is the root of the mailbox hierarchy.
///
/// Note that this may have special or no meaning depending on the e-mail
/// backend.
pub root_mailbox: String,
pub format: String,
pub identity: String,
pub extra_identities: Vec<String>,
pub read_only: bool,
pub display_name: Option<String>,
#[serde(default)]
pub order: (SortField, SortOrder),
pub subscribed_mailboxes: Vec<String>,
#[serde(default)]
pub mailboxes: HashMap<String, MailboxConf>,
#[serde(default)]
pub manual_refresh: bool,
#[serde(flatten)]
pub extra: HashMap<String, String>,
}
impl AccountSettings {
/// Create the account's display name from fields
/// [`AccountSettings::identity`] and [`AccountSettings::display_name`].
pub fn make_display_name(&self) -> Address {
Address::new(self.display_name.clone(), self.identity.clone())
}
pub fn order(&self) -> Option<(SortField, SortOrder)> {
Some(self.order)
}
#[cfg(feature = "vcard")]
pub fn vcard_folder(&self) -> Option<&str> {
self.extra.get("vcard_folder").map(String::as_str)
}
/// Get the server password, either directly from the `server_password`
/// settings value, or by running the `server_password_command` and reading
/// the output.
pub fn server_password(&self) -> Result<String> {
if let Some(cmd) = self.extra.get("server_password_command") {
let output = std::process::Command::new("sh")
.args(["-c", cmd])
.stdin(std::process::Stdio::piped())
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.output()?;
if output.status.success() {
Ok(std::str::from_utf8(&output.stdout)?.trim_end().to_string())
} else {
Err(Error::new(format!(
"({}) server_password_command `{}` returned {}: {}",
self.name,
cmd,
output.status,
String::from_utf8_lossy(&output.stderr)
)))
}
} else if let Some(pass) = self.extra.get("server_password") {
Ok(pass.to_owned())
} else {
Err(Error::new(
"Configuration error: connection requires either server_password or \
server_password_command",
))
}
}
pub fn validate_config(&mut self) -> Result<()> {
#[cfg(feature = "vcard")]
{
if let Some(folder) = self.extra.remove("vcard_folder") {
let path = Path::new(&folder);
if !matches!(path.try_exists(), Ok(true)) {
return Err(Error::new(format!(
"`vcard_folder` path {} does not exist",
path.display()
))
.set_details("`vcard_folder` must be a path of a folder containing .vcf files")
.set_kind(ErrorKind::Configuration));
}
if !path.is_dir() {
return Err(Error::new(format!(
"`vcard_folder` path {} is not a directory",
path.display()
))
.set_details("`vcard_folder` must be a path of a folder containing .vcf files")
.set_kind(ErrorKind::Configuration));
}
}
}
Ok(())
}
}
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(default)]
pub struct MailboxConf {
#[serde(alias = "rename")]
pub alias: Option<String>,
#[serde(default = "false_val")]
pub autoload: bool,
#[serde(default)]
pub subscribe: ToggleFlag,
#[serde(default)]
pub ignore: ToggleFlag,
#[serde(default = "none")]
pub usage: Option<SpecialUsageMailbox>,
#[serde(default = "none")]
pub sort_order: Option<usize>,
#[serde(default = "none")]
pub encoding: Option<String>,
#[serde(flatten)]
pub extra: HashMap<String, String>,
}
impl Default for MailboxConf {
fn default() -> Self {
Self {
alias: None,
autoload: false,
subscribe: ToggleFlag::Unset,
ignore: ToggleFlag::Unset,
usage: None,
sort_order: None,
encoding: None,
extra: HashMap::default(),
}
}
}
impl MailboxConf {
pub fn alias(&self) -> Option<&str> {
self.alias.as_deref()
}
}
pub const fn true_val() -> bool {
true
}
pub const fn false_val() -> bool {
false
}
pub const fn none<T>() -> Option<T> {
None
}
pub use config_field_types::*;
pub mod config_field_types {
use serde::{Deserialize, Deserializer, Serialize, Serializer};
macro_rules! named_unit_variant {
($variant:ident) => {
pub mod $variant {
pub fn deserialize<'de, D>(deserializer: D) -> Result<(), D::Error>
where
D: serde::Deserializer<'de>,
{
struct V;
impl<'de> serde::de::Visitor<'de> for V {
type Value = ();
fn expecting(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
f.write_str(concat!("\"", stringify!($variant), "\""))
}
fn visit_str<E: serde::de::Error>(
self,
value: &str,
) -> Result<Self::Value, E> {
if value == stringify!($variant) {
Ok(())
} else {
Err(E::invalid_value(serde::de::Unexpected::Str(value), &self))
}
}
}
deserializer.deserialize_str(V)
}
}
};
}
pub mod strings {
named_unit_variant!(ask);
}
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub enum ToggleFlag {
#[default]
Unset,
InternalVal(bool),
False,
True,
}
impl From<bool> for ToggleFlag {
fn from(val: bool) -> Self {
if val {
Self::True
} else {
Self::False
}
}
}
impl ToggleFlag {
pub fn is_unset(&self) -> bool {
Self::Unset == *self
}
pub fn is_internal(&self) -> bool {
matches!(self, Self::InternalVal(_))
}
pub fn is_false(&self) -> bool {
matches!(self, Self::False | Self::InternalVal(false))
}
pub fn is_true(&self) -> bool {
matches!(self, Self::True | Self::InternalVal(true))
}
}
impl Serialize for ToggleFlag {
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
where
S: Serializer,
{
match self {
Self::Unset | Self::InternalVal(_) => serializer.serialize_none(),
Self::False => serializer.serialize_bool(false),
Self::True => serializer.serialize_bool(true),
}
}
}
impl<'de> Deserialize<'de> for ToggleFlag {
fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
where
D: Deserializer<'de>,
{
#[derive(Deserialize)]
#[serde(untagged)]
enum InnerToggleFlag {
Bool(bool),
}
let s = <InnerToggleFlag>::deserialize(deserializer);
Ok(
match s.map_err(|err| {
serde::de::Error::custom(format!(
r#"expected one of "true", "false", found `{}`"#,
err
))
})? {
InnerToggleFlag::Bool(true) => Self::True,
InnerToggleFlag::Bool(false) => Self::False,
},
)
}
}
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub enum ActionFlag {
InternalVal(bool),
False,
True,
#[default]
Ask,
}
impl From<bool> for ActionFlag {
fn from(val: bool) -> Self {
if val {
Self::True
} else {
Self::False
}
}
}
impl ActionFlag {
pub fn is_internal(&self) -> bool {
matches!(self, Self::InternalVal(_))
}
pub fn is_ask(&self) -> bool {
matches!(self, Self::Ask)
}
pub fn is_false(&self) -> bool {
matches!(self, Self::False | Self::InternalVal(false))
}
pub fn is_true(&self) -> bool {
matches!(self, Self::True | Self::InternalVal(true))
}
}
impl Serialize for ActionFlag {
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
where
S: Serializer,
{
match self {
Self::InternalVal(_) => serializer.serialize_none(),
Self::False => serializer.serialize_bool(false),
Self::True => serializer.serialize_bool(true),
Self::Ask => serializer.serialize_str("ask"),
}
}
}
impl<'de> Deserialize<'de> for ActionFlag {
fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
where
D: Deserializer<'de>,
{
#[derive(Deserialize)]
#[serde(untagged)]
enum InnerActionFlag {
Bool(bool),
#[serde(with = "strings::ask")]
Ask,
}
let s = <InnerActionFlag>::deserialize(deserializer);
Ok(
match s.map_err(|err| {
serde::de::Error::custom(format!(
r#"expected one of "true", "false", "ask", found `{}`"#,
err
))
})? {
InnerActionFlag::Bool(true) => Self::True,
InnerActionFlag::Bool(false) => Self::False,
InnerActionFlag::Ask => Self::Ask,
},
)
}
}
}