use crate::{ log, password_security::config::{ decrypt_str_or_original, decrypt_vec_or_original, encrypt_str_or_original, encrypt_vec_or_original, }, }; use anyhow::Result; use directories_next::ProjectDirs; use rand::Rng; use serde_derive::{Deserialize, Serialize}; use sodiumoxide::crypto::sign; use std::{ collections::HashMap, fs, net::{IpAddr, Ipv4Addr, SocketAddr}, path::{Path, PathBuf}, sync::{Arc, Mutex, RwLock}, time::SystemTime, }; pub const RENDEZVOUS_TIMEOUT: u64 = 12_000; pub const CONNECT_TIMEOUT: u64 = 18_000; pub const REG_INTERVAL: i64 = 12_000; pub const COMPRESS_LEVEL: i32 = 3; const SERIAL: i32 = 3; const PASSWORD_ENC_VERSION: &'static str = "00"; // 128x128 #[cfg(target_os = "macos")] // 128x128 on 160x160 canvas, then shrink to 128, mac looks better with padding pub const ICON: &str = " "; #[cfg(not(target_os = "macos"))] // 128x128 no padding pub const ICON: &str = " "; #[cfg(target_os = "macos")] lazy_static::lazy_static! { pub static ref ORG: Arc> = Arc::new(RwLock::new("com.carriez".to_owned())); } type Size = (i32, i32, i32, i32); lazy_static::lazy_static! { static ref CONFIG: Arc> = Arc::new(RwLock::new(Config::load())); static ref CONFIG2: Arc> = Arc::new(RwLock::new(Config2::load())); static ref LOCAL_CONFIG: Arc> = Arc::new(RwLock::new(LocalConfig::load())); pub static ref ONLINE: Arc>> = Default::default(); pub static ref PROD_RENDEZVOUS_SERVER: Arc> = Default::default(); pub static ref APP_NAME: Arc> = Arc::new(RwLock::new("RustDesk".to_owned())); } #[cfg(target_os = "android")] lazy_static::lazy_static! { pub static ref APP_DIR: Arc> = Arc::new(RwLock::new("/data/user/0/com.carriez.flutter_hbb/app_flutter".to_owned())); } #[cfg(target_os = "ios")] lazy_static::lazy_static! { pub static ref APP_DIR: Arc> = Default::default(); } #[cfg(any(target_os = "android", target_os = "ios"))] lazy_static::lazy_static! { pub static ref APP_HOME_DIR: Arc> = Default::default(); } const CHARS: &'static [char] = &[ '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'm', 'n', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', ]; pub const RENDEZVOUS_SERVERS: &'static [&'static str] = &[ "rs-ny.rustdesk.com", "rs-sg.rustdesk.com", "rs-cn.rustdesk.com", ]; pub const RS_PUB_KEY: &'static str = "OeVuKk5nlHiXp+APNn0Y3pC1Iwpwn44JGqrQCsWqmBw="; pub const RENDEZVOUS_PORT: i32 = 21116; pub const RELAY_PORT: i32 = 21117; #[derive(Clone, Copy, PartialEq, Eq, Debug)] pub enum NetworkType { Direct, ProxySocks, } #[derive(Debug, Default, Serialize, Deserialize, Clone, PartialEq)] pub struct Config { #[serde(default)] pub id: String, #[serde(default)] password: String, #[serde(default)] salt: String, #[serde(default)] pub key_pair: (Vec, Vec), // sk, pk #[serde(default)] key_confirmed: bool, #[serde(default)] keys_confirmed: HashMap, } #[derive(Debug, Default, PartialEq, Serialize, Deserialize, Clone)] pub struct Socks5Server { #[serde(default)] pub proxy: String, #[serde(default)] pub username: String, #[serde(default)] pub password: String, } // more variable configs #[derive(Debug, Default, Serialize, Deserialize, Clone, PartialEq)] pub struct Config2 { #[serde(default)] rendezvous_server: String, #[serde(default)] nat_type: i32, #[serde(default)] serial: i32, #[serde(default)] socks: Option, // the other scalar value must before this #[serde(default)] pub options: HashMap, } #[derive(Debug, Default, Serialize, Deserialize, Clone, PartialEq)] pub struct PeerConfig { #[serde(default)] pub password: Vec, #[serde(default)] pub size: Size, #[serde(default)] pub size_ft: Size, #[serde(default)] pub size_pf: Size, #[serde(default)] pub view_style: String, // original (default), scale #[serde(default)] pub image_quality: String, #[serde(default)] pub custom_image_quality: Vec, #[serde(default)] pub show_remote_cursor: bool, #[serde(default)] pub lock_after_session_end: bool, #[serde(default)] pub privacy_mode: bool, #[serde(default)] pub port_forwards: Vec<(i32, String, i32)>, #[serde(default)] pub direct_failures: i32, #[serde(default)] pub disable_audio: bool, #[serde(default)] pub disable_clipboard: bool, #[serde(default)] pub enable_file_transfer: bool, #[serde(default)] pub show_quality_monitor: bool, // the other scalar value must before this #[serde(default)] pub options: HashMap, #[serde(default)] pub info: PeerInfoSerde, #[serde(default)] pub transfer: TransferSerde, } #[derive(Debug, PartialEq, Default, Serialize, Deserialize, Clone)] pub struct PeerInfoSerde { #[serde(default)] pub username: String, #[serde(default)] pub hostname: String, #[serde(default)] pub platform: String, } #[derive(Debug, Default, Serialize, Deserialize, Clone, PartialEq)] pub struct TransferSerde { #[serde(default)] pub write_jobs: Vec, #[serde(default)] pub read_jobs: Vec, } fn patch(path: PathBuf) -> PathBuf { if let Some(_tmp) = path.to_str() { #[cfg(windows)] return _tmp .replace( "system32\\config\\systemprofile", "ServiceProfiles\\LocalService", ) .into(); #[cfg(target_os = "macos")] return _tmp.replace("Application Support", "Preferences").into(); #[cfg(target_os = "linux")] { if _tmp == "/root" { if let Ok(output) = std::process::Command::new("whoami").output() { let user = String::from_utf8_lossy(&output.stdout) .to_string() .trim() .to_owned(); if user != "root" { return format!("/home/{}", user).into(); } } } } } path } impl Config2 { fn load() -> Config2 { let mut config = Config::load_::("2"); if let Some(mut socks) = config.socks { let (password, _, store) = decrypt_str_or_original(&socks.password, PASSWORD_ENC_VERSION); socks.password = password; config.socks = Some(socks); if store { config.store(); } } config } pub fn decrypt_password(&mut self) { if let Some(mut socks) = self.socks.clone() { socks.password = decrypt_str_or_original(&socks.password, PASSWORD_ENC_VERSION).0; self.socks = Some(socks); } } pub fn file() -> PathBuf { Config::file_("2") } fn store(&self) { let mut config = self.clone(); if let Some(mut socks) = config.socks { socks.password = encrypt_str_or_original(&socks.password, PASSWORD_ENC_VERSION); config.socks = Some(socks); } Config::store_(&config, "2"); } pub fn get() -> Config2 { return CONFIG2.read().unwrap().clone(); } pub fn set(cfg: Config2) -> bool { let mut lock = CONFIG2.write().unwrap(); if *lock == cfg { return false; } *lock = cfg; lock.store(); true } } pub fn load_path( file: PathBuf, ) -> T { let cfg = match confy::load_path(&file) { Ok(config) => config, Err(err) => { log::error!("Failed to load config: {}", err); T::default() } }; cfg } impl Config { fn load_( suffix: &str, ) -> T { let file = Self::file_(suffix); log::debug!("Configuration path: {}", file.display()); let cfg = load_path(file); if suffix.is_empty() { log::debug!("{:?}", cfg); } cfg } fn store_(config: &T, suffix: &str) { let file = Self::file_(suffix); if let Err(err) = confy::store_path(file, config) { log::error!("Failed to store config: {}", err); } } fn load() -> Config { let mut config = Config::load_::(""); let (password, _, store) = decrypt_str_or_original(&config.password, PASSWORD_ENC_VERSION); config.password = password; if store { config.store(); } config } pub fn decrypt_password(&mut self) { self.password = decrypt_str_or_original(&self.password, PASSWORD_ENC_VERSION).0; } fn store(&self) { let mut config = self.clone(); config.password = encrypt_str_or_original(&config.password, PASSWORD_ENC_VERSION); Config::store_(&config, ""); } pub fn file() -> PathBuf { Self::file_("") } fn file_(suffix: &str) -> PathBuf { let name = format!("{}{}", *APP_NAME.read().unwrap(), suffix); Config::with_extension(Self::path(name)) } pub fn get_home() -> PathBuf { #[cfg(any(target_os = "android", target_os = "ios"))] return Self::path(APP_HOME_DIR.read().unwrap().as_str()); if let Some(path) = dirs_next::home_dir() { patch(path) } else if let Ok(path) = std::env::current_dir() { path } else { std::env::temp_dir() } } pub fn path>(p: P) -> PathBuf { #[cfg(any(target_os = "android", target_os = "ios"))] { let mut path: PathBuf = APP_DIR.read().unwrap().clone().into(); path.push(p); return path; } #[cfg(not(target_os = "macos"))] let org = ""; #[cfg(target_os = "macos")] let org = ORG.read().unwrap().clone(); // /var/root for root if let Some(project) = ProjectDirs::from("", &org, &*APP_NAME.read().unwrap()) { let mut path = patch(project.config_dir().to_path_buf()); path.push(p); return path; } return "".into(); } #[allow(unreachable_code)] pub fn log_path() -> PathBuf { #[cfg(target_os = "macos")] { if let Some(path) = dirs_next::home_dir().as_mut() { path.push(format!("Library/Logs/{}", *APP_NAME.read().unwrap())); return path.clone(); } } #[cfg(target_os = "linux")] { let mut path = Self::get_home(); path.push(format!(".local/share/logs/{}", *APP_NAME.read().unwrap())); std::fs::create_dir_all(&path).ok(); return path; } if let Some(path) = Self::path("").parent() { let mut path: PathBuf = path.into(); path.push("log"); return path; } "".into() } pub fn ipc_path(postfix: &str) -> String { #[cfg(windows)] { // \\ServerName\pipe\PipeName // where ServerName is either the name of a remote computer or a period, to specify the local computer. // https://docs.microsoft.com/en-us/windows/win32/ipc/pipe-names format!( "\\\\.\\pipe\\{}\\query{}", *APP_NAME.read().unwrap(), postfix ) } #[cfg(not(windows))] { use std::os::unix::fs::PermissionsExt; #[cfg(target_os = "android")] let mut path: PathBuf = format!("{}/{}", *APP_DIR.read().unwrap(), *APP_NAME.read().unwrap()).into(); #[cfg(not(target_os = "android"))] let mut path: PathBuf = format!("/tmp/{}", *APP_NAME.read().unwrap()).into(); fs::create_dir(&path).ok(); fs::set_permissions(&path, fs::Permissions::from_mode(0o0777)).ok(); path.push(format!("ipc{}", postfix)); path.to_str().unwrap_or("").to_owned() } } pub fn icon_path() -> PathBuf { let mut path = Self::path("icons"); if fs::create_dir_all(&path).is_err() { path = std::env::temp_dir(); } path } #[inline] pub fn get_any_listen_addr() -> SocketAddr { SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 0) } pub fn get_rendezvous_server() -> String { let mut rendezvous_server = Self::get_option("custom-rendezvous-server"); if rendezvous_server.is_empty() { rendezvous_server = PROD_RENDEZVOUS_SERVER.read().unwrap().clone(); } if rendezvous_server.is_empty() { rendezvous_server = CONFIG2.read().unwrap().rendezvous_server.clone(); } if rendezvous_server.is_empty() { rendezvous_server = Self::get_rendezvous_servers() .drain(..) .next() .unwrap_or("".to_owned()); } if !rendezvous_server.contains(":") { rendezvous_server = format!("{}:{}", rendezvous_server, RENDEZVOUS_PORT); } rendezvous_server } pub fn get_rendezvous_servers() -> Vec { let s = Self::get_option("custom-rendezvous-server"); if !s.is_empty() { return vec![s]; } let s = PROD_RENDEZVOUS_SERVER.read().unwrap().clone(); if !s.is_empty() { return vec![s]; } let serial_obsolute = CONFIG2.read().unwrap().serial > SERIAL; if serial_obsolute { let ss: Vec = Self::get_option("rendezvous-servers") .split(",") .filter(|x| x.contains(".")) .map(|x| x.to_owned()) .collect(); if !ss.is_empty() { return ss; } } return RENDEZVOUS_SERVERS.iter().map(|x| x.to_string()).collect(); } pub fn reset_online() { *ONLINE.lock().unwrap() = Default::default(); } pub fn update_latency(host: &str, latency: i64) { ONLINE.lock().unwrap().insert(host.to_owned(), latency); let mut host = "".to_owned(); let mut delay = i64::MAX; for (tmp_host, tmp_delay) in ONLINE.lock().unwrap().iter() { if tmp_delay > &0 && tmp_delay < &delay { delay = tmp_delay.clone(); host = tmp_host.to_string(); } } if !host.is_empty() { let mut config = CONFIG2.write().unwrap(); if host != config.rendezvous_server { log::debug!("Update rendezvous_server in config to {}", host); log::debug!("{:?}", *ONLINE.lock().unwrap()); config.rendezvous_server = host; config.store(); } } } pub fn set_id(id: &str) { let mut config = CONFIG.write().unwrap(); if id == config.id { return; } config.id = id.into(); config.store(); } pub fn set_nat_type(nat_type: i32) { let mut config = CONFIG2.write().unwrap(); if nat_type == config.nat_type { return; } config.nat_type = nat_type; config.store(); } pub fn get_nat_type() -> i32 { CONFIG2.read().unwrap().nat_type } pub fn set_serial(serial: i32) { let mut config = CONFIG2.write().unwrap(); if serial == config.serial { return; } config.serial = serial; config.store(); } pub fn get_serial() -> i32 { std::cmp::max(CONFIG2.read().unwrap().serial, SERIAL) } fn get_auto_id() -> Option { #[cfg(any(target_os = "android", target_os = "ios"))] { return Some( rand::thread_rng() .gen_range(1_000_000_000..2_000_000_000) .to_string(), ); } let mut id = 0u32; #[cfg(not(any(target_os = "android", target_os = "ios")))] if let Ok(Some(ma)) = mac_address::get_mac_address() { for x in &ma.bytes()[2..] { id = (id << 8) | (*x as u32); } id = id & 0x1FFFFFFF; Some(id.to_string()) } else { None } } pub fn get_auto_password(length: usize) -> String { let mut rng = rand::thread_rng(); (0..length) .map(|_| CHARS[rng.gen::() % CHARS.len()]) .collect() } pub fn get_key_confirmed() -> bool { CONFIG.read().unwrap().key_confirmed } pub fn set_key_confirmed(v: bool) { let mut config = CONFIG.write().unwrap(); if config.key_confirmed == v { return; } config.key_confirmed = v; if !v { config.keys_confirmed = Default::default(); } config.store(); } pub fn get_host_key_confirmed(host: &str) -> bool { if let Some(true) = CONFIG.read().unwrap().keys_confirmed.get(host) { true } else { false } } pub fn set_host_key_confirmed(host: &str, v: bool) { if Self::get_host_key_confirmed(host) == v { return; } let mut config = CONFIG.write().unwrap(); config.keys_confirmed.insert(host.to_owned(), v); config.store(); } pub fn set_key_pair(pair: (Vec, Vec)) { let mut config = CONFIG.write().unwrap(); if config.key_pair == pair { return; } config.key_pair = pair; config.store(); } pub fn get_key_pair() -> (Vec, Vec) { // lock here to make sure no gen_keypair more than once let mut config = CONFIG.write().unwrap(); if config.key_pair.0.is_empty() { let (pk, sk) = sign::gen_keypair(); config.key_pair = (sk.0.to_vec(), pk.0.into()); config.store(); } config.key_pair.clone() } pub fn get_id() -> String { let mut id = CONFIG.read().unwrap().id.clone(); if id.is_empty() { if let Some(tmp) = Config::get_auto_id() { id = tmp; Config::set_id(&id); } } id } pub fn get_id_or(b: String) -> String { let a = CONFIG.read().unwrap().id.clone(); if a.is_empty() { b } else { a } } pub fn get_options() -> HashMap { CONFIG2.read().unwrap().options.clone() } pub fn set_options(v: HashMap) { let mut config = CONFIG2.write().unwrap(); if config.options == v { return; } config.options = v; config.store(); } pub fn get_option(k: &str) -> String { if let Some(v) = CONFIG2.read().unwrap().options.get(k) { v.clone() } else { "".to_owned() } } pub fn set_option(k: String, v: String) { let mut config = CONFIG2.write().unwrap(); let v2 = if v.is_empty() { None } else { Some(&v) }; if v2 != config.options.get(&k) { if v2.is_none() { config.options.remove(&k); } else { config.options.insert(k, v); } config.store(); } } pub fn update_id() { // to-do: how about if one ip register a lot of ids? let id = Self::get_id(); let mut rng = rand::thread_rng(); let new_id = rng.gen_range(1_000_000_000..2_000_000_000).to_string(); Config::set_id(&new_id); log::info!("id updated from {} to {}", id, new_id); } pub fn set_permanent_password(password: &str) { let mut config = CONFIG.write().unwrap(); if password == config.password { return; } config.password = password.into(); config.store(); } pub fn get_permanent_password() -> String { CONFIG.read().unwrap().password.clone() } pub fn set_salt(salt: &str) { let mut config = CONFIG.write().unwrap(); if salt == config.salt { return; } config.salt = salt.into(); config.store(); } pub fn get_salt() -> String { let mut salt = CONFIG.read().unwrap().salt.clone(); if salt.is_empty() { salt = Config::get_auto_password(6); Config::set_salt(&salt); } salt } pub fn set_socks(socks: Option) { let mut config = CONFIG2.write().unwrap(); if config.socks == socks { return; } config.socks = socks; config.store(); } pub fn get_socks() -> Option { CONFIG2.read().unwrap().socks.clone() } pub fn get_network_type() -> NetworkType { match &CONFIG2.read().unwrap().socks { None => NetworkType::Direct, Some(_) => NetworkType::ProxySocks, } } pub fn get() -> Config { return CONFIG.read().unwrap().clone(); } pub fn set(cfg: Config) -> bool { let mut lock = CONFIG.write().unwrap(); if *lock == cfg { return false; } *lock = cfg; lock.store(); true } fn with_extension(path: PathBuf) -> PathBuf { let ext = path.extension(); if let Some(ext) = ext { let ext = format!("{}.toml", ext.to_string_lossy()); path.with_extension(&ext) } else { path.with_extension("toml") } } } const PEERS: &str = "peers"; impl PeerConfig { pub fn load(id: &str) -> PeerConfig { let _ = CONFIG.read().unwrap(); // for lock match confy::load_path(&Self::path(id)) { Ok(config) => { let mut config: PeerConfig = config; let mut store = false; let (password, _, store2) = decrypt_vec_or_original(&config.password, PASSWORD_ENC_VERSION); config.password = password; store = store || store2; config.options.get_mut("rdp_password").map(|v| { let (password, _, store2) = decrypt_str_or_original(v, PASSWORD_ENC_VERSION); *v = password; store = store || store2; }); config.options.get_mut("os-password").map(|v| { let (password, _, store2) = decrypt_str_or_original(v, PASSWORD_ENC_VERSION); *v = password; store = store || store2; }); if store { config.store(id); } config } Err(err) => { log::error!("Failed to load config: {}", err); Default::default() } } } pub fn store(&self, id: &str) { let _ = CONFIG.read().unwrap(); // for lock let mut config = self.clone(); config.password = encrypt_vec_or_original(&config.password, PASSWORD_ENC_VERSION); config .options .get_mut("rdp_password") .map(|v| *v = encrypt_str_or_original(v, PASSWORD_ENC_VERSION)); config .options .get_mut("os-password") .map(|v| *v = encrypt_str_or_original(v, PASSWORD_ENC_VERSION)); if let Err(err) = confy::store_path(Self::path(id), config) { log::error!("Failed to store config: {}", err); } } pub fn remove(id: &str) { fs::remove_file(&Self::path(id)).ok(); } fn path(id: &str) -> PathBuf { let path: PathBuf = [PEERS, id].iter().collect(); Config::with_extension(Config::path(path)) } pub fn peers() -> Vec<(String, SystemTime, PeerConfig)> { if let Ok(peers) = Config::path(PEERS).read_dir() { if let Ok(peers) = peers .map(|res| res.map(|e| e.path())) .collect::, _>>() { let mut peers: Vec<_> = peers .iter() .filter(|p| { p.is_file() && p.extension().map(|p| p.to_str().unwrap_or("")) == Some("toml") }) .map(|p| { let t = crate::get_modified_time(&p); let id = p .file_stem() .map(|p| p.to_str().unwrap_or("")) .unwrap_or("") .to_owned(); let c = PeerConfig::load(&id); if c.info.platform.is_empty() { fs::remove_file(&p).ok(); } (id, t, c) }) .filter(|p| !p.2.info.platform.is_empty()) .collect(); peers.sort_unstable_by(|a, b| b.1.cmp(&a.1)); return peers; } } Default::default() } } #[derive(Debug, Default, Serialize, Deserialize, Clone)] pub struct LocalConfig { #[serde(default)] remote_id: String, // latest used one #[serde(default)] size: Size, #[serde(default)] pub fav: Vec, #[serde(default)] options: HashMap, } impl LocalConfig { fn load() -> LocalConfig { Config::load_::("_local") } fn store(&self) { Config::store_(self, "_local"); } pub fn get_size() -> Size { LOCAL_CONFIG.read().unwrap().size } pub fn set_size(x: i32, y: i32, w: i32, h: i32) { let mut config = LOCAL_CONFIG.write().unwrap(); let size = (x, y, w, h); if size == config.size || size.2 < 300 || size.3 < 300 { return; } config.size = size; config.store(); } pub fn set_remote_id(remote_id: &str) { let mut config = LOCAL_CONFIG.write().unwrap(); if remote_id == config.remote_id { return; } config.remote_id = remote_id.into(); config.store(); } pub fn get_remote_id() -> String { LOCAL_CONFIG.read().unwrap().remote_id.clone() } pub fn set_fav(fav: Vec) { let mut lock = LOCAL_CONFIG.write().unwrap(); if lock.fav == fav { return; } lock.fav = fav; lock.store(); } pub fn get_fav() -> Vec { LOCAL_CONFIG.read().unwrap().fav.clone() } pub fn get_option(k: &str) -> String { if let Some(v) = LOCAL_CONFIG.read().unwrap().options.get(k) { v.clone() } else { "".to_owned() } } pub fn set_option(k: String, v: String) { let mut config = LOCAL_CONFIG.write().unwrap(); let v2 = if v.is_empty() { None } else { Some(&v) }; if v2 != config.options.get(&k) { if v2.is_none() { config.options.remove(&k); } else { config.options.insert(k, v); } config.store(); } } } #[derive(Debug, Default, Serialize, Deserialize, Clone)] pub struct DiscoveryPeer { pub id: String, #[serde(with = "serde_with::rust::map_as_tuple_list")] pub ip_mac: HashMap, pub username: String, pub hostname: String, pub platform: String, pub online: bool, } impl DiscoveryPeer { pub fn is_same_peer(&self, other: &DiscoveryPeer) -> bool { self.id == other.id && self.username == other.username } } #[derive(Debug, Default, Serialize, Deserialize, Clone)] pub struct LanPeers { pub peers: Vec, } impl LanPeers { pub fn load() -> LanPeers { let _ = CONFIG.read().unwrap(); // for lock match confy::load_path(&Config::file_("_lan_peers")) { Ok(peers) => peers, Err(err) => { log::error!("Failed to load lan peers: {}", err); Default::default() } } } pub fn store(peers: &Vec) { let f = LanPeers { peers: peers.clone(), }; if let Err(err) = confy::store_path(Config::file_("_lan_peers"), f) { log::error!("Failed to store lan peers: {}", err); } } pub fn modify_time() -> crate::ResultType { let p = Config::file_("_lan_peers"); Ok(fs::metadata(p)? .modified()? .duration_since(SystemTime::UNIX_EPOCH)? .as_millis() as _) } } #[derive(Debug, Default, Serialize, Deserialize, Clone)] pub struct HwCodecConfig { #[serde(default)] pub options: HashMap, } impl HwCodecConfig { pub fn load() -> HwCodecConfig { Config::load_::("_hwcodec") } pub fn store(&self) { Config::store_(self, "_hwcodec"); } pub fn remove() { std::fs::remove_file(Config::file_("_hwcodec")).ok(); } } #[cfg(test)] mod tests { use super::*; #[test] fn test_serialize() { let cfg: Config = Default::default(); let res = toml::to_string_pretty(&cfg); assert!(res.is_ok()); let cfg: PeerConfig = Default::default(); let res = toml::to_string_pretty(&cfg); assert!(res.is_ok()); } }