liminal-rs 0.2.1

A conversation-based messaging bus built on beamr
Documentation
/// Error taxonomy for haematite-backed durability operations.
#[derive(Debug, thiserror::Error)]
pub enum DurabilityError {
    /// Haematite returned a store-level failure.
    ///
    /// The umbrella `From<haematite::ApiError>` conversion lives in
    /// [`super::store`] because it routes the optimistic-concurrency variants to
    /// their dedicated cases rather than wrapping them here.
    #[error("haematite store error: {0}")]
    StoreError(haematite::ApiError),

    /// An append observed a different stream sequence than the caller expected.
    #[error("sequence conflict: expected {expected}, actual {actual}")]
    SequenceConflict {
        /// Caller-provided expected stream sequence.
        expected: u64,
        /// Actual stream sequence reported by haematite.
        actual: u64,
    },

    /// A cursor checkpoint attempted to move from a stale stored value.
    #[error("cursor regression: stored {stored}, attempted {attempted}")]
    CursorRegression {
        /// Current value stored for the cursor.
        stored: u64,
        /// Value the caller attempted to checkpoint from.
        attempted: u64,
    },

    /// A producer idempotency key collided with an existing dedup entry.
    #[error("dedup key collision for key {key}")]
    DedupCollision {
        /// Idempotency key that collided.
        key: String,
    },

    /// Durability configuration failed validation.
    #[error("configuration error: {0}")]
    ConfigError(String),

    /// Persisted envelope bytes could not be encoded or decoded.
    #[error("envelope serialization error: {0}")]
    EnvelopeError(String),
}

impl From<haematite::SequenceConflict> for DurabilityError {
    fn from(error: haematite::SequenceConflict) -> Self {
        Self::SequenceConflict {
            expected: error.expected,
            actual: error.actual,
        }
    }
}

impl From<haematite::CasMismatch> for DurabilityError {
    fn from(error: haematite::CasMismatch) -> Self {
        // The real `CasMismatch` carries `Option<u64>` to distinguish absent
        // (`None`) from stored-zero (`Some(0)`). The cursor contract treats an
        // absent key as the value 0 (see `HaematiteStore::cas`), so a `None`
        // collapses to 0 here when reporting a regression.
        Self::CursorRegression {
            stored: error.actual.unwrap_or(0),
            attempted: error.expected.unwrap_or(0),
        }
    }
}