cfgd-core 0.4.0

Core library for cfgd — shared types, providers, reconciler, state
Documentation
//! Snapshot tests for the output renderer.
//!
//! Goldens live in `snapshots/`. Refresh by running:
//!   INSTA_UPDATE=always cargo test -p cfgd-core --features test-helpers output::tests
//!
//! Organized by topic covering ~91 cases — baseline, verbosity,
//! status × role, themes, indent, corners, and regression.

#[cfg(feature = "test-helpers")]
mod baseline;
#[cfg(feature = "test-helpers")]
mod corners;
#[cfg(feature = "test-helpers")]
mod indent;
#[cfg(feature = "test-helpers")]
mod regression;
#[cfg(feature = "test-helpers")]
mod status_role;
#[cfg(feature = "test-helpers")]
mod themes;
#[cfg(feature = "test-helpers")]
mod themes_raw;
#[cfg(feature = "test-helpers")]
mod verbosity;

/// Macro: build a Printer via `for_test_doc()`, run the body with `&p` and
/// `&cap`, then assert against `snapshots/<bucket>/<name>.txt`.
#[macro_export]
macro_rules! golden_doc {
    ($bucket:ident, $name:ident, |$p:ident, $cap:ident| $body:block) => {
        #[test]
        fn $name() {
            let ($p, $cap) = $crate::output::Printer::for_test_doc();
            $body
            $p.flush();
            $cap.assert_human_snapshot(&format!(
                "{}/{}.txt",
                stringify!($bucket),
                stringify!($name)
            ));
        }
    };
}

/// Macro: like `golden_doc!` but with explicit Verbosity. Used by the
/// verbosity tests to exercise verbosity gating without `for_test_doc()`'s
/// default Normal. Strips ANSI codes before snapshot comparison.
#[macro_export]
macro_rules! golden_at {
    ($bucket:ident, $name:ident, $verbosity:expr, |$p:ident| $body:block) => {
        #[test]
        fn $name() {
            let ($p, buf) = $crate::output::Printer::for_test_at($verbosity);
            $body
            $p.flush();
            let raw = buf.lock().unwrap().clone();
            let actual = $crate::output::strip_ansi(&raw);
            let path = std::path::Path::new("src/output/tests/snapshots")
                .join(stringify!($bucket))
                .join(format!("{}.txt", stringify!($name)));
            if std::env::var("INSTA_UPDATE").as_deref() == Ok("always") || !path.exists() {
                std::fs::create_dir_all(path.parent().unwrap()).unwrap();
                std::fs::write(&path, &actual).unwrap();
                return;
            }
            let expected = std::fs::read_to_string(&path).unwrap();
            // CRLF→LF: windows captures `\r\n`; committed snapshots use `\n`.
            let actual_norm = actual.replace("\r\n", "\n");
            let expected_norm = expected.replace("\r\n", "\n");
            pretty_assertions::assert_eq!(
                actual_norm, expected_norm, "snapshot mismatch: {}", stringify!($name));
        }
    };
}

/// Macro: like `golden_at!` but with an explicit Theme preset (Normal verbosity).
/// Used by the themes tests to render a representative Doc against every preset.
/// Strips ANSI codes before snapshot comparison so color-only theme differences
/// collapse — preset divergence comes from glyph swaps (e.g., `minimal`).
#[macro_export]
macro_rules! golden_themed {
    ($bucket:ident, $name:ident, $theme_name:expr, |$p:ident| $body:block) => {
        #[test]
        fn $name() {
            let ($p, buf) = $crate::output::Printer::for_test_with_theme(
                $crate::output::Theme::from_preset($theme_name),
                $crate::output::Verbosity::Normal,
            );
            $body
            $p.flush();
            let raw = buf.lock().unwrap().clone();
            let actual = $crate::output::strip_ansi(&raw);
            let path = std::path::Path::new("src/output/tests/snapshots")
                .join(stringify!($bucket))
                .join(format!("{}.txt", stringify!($name)));
            if std::env::var("INSTA_UPDATE").as_deref() == Ok("always") || !path.exists() {
                std::fs::create_dir_all(path.parent().unwrap()).unwrap();
                std::fs::write(&path, &actual).unwrap();
                return;
            }
            let expected = std::fs::read_to_string(&path).unwrap();
            // CRLF→LF: windows captures `\r\n`; committed snapshots use `\n`.
            let actual_norm = actual.replace("\r\n", "\n");
            let expected_norm = expected.replace("\r\n", "\n");
            pretty_assertions::assert_eq!(
                actual_norm, expected_norm, "snapshot mismatch: {}", stringify!($name));
        }
    };
}