diff --git a/Cargo.lock b/Cargo.lock index 075e5cef..754c2389 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -82,6 +82,7 @@ dependencies = [ "similar", "syn", "tempfile", + "termcolor", "uuid", "walkdir", ] diff --git a/cargo-insta/Cargo.toml b/cargo-insta/Cargo.toml index 9e95c6fd..2258d64f 100644 --- a/cargo-insta/Cargo.toml +++ b/cargo-insta/Cargo.toml @@ -30,8 +30,10 @@ tempfile = "3.5.0" semver = {version = "1.0.7", features = ["serde"]} lazy_static = "1.4.0" clap = { workspace=true } +itertools = "0.10.0" [dev-dependencies] walkdir = "2.3.1" similar= "2.2.1" itertools = "0.10.0" +termcolor = "1.1.2" diff --git a/cargo-insta/src/cargo.rs b/cargo-insta/src/cargo.rs index 15e2f07f..7e15f890 100644 --- a/cargo-insta/src/cargo.rs +++ b/cargo-insta/src/cargo.rs @@ -1,9 +1,11 @@ use std::path::PathBuf; pub(crate) use cargo_metadata::Package; +use itertools::Itertools; /// Find snapshot roots within a package -// (I'm not sure how necessary this is; relative to just using all paths?) +// We need this because paths are not always conventional — for example cargo +// can reference artifacts that are outside of the package root. pub(crate) fn find_snapshot_roots(package: &Package) -> Vec { let mut roots = Vec::new(); @@ -38,13 +40,19 @@ pub(crate) fn find_snapshot_roots(package: &Package) -> Vec { // we would encounter paths twice. If we don't skip them here we run // into issues where the existence of a build script causes a snapshot // to be picked up twice since the same path is determined. (GH-15) - roots.sort_by_key(|x| x.as_os_str().len()); - let mut reduced_roots = vec![]; - for root in roots { - if !reduced_roots.iter().any(|x| root.starts_with(x)) { - reduced_roots.push(root); - } - } - - reduced_roots + let canonical_roots: Vec<_> = roots + .iter() + .filter_map(|x| x.canonicalize().ok()) + .sorted_by_key(|x| x.as_os_str().len()) + .collect(); + + canonical_roots + .clone() + .into_iter() + .filter(|root| { + !canonical_roots + .iter() + .any(|x| root.starts_with(x) && root != x) + }) + .collect() } diff --git a/cargo-insta/src/cli.rs b/cargo-insta/src/cli.rs index e3eafafe..ac00787a 100644 --- a/cargo-insta/src/cli.rs +++ b/cargo-insta/src/cli.rs @@ -83,7 +83,7 @@ enum Command { #[derive(Args, Debug, Clone)] struct TargetArgs { - /// Path to Cargo.toml + /// Path to `Cargo.toml` #[arg(long, value_name = "PATH")] manifest_path: Option, /// Explicit path to the workspace root @@ -95,7 +95,7 @@ struct TargetArgs { /// Work on all packages in the workspace #[arg(long)] workspace: bool, - /// Alias for --workspace (deprecated) + /// Alias for `--workspace` (deprecated) #[arg(long)] all: bool, /// Also walk into ignored paths. diff --git a/cargo-insta/src/container.rs b/cargo-insta/src/container.rs index 6a2ded50..125b2fec 100644 --- a/cargo-insta/src/container.rs +++ b/cargo-insta/src/container.rs @@ -47,7 +47,9 @@ impl PendingSnapshot { /// single rust file. #[derive(Debug)] pub(crate) struct SnapshotContainer { - snapshot_path: PathBuf, + // Path of the pending snapshot file (generally a `.new` or `.pending-snap` file) + pending_path: PathBuf, + // Path of the target snapshot file (generally a `.snap` file) target_path: PathBuf, kind: SnapshotKind, snapshots: Vec, @@ -56,7 +58,7 @@ pub(crate) struct SnapshotContainer { impl SnapshotContainer { pub(crate) fn load( - snapshot_path: PathBuf, + pending_path: PathBuf, target_path: PathBuf, kind: SnapshotKind, ) -> Result> { @@ -68,7 +70,7 @@ impl SnapshotContainer { } else { Some(Snapshot::from_file(&target_path)?) }; - let new = Snapshot::from_file(&snapshot_path)?; + let new = Snapshot::from_file(&pending_path)?; snapshots.push(PendingSnapshot { id: 0, old, @@ -79,7 +81,7 @@ impl SnapshotContainer { None } SnapshotKind::Inline => { - let mut pending_vec = PendingInlineSnapshot::load_batch(&snapshot_path)?; + let mut pending_vec = PendingInlineSnapshot::load_batch(&pending_path)?; let mut have_new = false; let rv = if fs::metadata(&target_path).is_ok() { @@ -111,8 +113,8 @@ impl SnapshotContainer { // The runtime code will issue something like this: // PendingInlineSnapshot::new(None, None, line).save(pending_snapshots)?; if !have_new { - fs::remove_file(&snapshot_path) - .map_err(|e| ContentError::FileIo(e, snapshot_path.to_path_buf()))?; + fs::remove_file(&pending_path) + .map_err(|e| ContentError::FileIo(e, pending_path.to_path_buf()))?; } rv @@ -120,7 +122,7 @@ impl SnapshotContainer { }; Ok(SnapshotContainer { - snapshot_path, + pending_path, target_path, kind, snapshots, @@ -141,7 +143,7 @@ impl SnapshotContainer { pub(crate) fn snapshot_sort_key(&self) -> impl Ord + '_ { let path = self - .snapshot_path + .pending_path .file_name() .and_then(|x| x.to_str()) .unwrap_or_default(); @@ -201,9 +203,9 @@ impl SnapshotContainer { patcher.save()?; } if did_skip { - PendingInlineSnapshot::save_batch(&self.snapshot_path, &new_pending)?; + PendingInlineSnapshot::save_batch(&self.pending_path, &new_pending)?; } else { - try_removing_snapshot(&self.snapshot_path); + try_removing_snapshot(&self.pending_path); } } else { // should only be one or this is weird @@ -211,22 +213,22 @@ impl SnapshotContainer { for snapshot in self.snapshots.iter() { match snapshot.op { Operation::Accept => { - let snapshot = Snapshot::from_file(&self.snapshot_path).map_err(|e| { + let snapshot = Snapshot::from_file(&self.pending_path).map_err(|e| { // If it's an IO error, pass a ContentError back so // we get a slightly clearer error message match e.downcast::() { Ok(io_error) => Box::new(ContentError::FileIo( *io_error, - self.snapshot_path.to_path_buf(), + self.pending_path.to_path_buf(), )), Err(other_error) => other_error, } })?; snapshot.save(&self.target_path)?; - try_removing_snapshot(&self.snapshot_path); + try_removing_snapshot(&self.pending_path); } Operation::Reject => { - try_removing_snapshot(&self.snapshot_path); + try_removing_snapshot(&self.pending_path); } Operation::Skip => {} } diff --git a/cargo-insta/src/inline.rs b/cargo-insta/src/inline.rs index b0ee5150..5aee2f49 100644 --- a/cargo-insta/src/inline.rs +++ b/cargo-insta/src/inline.rs @@ -149,19 +149,22 @@ impl FilePatcher { impl Visitor { fn scan_nested_macros(&mut self, tokens: &[TokenTree]) { for idx in 0..tokens.len() { + // Look for the start of a macro (potential snapshot location) if let Some(TokenTree::Ident(_)) = tokens.get(idx) { if let Some(TokenTree::Punct(ref punct)) = tokens.get(idx + 1) { if punct.as_char() == '!' { if let Some(TokenTree::Group(ref group)) = tokens.get(idx + 2) { + // Found a macro, determine its indentation let indentation = scan_for_path_start(tokens, idx); + // Extract tokens from the macro arguments let tokens: Vec<_> = group.stream().into_iter().collect(); + // Try to extract a snapshot, passing the calculated indentation self.try_extract_snapshot(&tokens, indentation); } } } } } - for token in tokens { // recurse into groups if let TokenTree::Group(group) = token { diff --git a/cargo-insta/src/main.rs b/cargo-insta/src/main.rs index 033b5706..487b35a5 100644 --- a/cargo-insta/src/main.rs +++ b/cargo-insta/src/main.rs @@ -1,3 +1,6 @@ +#![warn(clippy::doc_markdown)] +#![warn(rustdoc::all)] + //!
//! //!

cargo-insta: review tool for insta, a snapshot testing library for Rust

