// Copyright 2023 The Jujutsu Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #![allow(missing_docs)] use std::fs::File; use std::path::PathBuf; use rustix::fs::FlockOperation; pub struct FileLock { path: PathBuf, file: File, } impl FileLock { pub fn lock(path: PathBuf) -> FileLock { loop { // Create lockfile, or open pre-existing one let file = File::create(&path).expect("failed to open lockfile"); // If the lock was already held, wait for it to be released rustix::fs::flock(&file, FlockOperation::LockExclusive) .expect("failed to lock lockfile"); let stat = rustix::fs::fstat(&file).expect("failed to stat lockfile"); if stat.st_nlink == 0 { // Lockfile was deleted, probably by the previous holder's `Drop` impl; create a // new one so our ownership is visible, rather than hidden in an // unlinked file. Not always necessary, since the previous // holder might have exited abruptly. continue; } return Self { path, file }; } } } impl Drop for FileLock { fn drop(&mut self) { // Removing the file isn't strictly necessary, but reduces confusion. _ = std::fs::remove_file(&self.path); // Unblock any processes that tried to acquire the lock while we held it. // They're responsible for creating and locking a new lockfile, since we // just deleted this one. _ = rustix::fs::flock(&self.file, FlockOperation::Unlock); } }