use notify::{
event::{EventKind, ModifyKind},
Watcher,
};
use serde::{de::DeserializeOwned, Serialize};
use std::{
fmt, fs,
io::Write,
path::{Path, PathBuf},
sync::Mutex,
};
#[cfg(feature = "subscription")]
mod subscription;
#[cfg(feature = "subscription")]
pub use subscription::*;
#[cfg(all(feature = "dbus", feature = "subscription"))]
pub mod dbus;
#[cfg(feature = "macro")]
pub use cosmic_config_derive;
#[cfg(feature = "calloop")]
pub mod calloop;
#[derive(Debug)]
pub enum Error {
AtomicWrites(atomicwrites::Error<std::io::Error>),
InvalidName(String),
Io(std::io::Error),
NoConfigDirectory,
Notify(notify::Error),
Ron(ron::Error),
RonSpanned(ron::error::SpannedError),
GetKey(String, std::io::Error),
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Self::AtomicWrites(err) => err.fmt(f),
Self::InvalidName(name) => write!(f, "invalid config name '{}'", name),
Self::Io(err) => err.fmt(f),
Self::NoConfigDirectory => write!(f, "cosmic config directory not found"),
Self::Notify(err) => err.fmt(f),
Self::Ron(err) => err.fmt(f),
Self::RonSpanned(err) => err.fmt(f),
Self::GetKey(key, err) => write!(f, "failed to get key '{}': {}", key, err),
}
}
}
impl std::error::Error for Error {}
impl From<atomicwrites::Error<std::io::Error>> for Error {
fn from(f: atomicwrites::Error<std::io::Error>) -> Self {
Self::AtomicWrites(f)
}
}
impl From<std::io::Error> for Error {
fn from(f: std::io::Error) -> Self {
Self::Io(f)
}
}
impl From<notify::Error> for Error {
fn from(f: notify::Error) -> Self {
Self::Notify(f)
}
}
impl From<ron::Error> for Error {
fn from(f: ron::Error) -> Self {
Self::Ron(f)
}
}
impl From<ron::error::SpannedError> for Error {
fn from(f: ron::error::SpannedError) -> Self {
Self::RonSpanned(f)
}
}
pub trait ConfigGet {
fn get<T: DeserializeOwned>(&self, key: &str) -> Result<T, Error>;
}
pub trait ConfigSet {
fn set<T: Serialize>(&self, key: &str, value: T) -> Result<(), Error>;
}
#[derive(Clone, Debug)]
pub struct Config {
system_path: Option<PathBuf>,
user_path: Option<PathBuf>,
}
fn sanitize_name(name: &str) -> Result<&Path, Error> {
let path = Path::new(name);
if path
.components()
.all(|x| matches!(x, std::path::Component::Normal(_)))
{
Ok(path)
} else {
Err(Error::InvalidName(name.to_owned()))
}
}
impl Config {
pub fn libcosmic() -> Result<Self, Error> {
Self::new("com.system76.libcosmic", 1)
}
pub fn system(name: &str, version: u64) -> Result<Self, Error> {
let path = sanitize_name(name)?.join(format!("v{version}"));
#[cfg(unix)]
let system_path = xdg::BaseDirectories::with_prefix("cosmic")
.map_err(std::io::Error::from)?
.find_data_file(path);
#[cfg(windows)]
let system_path =
known_folders::get_known_folder_path(known_folders::KnownFolder::ProgramFilesCommon)
.map(|x| x.join("COSMIC").join(&path));
Ok(Self {
system_path,
user_path: None,
})
}
pub fn new(name: &str, version: u64) -> Result<Self, Error> {
let path = sanitize_name(name)?.join(format!("v{}", version));
#[cfg(unix)]
let system_path = xdg::BaseDirectories::with_prefix("cosmic")
.map_err(std::io::Error::from)?
.find_data_file(&path);
#[cfg(windows)]
let system_path =
known_folders::get_known_folder_path(known_folders::KnownFolder::ProgramFilesCommon)
.map(|x| x.join("COSMIC").join(&path));
let cosmic_user_path = dirs::config_dir()
.ok_or(Error::NoConfigDirectory)?
.join("cosmic");
let user_path = cosmic_user_path.join(path);
fs::create_dir_all(&user_path)?;
Ok(Self {
system_path,
user_path: Some(user_path),
})
}
pub fn with_custom_path(name: &str, version: u64, custom_path: PathBuf) -> Result<Self, Error> {
let path = sanitize_name(name)?.join(format!("v{version}"));
let cosmic_user_path = custom_path.join("cosmic");
let user_path = cosmic_user_path.join(path);
fs::create_dir_all(&user_path)?;
Ok(Self {
system_path: None,
user_path: Some(user_path),
})
}
pub fn new_state(name: &str, version: u64) -> Result<Self, Error> {
let path = sanitize_name(name)?.join(format!("v{}", version));
let cosmic_user_path = dirs::state_dir()
.ok_or(Error::NoConfigDirectory)?
.join("cosmic");
let user_path = cosmic_user_path.join(path);
fs::create_dir_all(&user_path)?;
Ok(Self {
system_path: None,
user_path: Some(user_path),
})
}
pub fn transaction<'a>(&'a self) -> ConfigTransaction<'a> {
ConfigTransaction {
config: self,
updates: Mutex::new(Vec::new()),
}
}
pub fn watch<F>(&self, f: F) -> Result<notify::RecommendedWatcher, Error>
where
F: Fn(&Self, &[String]) + Send + Sync + 'static,
{
let watch_config = self.clone();
let Some(user_path) = self.user_path.as_ref() else {
return Err(Error::NoConfigDirectory);
};
let user_path_clone = user_path.clone();
let mut watcher =
notify::recommended_watcher(move |event_res: Result<notify::Event, notify::Error>| {
match &event_res {
Ok(event) => {
match &event.kind {
EventKind::Access(_) | EventKind::Modify(ModifyKind::Metadata(_)) => {
return;
}
_ => {}
}
let mut keys = Vec::new();
for path in &event.paths {
match path.strip_prefix(&user_path_clone) {
Ok(key_path) => {
if let Some(key) = key_path.to_str() {
if key.starts_with(".atomicwrite") {
continue;
}
keys.push(key.to_string());
}
}
Err(_err) => {
}
}
}
if !keys.is_empty() {
f(&watch_config, &keys);
}
}
Err(_err) => {
}
}
})?;
watcher.watch(user_path, notify::RecursiveMode::NonRecursive)?;
Ok(watcher)
}
fn default_path(&self, key: &str) -> Result<PathBuf, Error> {
let Some(system_path) = self.system_path.as_ref() else {
return Err(Error::NoConfigDirectory);
};
Ok(system_path.join(sanitize_name(key)?))
}
fn key_path(&self, key: &str) -> Result<PathBuf, Error> {
let Some(user_path) = self.user_path.as_ref() else {
return Err(Error::NoConfigDirectory);
};
Ok(user_path.join(sanitize_name(key)?))
}
}
impl ConfigGet for Config {
fn get<T: DeserializeOwned>(&self, key: &str) -> Result<T, Error> {
let key_path = self.key_path(key);
let data = match key_path {
Ok(key_path) if key_path.is_file() => {
fs::read_to_string(key_path).map_err(|err| Error::GetKey(key.to_string(), err))?
}
_ => {
let default_path = self.default_path(key)?;
fs::read_to_string(default_path)
.map_err(|err| Error::GetKey(key.to_string(), err))?
}
};
let t = ron::from_str(&data)?;
Ok(t)
}
}
impl ConfigSet for Config {
fn set<T: Serialize>(&self, key: &str, value: T) -> Result<(), Error> {
let tx = self.transaction();
tx.set(key, value)?;
tx.commit()
}
}
#[must_use = "Config transaction must be committed"]
pub struct ConfigTransaction<'a> {
config: &'a Config,
updates: Mutex<Vec<(PathBuf, String)>>,
}
impl<'a> ConfigTransaction<'a> {
pub fn commit(self) -> Result<(), Error> {
let mut updates = self.updates.lock().unwrap();
for (key_path, data) in updates.drain(..) {
atomicwrites::AtomicFile::new(
key_path,
atomicwrites::OverwriteBehavior::AllowOverwrite,
)
.write(|file| file.write_all(data.as_bytes()))?;
}
Ok(())
}
}
impl<'a> ConfigSet for ConfigTransaction<'a> {
fn set<T: Serialize>(&self, key: &str, value: T) -> Result<(), Error> {
let key_path = self.config.key_path(key)?;
let data = ron::ser::to_string_pretty(&value, ron::ser::PrettyConfig::new())?;
{
let mut updates = self.updates.lock().unwrap();
updates.push((key_path, data));
}
Ok(())
}
}
pub trait CosmicConfigEntry
where
Self: Sized,
{
const VERSION: u64;
fn write_entry(&self, config: &Config) -> Result<(), crate::Error>;
fn get_entry(config: &Config) -> Result<Self, (Vec<crate::Error>, Self)>;
fn update_keys<T: AsRef<str>>(
&mut self,
config: &Config,
changed_keys: &[T],
) -> (Vec<crate::Error>, Vec<&'static str>);
}
#[derive(Debug)]
pub struct Update<T> {
pub errors: Vec<crate::Error>,
pub keys: Vec<&'static str>,
pub config: T,
}