insta 1.47.2

A snapshot testing library for Rust
Documentation
//! Provides the [`Comparator`] trait, which provides a mechanism for specifying
//! how [`Snapshot`] data should be compared.

use crate::snapshot::{Snapshot, SnapshotContents, TextSnapshotKind};

/// Allows specific behavior to be invoked when [`Snapshot`]s are compared.
///
/// This is intended for when custom `Snapshot` comparison behavior is
/// desired. For example, two binary files that contain the same logical data
/// but have different representations on disk (as might be the case with
/// compressed images) could be compared with a `Comparator` that decompresses
/// `Snapshot` data before comparing it.
///
/// To make a custom `Comparator` active, pass it to
/// [`crate::settings::Settings::set_comparator`] or call [`with_settings!`] and
/// provide an appropriate `Comparator` instance.
///
/// This trait requires `'static` so that implementing structs can be stored in
/// [`crate::settings::Settings`].
// TODO: `Send + Sync` is required because `Settings` currently uses `Arc`
// internally. Consider removing these bounds if `Settings` switches to `Rc`
// in the next breaking change.
pub trait Comparator: Send + Sync + 'static {
    /// Returns `true` if the contents of `reference` and `test` match.
    ///
    /// This is the standard comparison used by [`assert_snapshot!`].
    fn matches(&self, reference: &Snapshot, test: &Snapshot) -> bool;

    /// Returns `true` if `reference` and `test` match fully, including metadata.
    ///
    /// This is used when `INSTA_REQUIRE_FULL_MATCH` is enabled. The default
    /// implementation delegates to [`matches`](Self::matches).
    fn matches_fully(&self, reference: &Snapshot, test: &Snapshot) -> bool {
        self.matches(reference, test)
    }

    /// Returns a type-erased clone of `self`.
    ///
    /// This is needed so that [`crate::settings::Settings`] (which provides the
    /// usual mechanism for setting a custom `Comparator`) can implement
    /// [`Clone`].
    fn dyn_clone(&self) -> Box<dyn Comparator>;
}

/// Provides default comparison semantics for [`Snapshot`]s. Binary snapshots
/// are compared on the basis of their contents (including file extension). Text
/// snapshots are compared on the basis of their deserialized representation.
#[derive(Clone)]
pub struct DefaultComparator;

impl Comparator for DefaultComparator {
    fn matches(&self, reference: &Snapshot, test: &Snapshot) -> bool {
        reference.contents() == test.contents()
            // For binary snapshots the extension also needs to be the same:
            && reference.metadata().snapshot_kind == test.metadata().snapshot_kind
    }

    fn matches_fully(&self, reference: &Snapshot, test: &Snapshot) -> bool {
        match (reference.contents(), test.contents()) {
            (SnapshotContents::Text(ref_contents), SnapshotContents::Text(test_contents)) => {
                // Note that we previously would match the exact values of the
                // unnormalized text. But that's too strict — it means we can
                // never match a snapshot that has leading/trailing whitespace.
                // So instead we check it matches on the latest format.
                // Generally those should be the same — latest should be doing
                // the minimum normalization; if they diverge we could update
                // this to be stricter.
                let contents_match_exact = ref_contents.matches_latest(test_contents);
                match ref_contents.kind {
                    TextSnapshotKind::File => {
                        reference.metadata().trim_for_persistence()
                            == test.metadata().trim_for_persistence()
                            && contents_match_exact
                    }
                    TextSnapshotKind::Inline => contents_match_exact,
                }
            }
            _ => self.matches(reference, test),
        }
    }

    fn dyn_clone(&self) -> Box<dyn Comparator> {
        Box::new(self.clone())
    }
}

#[cfg(test)]
mod test {
    use super::DefaultComparator;

    use crate::comparator::Comparator;
    use crate::snapshot::{
        MetaData, Snapshot, SnapshotContents, TextSnapshotContents, TextSnapshotKind,
    };

    const TEXT: &str =
        "The sky above the port was the color of a television, tuned to a dead channel.";

    #[test]
    fn default_comparator_matches() {
        let comparator = DefaultComparator;
        let a = Snapshot::from_components(
            String::from("test"),
            None,
            MetaData::default(),
            SnapshotContents::Text(TextSnapshotContents::new(
                String::from(TEXT),
                TextSnapshotKind::Inline,
            )),
        );
        let b = a.clone();
        assert!(comparator.matches(&a, &b));
        assert!(comparator.matches_fully(&a, &b));
    }

    #[test]
    fn default_comparator_matches_fully() {
        let comparator = DefaultComparator;
        let a = Snapshot::from_components(
            String::from("test"),
            None,
            MetaData::default(),
            SnapshotContents::Text(TextSnapshotContents::new(
                String::from(TEXT),
                TextSnapshotKind::File,
            )),
        );
        let mut b = Snapshot::from_components(
            String::from("test"),
            None,
            MetaData::default(),
            SnapshotContents::Text(TextSnapshotContents::new(
                String::from(TEXT),
                TextSnapshotKind::Inline,
            )),
        );
        b.metadata.description = Some(String::from("wintermute")); // Differs from None in a.

        // Comparing contents alone passes.
        assert!(comparator.matches(&a, &b));

        // Comparing contents alone still passes.
        assert!(comparator.matches_fully(&a, &a));
        // Comparing snapshots with differing metadata fails.
        assert!(!comparator.matches_fully(&a, &b));
    }
}