tsink 0.10.2

A lightweight embedded time-series database with a straightforward API
Documentation
use super::*;
use std::time::{Duration, Instant};

const DATA_PATH_LOCK_FILE_NAME: &str = ".tsink.lock";
const LOCK_ACQUIRE_RETRY_TIMEOUT: Duration = Duration::from_secs(1);
const LOCK_ACQUIRE_RETRY_INTERVAL: Duration = Duration::from_millis(10);

#[derive(Debug)]
pub(super) struct DataPathProcessLock {
    _lock_path: PathBuf,
    lock_file: std::fs::File,
}

impl DataPathProcessLock {
    pub(super) fn acquire(data_path: &Path) -> Result<Self> {
        std::fs::create_dir_all(data_path)?;
        let lock_path = data_path.join(DATA_PATH_LOCK_FILE_NAME);
        let lock_file = std::fs::OpenOptions::new()
            .read(true)
            .write(true)
            .create(true)
            .truncate(false)
            .open(&lock_path)
            .map_err(|source| TsinkError::IoWithPath {
                path: lock_path.clone(),
                source,
            })?;

        let deadline = Instant::now() + LOCK_ACQUIRE_RETRY_TIMEOUT;
        loop {
            match lock_file.try_lock() {
                Ok(()) => break,
                Err(std::fs::TryLockError::WouldBlock) => {
                    if Instant::now() >= deadline {
                        return Err(TsinkError::InvalidConfiguration(format!(
                            "data path {} is already locked by another tsink process ({})",
                            data_path.display(),
                            lock_path.display()
                        )));
                    }
                    std::thread::sleep(LOCK_ACQUIRE_RETRY_INTERVAL);
                }
                Err(std::fs::TryLockError::Error(source)) => {
                    return Err(TsinkError::IoWithPath {
                        path: lock_path.clone(),
                        source,
                    });
                }
            }
        }

        Ok(Self {
            _lock_path: lock_path,
            lock_file,
        })
    }
}

impl Drop for DataPathProcessLock {
    fn drop(&mut self) {
        let _ = self.lock_file.unlock();
    }
}