diff --git a/serial_test/Cargo.toml b/serial_test/Cargo.toml index 7665da5..525feed 100644 --- a/serial_test/Cargo.toml +++ b/serial_test/Cargo.toml @@ -14,17 +14,19 @@ keywords = ["sequential"] lazy_static = "1.2" parking_lot = "^0.12" serial_test_derive = { version = "~0.8.0", path = "../serial_test_derive" } -fslock = {version = "0.2", optional = true} -document-features = {version = "0.2", optional=true} -log = {version = "0.4", optional = true} -futures = {version = "^0.3", default_features = false, features = ["executor"] } +fslock = { version = "0.2", optional = true } +document-features = { version = "0.2", optional = true } +log = { version = "0.4", optional = true } +futures = { version = "^0.3", default_features = false, features = [ + "executor", +] } [dev-dependencies] itertools = "0.10" tokio = { version = "^1.17", features = ["macros", "rt"] } [features] -default = ["logging"] +default = ["logging", "timeout"] ## Switches on debug logging (and requires the `log` package) logging = ["log"] @@ -32,10 +34,14 @@ logging = ["log"] ## The file_locks feature unlocks the `file_serial`/`file_parallel` macros file_locks = ["fslock"] +## The `timeout` feature lets tests time out after a certain amount of time +## if not enabled tests will wait indefinetly to be started +timeout = [] + docsrs = ["document-features"] # docs.rs-specific configuration [package.metadata.docs.rs] all-features = true # defines the configuration attribute `docsrs` -rustdoc-args = ["--cfg", "docsrs"] \ No newline at end of file +rustdoc-args = ["--cfg", "docsrs"] diff --git a/serial_test/src/code_lock.rs b/serial_test/src/code_lock.rs index 0835335..781dd82 100644 --- a/serial_test/src/code_lock.rs +++ b/serial_test/src/code_lock.rs @@ -1,10 +1,9 @@ use crate::rwlock::{Locks, MutexGuardWrapper}; use lazy_static::lazy_static; #[cfg(feature = "logging")] -use log::debug; -use parking_lot::{Mutex, RwLock}; +use log::{debug, warn}; +use parking_lot::RwLock; use std::{ - cell::RefCell, collections::HashMap, ops::{Deref, DerefMut}, sync::{atomic::AtomicU32, Arc}, @@ -46,11 +45,14 @@ impl UniqueReentrantMutex { lazy_static! { pub(crate) static ref LOCKS: Arc>> = Arc::new(RwLock::new(HashMap::new())); - static ref MAX_WAIT: Arc>> = - Arc::new(Mutex::new(RefCell::new(Duration::from_secs(60)))); static ref MUTEX_ID: Arc = Arc::new(AtomicU32::new(1)); } +#[cfg(feature = "timeout")] +lazy_static! { + static ref MAX_WAIT: Arc> = Arc::new(RwLock::new(Duration::from_secs(60))); +} + impl Default for UniqueReentrantMutex { fn default() -> Self { Self { @@ -67,20 +69,23 @@ impl Default for UniqueReentrantMutex { /// /// However, sometimes if you've got a *lot* of serial tests it might theoretically not be enough, /// hence this method. +/// +/// This function is only available when the `timeout` feature is enabled. +#[cfg(feature = "timeout")] pub fn set_max_wait(max_wait: Duration) { - MAX_WAIT.lock().replace(max_wait); + *MAX_WAIT.write() = max_wait; } pub(crate) fn wait_duration() -> Duration { - *MAX_WAIT.lock().borrow() + *MAX_WAIT.read() } pub(crate) fn check_new_key(name: &str) { let start = Instant::now(); loop { #[cfg(feature = "logging")] - { - let duration = Instant::now() - start; + if wait_duration().is_some() { + let duration = start.elapsed(); debug!("Waiting for '{}' {:?}", name, duration); } // Check if a new key is needed. Just need a read lock, which can be done in sync with everyone else @@ -105,9 +110,12 @@ pub(crate) fn check_new_key(name: &str) { // If the try_lock fails, then go around the loop again // Odds are another test was also locking on the write and has now written the key - let duration = Instant::now() - start; - if duration >= wait_duration() { - panic!("check_new_key timed out!"); + #[cfg(feature = "timeout")] + if let Some(wait) = wait_duration() { + let duration = start.elapsed(); + if duration > wait { + panic!("Timeout waiting for '{}' {:?}", name, duration); + } } } }