diagweave 0.1.1

Core runtime and macros re-export for diagweave error algebra and report diagweaveing.
Documentation
//! Error transformation methods for Report.
//!
//! This module contains the `map_err` method and related functionality
//! for transforming the inner error type while preserving diagnostic data.

use core::error::Error;

use super::types::build_origin_source_chain;
use super::{Report, SeverityState};
use alloc::boxed::Box;

impl<E, State> Report<E, State>
where
    State: SeverityState,
{
    /// Maps the inner error type while preserving all diagnostic data.
    ///
    /// When source chain accumulation is enabled via [`ReportOptions::accumulate_src_chain`],
    /// this method also accumulates the origin source error chain.
    ///
    /// # Source Chain Accumulation
    ///
    /// If `accumulate_src_chain` is `true`:
    /// - The current report's origin source chain (if any) is preserved
    /// - The old inner error is added as a source of the new outer error
    /// - The resulting chain reflects: `outer -> old_inner -> ...old sources`
    ///
    /// If `accumulate_src_chain` is `false`:
    /// - Only the error type is transformed
    /// - No source chain manipulation occurs
    ///
    /// # Example
    ///
    /// ```rust
    /// use diagweave::prelude::Report;
    /// use diagweave::Error;
    ///
    /// #[derive(Debug, Error)]
    /// #[display("inner error")]
    /// struct InnerError;
    ///
    /// #[derive(Debug, Error)]
    /// #[display("outer error")]
    /// struct OuterError;
    ///
    /// impl From<InnerError> for OuterError {
    ///     fn from(e: InnerError) -> Self {
    ///         OuterError
    ///     }
    /// }
    ///
    /// let inner_error = InnerError;
    /// let inner_report = Report::new(inner_error);
    ///
    /// // Transform error type while preserving diagnostics
    /// let report: Report<OuterError> = inner_report.map_err(|e| OuterError::from(e));
    ///
    /// // Control source chain accumulation per-report
    /// let _report = report.set_accumulate_src_chain(false); // Disable accumulation
    /// ```
    ///
    /// # Preserved Data
    ///
    /// The following diagnostic data is preserved during transformation:
    /// - Attachments (notes and payloads)
    /// - Context (business and system)
    /// - Display causes
    /// - Stack traces
    /// - Metadata (error code, category, retryable)
    /// - Diagnostic source errors
    /// - Trace context (when `trace` feature is enabled)
    ///
    /// # Performance
    ///
    /// When source chain accumulation is disabled (the default), this method
    /// performs minimal work - it simply transforms the error type and moves
    /// the diagnostic data. When accumulation is enabled, there is additional
    /// overhead for building and preserving the source chain.
    ///
    /// # Type Constraints
    ///
    /// Both the input error type `E` and output error type `Outer` must implement:
    /// - `Error` - The core error trait
    /// - `Send + Sync` - Required for thread safety
    /// - `'static` - Required for storing in the report
    pub fn map_err<Outer>(self, map: impl FnOnce(E) -> Outer) -> Report<Outer, State>
    where
        E: Error + Send + Sync + 'static,
        Outer: Error + Send + Sync + 'static,
    {
        let Self { inner, data } = self;

        let super::ReportData {
            metadata,
            options,
            #[cfg(feature = "trace")]
            trace,
            bag,
        } = *data;

        // Check if source chain accumulation is enabled for this report
        if options.resolve_accumulate_src_chain() {
            // Build origin source chain with the old inner as the new root
            let origin_source_errors = build_origin_source_chain(&inner, bag.inner());

            // Now create the outer error
            let outer = map(inner);

            // Build new bag with the origin source chain using the helper method
            let new_bag = bag.with_origin_srcs(origin_source_errors);

            Report {
                inner: outer,
                data: Box::new(super::ReportData {
                    metadata,
                    options,
                    #[cfg(feature = "trace")]
                    trace,
                    bag: new_bag,
                }),
            }
        } else {
            // Simple transformation without source chain accumulation
            let outer = map(inner);
            Report {
                inner: outer,
                data: Box::new(super::ReportData {
                    metadata,
                    options,
                    #[cfg(feature = "trace")]
                    trace,
                    bag,
                }),
            }
        }
    }
}


#[cfg(test)]
mod tests {
    use super::*;
    use crate::report::{HasSeverity, MissingSeverity, Report, Severity};

    #[derive(Debug)]
    struct TestError;

    impl core::fmt::Display for TestError {
        fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
            f.write_str("test error")
        }
    }

    impl Error for TestError {}

    #[derive(Debug)]
    struct OuterError;

    impl core::fmt::Display for OuterError {
        fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
            f.write_str("outer error")
        }
    }

    impl Error for OuterError {}

    impl From<TestError> for OuterError {
        fn from(_: TestError) -> Self {
            OuterError
        }
    }

    #[test]
    fn test_map_err_preserves_severity() {
        let report: Report<TestError, MissingSeverity> = Report::new(TestError);
        let mapped: Report<OuterError, MissingSeverity> =
            Report::<TestError, MissingSeverity>::map_err(report, |_| OuterError);
        assert!(Report::<OuterError, MissingSeverity>::severity(&mapped).is_none());
    }

    #[test]
    fn test_map_err_with_severity() {
        let report = Report::<TestError, MissingSeverity>::new(TestError);
        let report = Report::<TestError, MissingSeverity>::set_severity(report, Severity::Error);
        let mapped: Report<OuterError, HasSeverity> = report.map_err(|_| OuterError);
        assert_eq!(mapped.severity(), Some(Severity::Error));
    }
}