timewall 1.5.0

All-in-one tool for Apple dynamic HEIF wallpapers on GNU/Linux
use std::{
    env, fs,
    path::{Path, PathBuf},
};

use directories::ProjectDirs;

use crate::constants::{APP_NAME, APP_QUALIFIER};

/// Abstraction over a pidfile containing PID of the last ran wallpaper setter process.
pub struct SetterPidFile {
    pidfile_path: PathBuf,
}

impl SetterPidFile {
    /// Find user's runtime directory and load instance from there.
    pub fn find() -> Self {
        let runtime_dir = if let Result::Ok(path_str) = env::var("TIMEWALL_RUNTIME_DIR") {
            PathBuf::from(path_str)
        } else {
            match ProjectDirs::from(APP_QUALIFIER, "", APP_NAME) {
                Some(app_dirs) => app_dirs
                    .runtime_dir()
                    .map_or_else(|| env::temp_dir().join(APP_NAME), Path::to_path_buf),
                None => panic!("couldn't determine user's home directory"),
            }
        };
        Self::load(runtime_dir.join("last_setter.pid"))
    }

    /// Load instance from given path.
    fn load<P: AsRef<Path>>(pid_path: P) -> Self {
        let pid_path = pid_path.as_ref();
        let parent_dir = pid_path.parent().unwrap();

        if !parent_dir.exists() {
            fs::create_dir_all(parent_dir).expect("couldn't create runtime directory");
        }

        Self {
            pidfile_path: pid_path.to_path_buf(),
        }
    }

    /// Save the PID.
    pub fn save(&self, pid: u32) {
        fs::write(&self.pidfile_path, pid.to_string()).expect("couldn't write setter pidfile");
    }

    /// Read the PID value if it exists.
    pub fn read(&self) -> Option<u32> {
        if self.pidfile_path.exists() {
            Some(
                fs::read_to_string(&self.pidfile_path)
                    .unwrap()
                    .parse::<u32>()
                    .unwrap(),
            )
        } else {
            None
        }
    }

    /// Remove the pidfile.
    pub fn clear(&self) {
        fs::remove_file(&self.pidfile_path).expect("cound't remove setter pidfile");
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use assert_fs::prelude::*;
    use assert_fs::TempDir;
    use predicates::prelude::*;
    use rstest::*;

    #[fixture]
    fn tmp_dir() -> TempDir {
        assert_fs::TempDir::new().unwrap()
    }

    #[rstest]
    fn test_setter_pidfile_load_not_exists(tmp_dir: TempDir) {
        let fake_runtime_dir = tmp_dir.child("fake_runtime_dir");
        let pidfile_path = fake_runtime_dir.child("last_setter.pid");

        SetterPidFile::load(&pidfile_path);

        fake_runtime_dir.assert(predicate::path::exists());
    }

    #[rstest]
    fn test_setter_pidfile_save_read(tmp_dir: TempDir) {
        let pidfile_path = tmp_dir.child("test.pid");

        let pidfile = SetterPidFile::load(&pidfile_path);
        pidfile_path.assert(predicate::path::missing());
        assert_eq!(pidfile.read(), None);

        pidfile.save(1234);
        assert_eq!(pidfile.read(), Some(1234));

        pidfile.save(420);
        assert_eq!(pidfile.read(), Some(420));
    }

    #[rstest]
    fn test_setter_pidfile_clear(tmp_dir: TempDir) {
        let pidfile_path = tmp_dir.child("test.pid");

        let last_pid = SetterPidFile::load(&pidfile_path);
        last_pid.save(1234);
        assert_eq!(last_pid.read(), Some(1234));

        last_pid.clear();
        assert_eq!(last_pid.read(), None);
    }
}