meli/src/conf.rs

1423 lines
47 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/>.
*/
/*! Configuration logic and `config.toml` interfaces. */
extern crate bincode;
extern crate serde;
extern crate toml;
extern crate xdg;
use std::{
collections::HashSet,
io::Read,
process::{Command, Stdio},
};
use melib::{backends::TagHash, search::Query, StderrLogger};
use crate::{conf::deserializers::non_empty_string, terminal::Color};
#[rustfmt::skip]
mod overrides;
pub use overrides::*;
pub mod composing;
pub mod notifications;
pub mod pager;
pub mod pgp;
pub mod tags;
#[macro_use]
pub mod shortcuts;
mod listing;
pub mod terminal;
mod themes;
pub use themes::*;
pub mod accounts;
use std::{
collections::HashMap,
env,
fs::OpenOptions,
io::{self, BufRead, Write},
os::unix::fs::PermissionsExt,
path::{Path, PathBuf},
};
use indexmap::IndexMap;
pub use melib::thread::{SortField, SortOrder};
use melib::{
conf::{AccountSettings, MailboxConf, ToggleFlag},
error::*,
};
use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
pub use self::{accounts::Account, composing::*, pgp::*, shortcuts::*, tags::*};
use self::{
default_vals::*, listing::ListingSettings, notifications::NotificationsSettings,
terminal::TerminalSettings,
};
use crate::pager::PagerSettings;
#[macro_export]
macro_rules! split_command {
($cmd:expr) => {{
$cmd.split_whitespace().collect::<Vec<&str>>()
}};
}
#[macro_export]
macro_rules! account_settings {
($context:ident[$account_hash:expr].$setting:ident.$field:ident) => {{
$context.accounts[&$account_hash]
.settings
.conf_override
.$setting
.$field
.as_ref()
.unwrap_or(&$context.settings.$setting.$field)
}};
}
#[macro_export]
macro_rules! mailbox_settings {
($context:ident[$account_hash:expr][$mailbox_path:expr].$setting:ident.$field:ident) => {{
$context.accounts[&$account_hash][$mailbox_path]
.conf
.conf_override
.$setting
.$field
.as_ref()
.or($context.accounts[&$account_hash]
.settings
.conf_override
.$setting
.$field
.as_ref())
.unwrap_or(&$context.settings.$setting.$field)
}};
}
#[derive(Default, Debug, Clone, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct MailUIConf {
#[serde(default)]
pub pager: PagerSettingsOverride,
#[serde(default)]
pub listing: ListingSettingsOverride,
#[serde(default)]
pub notifications: NotificationsSettingsOverride,
#[serde(default)]
pub shortcuts: ShortcutsOverride,
#[serde(default)]
pub composing: ComposingSettingsOverride,
#[serde(default)]
pub identity: Option<String>,
#[serde(default)]
pub tags: TagsSettingsOverride,
#[serde(default)]
pub themes: Option<Themes>,
#[serde(default)]
pub pgp: PGPSettingsOverride,
}
#[derive(Debug, Default, Clone, Serialize, Deserialize)]
#[serde(default)]
pub struct FileMailboxConf {
#[serde(flatten)]
pub conf_override: MailUIConf,
#[serde(default = "false_val")]
pub collapsed: bool,
#[serde(flatten)]
pub mailbox_conf: MailboxConf,
}
impl FileMailboxConf {
pub fn conf_override(&self) -> &MailUIConf {
&self.conf_override
}
pub fn mailbox_conf(&self) -> &MailboxConf {
&self.mailbox_conf
}
}
use crate::conf::deserializers::extra_settings;
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
pub struct FileAccount {
pub root_mailbox: String,
pub format: String,
pub identity: String,
#[serde(default)]
pub extra_identities: Vec<String>,
#[serde(default = "none")]
pub display_name: Option<String>,
#[serde(default = "false_val")]
pub read_only: bool,
#[serde(default)]
pub subscribed_mailboxes: Vec<String>,
#[serde(default)]
pub mailboxes: IndexMap<String, FileMailboxConf>,
#[serde(default)]
pub search_backend: SearchBackend,
#[serde(default)]
pub order: (SortField, SortOrder),
#[serde(default = "false_val")]
pub manual_refresh: bool,
#[serde(default = "none")]
pub refresh_command: Option<String>,
#[serde(flatten)]
pub conf_override: MailUIConf,
#[serde(flatten)]
#[serde(deserialize_with = "extra_settings")]
pub extra: IndexMap<String, String>, /* use custom deserializer to convert any given value
* (eg bool, number, etc) to string */
}
impl FileAccount {
pub fn mailboxes(&self) -> &IndexMap<String, FileMailboxConf> {
&self.mailboxes
}
pub fn mailbox(&self) -> &str {
&self.root_mailbox
}
pub fn search_backend(&self) -> &SearchBackend {
&self.search_backend
}
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct FileSettings {
pub accounts: IndexMap<String, FileAccount>,
#[serde(default)]
pub pager: PagerSettings,
#[serde(default)]
pub listing: ListingSettings,
#[serde(default)]
pub notifications: NotificationsSettings,
#[serde(default)]
pub shortcuts: Shortcuts,
pub composing: ComposingSettings,
#[serde(default)]
pub tags: TagsSettings,
#[serde(default)]
pub pgp: PGPSettings,
#[serde(default)]
pub terminal: TerminalSettings,
#[serde(default)]
pub log: LogSettings,
}
#[derive(Debug, Clone, Default, Serialize)]
pub struct AccountConf {
pub account: AccountSettings,
pub conf: FileAccount,
pub conf_override: MailUIConf,
pub mailbox_confs: IndexMap<String, FileMailboxConf>,
}
impl AccountConf {
pub fn account(&self) -> &AccountSettings {
&self.account
}
pub fn account_mut(&mut self) -> &mut AccountSettings {
&mut self.account
}
pub fn conf(&self) -> &FileAccount {
&self.conf
}
pub fn conf_mut(&mut self) -> &mut FileAccount {
&mut self.conf
}
}
impl From<FileAccount> for AccountConf {
fn from(x: FileAccount) -> Self {
let format = x.format.to_lowercase();
let root_mailbox = x.root_mailbox.clone();
let identity = x.identity.clone();
let display_name = x.display_name.clone();
let order = x.order;
let mailboxes = x
.mailboxes
.iter()
.map(|(k, v)| (k.clone(), v.mailbox_conf.clone()))
.collect();
let acc = AccountSettings {
name: String::new(),
root_mailbox,
format,
identity,
extra_identities: x.extra_identities.clone(),
read_only: x.read_only,
display_name,
order,
subscribed_mailboxes: x.subscribed_mailboxes.clone(),
mailboxes,
manual_refresh: x.manual_refresh,
extra: x.extra.clone().into_iter().collect(),
};
let mailbox_confs = x.mailboxes.clone();
AccountConf {
account: acc,
conf_override: x.conf_override.clone(),
conf: x,
mailbox_confs,
}
}
}
pub fn get_config_file() -> Result<PathBuf> {
let xdg_dirs = xdg::BaseDirectories::with_prefix("meli").map_err(|err| {
Error::new(format!(
"Could not detect XDG directories for user: {}",
err
))
.set_source(Some(std::sync::Arc::new(Box::new(err))))
})?;
match env::var("MELI_CONFIG") {
Ok(path) => Ok(PathBuf::from(path)),
Err(_) => Ok(xdg_dirs
.place_config_file("config.toml")
.chain_err_summary(|| {
format!(
"Cannot create configuration directory in {}",
xdg_dirs.get_config_home().display()
)
})?),
}
}
pub fn get_included_configs(conf_path: PathBuf) -> Result<Vec<PathBuf>> {
const M4_PREAMBLE: &str = r#"divert(-1)dnl
define(`include', `divert(0)$1
divert(-1)
')dnl
changequote(`"', `"')dnl
"#;
let mut ret = vec![];
let prefix = conf_path.parent().unwrap().to_path_buf();
let mut stack = vec![(None::<PathBuf>, conf_path)];
let mut contents = String::new();
while let Some((parent, p)) = stack.pop() {
if !p.exists() || p.is_dir() {
return Err(format!(
"Path {}{included}{in_parent} {msg}.",
p.display(),
included = if parent.is_some() {
" which is included in "
} else {
""
},
in_parent = if let Some(parent) = parent {
std::borrow::Cow::Owned(parent.display().to_string())
} else {
std::borrow::Cow::Borrowed("")
},
msg = if !p.exists() {
"does not exist"
} else {
"is a directory, not a text file"
}
)
.into());
}
contents.clear();
let mut file = std::fs::File::open(&p)?;
file.read_to_string(&mut contents)?;
let mut handle = match Command::new("m4")
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.spawn()
{
Ok(handle) => handle,
Err(error) => match error.kind() {
io::ErrorKind::NotFound => {
return Err("`m4` executable not found. Please install.".into())
}
_ => return Err(error.into()),
},
};
let mut stdin = handle.stdin.take().unwrap();
stdin.write_all(M4_PREAMBLE.as_bytes())?;
stdin.write_all(contents.as_bytes())?;
drop(stdin);
let stdout = handle.wait_with_output()?.stdout.clone();
for subpath in stdout.lines() {
let subpath = subpath?;
let path = &Path::new(&subpath);
if path.is_absolute() {
stack.push((Some(p.clone()), path.to_path_buf()));
} else {
stack.push((Some(p.clone()), prefix.join(path)));
}
}
ret.push(p);
}
Ok(ret)
}
pub fn expand_config(conf_path: PathBuf) -> Result<String> {
let _paths = get_included_configs(conf_path.clone())?;
const M4_PREAMBLE: &str = r#"define(`builtin_include', defn(`include'))dnl
define(`include', `builtin_include(substr($1,1,decr(decr(len($1)))))dnl')dnl
"#;
let mut contents = String::new();
contents.clear();
let mut file = std::fs::File::open(&conf_path)?;
file.read_to_string(&mut contents)?;
let mut handle = Command::new("m4")
.current_dir(conf_path.parent().unwrap_or(Path::new("/")))
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.spawn()?;
let mut stdin = handle.stdin.take().unwrap();
stdin.write_all(M4_PREAMBLE.as_bytes())?;
stdin.write_all(contents.as_bytes())?;
drop(stdin);
let stdout = handle.wait_with_output()?.stdout;
Ok(String::from_utf8_lossy(&stdout).to_string())
}
struct Ask {
message: String,
}
impl Ask {
fn run(self) -> bool {
let mut buffer = String::new();
let stdin = io::stdin();
let mut handle = stdin.lock();
print!("{} [Y/n] ", &self.message);
let _ = io::stdout().flush();
loop {
buffer.clear();
handle
.read_line(&mut buffer)
.expect("Could not read from stdin.");
match buffer.trim() {
"" | "Y" | "y" | "yes" | "YES" | "Yes" => {
return true;
}
"n" | "N" | "no" | "No" | "NO" => {
return false;
}
_ => {
print!("\n{} [Y/n] ", &self.message);
let _ = io::stdout().flush();
}
}
}
}
}
impl FileSettings {
pub fn new() -> Result<FileSettings> {
let config_path = get_config_file()?;
if !config_path.exists() {
let path_string = config_path.display().to_string();
if path_string.is_empty() {
return Err(Error::new("No configuration found."));
}
#[cfg(not(test))]
let ask = Ask {
message: format!(
"No configuration found. Would you like to generate one in {}?",
path_string
),
};
#[cfg(not(test))]
if ask.run() {
create_config_file(&config_path)?;
return Err(Error::new(
"Edit the sample configuration and relaunch meli.",
));
}
#[cfg(test)]
return Ok(FileSettings::default());
#[cfg(not(test))]
return Err(Error::new("No configuration file found."));
}
FileSettings::validate(config_path, true, false)
}
pub fn validate(path: PathBuf, interactive: bool, clear_extras: bool) -> Result<Self> {
let s = pp::pp(&path)?;
let map: toml::map::Map<String, toml::value::Value> =
toml::from_str(&s).map_err(|err| {
Error::new(format!(
"{}:\nConfig file is invalid TOML: {}",
path.display(),
err
))
})?;
/*
* Check that a global composing option is set and return a user-friendly
* error message because the default serde one is confusing.
*/
if !map.contains_key("composing") {
let err_msg = r#"You must set a global `composing` option. If you override `composing` in each account, you can use a dummy global like follows:
[composing]
send_mail = 'false'
This is required so that you don't accidentally start meli and find out later that you can't send emails."#;
if interactive {
println!("{}", err_msg);
let ask = Ask {
message: format!(
"Would you like to append this dummy value in your configuration file {} \
and continue?",
path.display()
),
};
if ask.run() {
let mut file = OpenOptions::new().append(true).open(&path)?;
file.write_all("[composing]\nsend_mail = 'false'\n".as_bytes())
.map_err(|err| {
Error::new(format!("Could not append to {}: {}", path.display(), err))
})?;
return FileSettings::validate(path, interactive, clear_extras);
}
}
return Err(Error::new(format!(
"{}\n\nEdit the {} and relaunch meli.",
if interactive { "" } else { err_msg },
path.display()
)));
}
let mut s: FileSettings = toml::from_str(&s).map_err(|err| {
Error::new(format!(
"{}:\nConfig file contains errors: {}",
path.display(),
err
))
})?;
let backends = melib::backends::Backends::new();
let Themes {
light: default_light,
dark: default_dark,
..
} = Themes::default();
for (k, v) in default_light.keys.into_iter() {
if !s.terminal.themes.light.contains_key(&k) {
s.terminal.themes.light.insert(k, v);
}
}
for theme in s.terminal.themes.other_themes.values_mut() {
for (k, v) in default_dark.keys.clone().into_iter() {
if !theme.contains_key(&k) {
theme.insert(k, v);
}
}
}
for (k, v) in default_dark.keys.into_iter() {
if !s.terminal.themes.dark.contains_key(&k) {
s.terminal.themes.dark.insert(k, v);
}
}
match s.terminal.theme.as_str() {
"dark" | "light" => {}
t if s.terminal.themes.other_themes.contains_key(t) => {}
t => {
return Err(Error::new(format!("Theme `{}` was not found.", t)));
}
}
s.terminal.themes.validate()?;
for (name, acc) in s.accounts.iter_mut() {
let FileAccount {
root_mailbox,
format,
identity,
extra_identities,
read_only,
display_name,
order,
subscribed_mailboxes,
mailboxes,
extra,
manual_refresh,
refresh_command: _,
search_backend: _,
conf_override: _,
} = acc.clone();
let lowercase_format = format.to_lowercase();
let mut s = AccountSettings {
name: name.to_string(),
root_mailbox,
format: format.clone(),
identity,
extra_identities,
read_only,
display_name,
order,
subscribed_mailboxes,
manual_refresh,
mailboxes: mailboxes
.into_iter()
.map(|(k, v)| (k, v.mailbox_conf))
.collect(),
extra: extra.into_iter().collect(),
};
backends.validate_config(&lowercase_format, &mut s)?;
if !s.extra.is_empty() {
return Err(Error::new(format!(
"Unrecognised configuration values: {:?}",
s.extra
)));
}
if clear_extras {
acc.extra.clear();
}
}
Ok(s)
}
}
#[derive(Debug, Clone, Default, Serialize)]
pub struct Settings {
pub accounts: IndexMap<String, AccountConf>,
pub pager: PagerSettings,
pub listing: ListingSettings,
pub notifications: NotificationsSettings,
pub shortcuts: Shortcuts,
pub tags: TagsSettings,
pub composing: ComposingSettings,
pub pgp: PGPSettings,
pub terminal: TerminalSettings,
pub log: LogSettings,
#[serde(skip)]
_logger: StderrLogger,
}
impl Settings {
pub fn new() -> Result<Settings> {
let fs = FileSettings::new()?;
let mut s: IndexMap<String, AccountConf> = IndexMap::new();
for (id, x) in fs.accounts {
let mut ac = AccountConf::from(x);
ac.account.name = id.clone();
s.insert(id, ac);
}
let mut _logger = StderrLogger::new(fs.log.maximum_level);
if let Some(ref log_path) = fs.log.log_file {
_logger.change_log_dest(log_path.into());
}
Ok(Settings {
accounts: s,
pager: fs.pager,
listing: fs.listing,
notifications: fs.notifications,
shortcuts: fs.shortcuts,
tags: fs.tags,
composing: fs.composing,
pgp: fs.pgp,
terminal: fs.terminal,
log: fs.log,
_logger,
})
}
pub fn without_accounts() -> Result<Settings> {
let fs = FileSettings::new()?;
let mut _logger = StderrLogger::new(fs.log.maximum_level);
if let Some(ref log_path) = fs.log.log_file {
_logger.change_log_dest(log_path.into());
}
Ok(Settings {
accounts: IndexMap::new(),
pager: fs.pager,
listing: fs.listing,
notifications: fs.notifications,
shortcuts: fs.shortcuts,
tags: fs.tags,
composing: fs.composing,
pgp: fs.pgp,
terminal: fs.terminal,
log: fs.log,
_logger,
})
}
}
#[derive(Copy, Debug, Clone, Hash, PartialEq, Eq, Default)]
pub enum IndexStyle {
Plain,
Threaded,
#[default]
Compact,
Conversations,
}
/*
* Deserialize default functions
*/
mod default_vals {
pub(in crate::conf) fn false_val<T: std::convert::From<bool>>() -> T {
false.into()
}
pub(in crate::conf) fn true_val<T: std::convert::From<bool>>() -> T {
true.into()
}
pub(in crate::conf) fn zero_val<T: std::convert::From<usize>>() -> T {
0.into()
}
pub(in crate::conf) fn eighty_val<T: std::convert::From<usize>>() -> T {
80.into()
}
pub(in crate::conf) fn none<T>() -> Option<T> {
None
}
pub(in crate::conf) fn internal_value_false<T: std::convert::From<super::ToggleFlag>>() -> T {
super::ToggleFlag::InternalVal(false).into()
}
pub(in crate::conf) fn internal_value_true<T: std::convert::From<super::ToggleFlag>>() -> T {
super::ToggleFlag::InternalVal(true).into()
}
pub(in crate::conf) fn ask<T: std::convert::From<super::ToggleFlag>>() -> T {
super::ToggleFlag::Ask.into()
}
}
mod deserializers {
use serde::{Deserialize, Deserializer};
pub(in crate::conf) fn non_empty_string<'de, D, T: std::convert::From<Option<String>>>(
deserializer: D,
) -> std::result::Result<T, D::Error>
where
D: Deserializer<'de>,
{
let s = <String>::deserialize(deserializer)?;
if s.is_empty() {
Ok(None.into())
} else {
Ok(Some(s).into())
}
}
use toml::Value;
fn any_of<'de, D>(deserializer: D) -> std::result::Result<String, D::Error>
where
D: Deserializer<'de>,
{
let v: Value = Deserialize::deserialize(deserializer)?;
let mut ret = v.to_string();
if ret.starts_with('"') && ret.ends_with('"') {
ret.drain(0..1).count();
ret.drain(ret.len() - 1..).count();
}
Ok(ret)
}
use indexmap::IndexMap;
pub(in crate::conf) fn extra_settings<'de, D>(
deserializer: D,
) -> std::result::Result<IndexMap<String, String>, D::Error>
where
D: Deserializer<'de>,
{
/* Why is this needed? If the user gives a configuration value such as key =
* true, the parsing will fail since it expects string values. We
* want to accept key = true as well as key = "true". */
#[derive(Deserialize)]
struct Wrapper(#[serde(deserialize_with = "any_of")] String);
let v = <IndexMap<String, Wrapper>>::deserialize(deserializer)?;
Ok(v.into_iter().map(|(k, Wrapper(v))| (k, v)).collect())
}
}
impl<'de> Deserialize<'de> for IndexStyle {
fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let s = <String>::deserialize(deserializer)?;
match s.as_str() {
"Plain" | "plain" => Ok(IndexStyle::Plain),
"Threaded" | "threaded" => Ok(IndexStyle::Threaded),
"Compact" | "compact" => Ok(IndexStyle::Compact),
"Conversations" | "conversations" => Ok(IndexStyle::Conversations),
_ => Err(de::Error::custom("invalid `index_style` value")),
}
}
}
impl Serialize for IndexStyle {
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
where
S: Serializer,
{
match self {
IndexStyle::Plain => serializer.serialize_str("plain"),
IndexStyle::Threaded => serializer.serialize_str("threaded"),
IndexStyle::Compact => serializer.serialize_str("compact"),
IndexStyle::Conversations => serializer.serialize_str("conversations"),
}
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq, Default)]
pub enum SearchBackend {
None,
#[default]
Auto,
#[cfg(feature = "sqlite3")]
Sqlite3,
}
impl<'de> Deserialize<'de> for SearchBackend {
fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let s = <String>::deserialize(deserializer)?;
match s.as_str() {
#[cfg(feature = "sqlite3")]
sqlite3
if sqlite3.eq_ignore_ascii_case("sqlite3")
|| sqlite3.eq_ignore_ascii_case("sqlite") =>
{
Ok(SearchBackend::Sqlite3)
}
none if none.eq_ignore_ascii_case("none")
|| none.eq_ignore_ascii_case("nothing")
|| none.is_empty() =>
{
Ok(SearchBackend::None)
}
auto if auto.eq_ignore_ascii_case("auto") => Ok(SearchBackend::Auto),
_ => Err(de::Error::custom("invalid `search_backend` value")),
}
}
}
impl Serialize for SearchBackend {
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
where
S: Serializer,
{
match self {
#[cfg(feature = "sqlite3")]
SearchBackend::Sqlite3 => serializer.serialize_str("sqlite3"),
SearchBackend::None => serializer.serialize_str("none"),
SearchBackend::Auto => serializer.serialize_str("auto"),
}
}
}
pub fn create_config_file(p: &Path) -> Result<()> {
let mut file = OpenOptions::new()
.write(true)
.create_new(true)
.open(p)
.chain_err_summary(|| format!("Cannot create configuration file in {}", p.display()))?;
file.write_all(include_bytes!("../docs/samples/sample-config.toml"))
.and_then(|()| file.flush())
.chain_err_summary(|| format!("Could not write to configuration file {}", p.display()))?;
println!("Written example configuration to {}", p.display());
let set_permissions = |file: std::fs::File| -> Result<()> {
let metadata = file.metadata()?;
let mut permissions = metadata.permissions();
permissions.set_mode(0o600); // Read/write for owner only.
file.set_permissions(permissions)?;
Ok(())
};
if let Err(err) = set_permissions(file) {
println!(
"Warning: Could not set permissions of {} to 0o600: {}",
p.display(),
err
);
}
Ok(())
}
mod pp {
//! Preprocess configuration files by unfolding `include` macros.
use std::{
io::Read,
path::{Path, PathBuf},
};
use melib::{
error::{Error, Result},
parsec::*,
ShellExpandTrait,
};
/// Try to parse line into a path to be included.
fn include_directive<'a>() -> impl Parser<'a, Option<&'a str>> {
move |input: &'a str| {
enum State {
Start,
Path,
}
use State::*;
let mut state = State::Start;
let mut i = 0;
while i < input.len() {
match (&state, input.as_bytes()[i]) {
(Start, b'#') => {
return Ok(("", None));
}
(Start, b) if (b as char).is_whitespace() => { /* consume */ }
(Start, _) if input.as_bytes()[i..].starts_with(b"include(") => {
i += "include(".len();
state = Path;
continue;
}
(Start, _) => {
return Ok(("", None));
}
(Path, b'"') | (Path, b'\'') | (Path, b'`') => {
let mut end = i + 1;
while end < input.len() && input.as_bytes()[end] != input.as_bytes()[i] {
end += 1;
}
if end == input.len() {
return Err(input);
}
let ret = &input[i + 1..end];
end += 1;
if end < input.len() && input.as_bytes()[end] != b')' {
/* Nothing else allowed in line */
return Err(input);
}
end += 1;
while end < input.len() {
if !(input.as_bytes()[end] as char).is_whitespace() {
/* Nothing else allowed in line */
return Err(input);
}
end += 1;
}
return Ok(("", Some(ret)));
}
(Path, _) => return Err(input),
}
i += 1;
}
Ok(("", None))
}
}
/// Expands `include` macros in path.
fn pp_helper(path: &Path, level: u8) -> Result<String> {
if level > 7 {
return Err(Error::new(format!(
"Maximum recursion limit reached while unfolding include directives in {}. Have \
you included a config file within itself?",
path.display()
)));
}
let mut contents = String::new();
let mut file = std::fs::File::open(path)?;
file.read_to_string(&mut contents)?;
let mut ret = String::with_capacity(contents.len());
for (i, l) in contents.lines().enumerate() {
if let (_, Some(sub_path)) = include_directive().parse(l).map_err(|l| {
Error::new(format!(
"Malformed include directive in line {} of file {}: {}\nConfiguration uses \
the standard m4 macro include(\"filename\").",
i,
path.display(),
l
))
})? {
let mut p = Path::new(sub_path).expand();
if p.is_relative() {
/* We checked that path is ok above so we can do unwrap here */
let prefix = path.parent().unwrap();
p = prefix.join(p)
}
ret.push_str(&pp_helper(&p, level + 1)?);
} else {
ret.push_str(l);
ret.push('\n');
}
}
Ok(ret)
}
/// Expands `include` macros in configuration file and other configuration
/// files (eg. themes) in the filesystem.
pub fn pp<P: AsRef<Path>>(path: P) -> Result<String> {
let p_buf: PathBuf = if path.as_ref().is_relative() {
path.as_ref().expand().canonicalize()?
} else {
path.as_ref().expand()
};
let mut ret = super::expand_config(p_buf)?;
if let Ok(xdg_dirs) = xdg::BaseDirectories::with_prefix("meli") {
for theme_mailbox in xdg_dirs.find_config_files("themes") {
let read_dir = std::fs::read_dir(theme_mailbox)?;
for theme in read_dir {
let file = theme?.path();
if let Some(extension) = file.extension() {
if extension == "toml" {
ret.push_str(&pp_helper(&file, 0)?);
}
}
}
}
}
Ok(ret)
}
}
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct LogSettings {
#[serde(default)]
pub log_file: Option<PathBuf>,
#[serde(default)]
pub maximum_level: melib::LogLevel,
}
pub use dotaddressable::*;
mod dotaddressable {
use super::*;
pub trait DotAddressable: Serialize {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
if !path.is_empty() {
Err(Error::new(format!(
"{} has no fields, it is of type {}",
parent_field,
std::any::type_name::<Self>()
)))
} else {
Ok(toml::to_string(self).map_err(|err| err.to_string())?)
}
}
}
impl DotAddressable for bool {}
impl DotAddressable for String {}
impl DotAddressable for char {}
impl DotAddressable for IndexStyle {}
impl DotAddressable for u64 {}
impl DotAddressable for TagHash {}
impl DotAddressable for crate::terminal::Color {}
impl DotAddressable for crate::terminal::Attr {}
impl DotAddressable for crate::terminal::Key {}
impl DotAddressable for usize {}
impl DotAddressable for Query {}
impl DotAddressable for melib::LogLevel {}
impl DotAddressable for PathBuf {}
impl DotAddressable for ToggleFlag {}
impl DotAddressable for SearchBackend {}
impl DotAddressable for melib::SpecialUsageMailbox {}
impl<T: DotAddressable> DotAddressable for Option<T> {}
impl<T: DotAddressable> DotAddressable for Vec<T> {}
impl<K: DotAddressable + std::cmp::Eq + std::hash::Hash, V: DotAddressable> DotAddressable
for HashMap<K, V>
{
}
impl<K: DotAddressable + std::cmp::Eq + std::hash::Hash, V: DotAddressable> DotAddressable
for IndexMap<K, V>
{
}
impl<K: DotAddressable + std::cmp::Eq + std::hash::Hash> DotAddressable for HashSet<K> {}
impl DotAddressable for (SortField, SortOrder) {}
impl DotAddressable for LogSettings {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"log_file" => self.log_file.lookup(field, tail),
"maximum_level" => self.maximum_level.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for Settings {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"accounts" => self.accounts.lookup(field, tail),
"pager" => self.pager.lookup(field, tail),
"listing" => self.listing.lookup(field, tail),
"notifications" => self.notifications.lookup(field, tail),
"shortcuts" => self.shortcuts.lookup(field, tail),
"tags" => Err(Error::new("unimplemented")),
"composing" => Err(Error::new("unimplemented")),
"pgp" => Err(Error::new("unimplemented")),
"terminal" => self.terminal.lookup(field, tail),
"log" => self.log.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for AccountConf {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"account" => self.account.lookup(field, tail),
"conf" => self.conf.lookup(field, tail),
"conf_override" => self.conf_override.lookup(field, tail),
"mailbox_confs" => self.mailbox_confs.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for MailUIConf {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let _tail = &path[1..];
match *field {
"pager" => Err(Error::new("unimplemented")), /* self.pager.lookup(field, */
// tail),
"listing" => Err(Error::new("unimplemented")), /* self.listing.lookup(field, tail), */
"notifications" => Err(Error::new("unimplemented")), /* self.notifications.lookup(field, tail), */
"shortcuts" => Err(Error::new("unimplemented")), /* self.shortcuts. */
// lookup(field,
// tail),
"composing" => Err(Error::new("unimplemented")), /* self.composing.lookup(field, tail), */
"identity" => Err(Error::new("unimplemented")), /* self.identity. */
// lookup(field,
// tail)<String>,
"tags" => Err(Error::new("unimplemented")), /* self.tags.lookup(field, */
// tail),
"themes" => Err(Error::new("unimplemented")), /* self.themes. */
// lookup(field,
// tail)<Themes>,
"pgp" => Err(Error::new("unimplemented")), //self.pgp.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for FileMailboxConf {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"conf_override" => self.conf_override.lookup(field, tail),
"mailbox_conf" => self.mailbox_conf.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for FileAccount {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"root_mailbox" => self.root_mailbox.lookup(field, tail),
"format" => self.format.lookup(field, tail),
"identity" => self.identity.lookup(field, tail),
"display_name" => self.display_name.lookup(field, tail),
"read_only" => self.read_only.lookup(field, tail),
"subscribed_mailboxes" => self.subscribed_mailboxes.lookup(field, tail),
"mailboxes" => self.mailboxes.lookup(field, tail),
"search_backend" => self.search_backend.lookup(field, tail),
"manual_refresh" => self.manual_refresh.lookup(field, tail),
"refresh_command" => self.refresh_command.lookup(field, tail),
"conf_override" => self.conf_override.lookup(field, tail),
"extra" => self.extra.lookup(field, tail),
"order" => self.order.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for melib::AccountSettings {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"name" => self.name.lookup(field, tail),
"root_mailbox" => self.root_mailbox.lookup(field, tail),
"format" => self.format.lookup(field, tail),
"identity" => self.identity.lookup(field, tail),
"read_only" => self.read_only.lookup(field, tail),
"display_name" => self.display_name.lookup(field, tail),
"subscribed_mailboxes" => self.subscribed_mailboxes.lookup(field, tail),
"mailboxes" => self.mailboxes.lookup(field, tail),
"manual_refresh" => self.manual_refresh.lookup(field, tail),
"extra" => self.extra.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
impl DotAddressable for melib::MailboxConf {
fn lookup(&self, parent_field: &str, path: &[&str]) -> Result<String> {
match path.first() {
Some(field) => {
let tail = &path[1..];
match *field {
"alias" => self.alias.lookup(field, tail),
"autoload" => self.autoload.lookup(field, tail),
"subscribe" => self.subscribe.lookup(field, tail),
"ignore" => self.ignore.lookup(field, tail),
"usage" => self.usage.lookup(field, tail),
"extra" => self.extra.lookup(field, tail),
other => Err(Error::new(format!(
"{} has no field named {}",
parent_field, other
))),
}
}
None => Ok(toml::to_string(self).map_err(|err| err.to_string())?),
}
}
}
}
#[test]
fn test_config_parse() {
use std::{fmt::Write, fs, io::prelude::*, path::PathBuf};
struct ConfigFile {
path: PathBuf,
file: fs::File,
}
const TEST_CONFIG: &str = r#"
[accounts.account-name]
root_mailbox = "/path/to/root/mailbox"
format = "Maildir"
listing.index_style = "Conversations" # or [plain, threaded, compact]
identity="email@example.com"
display_name = "Name"
subscribed_mailboxes = ["INBOX", "INBOX/Sent", "INBOX/Drafts", "INBOX/Junk"]
# Set mailbox-specific settings
[accounts.account-name.mailboxes]
"INBOX" = { rename="Inbox" }
"drafts" = { rename="Drafts" }
"foobar-devel" = { ignore = true } # don't show notifications for this mailbox
# Setting up an mbox account
[accounts.mbox]
root_mailbox = "/var/mail/username"
format = "mbox"
listing.index_style = "Compact"
identity="username@hostname.local"
"#;
const EXTRA_CONFIG: &str = r#"
[accounts.mbox]
root_mailbox = "/"
format = "mbox"
index_style = "Compact"
identity="username@hostname.local"
[composing]
send_mail = 'false'
"#;
const IMAP_CONFIG: &str = r#"
[accounts.imap]
root_mailbox = "INBOX"
format = "imap"
identity="username@example.com"
server_username = "null"
server_hostname = "example.com"
server_password_command = "false"
[composing]
send_mail = 'false'
"#;
const EXAMPLE_CONFIG: &str = include_str!("../docs/samples/sample-config.toml");
impl ConfigFile {
fn new(content: &str) -> std::result::Result<Self, std::io::Error> {
let mut f = fs::File::open("/dev/urandom")?;
let mut buf = [0u8; 16];
f.read_exact(&mut buf)?;
let mut filename = String::with_capacity(2 * 16);
for byte in &buf {
write!(&mut filename, "{:02X}", byte).unwrap();
}
let mut path = std::env::temp_dir();
path.push(&*filename);
let mut file = OpenOptions::new()
.create_new(true)
.append(true)
.open(&path)?;
file.write_all(content.as_bytes())?;
Ok(ConfigFile { path, file })
}
}
impl Drop for ConfigFile {
fn drop(&mut self) {
let _ = fs::remove_file(&self.path);
}
}
let mut new_file = ConfigFile::new(TEST_CONFIG).unwrap();
let err = FileSettings::validate(new_file.path.clone(), false, true).unwrap_err();
assert!(err.summary.as_ref().starts_with(
"You must set a global `composing` option. If you override `composing` in each account, \
you can use a dummy global like follows"
));
new_file
.file
.write_all("[composing]\nsend_mail = 'false'\n".as_bytes())
.unwrap();
let err = FileSettings::validate(new_file.path.clone(), false, true).unwrap_err();
assert_eq!(
err.summary.as_ref(),
"Configuration error (account-name): root_mailbox `/path/to/root/mailbox` is not a valid \
directory."
);
/* Test unrecognised configuration entries error */
let new_file = ConfigFile::new(EXTRA_CONFIG).unwrap();
let err = FileSettings::validate(new_file.path.clone(), false, true).unwrap_err();
assert_eq!(
err.summary.as_ref(),
"Unrecognised configuration values: {\"index_style\": \"Compact\"}"
);
/* Test IMAP config */
let new_file = ConfigFile::new(IMAP_CONFIG).unwrap();
FileSettings::validate(new_file.path.clone(), false, true)
.expect("could not parse IMAP config");
/* Test sample config */
let example_config = EXAMPLE_CONFIG.replace("\n#", "\n");
let re = regex::Regex::new(r#"root_mailbox\s*=\s*"[^"]*""#).unwrap();
let example_config = re.replace_all(
&example_config,
&format!(
r#"root_mailbox = "{}""#,
std::env::temp_dir().to_str().unwrap()
),
);
let new_file = ConfigFile::new(&example_config).unwrap();
let config = FileSettings::validate(new_file.path.clone(), false, true)
.expect("Could not parse example config!");
for (accname, acc) in config.accounts.iter() {
if !acc.extra.is_empty() {
panic!(
"In example config, account `{}` has unrecognised configuration entries: {:?}",
accname, acc.extra
);
}
}
}