diff --git a/cargo-insta/src/utils.rs b/cargo-insta/src/utils.rs index 74930394..53d5b621 100644 --- a/cargo-insta/src/utils.rs +++ b/cargo-insta/src/utils.rs @@ -48,7 +48,7 @@ lazy_static! { pub static ref INSTA_VERSION: Version = read_insta_version().unwrap(); } -/// cargo-insta version +/// `cargo-insta` version // We could put this in a lazy_static pub(crate) fn cargo_insta_version() -> String { env!("CARGO_PKG_VERSION").to_string() diff --git a/cargo-insta/src/walk.rs b/cargo-insta/src/walk.rs index d94c651a..02751990 100644 --- a/cargo-insta/src/walk.rs +++ b/cargo-insta/src/walk.rs @@ -28,23 +28,25 @@ pub(crate) fn find_pending_snapshots<'a>( flags: FindFlags, ) -> impl Iterator>> + 'a { make_snapshot_walker(package_root, extensions, flags) - .filter_map(|e| e.ok()) - .filter_map(move |e| { - let fname = e.file_name().to_string_lossy(); - if fname.ends_with(".new") { - let new_path = e.into_path(); - let old_path = new_path.clone().with_extension(""); + .filter_map(Result::ok) + .filter_map(|entry| { + let fname = entry.file_name().to_string_lossy(); + let path = entry.clone().into_path(); + + #[allow(clippy::manual_map)] + if let Some(new_fname) = fname.strip_suffix(".new") { Some(SnapshotContainer::load( - new_path, - old_path, + path.clone(), + path.with_file_name(new_fname), SnapshotKind::File, )) - } else if fname.starts_with('.') && fname.ends_with(".pending-snap") { - let mut target_path = e.path().to_path_buf(); - target_path.set_file_name(&fname[1..fname.len() - 13]); + } else if let Some(new_fname) = fname + .strip_prefix('.') + .and_then(|f| f.strip_suffix(".pending-snap")) + { Some(SnapshotContainer::load( - e.path().to_path_buf(), - target_path, + path.clone(), + path.with_file_name(new_fname), SnapshotKind::Inline, )) } else { diff --git a/cargo-insta/tests/main.rs b/cargo-insta/tests/main.rs index 91cd22c6..1e1db78f 100644 --- a/cargo-insta/tests/main.rs +++ b/cargo-insta/tests/main.rs @@ -1,25 +1,41 @@ /// Integration tests which allow creating a full repo, running `cargo-insta` /// and then checking the output. /// +/// Often we want to see output from the test commands we run here; for example +/// a `dbg!` statement we add while debugging. Cargo by default hides the output +/// of passing tests. +/// - Like any test, to forward the output of an outer test (i.e. one of the +/// `#[test]`s in this file) to the terminal, pass `--nocapture` to the test +/// runner, like `cargo insta test -- --nocapture`. +/// - To forward the output of an inner test (i.e. the test commands we create +/// and run within an outer test) to the output of an outer test, pass +/// `--nocapture` in the command we create; for example `.args(["test", +/// "--accept", "--", "--nocapture"])`. We then also need to pass +/// `--nocapture` to the outer test to forward that to the terminal. +/// - If a test fails on `assert_success` or `assert_failure`, we print the +/// output of the inner test (bypassing the capturing). We color it to +/// discriminate it from the output of the outer test. +/// /// We can write more docs if that would be helpful. For the moment one thing to /// be aware of: it seems the packages must have different names, or we'll see /// interference between the tests. /// -/// (That seems to be because they all share the same `target` directory, which -/// cargo will confuse for each other if they share the same name. I haven't -/// worked out why — this is the case even if the files are the same between two -/// tests but with different commands — and those files exist in different -/// temporary workspace dirs. (We could try to enforce different names, or give -/// up using a consistent target directory for a cache, but it would slow down -/// repeatedly running the tests locally. To demonstrate the effect, name crates -/// the same...). This also causes issues when running the same tests -/// concurrently. +/// > That seems to be because they all share the same `target` directory, which +/// > cargo will confuse for each other if they share the same name. I haven't +/// > worked out why — this is the case even if the files are the same between two +/// > tests but with different commands — and those files exist in different +/// > temporary workspace dirs. (We could try to enforce different names, or give +/// > up using a consistent target directory for a cache, but it would slow down +/// > repeatedly running the tests locally. To demonstrate the effect, name crates +/// > the same... This also causes issues when running the same tests +/// > concurrently. use std::collections::HashMap; -use std::env; use std::fs; use std::path::{Path, PathBuf}; use std::process::Command; +use std::{env, fs::remove_dir_all}; +use console::style; use ignore::WalkBuilder; use insta::assert_snapshot; use itertools::Itertools; @@ -47,7 +63,7 @@ impl TestFiles { } } -/// Path of the insta crate in this repo, which we use as a dependency in the test project +/// Path of the [`insta`] crate in this repo, which we use as a dependency in the test project fn insta_path() -> PathBuf { PathBuf::from(env!("CARGO_MANIFEST_DIR")) .parent() @@ -67,20 +83,28 @@ fn target_dir() -> PathBuf { } fn assert_success(output: &std::process::Output) { - // Print stderr. Cargo test hides this when tests are successful, but if a - // test successfully exectues a command but then fails (e.g. on a snapshot), - // we would otherwise lose any output from the command such as `dbg!` - // statements. - eprint!("{}", String::from_utf8_lossy(&output.stderr)); - eprint!("{}", String::from_utf8_lossy(&output.stdout)); assert!( output.status.success(), "Tests failed: {}\n{}", - String::from_utf8_lossy(&output.stdout), - String::from_utf8_lossy(&output.stderr) + // Color the inner output so we can tell what's coming from there vs our own + // output + // Should we also do something like indent them? Or add a prefix? + format_args!("{}", style(String::from_utf8_lossy(&output.stdout)).green()), + format_args!("{}", style(String::from_utf8_lossy(&output.stderr)).red()) ); } +fn assert_failure(output: &std::process::Output) { + assert!( + !output.status.success(), + "Tests unexpectedly succeeded: {}\n{}", + // Color the inner output so we can tell what's coming from there vs our own + // output + // Should we also do something like indent them? Or add a prefix? + format_args!("{}", style(String::from_utf8_lossy(&output.stdout)).green()), + format_args!("{}", style(String::from_utf8_lossy(&output.stderr)).red()) + ); +} struct TestProject { /// Temporary directory where the project is created workspace_dir: PathBuf, @@ -110,23 +134,27 @@ impl TestProject { workspace_dir, } } - fn cmd(&self) -> Command { - let mut command = Command::new(env!("CARGO_BIN_EXE_cargo-insta")); + fn clean_env(cmd: &mut Command) { // Remove environment variables so we don't inherit anything (such as // `INSTA_FORCE_PASS` or `CARGO_INSTA_*`) from a cargo-insta process // which runs this integration test. for (key, _) in env::vars() { if key.starts_with("CARGO_INSTA") || key.starts_with("INSTA") { - command.env_remove(&key); + cmd.env_remove(&key); } } // Turn off CI flag so that cargo insta test behaves as we expect // under normal operation - command.env("CI", "0"); + cmd.env("CI", "0"); // And any others that can affect the output - command.env_remove("CARGO_TERM_COLOR"); - command.env_remove("CLICOLOR_FORCE"); - command.env_remove("RUSTDOCFLAGS"); + cmd.env_remove("CARGO_TERM_COLOR"); + cmd.env_remove("CLICOLOR_FORCE"); + cmd.env_remove("RUSTDOCFLAGS"); + } + + fn insta_cmd(&self) -> Command { + let mut command = Command::new(env!("CARGO_BIN_EXE_cargo-insta")); + Self::clean_env(&mut command); command.current_dir(self.workspace_dir.as_path()); // Use the same target directory as other tests, consistent across test @@ -229,8 +257,8 @@ fn test_json_snapshot() { .create_project(); let output = test_project - .cmd() - .args(["test", "--accept"]) + .insta_cmd() + .args(["test", "--accept", "--", "--nocapture"]) .output() .unwrap(); @@ -298,7 +326,7 @@ fn test_yaml_snapshot() { .create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept"]) .output() .unwrap(); @@ -373,7 +401,7 @@ fn test_trailing_comma_in_inline_snapshot() { .create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept"]) .output() .unwrap(); @@ -488,7 +516,7 @@ fn test_root() { } /// Check that in a workspace with a default root crate, running `cargo insta -/// test --workspace --accept` will update snapsnots in both the root crate and the +/// test --workspace --accept` will update snapshots in both the root crate and the /// member crate. #[test] fn test_root_crate_workspace_accept() { @@ -496,7 +524,7 @@ fn test_root_crate_workspace_accept() { workspace_with_root_crate("root-crate-workspace-accept".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept", "--workspace"]) .output() .unwrap(); @@ -531,7 +559,7 @@ fn test_root_crate_workspace() { workspace_with_root_crate("root-crate-workspace".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() // Need to disable colors to assert the output below .args(["test", "--workspace", "--color=never"]) .output() @@ -546,13 +574,13 @@ fn test_root_crate_workspace() { } /// Check that in a workspace with a default root crate, running `cargo insta -/// test --accept` will only update snapsnots in the root crate +/// test --accept` will only update snapshots in the root crate #[test] fn test_root_crate_no_all() { let test_project = workspace_with_root_crate("root-crate-no-all".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept"]) .output() .unwrap(); @@ -654,7 +682,7 @@ fn test_virtual_manifest_all() { workspace_with_virtual_manifest("virtual-manifest-all".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept", "--workspace"]) .output() .unwrap(); @@ -691,7 +719,7 @@ fn test_virtual_manifest_default() { workspace_with_virtual_manifest("virtual-manifest-default".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept"]) .output() .unwrap(); @@ -728,7 +756,7 @@ fn test_virtual_manifest_single_crate() { workspace_with_virtual_manifest("virtual-manifest-single".to_string()).create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept", "-p", "virtual-manifest-single-member-1"]) .output() .unwrap(); @@ -791,7 +819,7 @@ fn test_old_yaml_format() { // Run the test with --force-update-snapshots and --accept let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept", "--", "--nocapture"]) .output() .unwrap(); @@ -856,7 +884,7 @@ Hello, world! // Test with current insta version let output_current = test_current_insta - .cmd() + .insta_cmd() .args(["test", "--accept", "--force-update-snapshots"]) .output() .unwrap(); @@ -865,7 +893,7 @@ Hello, world! // Test with insta 1.40.0 let output_1_40_0 = test_insta_1_40_0 - .cmd() + .insta_cmd() .args(["test", "--accept", "--force-update-snapshots"]) .output() .unwrap(); @@ -937,7 +965,7 @@ fn test_linebreaks() { // Run the test with --force-update-snapshots and --accept let output = test_project - .cmd() + .insta_cmd() .args([ "test", "--force-update-snapshots", @@ -1005,7 +1033,7 @@ fn test_excessive_hashes() { // Run the test with --force-update-snapshots and --accept let output = test_project - .cmd() + .insta_cmd() .args([ "test", "--force-update-snapshots", @@ -1059,7 +1087,7 @@ fn test_wrong_indent_force() { // Confirm the test passes despite the indent let output = test_project - .cmd() + .insta_cmd() .args(["test", "--check", "--", "--nocapture"]) .output() .unwrap(); @@ -1068,7 +1096,7 @@ fn test_wrong_indent_force() { // Then run the test with --force-update-snapshots and --accept to confirm // the new snapshot is written let output = test_project - .cmd() + .insta_cmd() .args([ "test", "--force-update-snapshots", @@ -1130,7 +1158,7 @@ fn test_hashtag_escape() { .create_project(); let output = test_project - .cmd() + .insta_cmd() .args(["test", "--accept"]) .output() .unwrap(); @@ -1152,3 +1180,240 @@ fn test_hashtag_escape() { } "####); } + +// Can't get the test binary discovery to work, don't have a windows machine to +// hand, others are welcome to fix it. (No specific reason to think that insta +// doesn't work on windows, just that the test doesn't work.) +#[cfg(not(target_os = "windows"))] +#[test] +fn test_insta_workspace_root() { + // This function locates the compiled test binary in the target directory. + // It's necessary because the exact filename of the test binary includes a hash + // that we can't predict, so we need to search for it. + fn find_test_binary(dir: &Path) -> PathBuf { + dir.join("target/debug/deps") + .read_dir() + .unwrap() + .filter_map(Result::ok) + .find(|entry| { + let file_name = entry.file_name(); + let file_name_str = file_name.to_str().unwrap_or(""); + // We're looking for a file that: + file_name_str.starts_with("insta_workspace_root_test-") // Matches our test name + && !file_name_str.contains('.') // Doesn't have an extension (it's the executable, not a metadata file) + && entry.metadata().map(|m| m.is_file()).unwrap_or(false) // Is a file, not a directory + }) + .map(|entry| entry.path()) + .expect("Failed to find test binary") + } + + fn run_test_binary( + binary_path: &Path, + current_dir: &Path, + env: Option<(&str, &str)>, + ) -> std::process::Output { + let mut cmd = Command::new(binary_path); + TestProject::clean_env(&mut cmd); + cmd.current_dir(current_dir); + if let Some((key, value)) = env { + cmd.env(key, value); + } + cmd.output().unwrap() + } + + let test_project = TestFiles::new() + .add_file( + "Cargo.toml", + r#" + [package] + name = "insta_workspace_root_test" + version = "0.1.0" + edition = "2021" + + [dependencies] + insta = { path = '$PROJECT_PATH' } + "# + .to_string(), + ) + .add_file( + "src/lib.rs", + r#" + #[cfg(test)] + mod tests { + use insta::assert_snapshot; + + #[test] + fn test_snapshot() { + assert_snapshot!("Hello, world!"); + } + } + "# + .to_string(), + ) + .create_project(); + + let mut cargo_cmd = Command::new("cargo"); + TestProject::clean_env(&mut cargo_cmd); + let output = cargo_cmd + .args(["test", "--no-run"]) + .current_dir(&test_project.workspace_dir) + .output() + .unwrap(); + assert_success(&output); + + let test_binary_path = find_test_binary(&test_project.workspace_dir); + + // Run the test without snapshot (should fail) + assert_failure(&run_test_binary( + &test_binary_path, + &test_project.workspace_dir, + None, + )); + + // Create the snapshot + assert_success(&run_test_binary( + &test_binary_path, + &test_project.workspace_dir, + Some(("INSTA_UPDATE", "always")), + )); + + // Verify snapshot creation + assert!(test_project.workspace_dir.join("src/snapshots").exists()); + assert!(test_project + .workspace_dir + .join("src/snapshots/insta_workspace_root_test__tests__snapshot.snap") + .exists()); + + // Move the workspace + let moved_workspace = { + let moved_workspace = PathBuf::from("/tmp/cargo-insta-test-moved"); + remove_dir_all(&moved_workspace).ok(); + fs::create_dir(&moved_workspace).unwrap(); + fs::rename(&test_project.workspace_dir, &moved_workspace).unwrap(); + moved_workspace + }; + let moved_binary_path = find_test_binary(&moved_workspace); + + // Run test in moved workspace without INSTA_WORKSPACE_ROOT (should fail) + assert_failure(&run_test_binary(&moved_binary_path, &moved_workspace, None)); + + // Run test in moved workspace with INSTA_WORKSPACE_ROOT (should pass) + assert_success(&run_test_binary( + &moved_binary_path, + &moved_workspace, + Some(("INSTA_WORKSPACE_ROOT", moved_workspace.to_str().unwrap())), + )); +} + +#[test] +fn test_external_test_path() { + let test_project = TestFiles::new() + .add_file( + "proj/Cargo.toml", + r#" +[package] +name = "external_test_path" +version = "0.1.0" +edition = "2021" + +[dependencies] +insta = { path = '$PROJECT_PATH' } + +[[test]] +name = "tlib" +path = "../tests/lib.rs" +"# + .to_string(), + ) + .add_file( + "proj/src/lib.rs", + r#" +pub fn hello() -> String { + "Hello, world!".to_string() +} +"# + .to_string(), + ) + .add_file( + "tests/lib.rs", + r#" +use external_test_path::hello; + +#[test] +fn test_hello() { + insta::assert_snapshot!(hello()); +} +"# + .to_string(), + ) + .create_project(); + + // Change to the proj directory for running cargo commands + let proj_dir = test_project.workspace_dir.join("proj"); + + // Initially, the test should fail + let output = test_project + .insta_cmd() + .current_dir(&proj_dir) + .args(["test", "--"]) + .output() + .unwrap(); + + assert_failure(&output); + + // Verify that the snapshot was created in the correct location + assert_snapshot!(TestProject::current_file_tree(&test_project.workspace_dir), @r" + proj + proj/Cargo.lock + proj/Cargo.toml + proj/src + proj/src/lib.rs + tests + tests/lib.rs + tests/snapshots + tests/snapshots/tlib__hello.snap.new + "); + + // Run cargo insta accept + let output = test_project + .insta_cmd() + .current_dir(&proj_dir) + .args(["test", "--accept"]) + .output() + .unwrap(); + + assert_success(&output); + + // Verify that the snapshot was created in the correct location + assert_snapshot!(TestProject::current_file_tree(&test_project.workspace_dir), @r" + proj + proj/Cargo.lock + proj/Cargo.toml + proj/src + proj/src/lib.rs + tests + tests/lib.rs + tests/snapshots + tests/snapshots/tlib__hello.snap + "); + + // Run the test again, it should pass now + let output = Command::new(env!("CARGO_BIN_EXE_cargo-insta")) + .current_dir(&proj_dir) + .args(["test"]) + .output() + .unwrap(); + + assert_success(&output); + + let snapshot_path = test_project + .workspace_dir + .join("tests/snapshots/tlib__hello.snap"); + assert_snapshot!(fs::read_to_string(snapshot_path).unwrap(), @r#" + --- + source: "../tests/lib.rs" + expression: hello() + --- + Hello, world! + "#); +} diff --git a/insta/src/content/json.rs b/insta/src/content/json.rs index 2141226a..e4dd5e67 100644 --- a/insta/src/content/json.rs +++ b/insta/src/content/json.rs @@ -29,7 +29,7 @@ pub struct Serializer { } impl Serializer { - /// Creates a new serializer that writes into the given writer. + /// Creates a new [`Serializer`] that writes into the given writer. pub fn new() -> Serializer { Serializer { out: String::new(), diff --git a/insta/src/content/mod.rs b/insta/src/content/mod.rs index 8fe5e14e..3be7144c 100644 --- a/insta/src/content/mod.rs +++ b/insta/src/content/mod.rs @@ -24,7 +24,6 @@ pub enum Error { UnexpectedDataType, #[cfg(feature = "_cargo_insta_internal")] MissingField, - #[cfg(feature = "_cargo_insta_internal")] FileIo(std::io::Error, std::path::PathBuf), } @@ -39,7 +38,6 @@ impl fmt::Display for Error { } #[cfg(feature = "_cargo_insta_internal")] Error::MissingField => f.write_str("A required field was missing"), - #[cfg(feature = "_cargo_insta_internal")] Error::FileIo(e, p) => { f.write_str(format!("File error for {:?}: {}", p.display(), e).as_str()) } @@ -57,7 +55,7 @@ impl std::error::Error for Error {} /// /// Some enum variants are intentionally not exposed to user code. /// It's generally recommended to construct content objects by -/// using the [`From`](std::convert::From) trait and by using the +/// using the [`From`] trait and by using the /// accessor methods to assert on it. /// /// While matching on the content is possible in theory it is @@ -65,12 +63,12 @@ impl std::error::Error for Error {} /// enum holds variants that can "wrap" values where it's not /// expected. For instance if a field holds an `Option` /// you cannot use pattern matching to extract the string as it -/// will be contained in an internal `Some` variant that is not -/// exposed. On the other hand the `as_str` method will +/// will be contained in an internal [`Some`] variant that is not +/// exposed. On the other hand the [`Content::as_str`] method will /// automatically resolve such internal wrappers. /// /// If you do need to pattern match you should use the -/// `resolve_inner` method to resolve such internal wrappers. +/// [`Content::resolve_inner`] method to resolve such internal wrappers. #[derive(Debug, Clone, PartialEq, PartialOrd)] pub enum Content { Bool(bool), @@ -196,7 +194,7 @@ impl Content { } } - /// Mutable version of [`resolve_inner`](Self::resolve_inner). + /// Mutable version of [`Self::resolve_inner`]. pub fn resolve_inner_mut(&mut self) -> &mut Content { match *self { Content::Some(ref mut v) diff --git a/insta/src/content/serialization.rs b/insta/src/content/serialization.rs index b0eeea6d..e9223e2a 100644 --- a/insta/src/content/serialization.rs +++ b/insta/src/content/serialization.rs @@ -19,7 +19,7 @@ pub enum Key<'a> { } impl<'a> Key<'a> { - /// Needed because std::mem::discriminant is not Ord + /// Needed because [`std::mem::discriminant`] is not [`Ord`] fn discriminant(&self) -> usize { match self { Key::Bool(_) => 1, diff --git a/insta/src/content/yaml/vendored/emitter.rs b/insta/src/content/yaml/vendored/emitter.rs index d04b4fa9..2ea1f2c2 100644 --- a/insta/src/content/yaml/vendored/emitter.rs +++ b/insta/src/content/yaml/vendored/emitter.rs @@ -1,6 +1,5 @@ use crate::content::yaml::vendored::yaml::{Hash, Yaml}; -use std::convert::From; use std::error::Error; use std::fmt::{self, Display}; @@ -39,7 +38,7 @@ pub struct YamlEmitter<'a> { pub type EmitResult = Result<(), EmitError>; -// from serialize::json +/// From [`serialize::json`] fn escape_str(wr: &mut dyn fmt::Write, v: &str) -> Result<(), fmt::Error> { wr.write_str("\"")?; @@ -103,7 +102,7 @@ fn escape_str(wr: &mut dyn fmt::Write, v: &str) -> Result<(), fmt::Error> { } impl<'a> YamlEmitter<'a> { - pub fn new(writer: &'a mut dyn fmt::Write) -> YamlEmitter { + pub fn new(writer: &'a mut dyn fmt::Write) -> YamlEmitter<'a> { YamlEmitter { writer, best_indent: 2, @@ -249,20 +248,22 @@ impl<'a> YamlEmitter<'a> { } } +#[allow(clippy::doc_markdown)] // \` is recognised as unbalanced backticks /// Check if the string requires quoting. +/// /// Strings starting with any of the following characters must be quoted. -/// :, &, *, ?, |, -, <, >, =, !, %, @ +/// `:`, `&`, `*`, `?`, `|`, `-`, `<`, `>`, `=`, `!`, `%`, `@` /// Strings containing any of the following characters must be quoted. -/// {, }, [, ], ,, #, ` +/// `{`, `}`, `\[`, `\]`, `,`, `#`, `\`` /// /// If the string contains any of the following control characters, it must be escaped with double quotes: -/// \0, \x01, \x02, \x03, \x04, \x05, \x06, \a, \b, \t, \n, \v, \f, \r, \x0e, \x0f, \x10, \x11, \x12, \x13, \x14, \x15, \x16, \x17, \x18, \x19, \x1a, \e, \x1c, \x1d, \x1e, \x1f, \N, \_, \L, \P +/// `\0`, `\x01`, `\x02`, `\x03`, `\x04`, `\x05`, `\x06`, `\a`, `\b`, `\t`, `\n, `\v, `\f`, `\r`, `\x0e`, `\x0f`, `\x10`, `\x11`, `\x12`, `\x13`, `\x14`, `\x15`, `\x16`, `\x17`, `\x18`, `\x19`, `\x1a`, `\e`, `\x1c`, `\x1d`, `\x1e`, `\x1f`, `\N`, `\_`, `\L`, `\P` /// /// Finally, there are other cases when the strings must be quoted, no matter if you're using single or double quotes: -/// * When the string is true or false (otherwise, it would be treated as a boolean value); -/// * When the string is null or ~ (otherwise, it would be considered as a null value); -/// * When the string looks like a number, such as integers (e.g. 2, 14, etc.), floats (e.g. 2.6, 14.9) and exponential numbers (e.g. 12e7, etc.) (otherwise, it would be treated as a numeric value); -/// * When the string looks like a date (e.g. 2014-12-31) (otherwise it would be automatically converted into a Unix timestamp). +/// * When the string is `true` or `false` (otherwise, it would be treated as a boolean value); +/// * When the string is `null` or `~` (otherwise, it would be considered as a null value); +/// * When the string looks like a number, such as integers (e.g. `2`, `14`, etc.), floats (e.g. `2.6`, `14.9`) and exponential numbers (e.g. `12e7`, etc.) (otherwise, it would be treated as a numeric value); +/// * When the string looks like a date (e.g. `2014-12-31`) (otherwise it would be automatically converted into a Unix timestamp). fn need_quotes(string: &str) -> bool { fn need_quotes_spaces(string: &str) -> bool { string.starts_with(' ') || string.ends_with(' ') diff --git a/insta/src/content/yaml/vendored/parser.rs b/insta/src/content/yaml/vendored/parser.rs index 1656bba9..b21ecb07 100644 --- a/insta/src/content/yaml/vendored/parser.rs +++ b/insta/src/content/yaml/vendored/parser.rs @@ -29,8 +29,8 @@ enum State { End, } -/// `Event` is used with the low-level event base parsing API, -/// see `EventReceiver` trait. +/// [`Event`] is used with the low-level event base parsing API, +/// see [`EventReceiver`] trait. #[derive(Clone, PartialEq, Debug, Eq)] pub enum Event { /// Reserved for internal use @@ -40,7 +40,7 @@ pub enum Event { DocumentEnd, /// Refer to an anchor ID Alias(usize), - /// Value, style, anchor_id, tag + /// Value, style, anchor ID, tag Scalar(String, TScalarStyle, usize, Option), /// Anchor ID SequenceStart(usize), diff --git a/insta/src/content/yaml/vendored/yaml.rs b/insta/src/content/yaml/vendored/yaml.rs index df502b2b..96787f76 100644 --- a/insta/src/content/yaml/vendored/yaml.rs +++ b/insta/src/content/yaml/vendored/yaml.rs @@ -14,8 +14,8 @@ use std::vec; /// access your YAML document. #[derive(Clone, PartialEq, PartialOrd, Debug, Eq, Ord, Hash)] pub enum Yaml { - /// Float types are stored as String and parsed on demand. - /// Note that f64 does NOT implement Eq trait and can NOT be stored in BTreeMap. + /// Float types are stored as [`String`] and parsed on demand. + /// Note that [`f64'] does NOT implement [`Eq'] trait and can NOT be stored in [`BTreeMap`]. Real(string::String), /// YAML int is stored as i64. Integer(i64), diff --git a/insta/src/env.rs b/insta/src/env.rs index 807f8d36..14f02b93 100644 --- a/insta/src/env.rs +++ b/insta/src/env.rs @@ -12,18 +12,16 @@ use crate::{ lazy_static::lazy_static! { static ref WORKSPACES: Mutex>> = Mutex::new(BTreeMap::new()); - static ref TOOL_CONFIGS: Mutex>> = Mutex::new(BTreeMap::new()); + static ref TOOL_CONFIGS: Mutex>> = Mutex::new(BTreeMap::new()); } -pub fn get_tool_config(manifest_dir: &str) -> Arc { - let mut configs = TOOL_CONFIGS.lock().unwrap(); - if let Some(rv) = configs.get(manifest_dir) { - return rv.clone(); - } - let config = - Arc::new(ToolConfig::from_manifest_dir(manifest_dir).expect("failed to load tool config")); - configs.insert(manifest_dir.to_string(), config.clone()); - config +pub fn get_tool_config(workspace_dir: &Path) -> Arc { + TOOL_CONFIGS + .lock() + .unwrap() + .entry(workspace_dir.to_path_buf()) + .or_insert_with(|| ToolConfig::from_workspace(workspace_dir).unwrap().into()) + .clone() } /// The test runner to use. @@ -125,11 +123,6 @@ pub struct ToolConfig { } impl ToolConfig { - /// Loads the tool config for a specific manifest. - pub fn from_manifest_dir(manifest_dir: &str) -> Result { - ToolConfig::from_workspace(&get_cargo_workspace(manifest_dir)) - } - /// Loads the tool config from a cargo workspace. pub fn from_workspace(workspace_dir: &Path) -> Result { let mut cfg = None; @@ -328,7 +321,7 @@ impl ToolConfig { self.snapshot_update } - /// Returns the value of glob_fail_fast + /// Returns whether the glob should fail fast, as snapshot failures within the glob macro will appear only at the end of execution unless `glob_fail_fast` is set. #[cfg(feature = "glob")] pub fn glob_fail_fast(&self) -> bool { self.glob_fail_fast @@ -411,43 +404,61 @@ pub fn snapshot_update_behavior(tool_config: &ToolConfig, unseen: bool) -> Snaps /// Returns the cargo workspace for a manifest pub fn get_cargo_workspace(manifest_dir: &str) -> Arc { - // we really do not care about poisoning here. - let mut workspaces = WORKSPACES.lock().unwrap_or_else(|x| x.into_inner()); - if let Some(rv) = workspaces.get(manifest_dir) { - rv.clone() - } else { - // If INSTA_WORKSPACE_ROOT environment variable is set, use the value - // as-is. This is useful for those users where the compiled in - // CARGO_MANIFEST_DIR points to some transient location. This can easily - // happen if the user builds the test in one directory but then tries to - // run it in another: even if sources are available in the new - // directory, in the past we would always go with the compiled-in value. - // The compiled-in directory may not even exist anymore. - let path = if let Ok(workspace_root) = std::env::var("INSTA_WORKSPACE_ROOT") { - Arc::new(PathBuf::from(workspace_root)) - } else { + // If INSTA_WORKSPACE_ROOT environment variable is set, use the value as-is. + // This is useful where CARGO_MANIFEST_DIR at compilation points to some + // transient location. This can easily happen when building the test in one + // directory but running it in another. + if let Ok(workspace_root) = env::var("INSTA_WORKSPACE_ROOT") { + return PathBuf::from(workspace_root).into(); + } + + let error_message = || { + format!( + "`cargo metadata --format-version=1 --no-deps` in path `{}`", + manifest_dir + ) + }; + + WORKSPACES + .lock() + // we really do not care about poisoning here. + .unwrap() + .entry(manifest_dir.to_string()) + .or_insert_with(|| { let output = std::process::Command::new( - env::var("CARGO") - .ok() - .unwrap_or_else(|| "cargo".to_string()), + env::var("CARGO").unwrap_or_else(|_| "cargo".to_string()), ) - .arg("metadata") - .arg("--format-version=1") - .arg("--no-deps") + .args(["metadata", "--format-version=1", "--no-deps"]) .current_dir(manifest_dir) .output() - .unwrap(); - let docs = crate::content::yaml::vendored::yaml::YamlLoader::load_from_str( + .unwrap_or_else(|e| panic!("failed to run {}\n\n{}", error_message(), e)); + + crate::content::yaml::vendored::yaml::YamlLoader::load_from_str( std::str::from_utf8(&output.stdout).unwrap(), ) - .unwrap(); - let manifest = docs.first().expect("Unable to parse cargo manifest"); - let workspace_root = PathBuf::from(manifest["workspace_root"].as_str().unwrap()); - Arc::new(workspace_root) - }; - workspaces.insert(manifest_dir.to_string(), path.clone()); - path - } + .map_err(|e| e.to_string()) + .and_then(|docs| { + docs.into_iter() + .next() + .ok_or_else(|| "No content found in yaml".to_string()) + }) + .and_then(|metadata| { + metadata["workspace_root"] + .clone() + .into_string() + .ok_or_else(|| "Couldn't find `workspace_root`".to_string()) + }) + .map(|path| PathBuf::from(path).into()) + .unwrap_or_else(|e| { + panic!( + "failed to parse cargo metadata output from {}: {}\n\n{:?}", + error_message(), + e, + output.stdout + ) + }) + }) + .clone() } #[cfg(feature = "_cargo_insta_internal")] @@ -480,7 +491,7 @@ impl std::str::FromStr for UnreferencedSnapshots { } } -/// Memoizes a snapshot file in the reference file. +/// Memoizes a snapshot file in the reference file, as part of removing unreferenced snapshots. pub fn memoize_snapshot_file(snapshot_file: &Path) { if let Ok(path) = env::var("INSTA_SNAPSHOT_REFERENCES_FILE") { let mut f = fs::OpenOptions::new() diff --git a/insta/src/glob.rs b/insta/src/glob.rs index 53579de4..faec3a8b 100644 --- a/insta/src/glob.rs +++ b/insta/src/glob.rs @@ -15,9 +15,9 @@ pub(crate) struct GlobCollector { pub(crate) show_insta_hint: bool, } -// the glob stack holds failure count + an indication if cargo insta review -// should be run. lazy_static::lazy_static! { + /// the glob stack holds failure count and an indication if `cargo insta review` + /// should be run. pub(crate) static ref GLOB_STACK: Mutex> = Mutex::default(); } @@ -38,7 +38,7 @@ lazy_static::lazy_static! { }; } -pub fn glob_exec(manifest_dir: &str, base: &Path, pattern: &str, mut f: F) { +pub fn glob_exec(workspace_dir: &Path, base: &Path, pattern: &str, mut f: F) { // If settings.allow_empty_glob() == true and `base` doesn't exist, skip // everything. This is necessary as `base` is user-controlled via `glob!/3` // and may not exist. @@ -61,7 +61,7 @@ pub fn glob_exec(manifest_dir: &str, base: &Path, pattern: &str GLOB_STACK.lock().unwrap().push(GlobCollector { failed: 0, show_insta_hint: false, - fail_fast: get_tool_config(manifest_dir).glob_fail_fast(), + fail_fast: get_tool_config(workspace_dir).glob_fail_fast(), }); // step 1: collect all matching files diff --git a/insta/src/lib.rs b/insta/src/lib.rs index 574d3807..142256b2 100644 --- a/insta/src/lib.rs +++ b/insta/src/lib.rs @@ -1,3 +1,6 @@ +#![warn(clippy::doc_markdown)] +#![warn(rustdoc::all)] + //!
//! //!

