bender 0.32.0

A dependency management tool for hardware projects.
// Copyright (c) 2025 ETH Zurich

//! Cross-process filesystem advisory locks.
//!
//! Bender uses these to serialize concurrent invocations against the same git
//! database and checkout. The lock is taken on a sentinel file in
//! `<database>/git/locks/<name>-<hash>.lock` and released automatically when
//! the [`FsLock`] guard is dropped.

#![deny(missing_docs)]

use std::fs::{File, OpenOptions, TryLockError};
use std::path::{Path, PathBuf};

use miette::{Context as _, IntoDiagnostic as _};

use crate::Result;

/// An exclusive, cross-process advisory lock held on a sentinel file.
///
/// The lock is released when this guard is dropped (or when the process exits).
pub struct FsLock {
    file: Option<File>,
    path: PathBuf,
}

impl FsLock {
    /// Acquire an exclusive lock on `path`, creating the file if missing.
    ///
    /// If the lock is contended, an info message is logged so the user can see
    /// why bender is waiting, and the call then blocks until the lock is
    /// available. The actual lock acquisition runs on a blocking worker so it
    /// does not stall the tokio runtime.
    pub async fn acquire_exclusive(path: PathBuf) -> Result<Self> {
        if let Some(parent) = path.parent() {
            std::fs::create_dir_all(parent)
                .into_diagnostic()
                .wrap_err_with(|| format!("Failed to create lock directory {:?}.", parent))?;
        }
        let file = OpenOptions::new()
            .create(true)
            .read(true)
            .write(true)
            .truncate(false)
            .open(&path)
            .into_diagnostic()
            .wrap_err_with(|| format!("Failed to open lock file {:?}.", path))?;

        let path_for_blocking = path.clone();
        let file = tokio::task::spawn_blocking(move || -> Result<File> {
            match file.try_lock() {
                Ok(()) => Ok(file),
                Err(TryLockError::WouldBlock) => {
                    log::info!("waiting for lock on {:?}", path_for_blocking);
                    file.lock().into_diagnostic().wrap_err_with(|| {
                        format!("Failed to acquire lock on {:?}.", path_for_blocking)
                    })?;
                    Ok(file)
                }
                Err(TryLockError::Error(e)) => Err(e)
                    .into_diagnostic()
                    .wrap_err_with(|| format!("Failed to try-lock {:?}.", path_for_blocking)),
            }
        })
        .await
        .into_diagnostic()
        .wrap_err("Lock acquisition task panicked.")??;

        Ok(FsLock {
            file: Some(file),
            path,
        })
    }

    /// The path of the lock file.
    pub fn path(&self) -> &Path {
        &self.path
    }
}

impl Drop for FsLock {
    fn drop(&mut self) {
        if let Some(file) = self.file.take() {
            // Best-effort: errors here are not actionable, and the OS releases
            // the lock automatically when the file handle closes.
            let _ = file.unlock();
        }
    }
}