use crate::DEFAULT_PORT;
use serde::{Deserialize, Serialize};
use std::convert::TryFrom;
use std::fs;
use std::net::{SocketAddr, ToSocketAddrs};
use std::path::Path;
use toml::value::Array;
use toml::Value;
#[derive(Debug, Deserialize, Serialize)]
struct TOMLConfig {
audio: Option<AudioConfig>,
servers: Option<Array>,
}
#[derive(Clone, Debug, Default)]
pub struct Config {
pub audio: AudioConfig,
pub servers: Vec<ServerConfig>,
}
impl Config {
pub fn write_default_cfg(&self, create: bool) -> Result<(), std::io::Error> {
let path = if create {
get_creatable_cfg_path()
} else {
get_cfg_path()
};
let path = std::path::Path::new(&path);
if !create && !path.exists() {
return Ok(());
}
fs::write(
path,
toml::to_string(&TOMLConfig::from(self.clone())).unwrap(),
)
}
}
#[derive(Clone, Debug, Default, Deserialize, Serialize)]
pub struct AudioConfig {
pub input_volume: Option<f32>,
pub output_volume: Option<f32>,
}
#[derive(Clone, Debug, Deserialize, Serialize)]
pub struct ServerConfig {
pub name: String,
pub host: String,
pub port: Option<u16>,
pub username: Option<String>,
pub password: Option<String>,
}
impl ServerConfig {
pub fn to_socket_addr(&self) -> Option<SocketAddr> {
match (self.host.as_str(), self.port.unwrap_or(DEFAULT_PORT))
.to_socket_addrs()
.map(|mut e| e.next())
{
Ok(Some(addr)) => Some(addr),
_ => None,
}
}
}
pub fn get_cfg_path() -> String {
if let Ok(var) = std::env::var("XDG_CONFIG_HOME") {
let path = format!("{}/mumdrc", var);
if Path::new(&path).exists() {
return path;
}
} else if let Ok(var) = std::env::var("HOME") {
let path = format!("{}/.config/mumdrc", var);
if Path::new(&path).exists() {
return path;
}
}
"/etc/mumdrc".to_string()
}
pub fn get_creatable_cfg_path() -> String {
if let Ok(var) = std::env::var("XDG_CONFIG_HOME") {
let path = format!("{}/mumdrc", var);
if !Path::new(&path).exists() {
return path;
}
} else if let Ok(var) = std::env::var("HOME") {
let path = format!("{}/.config/mumdrc", var);
if !Path::new(&path).exists() {
return path;
}
}
"/etc/mumdrc".to_string()
}
pub fn cfg_exists() -> bool {
if let Ok(var) = std::env::var("XDG_CONFIG_HOME") {
let path = format!("{}/mumdrc", var);
if Path::new(&path).exists() {
return true;
}
} else if let Ok(var) = std::env::var("HOME") {
let path = format!("{}/.config/mumdrc", var);
if Path::new(&path).exists() {
return true;
}
} else if Path::new("/etc/mumdrc").exists() {
return true;
}
false
}
impl TryFrom<TOMLConfig> for Config {
type Error = toml::de::Error;
fn try_from(config: TOMLConfig) -> Result<Self, Self::Error> {
Ok(Config {
audio: config.audio.unwrap_or_default(),
servers: config
.servers
.map(|servers| {
servers
.into_iter()
.map(|s| s.try_into::<ServerConfig>())
.collect()
})
.transpose()?
.unwrap_or(Vec::new()),
})
}
}
impl From<Config> for TOMLConfig {
fn from(config: Config) -> Self {
TOMLConfig {
audio: if config.audio.output_volume.is_some() || config.audio.input_volume.is_some() {
Some(config.audio)
} else {
None
},
servers: Some(
config
.servers
.into_iter()
.map(|s| Value::try_from::<ServerConfig>(s).unwrap())
.collect(),
),
}
}
}
pub fn read_default_cfg() -> Config {
Config::try_from(
toml::from_str::<TOMLConfig>(&match fs::read_to_string(get_cfg_path()) {
Ok(f) => f,
Err(_) => return Config::default(),
})
.expect("invalid TOML in config file"), )
.expect("invalid config in TOML") }