insta: a snapshot testing library for Rust

@@ -7,7 +10,7 @@ //! //! Snapshots tests (also sometimes called approval tests) are tests that //! assert values against a reference value (the snapshot). This is similar -//! to how `assert_eq!` lets you compare a value against a reference value but +//! to how [`assert_eq!`] lets you compare a value against a reference value but //! unlike simple string assertions, snapshot tests let you test against complex //! values and come with comprehensive tools to review changes. //! @@ -46,7 +49,8 @@ //! ``` //! //! The recommended flow is to run the tests once, have them fail and check -//! if the result is okay. By default the new snapshots are stored next +//! if the result is okay. +//! By default, the new snapshots are stored next //! to the old ones with the extra `.new` extension. Once you are satisfied //! move the new files over. To simplify this workflow you can use //! `cargo insta review` (requires @@ -80,7 +84,7 @@ //! [`Display`](std::fmt::Display) outputs, often strings. //! - [`assert_debug_snapshot!`] for comparing [`Debug`] outputs of values. //! -//! The following macros require the use of serde's [`Serialize`](serde::Serialize): +//! The following macros require the use of [`serde::Serialize`]: //! #![cfg_attr( feature = "csv", @@ -170,11 +174,11 @@ //! //! The following features exist: //! -//! * `csv`: enables CSV support (via serde) -//! * `json`: enables JSON support (via serde) -//! * `ron`: enables RON support (via serde) -//! * `toml`: enables TOML support (via serde) -//! * `yaml`: enables YAML support (via serde) +//! * `csv`: enables CSV support (via [`serde`]) +//! * `json`: enables JSON support (via [`serde`]) +//! * `ron`: enables RON support (via [`serde`]) +//! * `toml`: enables TOML support (via [`serde`]) +//! * `yaml`: enables YAML support (via [`serde`]) //! * `redactions`: enables support for redactions //! * `filters`: enables support for filters //! * `glob`: enables support for globbing ([`glob!`]) @@ -184,13 +188,14 @@ //! limited capacity. You will receive a deprecation warning if you are not //! opting into them but for now the macros will continue to function. //! -//! Enabling any of the serde based formats enables the hidden `serde` feature -//! which gates some serde specific APIs such as [`Settings::set_info`]. +//! Enabling any of the [`serde`] based formats enables the hidden `serde` feature +//! which gates some [`serde`] specific APIs such as [`Settings::set_info`]. //! //! # Dependencies //! -//! `insta` tries to be light in dependencies but this is tricky to accomplish -//! given what it tries to do. By default it currently depends on `serde` for +//! [`insta`] tries to be light in dependencies but this is tricky to accomplish +//! given what it tries to do. +//! By default, it currently depends on [`serde`] for //! the [`assert_toml_snapshot!`] and [`assert_yaml_snapshot!`] macros. In the //! future this default dependencies will be removed. To already benefit from //! this optimization you can disable the default features and manually opt into @@ -201,7 +206,7 @@ //! There are some settings that can be changed on a per-thread (and thus //! per-test) basis. For more information see [Settings]. //! -//! Additionally Insta will load a YAML config file with settings that change +//! Additionally, Insta will load a YAML config file with settings that change //! the behavior of insta between runs. It's loaded from any of the following //! locations: `.config/insta.yaml`, `insta.yaml` and `.insta.yaml` from the //! workspace root. The following config options exist: @@ -246,8 +251,8 @@ //! //! Insta benefits from being compiled in release mode, even as dev dependency. //! It will compile slightly slower once, but use less memory, have faster diffs -//! and just generally be more fun to use. To achieve that, opt `insta` and -//! `similar` (the diffing library) into higher optimization in your +//! and just generally be more fun to use. To achieve that, opt [`insta`] and +//! [`similar`] (the diffing library) into higher optimization in your //! `Cargo.toml`: //! //! ```yaml @@ -258,8 +263,10 @@ //! opt-level = 3 //! ``` //! -//! You can also disable the default features of `insta` which will cut down on +//! You can also disable the default features of [`insta`] which will cut down on //! the compile time a bit by removing some quality of life features. +//! +//! [`insta`]: https://docs.rs/insta #![cfg_attr(docsrs, feature(doc_cfg))] #[macro_use] diff --git a/insta/src/macros.rs b/insta/src/macros.rs index ba837e40..222ef3c6 100644 --- a/insta/src/macros.rs +++ b/insta/src/macros.rs @@ -15,11 +15,24 @@ macro_rules! _function_name { }}; } -/// Asserts a `Serialize` snapshot in CSV format. +#[doc(hidden)] +#[macro_export] +macro_rules! _get_workspace_root { + () => {{ + use std::env; + + // Note the `env!("CARGO_MANIFEST_DIR")` needs to be in the macro (in + // contrast to a function in insta) because the macro needs to capture + // the value in the caller library, an exclusive property of macros. + $crate::_macro_support::get_cargo_workspace(env!("CARGO_MANIFEST_DIR")) + }}; +} + +/// Asserts a [`serde::Serialize`] snapshot in CSV format. /// /// **Feature:** `csv` (disabled by default) /// -/// This works exactly like [`crate::assert_yaml_snapshot!`] +/// This works exactly like [`assert_yaml_snapshot!`](crate::assert_yaml_snapshot!) /// but serializes in [CSV](https://github.com/burntsushi/rust-csv) format instead of /// YAML. /// @@ -44,11 +57,11 @@ macro_rules! assert_csv_snapshot { }; } -/// Asserts a `Serialize` snapshot in TOML format. +/// Asserts a [`serde::Serialize`] snapshot in TOML format. /// /// **Feature:** `toml` (disabled by default) /// -/// This works exactly like [`crate::assert_yaml_snapshot!`] +/// This works exactly like [`assert_yaml_snapshot!`](crate::assert_yaml_snapshot!) /// but serializes in [TOML](https://github.com/alexcrichton/toml-rs) format instead of /// YAML. Note that TOML cannot represent all values due to limitations in the /// format. @@ -74,14 +87,14 @@ macro_rules! assert_toml_snapshot { }; } -/// Asserts a `Serialize` snapshot in YAML format. +/// Asserts a [`serde::Serialize`] snapshot in YAML format. /// /// **Feature:** `yaml` /// -/// The value needs to implement the `serde::Serialize` trait and the snapshot +/// The value needs to implement the [`serde::Serialize`] trait and the snapshot /// will be serialized in YAML format. This does mean that unlike the debug /// snapshot variant the type of the value does not appear in the output. -/// You can however use the `assert_ron_snapshot!` macro to dump out +/// You can however use the [`assert_ron_snapshot!`](crate::assert_ron_snapshot!) macro to dump out /// the value in [RON](https://github.com/ron-rs/ron/) format which retains some /// type information for more accurate comparisons. /// @@ -92,7 +105,7 @@ macro_rules! assert_toml_snapshot { /// assert_yaml_snapshot!(vec![1, 2, 3]); /// ``` /// -/// Unlike the [`crate::assert_debug_snapshot!`] +/// Unlike the [`assert_debug_snapshot!`](crate::assert_debug_snapshot!) /// macro, this one has a secondary mode where redactions can be defined. /// /// The third argument to the macro can be an object expression for redaction. @@ -128,11 +141,11 @@ macro_rules! assert_yaml_snapshot { }; } -/// Asserts a `Serialize` snapshot in RON format. +/// Asserts a [`serde::Serialize`] snapshot in RON format. /// /// **Feature:** `ron` (disabled by default) /// -/// This works exactly like [`assert_yaml_snapshot!`] +/// This works exactly like [`assert_yaml_snapshot!`](crate::assert_yaml_snapshot!) /// but serializes in [RON](https://github.com/ron-rs/ron/) format instead of /// YAML which retains some type information for more accurate comparisons. /// @@ -158,11 +171,11 @@ macro_rules! assert_ron_snapshot { }; } -/// Asserts a `Serialize` snapshot in JSON format. +/// Asserts a [`serde::Serialize`] snapshot in JSON format. /// /// **Feature:** `json` /// -/// This works exactly like [`assert_yaml_snapshot!`] but serializes in JSON format. +/// This works exactly like [`assert_yaml_snapshot!`](crate::assert_yaml_snapshot!) but serializes in JSON format. /// This is normally not recommended because it makes diffs less reliable, but it can /// be useful for certain specialized situations. /// @@ -188,11 +201,11 @@ macro_rules! assert_json_snapshot { }; } -/// Asserts a `Serialize` snapshot in compact JSON format. +/// Asserts a [`serde::Serialize`] snapshot in compact JSON format. /// /// **Feature:** `json` /// -/// This works exactly like [`assert_json_snapshot!`] but serializes into a single +/// This works exactly like [`assert_json_snapshot!`](crate::assert_json_snapshot!) but serializes into a single /// line for as long as the output is less than 120 characters. This can be useful /// in cases where you are working with small result outputs but comes at the cost /// of slightly worse diffing behavior. @@ -284,10 +297,10 @@ macro_rules! _prepare_snapshot_for_redaction { }; } -/// Asserts a `Debug` snapshot. +/// Asserts a [`Debug`] snapshot. /// -/// The value needs to implement the `fmt::Debug` trait. This is useful for -/// simple values that do not implement the `Serialize` trait, but does not +/// The value needs to implement the [`Debug`] trait. This is useful for +/// simple values that do not implement the [`serde::Serialize`] trait, but does not /// permit redactions. /// /// Debug is called with `"{:#?}"`, which means this uses pretty-print. @@ -298,10 +311,10 @@ macro_rules! assert_debug_snapshot { }; } -/// Asserts a `Debug` snapshot in compact format. +/// Asserts a [`Debug`] snapshot in compact format. /// -/// The value needs to implement the `fmt::Debug` trait. This is useful for -/// simple values that do not implement the `Serialize` trait, but does not +/// The value needs to implement the [`Debug`] trait. This is useful for +/// simple values that do not implement the [`serde::Serialize`] trait, but does not /// permit redactions. /// /// Debug is called with `"{:?}"`, which means this does not use pretty-print. @@ -349,7 +362,7 @@ macro_rules! _assert_snapshot_base { $name.into(), #[allow(clippy::redundant_closure_call)] &$transform(&$value), - env!("CARGO_MANIFEST_DIR"), + $crate::_get_workspace_root!().as_path(), $crate::_function_name!(), module_path!(), file!(), @@ -360,9 +373,9 @@ macro_rules! _assert_snapshot_base { }; } -/// Asserts a `Display` snapshot. +/// Asserts a [`Display`](std::fmt::Display) snapshot. /// -/// This is now deprecated, replaced by the more generic `assert_snapshot!()` +/// This is now deprecated, replaced by the more generic [`assert_snapshot!`](crate::assert_snapshot!) #[macro_export] #[deprecated = "use assert_snapshot!() instead"] macro_rules! assert_display_snapshot { @@ -371,10 +384,10 @@ macro_rules! assert_display_snapshot { }; } -/// Asserts a string snapshot. +/// Asserts a [`String`] snapshot. /// -/// This is the simplest of all assertion methods. It accepts any value that -/// implements `fmt::Display`. +/// This is the simplest of all assertion methods. +/// It accepts any value that implements [`Display`](std::fmt::Display). /// /// ```no_run /// # use insta::*; @@ -485,9 +498,13 @@ macro_rules! with_settings { #[cfg_attr(docsrs, doc(cfg(feature = "glob")))] #[macro_export] macro_rules! glob { + // TODO: I think we could remove the three-argument version of this macro + // and just support a pattern such as + // `glob!("../test_data/inputs/*.txt"...`. ($base_path:expr, $glob:expr, $closure:expr) => {{ use std::path::Path; - let base = $crate::_macro_support::get_cargo_workspace(env!("CARGO_MANIFEST_DIR")) + + let base = $crate::_get_workspace_root!() .join(Path::new(file!()).parent().unwrap()) .join($base_path) .to_path_buf(); @@ -495,7 +512,12 @@ macro_rules! glob { // we try to canonicalize but on some platforms (eg: wasm) that might not work, so // we instead silently fall back. let base = base.canonicalize().unwrap_or_else(|_| base); - $crate::_macro_support::glob_exec(env!("CARGO_MANIFEST_DIR"), &base, $glob, $closure); + $crate::_macro_support::glob_exec( + $crate::_get_workspace_root!().as_path(), + &base, + $glob, + $closure, + ); }}; ($glob:expr, $closure:expr) => {{ diff --git a/insta/src/output.rs b/insta/src/output.rs index 5c9d103b..089af78f 100644 --- a/insta/src/output.rs +++ b/insta/src/output.rs @@ -226,13 +226,8 @@ pub fn print_snapshot_summary( workspace_root: &Path, snapshot: &Snapshot, snapshot_file: Option<&Path>, - mut line: Option, + line: Option, ) { - // default to old assertion line from snapshot. - if line.is_none() { - line = snapshot.metadata().assertion_line(); - } - if let Some(snapshot_file) = snapshot_file { let snapshot_file = workspace_root .join(snapshot_file) @@ -255,11 +250,12 @@ pub fn print_snapshot_summary( println!( "Source: {}{}", style(value.display()).cyan(), - if let Some(line) = line { - format!(":{}", style(line).bold()) - } else { - "".to_string() - } + line.or( + // default to old assertion line from snapshot. + snapshot.metadata().assertion_line() + ) + .map(|line| format!(":{}", style(line).bold())) + .unwrap_or_default() ); } diff --git a/insta/src/redaction.rs b/insta/src/redaction.rs index 525b3294..423e7fec 100644 --- a/insta/src/redaction.rs +++ b/insta/src/redaction.rs @@ -136,7 +136,7 @@ where /// /// This is useful to force something like a set or map to be ordered to make /// it deterministic. This is necessary as insta's serialization support is -/// based on serde which does not have native set support. As a result vectors +/// based on [`serde`] which does not have native set support. As a result vectors /// (which need to retain order) and sets (which should be given a stable order) /// look the same. /// diff --git a/insta/src/runtime.rs b/insta/src/runtime.rs index 05476417..3679031f 100644 --- a/insta/src/runtime.rs +++ b/insta/src/runtime.rs @@ -8,14 +8,14 @@ use std::str; use std::sync::{Arc, Mutex}; use std::{borrow::Cow, env}; -use crate::output::SnapshotPrinter; use crate::settings::Settings; use crate::snapshot::{MetaData, PendingInlineSnapshot, Snapshot, SnapshotContents}; use crate::utils::{path_to_storage, style}; +use crate::{env::get_tool_config, output::SnapshotPrinter}; use crate::{ env::{ - get_cargo_workspace, get_tool_config, memoize_snapshot_file, snapshot_update_behavior, - OutputBehavior, SnapshotUpdateBehavior, ToolConfig, + memoize_snapshot_file, snapshot_update_behavior, OutputBehavior, SnapshotUpdateBehavior, + ToolConfig, }, snapshot::SnapshotKind, }; @@ -42,6 +42,17 @@ macro_rules! elog { writeln!(std::io::stderr(), $($arg)*).ok(); }) } +#[cfg(feature = "glob")] +macro_rules! print_or_panic { + ($fail_fast:expr, $($tokens:tt)*) => {{ + if (!$fail_fast) { + eprintln!($($tokens)*); + eprintln!(); + } else { + panic!($($tokens)*); + } + }} +} /// Special marker to use an automatic name. /// @@ -95,15 +106,14 @@ fn is_doctest(function_name: &str) -> bool { } fn detect_snapshot_name(function_name: &str, module_path: &str) -> Result { - let mut name = function_name; - // clean test name first - name = name.rsplit("::").next().unwrap(); - let mut test_prefixed = false; - if name.starts_with("test_") { - name = &name[5..]; - test_prefixed = true; - } + let name = function_name.rsplit("::").next().unwrap(); + + let (name, test_prefixed) = if let Some(stripped) = name.strip_prefix("test_") { + (stripped, true) + } else { + (name, false) + }; // next check if we need to add a suffix let name = add_suffix_to_snapshot_name(Cow::Borrowed(name)); @@ -164,11 +174,10 @@ fn get_snapshot_filename( assertion_file: &str, snapshot_name: &str, cargo_workspace: &Path, - base: &str, is_doctest: bool, ) -> PathBuf { let root = Path::new(cargo_workspace); - let base = Path::new(base); + let base = Path::new(assertion_file); Settings::with(|settings| { root.join(base.parent().unwrap()) .join(settings.snapshot_path()) @@ -180,8 +189,7 @@ fn get_snapshot_filename( write!( &mut f, "doctest_{}__", - Path::new(assertion_file) - .file_name() + base.file_name() .unwrap() .to_string_lossy() .replace('.', "_") @@ -202,12 +210,13 @@ fn get_snapshot_filename( }) } -/// A single snapshot including surrounding context which asserts and save the +/// The context around a snapshot, such as the reference value, location, etc. +/// (but not including the generated value). Responsible for saving the /// snapshot. #[derive(Debug)] struct SnapshotAssertionContext<'a> { tool_config: Arc, - cargo_workspace: Arc, + workspace: &'a Path, module_path: &'a str, snapshot_name: Option>, snapshot_file: Option, @@ -222,14 +231,13 @@ struct SnapshotAssertionContext<'a> { impl<'a> SnapshotAssertionContext<'a> { fn prepare( refval: ReferenceValue<'a>, - manifest_dir: &'a str, + workspace: &'a Path, function_name: &'a str, module_path: &'a str, assertion_file: &'a str, assertion_line: u32, ) -> Result, Box> { - let tool_config = get_tool_config(manifest_dir); - let cargo_workspace = get_cargo_workspace(manifest_dir); + let tool_config = get_tool_config(workspace); let snapshot_name; let mut duplication_key = None; let mut snapshot_file = None; @@ -257,8 +265,7 @@ impl<'a> SnapshotAssertionContext<'a> { module_path, assertion_file, &name, - &cargo_workspace, - assertion_file, + workspace, is_doctest, ); if fs::metadata(&file).is_ok() { @@ -279,7 +286,7 @@ impl<'a> SnapshotAssertionContext<'a> { snapshot_name = detect_snapshot_name(function_name, module_path) .ok() .map(Cow::Owned); - let mut pending_file = cargo_workspace.join(assertion_file); + let mut pending_file = workspace.join(assertion_file); pending_file.set_file_name(format!( ".{}.pending-snap", pending_file @@ -300,7 +307,7 @@ impl<'a> SnapshotAssertionContext<'a> { Ok(SnapshotAssertionContext { tool_config, - cargo_workspace, + workspace, module_path, snapshot_name, snapshot_file, @@ -315,8 +322,8 @@ impl<'a> SnapshotAssertionContext<'a> { /// Given a path returns the local path within the workspace. pub fn localize_path(&self, p: &Path) -> Option { - let workspace = self.cargo_workspace.canonicalize().ok()?; - let p = self.cargo_workspace.join(p).canonicalize().ok()?; + let workspace = self.workspace.canonicalize().ok()?; + let p = self.workspace.join(p).canonicalize().ok()?; p.strip_prefix(&workspace).ok().map(|x| x.to_path_buf()) } @@ -440,127 +447,113 @@ impl<'a> SnapshotAssertionContext<'a> { Ok(snapshot_update) } -} - -fn prevent_inline_duplicate(function_name: &str, assertion_file: &str, assertion_line: u32) { - let key = format!("{}|{}|{}", function_name, assertion_file, assertion_line); - let mut set = INLINE_DUPLICATES.lock().unwrap(); - if set.contains(&key) { - // drop the lock so we don't poison it - drop(set); - panic!( - "Insta does not allow inline snapshot assertions in loops. \ - Wrap your assertions in allow_duplicates! to change this." - ); - } - set.insert(key); -} -/// This prints the information about the snapshot -fn print_snapshot_info(ctx: &SnapshotAssertionContext, new_snapshot: &Snapshot) { - let mut printer = SnapshotPrinter::new( - ctx.cargo_workspace.as_path(), - ctx.old_snapshot.as_ref(), - new_snapshot, - ); - printer.set_line(Some(ctx.assertion_line)); - printer.set_snapshot_file(ctx.snapshot_file.as_deref()); - printer.set_title(Some("Snapshot Summary")); - printer.set_show_info(true); - match ctx.tool_config.output_behavior() { - OutputBehavior::Summary => { - printer.print(); - } - OutputBehavior::Diff => { - printer.set_show_diff(true); - printer.print(); + /// This prints the information about the snapshot + fn print_snapshot_info(&self, new_snapshot: &Snapshot) { + let mut printer = + SnapshotPrinter::new(self.workspace, self.old_snapshot.as_ref(), new_snapshot); + printer.set_line(Some(self.assertion_line)); + printer.set_snapshot_file(self.snapshot_file.as_deref()); + printer.set_title(Some("Snapshot Summary")); + printer.set_show_info(true); + match self.tool_config.output_behavior() { + OutputBehavior::Summary => { + printer.print(); + } + OutputBehavior::Diff => { + printer.set_show_diff(true); + printer.print(); + } + _ => {} } - _ => {} } -} - -#[cfg(feature = "glob")] -macro_rules! print_or_panic { - ($fail_fast:expr, $($tokens:tt)*) => {{ - if (!$fail_fast) { - eprintln!($($tokens)*); - eprintln!(); - } else { - panic!($($tokens)*); - } - }} -} -/// Finalizes the assertion based on the update result. -fn finalize_assertion(ctx: &SnapshotAssertionContext, update_result: SnapshotUpdateBehavior) { - // if we are in glob mode, we want to adjust the finalization - // so that we do not show the hints immediately. - let fail_fast = { - #[cfg(feature = "glob")] - { - if let Some(top) = crate::glob::GLOB_STACK.lock().unwrap().last() { - top.fail_fast - } else { + /// Finalizes the assertion when the snapshot comparison fails, potentially + /// panicking to fail the test + fn finalize(&self, update_result: SnapshotUpdateBehavior) { + // if we are in glob mode, we want to adjust the finalization + // so that we do not show the hints immediately. + let fail_fast = { + #[cfg(feature = "glob")] + { + if let Some(top) = crate::glob::GLOB_STACK.lock().unwrap().last() { + top.fail_fast + } else { + true + } + } + #[cfg(not(feature = "glob"))] + { true } - } - #[cfg(not(feature = "glob"))] + }; + + if fail_fast + && update_result == SnapshotUpdateBehavior::NewFile + && self.tool_config.output_behavior() != OutputBehavior::Nothing + && !self.is_doctest { - true + println!( + "{hint}", + hint = style("To update snapshots run `cargo insta review`").dim(), + ); } - }; - if fail_fast - && update_result == SnapshotUpdateBehavior::NewFile - && ctx.tool_config.output_behavior() != OutputBehavior::Nothing - && !ctx.is_doctest - { - println!( - "{hint}", - hint = style("To update snapshots run `cargo insta review`").dim(), - ); - } + if update_result != SnapshotUpdateBehavior::InPlace && !self.tool_config.force_pass() { + if fail_fast && self.tool_config.output_behavior() != OutputBehavior::Nothing { + let msg = if env::var("INSTA_CARGO_INSTA") == Ok("1".to_string()) { + "Stopped on the first failure." + } else { + "Stopped on the first failure. Run `cargo insta test` to run all snapshots." + }; + println!("{hint}", hint = style(msg).dim(),); + } - if update_result != SnapshotUpdateBehavior::InPlace && !ctx.tool_config.force_pass() { - if fail_fast && ctx.tool_config.output_behavior() != OutputBehavior::Nothing { - let msg = if env::var("INSTA_CARGO_INSTA") == Ok("1".to_string()) { - "Stopped on the first failure." - } else { - "Stopped on the first failure. Run `cargo insta test` to run all snapshots." - }; - println!("{hint}", hint = style(msg).dim(),); - } + // if we are in glob mode, count the failures and print the + // errors instead of panicking. The glob will then panic at + // the end. + #[cfg(feature = "glob")] + { + let mut stack = crate::glob::GLOB_STACK.lock().unwrap(); + if let Some(glob_collector) = stack.last_mut() { + glob_collector.failed += 1; + if update_result == SnapshotUpdateBehavior::NewFile + && self.tool_config.output_behavior() != OutputBehavior::Nothing + { + glob_collector.show_insta_hint = true; + } - // if we are in glob mode, count the failures and print the - // errors instead of panicking. The glob will then panic at - // the end. - #[cfg(feature = "glob")] - { - let mut stack = crate::glob::GLOB_STACK.lock().unwrap(); - if let Some(glob_collector) = stack.last_mut() { - glob_collector.failed += 1; - if update_result == SnapshotUpdateBehavior::NewFile - && ctx.tool_config.output_behavior() != OutputBehavior::Nothing - { - glob_collector.show_insta_hint = true; + print_or_panic!( + fail_fast, + "snapshot assertion from glob for '{}' failed in line {}", + self.snapshot_name.as_deref().unwrap_or("unnamed snapshot"), + self.assertion_line + ); + return; } - - print_or_panic!( - fail_fast, - "snapshot assertion from glob for '{}' failed in line {}", - ctx.snapshot_name.as_deref().unwrap_or("unnamed snapshot"), - ctx.assertion_line - ); - return; } + + panic!( + "snapshot assertion for '{}' failed in line {}", + self.snapshot_name.as_deref().unwrap_or("unnamed snapshot"), + self.assertion_line + ); } + } +} +fn prevent_inline_duplicate(function_name: &str, assertion_file: &str, assertion_line: u32) { + let key = format!("{}|{}|{}", function_name, assertion_file, assertion_line); + let mut set = INLINE_DUPLICATES.lock().unwrap(); + if set.contains(&key) { + // drop the lock so we don't poison it + drop(set); panic!( - "snapshot assertion for '{}' failed in line {}", - ctx.snapshot_name.as_deref().unwrap_or("unnamed snapshot"), - ctx.assertion_line + "Insta does not allow inline snapshot assertions in loops. \ + Wrap your assertions in allow_duplicates! to change this." ); } + set.insert(key); } fn record_snapshot_duplicate( @@ -572,8 +565,7 @@ fn record_snapshot_duplicate( if let Some(prev_snapshot) = results.get(key) { if prev_snapshot.contents() != snapshot.contents() { println!("Snapshots in allow-duplicates block do not match."); - let mut printer = - SnapshotPrinter::new(ctx.cargo_workspace.as_path(), Some(prev_snapshot), snapshot); + let mut printer = SnapshotPrinter::new(ctx.workspace, Some(prev_snapshot), snapshot); printer.set_line(Some(ctx.assertion_line)); printer.set_snapshot_file(ctx.snapshot_file.as_deref()); printer.set_title(Some("Differences in Block")); @@ -621,9 +613,9 @@ where /// assertion with a panic if needed. #[allow(clippy::too_many_arguments)] pub fn assert_snapshot( - refval: ReferenceValue<'_>, + refval: ReferenceValue, new_snapshot_value: &str, - manifest_dir: &str, + workspace: &Path, function_name: &str, module_path: &str, assertion_file: &str, @@ -632,15 +624,13 @@ pub fn assert_snapshot( ) -> Result<(), Box> { let ctx = SnapshotAssertionContext::prepare( refval, - manifest_dir, + workspace, function_name, module_path, assertion_file, assertion_line, )?; - let tool_config = get_tool_config(manifest_dir); - // apply filters if they are available #[cfg(feature = "filters")] let new_snapshot_value = @@ -653,7 +643,7 @@ pub fn assert_snapshot( let new_snapshot = ctx.new_snapshot(SnapshotContents::new(new_snapshot_value.into(), kind), expr); - // memoize the snapshot file if requested. + // memoize the snapshot file if requested, as part of potentially removing unreferenced snapshots if let Some(ref snapshot_file) = ctx.snapshot_file { memoize_snapshot_file(snapshot_file); } @@ -671,7 +661,7 @@ pub fn assert_snapshot( .old_snapshot .as_ref() .map(|x| { - if tool_config.require_full_match() { + if ctx.tool_config.require_full_match() { x.matches_fully(&new_snapshot) } else { x.matches(&new_snapshot) @@ -683,7 +673,7 @@ pub fn assert_snapshot( ctx.cleanup_passing()?; if matches!( - tool_config.snapshot_update(), + ctx.tool_config.snapshot_update(), crate::env::SnapshotUpdate::Force ) { // Avoid creating new files if contents match exactly. In @@ -700,14 +690,15 @@ pub fn assert_snapshot( } // otherwise print information and update snapshots. } else { - print_snapshot_info(&ctx, &new_snapshot); + ctx.print_snapshot_info(&new_snapshot); let update_result = ctx.update_snapshot(new_snapshot)?; - finalize_assertion(&ctx, update_result); + ctx.finalize(update_result); } Ok(()) } +#[allow(rustdoc::private_doc_tests)] /// Test snapshots in doctests. /// /// ``` diff --git a/insta/src/settings.rs b/insta/src/settings.rs index 7128cfe3..1a1f4425 100644 --- a/insta/src/settings.rs +++ b/insta/src/settings.rs @@ -176,7 +176,7 @@ impl Default for Settings { impl Settings { /// Returns the default settings. /// - /// It's recommended to use `clone_current` instead so that + /// It's recommended to use [`Self::clone_current`] instead so that /// already applied modifications are not discarded. pub fn new() -> Settings { Settings::default() @@ -196,7 +196,7 @@ impl Settings { /// Enables forceful sorting of maps before serialization. /// /// Note that this only applies to snapshots that undergo serialization - /// (eg: does not work for `assert_debug_snapshot!`.) + /// (eg: does not work for [`assert_debug_snapshot!`](crate::assert_debug_snapshot!).) /// /// The default value is `false`. pub fn set_sort_maps(&mut self, value: bool) { @@ -210,7 +210,7 @@ impl Settings { /// Disables prepending of modules to the snapshot filename. /// - /// By default the filename of a snapshot is `__.snap`. + /// By default, the filename of a snapshot is `__.snap`. /// Setting this flag to `false` changes the snapshot filename to just /// `.snap`. /// @@ -226,7 +226,7 @@ impl Settings { /// Allows the [`glob!`] macro to succeed if it matches no files. /// - /// By default the glob macro will fail the test if it does not find + /// By default, the glob macro will fail the test if it does not find /// any files to prevent accidental typos. This can be disabled when /// fixtures should be conditional. /// @@ -269,10 +269,10 @@ impl Settings { /// Sets the input file reference. /// - /// This value is completely unused by the snapshot testing system but - /// it lets you store some meta data with a snapshot that refers you back - /// to the input file. The path stored here is made relative to the - /// workspace root before storing with the snapshot. + /// This value is completely unused by the snapshot testing system but it + /// allows storing some metadata with a snapshot that refers back to the + /// input file. The path stored here is made relative to the workspace root + /// before storing with the snapshot. pub fn set_input_file>(&mut self, p: P) { self._private_inner_mut().input_file(p); } @@ -295,7 +295,7 @@ impl Settings { /// super useful by itself, particularly when working with loops and generated /// tests. In that case the `description` can be set as extra information. /// - /// See also [`set_info`](Self::set_info). + /// See also [`Self::set_info`]. pub fn set_description>(&mut self, value: S) { self._private_inner_mut().description(value); } @@ -320,7 +320,7 @@ impl Settings { /// As an example the input parameters to the function that creates the snapshot /// can be persisted here. /// - /// Alternatively you can use [`set_raw_info`](Self::set_raw_info) instead. + /// Alternatively you can use [`Self::set_raw_info`] instead. #[cfg(feature = "serde")] #[cfg_attr(docsrs, doc(cfg(feature = "serde")))] pub fn set_info(&mut self, s: &S) { @@ -329,7 +329,7 @@ impl Settings { /// Sets the info from a content object. /// - /// This works like [`set_info`](Self::set_info) but does not require `serde`. + /// This works like [`Self::set_info`] but does not require [`serde`]. pub fn set_raw_info(&mut self, content: &Content) { self._private_inner_mut().raw_info(content); } @@ -365,7 +365,7 @@ impl Settings { /// snapshots. /// /// Note that this only applies to snapshots that undergo serialization - /// (eg: does not work for `assert_debug_snapshot!`.) + /// (eg: does not work for [`assert_debug_snapshot!`](crate::assert_debug_snapshot!).) #[cfg(feature = "redactions")] #[cfg_attr(docsrs, doc(cfg(feature = "redactions")))] pub fn add_redaction>(&mut self, selector: &str, replacement: R) { @@ -384,7 +384,7 @@ impl Settings { /// /// This works similar to a redaction but instead of changing the value it /// asserts the value at a certain place. This function is internally - /// supposed to call things like `assert_eq!`. + /// supposed to call things like [`assert_eq!`]. /// /// This is a shortcut to `add_redaction(selector, dynamic_redaction(...))`; #[cfg(feature = "redactions")] @@ -437,7 +437,7 @@ impl Settings { /// /// The first argument is the [`regex`] pattern to apply, the second is a replacement /// string. The replacement string has the same functionality as the second argument - /// to [`Regex::replace`](regex::Regex::replace). + /// to [`regex::Regex::replace`]. /// /// This is useful to perform some cleanup procedures on the snapshot for unstable values. /// @@ -493,7 +493,7 @@ impl Settings { /// Runs a function with the current settings bound to the thread. /// - /// This is an alternative to [`bind_to_scope`](Settings::bind_to_scope) + /// This is an alternative to [`Self::bind_to_scope`]() /// which does not require holding on to a drop guard. The return value /// of the closure is passed through. /// @@ -510,7 +510,7 @@ impl Settings { f() } - /// Like `bind` but for futures. + /// Like [`Self::bind`] but for futures. /// /// This lets you bind settings for the duration of a future like this: /// @@ -579,7 +579,7 @@ impl Settings { } } -/// Returned from [`bind_to_scope`](Settings::bind_to_scope) +/// Returned from [`Settings::bind_to_scope`] #[must_use = "The guard is immediately dropped so binding has no effect. Use `let _guard = ...` to bind it."] pub struct SettingsBindDropGuard(Option>); diff --git a/insta/src/snapshot.rs b/insta/src/snapshot.rs index bf9b67e6..fe47d01d 100644 --- a/insta/src/snapshot.rs +++ b/insta/src/snapshot.rs @@ -418,6 +418,8 @@ impl Snapshot { fn as_content(&self) -> Content { let mut fields = vec![("module_name", Content::from(self.module_name.as_str()))]; + // Note this is currently never used, since this method is only used for + // inline snapshots if let Some(name) = self.snapshot_name.as_deref() { fields.push(("snapshot_name", Content::from(name))); } @@ -489,7 +491,8 @@ impl Snapshot { } let serialized_snapshot = self.serialize_snapshot(md); - fs::write(path, serialized_snapshot)?; + fs::write(path, serialized_snapshot) + .map_err(|e| content::Error::FileIo(e, path.to_path_buf()))?; Ok(()) } @@ -502,7 +505,7 @@ impl Snapshot { self.save_with_metadata(path, &self.metadata.trim_for_persistence()) } - /// Same as `save` but instead of writing a normal snapshot file this will write + /// Same as [`Self::save`] but instead of writing a normal snapshot file this will write /// a `.snap.new` file with additional information. /// /// The name of the new snapshot file is returned. @@ -1155,3 +1158,31 @@ fn test_ownership() { assert_debug_snapshot!(r, @"0..10"); assert_debug_snapshot!(r, @"0..10"); } + +#[test] +fn test_empty_lines() { + assert_snapshot!(r#"single line should fit on a single line"#, @"single line should fit on a single line"); + assert_snapshot!(r#"single line should fit on a single line, even if it's really really really really really really really really really long"#, @"single line should fit on a single line, even if it's really really really really really really really really really long"); + + assert_snapshot!(r#"multiline content starting on first line + + final line + "#, @r###" + multiline content starting on first line + + final line + + "###); + + assert_snapshot!(r#" + multiline content starting on second line + + final line + "#, @r###" + + multiline content starting on second line + + final line + + "###); +} diff --git a/insta/tests/snapshots/test_advanced__basic_suffixes@1.snap b/insta/tests/snapshots/test_advanced__basic_suffixes@1.snap new file mode 100644 index 00000000..1f01b3f4 --- /dev/null +++ b/insta/tests/snapshots/test_advanced__basic_suffixes@1.snap @@ -0,0 +1,5 @@ +--- +source: insta/tests/test_advanced.rs +expression: "&value" +--- +1 diff --git a/insta/tests/snapshots/test_advanced__basic_suffixes@2.snap b/insta/tests/snapshots/test_advanced__basic_suffixes@2.snap new file mode 100644 index 00000000..9f7dce28 --- /dev/null +++ b/insta/tests/snapshots/test_advanced__basic_suffixes@2.snap @@ -0,0 +1,5 @@ +--- +source: insta/tests/test_advanced.rs +expression: "&value" +--- +2 diff --git a/insta/tests/snapshots/test_advanced__basic_suffixes@3.snap b/insta/tests/snapshots/test_advanced__basic_suffixes@3.snap new file mode 100644 index 00000000..1cab7cd9 --- /dev/null +++ b/insta/tests/snapshots/test_advanced__basic_suffixes@3.snap @@ -0,0 +1,5 @@ +--- +source: insta/tests/test_advanced.rs +expression: "&value" +--- +3 diff --git a/insta/tests/snapshots/test_bugs__crlf.snap b/insta/tests/snapshots/test_basic__crlf.snap similarity index 63% rename from insta/tests/snapshots/test_bugs__crlf.snap rename to insta/tests/snapshots/test_basic__crlf.snap index e3c3075f..bd6aba5c 100644 --- a/insta/tests/snapshots/test_bugs__crlf.snap +++ b/insta/tests/snapshots/test_basic__crlf.snap @@ -1,5 +1,5 @@ --- -source: insta/tests/test_bugs.rs +source: insta/tests/test_basic.rs expression: "\"foo\\r\\nbar\\r\\nbaz\"" --- foo diff --git a/insta/tests/snapshots/test_bugs__trailing_crlf.snap b/insta/tests/snapshots/test_basic__trailing_crlf.snap similarity index 66% rename from insta/tests/snapshots/test_bugs__trailing_crlf.snap rename to insta/tests/snapshots/test_basic__trailing_crlf.snap index e61fca0b..cd7d27c4 100644 --- a/insta/tests/snapshots/test_bugs__trailing_crlf.snap +++ b/insta/tests/snapshots/test_basic__trailing_crlf.snap @@ -1,5 +1,5 @@ --- -source: insta/tests/test_bugs.rs +source: insta/tests/test_basic.rs expression: "\"foo\\r\\nbar\\r\\nbaz\\r\\n\"" --- foo diff --git a/insta/tests/snapshots/test_suffixes__basic_suffixes@1.snap b/insta/tests/snapshots/test_suffixes__basic_suffixes@1.snap deleted file mode 100644 index 0ebfdecc..00000000 --- a/insta/tests/snapshots/test_suffixes__basic_suffixes@1.snap +++ /dev/null @@ -1,5 +0,0 @@ ---- -source: insta/tests/test_suffixes.rs -expression: "&value" ---- -1 diff --git a/insta/tests/snapshots/test_suffixes__basic_suffixes@2.snap b/insta/tests/snapshots/test_suffixes__basic_suffixes@2.snap deleted file mode 100644 index 11ca6d60..00000000 --- a/insta/tests/snapshots/test_suffixes__basic_suffixes@2.snap +++ /dev/null @@ -1,5 +0,0 @@ ---- -source: insta/tests/test_suffixes.rs -expression: "&value" ---- -2 diff --git a/insta/tests/snapshots/test_suffixes__basic_suffixes@3.snap b/insta/tests/snapshots/test_suffixes__basic_suffixes@3.snap deleted file mode 100644 index 94346c75..00000000 --- a/insta/tests/snapshots/test_suffixes__basic_suffixes@3.snap +++ /dev/null @@ -1,5 +0,0 @@ ---- -source: insta/tests/test_suffixes.rs -expression: "&value" ---- -3 diff --git a/insta/tests/test_allow_duplicates.rs b/insta/tests/test_advanced.rs similarity index 53% rename from insta/tests/test_allow_duplicates.rs rename to insta/tests/test_advanced.rs index 3a862de6..f8b46645 100644 --- a/insta/tests/test_allow_duplicates.rs +++ b/insta/tests/test_advanced.rs @@ -1,5 +1,26 @@ use insta::{allow_duplicates, assert_debug_snapshot}; +#[cfg(feature = "filters")] +#[test] +fn test_basic_filter() { + use insta::{assert_snapshot, with_settings}; + with_settings!({filters => vec![ + (r"\b[[:xdigit:]]{8}\b", "[SHORT_HEX]") + ]}, { + assert_snapshot!("Hello DEADBEEF!", @"Hello [SHORT_HEX]!"); + }) +} + +#[cfg(feature = "json")] +#[test] +fn test_basic_suffixes() { + for value in [1, 2, 3] { + insta::with_settings!({snapshot_suffix => value.to_string()}, { + insta::assert_json_snapshot!(&value); + }); + } +} + #[test] fn test_basic_duplicates_passes() { allow_duplicates! { diff --git a/insta/tests/test_basic.rs b/insta/tests/test_basic.rs index 010612f9..33b67611 100644 --- a/insta/tests/test_basic.rs +++ b/insta/tests/test_basic.rs @@ -116,3 +116,22 @@ fn insta_sort_order() { insta::assert_yaml_snapshot!(m); }); } + +#[test] +fn test_crlf() { + insta::assert_snapshot!("foo\r\nbar\r\nbaz"); +} + +#[test] +fn test_trailing_crlf() { + insta::assert_snapshot!("foo\r\nbar\r\nbaz\r\n"); +} + +#[test] +fn test_trailing_crlf_inline() { + insta::assert_snapshot!("foo\r\nbar\r\nbaz\r\n", @r#" + foo + bar + baz + "#); +} diff --git a/insta/tests/test_bugs.rs b/insta/tests/test_bugs.rs deleted file mode 100644 index b0700fe7..00000000 --- a/insta/tests/test_bugs.rs +++ /dev/null @@ -1,18 +0,0 @@ -#[test] -fn test_crlf() { - insta::assert_snapshot!("foo\r\nbar\r\nbaz"); -} - -#[test] -fn test_trailing_crlf() { - insta::assert_snapshot!("foo\r\nbar\r\nbaz\r\n"); -} - -#[test] -fn test_trailing_crlf_inline() { - insta::assert_snapshot!("foo\r\nbar\r\nbaz\r\n", @r###" - foo - bar - baz - "###); -} diff --git a/insta/tests/test_filters.rs b/insta/tests/test_filters.rs deleted file mode 100644 index 148e2961..00000000 --- a/insta/tests/test_filters.rs +++ /dev/null @@ -1,12 +0,0 @@ -#![cfg(feature = "filters")] - -use insta::{assert_snapshot, with_settings}; - -#[test] -fn test_basic_filter() { - with_settings!({filters => vec![ - (r"\b[[:xdigit:]]{8}\b", "[SHORT_HEX]") - ]}, { - assert_snapshot!("Hello DEADBEEF!", @"Hello [SHORT_HEX]!"); - }) -} diff --git a/insta/tests/test_inline.rs b/insta/tests/test_inline.rs index 93f70126..fe936908 100644 --- a/insta/tests/test_inline.rs +++ b/insta/tests/test_inline.rs @@ -61,6 +61,11 @@ fn test_newline() { "###); } +#[test] +fn test_inline_debug_expr() { + assert_snapshot!("hello", "a debug expr", @"hello"); +} + #[cfg(feature = "csv")] #[test] fn test_csv_inline() { @@ -304,3 +309,14 @@ fn test_inline_snapshot_whitespace() { "###); } + +#[test] +fn test_indentation() { + assert_snapshot!("aaa\nbbb\nccc\nddd", @r" + aaa + bbb + ccc + ddd + " + ); +} diff --git a/insta/tests/test_suffixes.rs b/insta/tests/test_suffixes.rs deleted file mode 100644 index af0eb916..00000000 --- a/insta/tests/test_suffixes.rs +++ /dev/null @@ -1,9 +0,0 @@ -#[cfg(feature = "json")] -#[test] -fn test_basic_suffixes() { - for value in [1, 2, 3] { - insta::with_settings!({snapshot_suffix => value.to_string()}, { - insta::assert_json_snapshot!(&value); - }); - } -}