cross-stream 0.13.1

An event stream store for personal, local-first use, specializing in event sourcing.
Documentation
//! Shared error types used across the crate.

/// The crate's general-purpose error: any boxed, thread-safe [`std::error::Error`].
///
/// Fallible store operations such as [`append`](crate::Store::append) and
/// [`remove`](crate::Store::remove) return `Result<_, Error>`. Use
/// [`NotFound::is_not_found`] or [`has_not_found_io_error`] to test for the
/// not-found case.
pub type Error = Box<dyn std::error::Error + Send + Sync>;

/// Marker error indicating a requested resource does not exist.
#[derive(Debug)]
pub struct NotFound;

impl std::fmt::Display for NotFound {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "Not found")
    }
}

impl std::error::Error for NotFound {}

impl NotFound {
    /// Check if an error is our custom NotFound error
    pub fn is_not_found(err: &Error) -> bool {
        err.downcast_ref::<NotFound>().is_some()
    }
}

/// Check if an error has ErrorKind::NotFound in its chain
pub fn has_not_found_io_error(err: &Error) -> bool {
    // Check if it's directly an IO error with NotFound kind
    if let Some(io_err) = err.downcast_ref::<std::io::Error>() {
        return io_err.kind() == std::io::ErrorKind::NotFound;
    }

    // Check the error chain for IO errors with NotFound kind
    let mut source = err.source();
    while let Some(err) = source {
        if let Some(io_err) = err.downcast_ref::<std::io::Error>() {
            if io_err.kind() == std::io::ErrorKind::NotFound {
                return true;
            }
        }
        source = err.source();
    }

    false
}