// Copyright 2020 Google LLC // // 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. use std::fs::{File, OpenOptions}; use std::path::PathBuf; use std::time::Duration; use backoff::{retry, ExponentialBackoff}; pub struct FileLock { path: PathBuf, _file: File, } impl FileLock { pub fn lock(path: PathBuf) -> FileLock { let mut options = OpenOptions::new(); options.create_new(true); options.write(true); let try_write_lock_file = || match options.open(&path) { Ok(file) => Ok(FileLock { path: path.clone(), _file: file, }), Err(err) if err.kind() == std::io::ErrorKind::AlreadyExists => { Err(backoff::Error::Transient(err)) } Err(err) if cfg!(windows) && err.kind() == std::io::ErrorKind::PermissionDenied => { Err(backoff::Error::Transient(err)) } Err(err) => Err(backoff::Error::Permanent(err)), }; let backoff = ExponentialBackoff { initial_interval: Duration::from_millis(1), max_elapsed_time: Some(Duration::from_secs(10)), ..Default::default() }; match retry(backoff, try_write_lock_file) { Err(err) => panic!( "failed to create lock file {}: {}", path.to_string_lossy(), err ), Ok(file_lock) => file_lock, } } } impl Drop for FileLock { fn drop(&mut self) { std::fs::remove_file(&self.path).expect("failed to delete lock file"); } } #[cfg(test)] mod tests { use std::cmp::max; use std::{env, thread}; use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt}; use super::*; #[test] fn lock_basic() { let number: u32 = rand::random(); let lock_path = env::temp_dir().join(format!("test-{}.lock", number)); assert!(!lock_path.exists()); { let _lock = FileLock::lock(lock_path.clone()); assert!(lock_path.exists()); } assert!(!lock_path.exists()); } #[test] fn lock_concurrent() { let number: u32 = rand::random(); let data_path = env::temp_dir().join(format!("test-{}", number)); let lock_path = env::temp_dir().join(format!("test-{}.lock", number)); let mut data_file = OpenOptions::new() .create(true) .write(true) .open(data_path.clone()) .unwrap(); data_file.write_u32::(0).unwrap(); let num_threads = max(num_cpus::get(), 4); let mut threads = vec![]; for _ in 0..num_threads { let data_path = data_path.clone(); let lock_path = lock_path.clone(); let handle = thread::spawn(move || { let _lock = FileLock::lock(lock_path); let mut data_file = OpenOptions::new() .read(true) .open(data_path.clone()) .unwrap(); let value = data_file.read_u32::().unwrap(); thread::sleep(Duration::from_millis(1)); let mut data_file = OpenOptions::new().write(true).open(data_path).unwrap(); data_file.write_u32::(value + 1).unwrap(); }); threads.push(handle); } for thread in threads { thread.join().ok().unwrap(); } let mut data_file = OpenOptions::new().read(true).open(data_path).unwrap(); let value = data_file.read_u32::().unwrap(); assert_eq!(value, num_threads as u32); } }