rcman 0.1.9

Framework-agnostic settings management with schema, backup/restore, secrets and derive macro support
Documentation
//! Poison recovery extension traits for `std::sync` locks
//!
//! These traits provide poison-recovery methods for `RwLock`.

use crate::error::Result;
use std::sync::{RwLock, RwLockReadGuard, RwLockWriteGuard};

/// Extension trait for `RwLock` with poison recovery
pub trait RwLockExt<T> {
    /// Acquire a read lock, recovering from poison errors
    ///
    /// # Errors
    ///
    /// Returns an error if the lock is poisoned and recovery fails.
    fn read_recovered(&self) -> Result<RwLockReadGuard<'_, T>>;

    /// Acquire a write lock, recovering from poison errors
    ///
    /// # Errors
    ///
    /// Returns an error if the lock is poisoned and recovery fails.
    fn write_recovered(&self) -> Result<RwLockWriteGuard<'_, T>>;
}

impl<T> RwLockExt<T> for RwLock<T> {
    fn read_recovered(&self) -> Result<RwLockReadGuard<'_, T>> {
        match self.read() {
            Ok(guard) => Ok(guard),
            Err(poisoned) => {
                log::error!("RwLock was poisoned (read), recovering; state may be inconsistent");
                Ok(poisoned.into_inner())
            }
        }
    }

    fn write_recovered(&self) -> Result<RwLockWriteGuard<'_, T>> {
        match self.write() {
            Ok(guard) => Ok(guard),
            Err(poisoned) => {
                log::error!("RwLock was poisoned (write), recovering; state may be inconsistent");
                Ok(poisoned.into_inner())
            }
        }
    }
}