use std::collections::HashMap; use std::io::{Read, Write}; use std::net::SocketAddr; use std::path::PathBuf; use anyhow::Result; use serde::{Deserialize, Serialize, Serializer, Deserializer}; #[derive(Serialize, Deserialize, Debug, Clone)] pub struct CompanionConfig { pub pid: Option, pub imap: ImapConfig, #[serde(flatten)] pub users: LoginStaticConfig, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct ProviderConfig { pub pid: Option, pub imap: ImapConfig, pub lmtp: LmtpConfig, pub users: UserManagement, } #[derive(Serialize, Deserialize, Debug, Clone)] #[serde(tag = "user_driver")] pub enum UserManagement { Static(LoginStaticConfig), Ldap(LoginLdapConfig), } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct LmtpConfig { pub bind_addr: SocketAddr, pub hostname: String, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct ImapConfig { pub bind_addr: SocketAddr, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct LoginStaticConfig { pub user_list: PathBuf, } #[derive(Serialize, Deserialize, Debug, Clone)] #[serde(tag = "storage_driver")] pub enum LdapStorage { Garage(LdapGarageConfig), InMemory, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct LdapGarageConfig { pub s3_endpoint: String, pub k2v_endpoint: String, pub aws_region: String, pub aws_access_key_id_attr: String, pub aws_secret_access_key_attr: String, pub bucket_attr: Option, pub default_bucket: Option, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct LoginLdapConfig { // LDAP connection info pub ldap_server: String, #[serde(default)] pub pre_bind_on_login: bool, pub bind_dn: Option, pub bind_password: Option, pub search_base: String, // Schema-like info required for Aerogramme's logic pub username_attr: String, #[serde(default = "default_mail_attr")] pub mail_attr: String, pub crypto_root_attr: String, // Storage related thing #[serde(flatten)] pub storage: LdapStorage, } // ---- #[derive(Serialize, Deserialize, Debug, Clone)] #[serde(tag = "storage_driver")] pub enum StaticStorage { Garage(StaticGarageConfig), InMemory, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct StaticGarageConfig { pub s3_endpoint: String, pub k2v_endpoint: String, pub aws_region: String, pub aws_access_key_id: String, pub aws_secret_access_key: String, pub bucket: String, } pub type UserList = HashMap; #[derive(Serialize, Deserialize, Debug, Clone)] #[serde(tag = "crypto_root")] pub enum CryptographyRoot { PasswordProtected { root_blob: String, }, Keyring, ClearText { master_key: String, secret_key: String, } } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct UserEntry { #[serde(default)] pub email_addresses: Vec, pub password: String, #[serde(flatten)] pub crypto_root: CryptographyRoot, #[serde(flatten)] pub storage: StaticStorage, } #[derive(Serialize, Deserialize, Debug, Clone)] pub struct SetupEntry { #[serde(default)] pub email_addresses: Vec, #[serde(default)] pub clear_password: Option, #[serde(flatten)] pub storage: StaticStorage, } #[derive(Serialize, Deserialize, Debug, Clone)] #[serde(tag = "role")] pub enum AnyConfig { Companion(CompanionConfig), Provider(ProviderConfig), } // --- pub fn read_config(config_file: PathBuf) -> Result { let mut file = std::fs::OpenOptions::new() .read(true) .open(config_file.as_path())?; let mut config = String::new(); file.read_to_string(&mut config)?; Ok(toml::from_str(&config)?) } pub fn write_config(config_file: PathBuf, config: &T) -> Result<()> { let mut file = std::fs::OpenOptions::new() .write(true) .open(config_file.as_path())?; file.write_all(toml::to_string(config)?.as_bytes())?; Ok(()) } fn default_mail_attr() -> String { "mail".into() } fn as_base64(val: &T, serializer: &mut S) -> Result<(), S::Error> where T: AsRef<[u8]>, S: Serializer { serializer.serialize_str(&base64::encode(val.as_ref())) } fn from_base64(deserializer: &mut D) -> Result, D::Error> where D: Deserializer { use serde::de::Error; String::deserialize(deserializer) .and_then(|string| base64::decode(&string).map_err(|err| Error::custom(err.to_string()))) }