From 70776758c0ba1f739c1e46d85a987f2e39950f0f Mon Sep 17 00:00:00 2001 From: Ed Page Date: Mon, 20 May 2024 16:38:45 -0500 Subject: [PATCH 01/19] refactor(filter): Pull out pattern logic --- crates/snapbox/src/filter/mod.rs | 2 + crates/snapbox/src/filter/pattern.rs | 332 ++++++++++++++++++++++++ crates/snapbox/src/filter/redactions.rs | 328 +---------------------- 3 files changed, 336 insertions(+), 326 deletions(-) create mode 100644 crates/snapbox/src/filter/pattern.rs diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index 570a47e5..88c92df3 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -4,6 +4,7 @@ //! - Making snapshots consistent across platforms or conditional compilation //! - Focusing snapshots on the characteristics of the data being tested +mod pattern; mod redactions; #[cfg(test)] mod test; @@ -11,6 +12,7 @@ mod test; use crate::data::DataInner; use crate::Data; +pub(crate) use pattern::normalize_to_pattern; pub use redactions::RedactedValue; pub use redactions::Redactions; diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs new file mode 100644 index 00000000..6b6a6bf2 --- /dev/null +++ b/crates/snapbox/src/filter/pattern.rs @@ -0,0 +1,332 @@ +use super::Redactions; + +pub(crate) fn normalize_to_pattern(input: &str, pattern: &str, redactions: &Redactions) -> String { + if input == pattern { + return input.to_owned(); + } + + let input = redactions.redact(input); + + let mut normalized: Vec<&str> = Vec::new(); + let mut input_index = 0; + let input_lines: Vec<_> = crate::utils::LinesWithTerminator::new(&input).collect(); + let mut pattern_lines = crate::utils::LinesWithTerminator::new(pattern).peekable(); + 'outer: while let Some(pattern_line) = pattern_lines.next() { + if is_line_elide(pattern_line) { + if let Some(next_pattern_line) = pattern_lines.peek() { + for (index_offset, next_input_line) in + input_lines[input_index..].iter().copied().enumerate() + { + if line_matches(next_input_line, next_pattern_line, redactions) { + normalized.push(pattern_line); + input_index += index_offset; + continue 'outer; + } + } + // Give up doing further normalization + break; + } else { + // Give up doing further normalization + normalized.push(pattern_line); + // captured rest so don't copy remaining lines over + input_index = input_lines.len(); + break; + } + } else { + let Some(input_line) = input_lines.get(input_index) else { + // Give up doing further normalization + break; + }; + + if line_matches(input_line, pattern_line, redactions) { + input_index += 1; + normalized.push(pattern_line); + } else { + // Give up doing further normalization + break; + } + } + } + + normalized.extend(input_lines[input_index..].iter().copied()); + normalized.join("") +} + +fn is_line_elide(line: &str) -> bool { + line == "...\n" || line == "..." +} + +fn line_matches(mut input: &str, pattern: &str, redactions: &Redactions) -> bool { + if input == pattern { + return true; + } + + let pattern = redactions.clear(pattern); + let mut sections = pattern.split("[..]").peekable(); + while let Some(section) = sections.next() { + if let Some(remainder) = input.strip_prefix(section) { + if let Some(next_section) = sections.peek() { + if next_section.is_empty() { + input = ""; + } else if let Some(restart_index) = remainder.find(next_section) { + input = &remainder[restart_index..]; + } + } else { + return remainder.is_empty(); + } + } else { + return false; + } + } + + false +} + +#[cfg(test)] +mod test { + use std::path::PathBuf; + + use super::*; + + #[test] + fn empty() { + let input = ""; + let pattern = ""; + let expected = ""; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn literals_match() { + let input = "Hello\nWorld"; + let pattern = "Hello\nWorld"; + let expected = "Hello\nWorld"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn pattern_shorter() { + let input = "Hello\nWorld"; + let pattern = "Hello\n"; + let expected = "Hello\nWorld"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn input_shorter() { + let input = "Hello\n"; + let pattern = "Hello\nWorld"; + let expected = "Hello\n"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn all_different() { + let input = "Hello\nWorld"; + let pattern = "Goodbye\nMoon"; + let expected = "Hello\nWorld"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn middles_diverge() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\nMoon\nGoodbye"; + let expected = "Hello\nWorld\nGoodbye"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn elide_delimited_with_sub() { + let input = "Hello World\nHow are you?\nGoodbye World"; + let pattern = "Hello [..]\n...\nGoodbye [..]"; + let expected = "Hello [..]\n...\nGoodbye [..]"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn leading_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "...\nGoodbye"; + let expected = "...\nGoodbye"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn trailing_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n..."; + let expected = "Hello\n..."; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn middle_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n...\nGoodbye"; + let expected = "Hello\n...\nGoodbye"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn post_elide_diverge() { + let input = "Hello\nSun\nAnd\nWorld"; + let pattern = "Hello\n...\nMoon"; + let expected = "Hello\nSun\nAnd\nWorld"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn post_diverge_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nMoon\nGoodbye\n..."; + let expected = "Hello\nWorld\nGoodbye\nSir"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn inline_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nW[..]d\nGoodbye\nSir"; + let expected = "Hello\nW[..]d\nGoodbye\nSir"; + let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + assert_eq!(expected, actual); + } + + #[test] + fn line_matches_cases() { + let cases = [ + ("", "", true), + ("", "[..]", true), + ("hello", "hello", true), + ("hello", "goodbye", false), + ("hello", "[..]", true), + ("hello", "he[..]", true), + ("hello", "go[..]", false), + ("hello", "[..]o", true), + ("hello", "[..]e", false), + ("hello", "he[..]o", true), + ("hello", "he[..]e", false), + ("hello", "go[..]o", false), + ("hello", "go[..]e", false), + ( + "hello world, goodbye moon", + "hello [..], goodbye [..]", + true, + ), + ( + "hello world, goodbye moon", + "goodbye [..], goodbye [..]", + false, + ), + ( + "hello world, goodbye moon", + "goodbye [..], hello [..]", + false, + ), + ("hello world, goodbye moon", "hello [..], [..] moon", true), + ( + "hello world, goodbye moon", + "goodbye [..], [..] moon", + false, + ), + ("hello world, goodbye moon", "hello [..], [..] world", false), + ]; + for (line, pattern, expected) in cases { + let actual = line_matches(line, pattern, &Redactions::new()); + assert_eq!(expected, actual, "line={:?} pattern={:?}", line, pattern); + } + } + + #[test] + fn substitute_literal() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", "world").unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } + + #[test] + fn substitute_path() { + let input = "input: /home/epage"; + let pattern = "input: [HOME]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(sep).join("home").join("epage"); + sub.insert("[HOME]", redacted).unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } + + #[test] + fn substitute_overlapping_path() { + let input = "\ +a: /home/epage +b: /home/epage/snapbox"; + let pattern = "\ +a: [A] +b: [B]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(&sep).join("home").join("epage"); + sub.insert("[A]", redacted).unwrap(); + let redacted = PathBuf::from(sep) + .join("home") + .join("epage") + .join("snapbox"); + sub.insert("[B]", redacted).unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } + + #[test] + fn substitute_disabled() { + let input = "cargo"; + let pattern = "cargo[EXE]"; + let mut sub = Redactions::new(); + sub.insert("[EXE]", "").unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } + + #[test] + #[cfg(feature = "regex")] + fn substitute_regex_unnamed() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) + .unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } + + #[test] + #[cfg(feature = "regex")] + fn substitute_regex_named() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert( + "[OBJECT]", + regex::Regex::new("(?world)!").unwrap(), + ) + .unwrap(); + let actual = normalize_to_pattern(input, pattern, &sub); + assert_eq!(actual, pattern); + } +} diff --git a/crates/snapbox/src/filter/redactions.rs b/crates/snapbox/src/filter/redactions.rs index 8914900f..ac74cd84 100644 --- a/crates/snapbox/src/filter/redactions.rs +++ b/crates/snapbox/src/filter/redactions.rs @@ -94,7 +94,7 @@ impl Redactions { /// assert_eq!(output, "Hello [..]!"); /// ``` pub fn normalize(&self, input: &str, pattern: &str) -> String { - normalize(input, pattern, self) + super::normalize_to_pattern(input, pattern, self) } /// Apply redaction only, no pattern-dependent globs @@ -111,7 +111,7 @@ impl Redactions { input } - fn clear<'v>(&self, pattern: &'v str) -> Cow<'v, str> { + pub(crate) fn clear<'v>(&self, pattern: &'v str) -> Cow<'v, str> { if !self.unused.is_empty() && pattern.contains('[') { let mut pattern = pattern.to_owned(); replace_many(&mut pattern, self.unused.iter().map(|var| (var, ""))); @@ -315,254 +315,10 @@ fn validate_placeholder(placeholder: &'static str) -> crate::assert::Result<&'st Ok(placeholder) } -fn normalize(input: &str, pattern: &str, redactions: &Redactions) -> String { - if input == pattern { - return input.to_owned(); - } - - let input = redactions.redact(input); - - let mut normalized: Vec<&str> = Vec::new(); - let mut input_index = 0; - let input_lines: Vec<_> = crate::utils::LinesWithTerminator::new(&input).collect(); - let mut pattern_lines = crate::utils::LinesWithTerminator::new(pattern).peekable(); - 'outer: while let Some(pattern_line) = pattern_lines.next() { - if is_line_elide(pattern_line) { - if let Some(next_pattern_line) = pattern_lines.peek() { - for (index_offset, next_input_line) in - input_lines[input_index..].iter().copied().enumerate() - { - if line_matches(next_input_line, next_pattern_line, redactions) { - normalized.push(pattern_line); - input_index += index_offset; - continue 'outer; - } - } - // Give up doing further normalization - break; - } else { - // Give up doing further normalization - normalized.push(pattern_line); - // captured rest so don't copy remaining lines over - input_index = input_lines.len(); - break; - } - } else { - let Some(input_line) = input_lines.get(input_index) else { - // Give up doing further normalization - break; - }; - - if line_matches(input_line, pattern_line, redactions) { - input_index += 1; - normalized.push(pattern_line); - } else { - // Give up doing further normalization - break; - } - } - } - - normalized.extend(input_lines[input_index..].iter().copied()); - normalized.join("") -} - -fn is_line_elide(line: &str) -> bool { - line == "...\n" || line == "..." -} - -fn line_matches(mut input: &str, pattern: &str, redactions: &Redactions) -> bool { - if input == pattern { - return true; - } - - let pattern = redactions.clear(pattern); - let mut sections = pattern.split("[..]").peekable(); - while let Some(section) = sections.next() { - if let Some(remainder) = input.strip_prefix(section) { - if let Some(next_section) = sections.peek() { - if next_section.is_empty() { - input = ""; - } else if let Some(restart_index) = remainder.find(next_section) { - input = &remainder[restart_index..]; - } - } else { - return remainder.is_empty(); - } - } else { - return false; - } - } - - false -} - #[cfg(test)] mod test { use super::*; - #[test] - fn empty() { - let input = ""; - let pattern = ""; - let expected = ""; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn literals_match() { - let input = "Hello\nWorld"; - let pattern = "Hello\nWorld"; - let expected = "Hello\nWorld"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn pattern_shorter() { - let input = "Hello\nWorld"; - let pattern = "Hello\n"; - let expected = "Hello\nWorld"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn input_shorter() { - let input = "Hello\n"; - let pattern = "Hello\nWorld"; - let expected = "Hello\n"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn all_different() { - let input = "Hello\nWorld"; - let pattern = "Goodbye\nMoon"; - let expected = "Hello\nWorld"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn middles_diverge() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\nMoon\nGoodbye"; - let expected = "Hello\nWorld\nGoodbye"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn elide_delimited_with_sub() { - let input = "Hello World\nHow are you?\nGoodbye World"; - let pattern = "Hello [..]\n...\nGoodbye [..]"; - let expected = "Hello [..]\n...\nGoodbye [..]"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn leading_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "...\nGoodbye"; - let expected = "...\nGoodbye"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn trailing_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n..."; - let expected = "Hello\n..."; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn middle_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n...\nGoodbye"; - let expected = "Hello\n...\nGoodbye"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn post_elide_diverge() { - let input = "Hello\nSun\nAnd\nWorld"; - let pattern = "Hello\n...\nMoon"; - let expected = "Hello\nSun\nAnd\nWorld"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn post_diverge_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nMoon\nGoodbye\n..."; - let expected = "Hello\nWorld\nGoodbye\nSir"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn inline_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nW[..]d\nGoodbye\nSir"; - let expected = "Hello\nW[..]d\nGoodbye\nSir"; - let actual = normalize(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); - } - - #[test] - fn line_matches_cases() { - let cases = [ - ("", "", true), - ("", "[..]", true), - ("hello", "hello", true), - ("hello", "goodbye", false), - ("hello", "[..]", true), - ("hello", "he[..]", true), - ("hello", "go[..]", false), - ("hello", "[..]o", true), - ("hello", "[..]e", false), - ("hello", "he[..]o", true), - ("hello", "he[..]e", false), - ("hello", "go[..]o", false), - ("hello", "go[..]e", false), - ( - "hello world, goodbye moon", - "hello [..], goodbye [..]", - true, - ), - ( - "hello world, goodbye moon", - "goodbye [..], goodbye [..]", - false, - ), - ( - "hello world, goodbye moon", - "goodbye [..], hello [..]", - false, - ), - ("hello world, goodbye moon", "hello [..], [..] moon", true), - ( - "hello world, goodbye moon", - "goodbye [..], [..] moon", - false, - ), - ("hello world, goodbye moon", "hello [..], [..] world", false), - ]; - for (line, pattern, expected) in cases { - let actual = line_matches(line, pattern, &Redactions::new()); - assert_eq!(expected, actual, "line={:?} pattern={:?}", line, pattern); - } - } - #[test] fn test_validate_placeholder() { let cases = [ @@ -577,84 +333,4 @@ mod test { assert_eq!(expected, actual, "placeholder={:?}", placeholder); } } - - #[test] - fn substitute_literal() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", "world").unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } - - #[test] - fn substitute_path() { - let input = "input: /home/epage"; - let pattern = "input: [HOME]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(sep).join("home").join("epage"); - sub.insert("[HOME]", redacted).unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } - - #[test] - fn substitute_overlapping_path() { - let input = "\ -a: /home/epage -b: /home/epage/snapbox"; - let pattern = "\ -a: [A] -b: [B]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(&sep).join("home").join("epage"); - sub.insert("[A]", redacted).unwrap(); - let redacted = PathBuf::from(sep) - .join("home") - .join("epage") - .join("snapbox"); - sub.insert("[B]", redacted).unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } - - #[test] - fn substitute_disabled() { - let input = "cargo"; - let pattern = "cargo[EXE]"; - let mut sub = Redactions::new(); - sub.insert("[EXE]", "").unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } - - #[test] - #[cfg(feature = "regex")] - fn substitute_regex_unnamed() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) - .unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } - - #[test] - #[cfg(feature = "regex")] - fn substitute_regex_named() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert( - "[OBJECT]", - regex::Regex::new("(?world)!").unwrap(), - ) - .unwrap(); - let actual = normalize(input, pattern, &sub); - assert_eq!(actual, pattern); - } } From a0d8c54eb38920f9be5891e05e09dfcf948e1022 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Tue, 21 May 2024 15:25:16 -0500 Subject: [PATCH 02/19] refactor(filter): Remove a layer of abstraction --- crates/snapbox/src/filter/mod.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index 88c92df3..f3b05e49 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -142,7 +142,7 @@ impl Filter for FilterRedactions<'_> { DataInner::Binary(bin) => DataInner::Binary(bin), DataInner::Text(text) => { if let Some(pattern) = self.pattern.render() { - let lines = self.substitutions.normalize(&text, &pattern); + let lines = normalize_to_pattern(&text, &pattern, self.substitutions); DataInner::Text(lines) } else { DataInner::Text(text) @@ -167,7 +167,7 @@ impl Filter for FilterRedactions<'_> { #[cfg(feature = "term-svg")] DataInner::TermSvg(text) => { if let Some(pattern) = self.pattern.render() { - let lines = self.substitutions.normalize(&text, &pattern); + let lines = normalize_to_pattern(&text, &pattern, self.substitutions); DataInner::TermSvg(lines) } else { DataInner::TermSvg(text) @@ -220,7 +220,7 @@ fn normalize_value_matches( *act = serde_json::json!(VALUE_WILDCARD); } (String(act), String(exp)) => { - *act = substitutions.normalize(act, exp); + *act = normalize_to_pattern(act, exp, substitutions); } (Array(act), Array(exp)) => { let mut sections = exp.split(|e| e == VALUE_WILDCARD).peekable(); From e10b8d6f70158c5359eb8e6a1c35b1de6e1f7a57 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Tue, 21 May 2024 15:26:51 -0500 Subject: [PATCH 03/19] refactor(filter): Centralize all pattern normalization --- crates/snapbox/src/filter/mod.rs | 136 +------------------------- crates/snapbox/src/filter/pattern.rs | 138 +++++++++++++++++++++++++++ 2 files changed, 139 insertions(+), 135 deletions(-) diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index f3b05e49..e62a1f07 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -13,6 +13,7 @@ use crate::data::DataInner; use crate::Data; pub(crate) use pattern::normalize_to_pattern; +pub use pattern::FilterRedactions; pub use redactions::RedactedValue; pub use redactions::Redactions; @@ -119,69 +120,6 @@ fn normalize_paths_chars(data: impl Iterator) -> impl Iterator { - substitutions: &'a crate::Redactions, - pattern: &'a Data, -} - -impl<'a> FilterRedactions<'a> { - pub fn new(substitutions: &'a crate::Redactions, pattern: &'a Data) -> Self { - FilterRedactions { - substitutions, - pattern, - } - } -} - -impl Filter for FilterRedactions<'_> { - fn filter(&self, data: Data) -> Data { - let source = data.source; - let filters = data.filters; - let inner = match data.inner { - DataInner::Error(err) => DataInner::Error(err), - DataInner::Binary(bin) => DataInner::Binary(bin), - DataInner::Text(text) => { - if let Some(pattern) = self.pattern.render() { - let lines = normalize_to_pattern(&text, &pattern, self.substitutions); - DataInner::Text(lines) - } else { - DataInner::Text(text) - } - } - #[cfg(feature = "json")] - DataInner::Json(value) => { - let mut value = value; - if let DataInner::Json(exp) = &self.pattern.inner { - normalize_value_matches(&mut value, exp, self.substitutions); - } - DataInner::Json(value) - } - #[cfg(feature = "json")] - DataInner::JsonLines(value) => { - let mut value = value; - if let DataInner::Json(exp) = &self.pattern.inner { - normalize_value_matches(&mut value, exp, self.substitutions); - } - DataInner::JsonLines(value) - } - #[cfg(feature = "term-svg")] - DataInner::TermSvg(text) => { - if let Some(pattern) = self.pattern.render() { - let lines = normalize_to_pattern(&text, &pattern, self.substitutions); - DataInner::TermSvg(lines) - } else { - DataInner::TermSvg(text) - } - } - }; - Data { - inner, - source, - filters, - } - } -} - #[cfg(feature = "structured-data")] fn normalize_json_string(value: &mut serde_json::Value, op: fn(&str) -> String) { match value { @@ -203,75 +141,3 @@ fn normalize_json_string(value: &mut serde_json::Value, op: fn(&str) -> String) _ => {} } } - -#[cfg(feature = "structured-data")] -fn normalize_value_matches( - actual: &mut serde_json::Value, - expected: &serde_json::Value, - substitutions: &crate::Redactions, -) { - use serde_json::Value::*; - - const KEY_WILDCARD: &str = "..."; - const VALUE_WILDCARD: &str = "{...}"; - - match (actual, expected) { - (act, String(exp)) if exp == VALUE_WILDCARD => { - *act = serde_json::json!(VALUE_WILDCARD); - } - (String(act), String(exp)) => { - *act = normalize_to_pattern(act, exp, substitutions); - } - (Array(act), Array(exp)) => { - let mut sections = exp.split(|e| e == VALUE_WILDCARD).peekable(); - let mut processed = 0; - while let Some(expected_subset) = sections.next() { - // Process all values in the current section - if !expected_subset.is_empty() { - let actual_subset = &mut act[processed..processed + expected_subset.len()]; - for (a, e) in actual_subset.iter_mut().zip(expected_subset) { - normalize_value_matches(a, e, substitutions); - } - processed += expected_subset.len(); - } - - if let Some(next_section) = sections.peek() { - // If the next section has nothing in it, replace from processed to end with - // a single "{...}" - if next_section.is_empty() { - act.splice(processed.., vec![String(VALUE_WILDCARD.to_owned())]); - processed += 1; - } else { - let first = next_section.first().unwrap(); - // Replace everything up until the value we are looking for with - // a single "{...}". - if let Some(index) = act.iter().position(|v| v == first) { - act.splice(processed..index, vec![String(VALUE_WILDCARD.to_owned())]); - processed += 1; - } else { - // If we cannot find the value we are looking for return early - break; - } - } - } - } - } - (Object(act), Object(exp)) => { - let has_key_wildcard = - exp.get(KEY_WILDCARD).and_then(|v| v.as_str()) == Some(VALUE_WILDCARD); - for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { - let actual_key = substitutions.redact(&actual_key); - if let Some(expected_value) = exp.get(&actual_key) { - normalize_value_matches(&mut actual_value, expected_value, substitutions) - } else if has_key_wildcard { - continue; - } - act.insert(actual_key, actual_value); - } - if has_key_wildcard { - act.insert(KEY_WILDCARD.to_owned(), String(VALUE_WILDCARD.to_owned())); - } - } - (_, _) => {} - } -} diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 6b6a6bf2..eafca5a4 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -1,4 +1,142 @@ +use super::Filter; use super::Redactions; +use crate::data::DataInner; +use crate::Data; + +pub struct FilterRedactions<'a> { + substitutions: &'a crate::Redactions, + pattern: &'a Data, +} + +impl<'a> FilterRedactions<'a> { + pub fn new(substitutions: &'a crate::Redactions, pattern: &'a Data) -> Self { + FilterRedactions { + substitutions, + pattern, + } + } +} + +impl Filter for FilterRedactions<'_> { + fn filter(&self, data: Data) -> Data { + let source = data.source; + let filters = data.filters; + let inner = match data.inner { + DataInner::Error(err) => DataInner::Error(err), + DataInner::Binary(bin) => DataInner::Binary(bin), + DataInner::Text(text) => { + if let Some(pattern) = self.pattern.render() { + let lines = normalize_to_pattern(&text, &pattern, self.substitutions); + DataInner::Text(lines) + } else { + DataInner::Text(text) + } + } + #[cfg(feature = "json")] + DataInner::Json(value) => { + let mut value = value; + if let DataInner::Json(exp) = &self.pattern.inner { + normalize_value_matches(&mut value, exp, self.substitutions); + } + DataInner::Json(value) + } + #[cfg(feature = "json")] + DataInner::JsonLines(value) => { + let mut value = value; + if let DataInner::Json(exp) = &self.pattern.inner { + normalize_value_matches(&mut value, exp, self.substitutions); + } + DataInner::JsonLines(value) + } + #[cfg(feature = "term-svg")] + DataInner::TermSvg(text) => { + if let Some(pattern) = self.pattern.render() { + let lines = normalize_to_pattern(&text, &pattern, self.substitutions); + DataInner::TermSvg(lines) + } else { + DataInner::TermSvg(text) + } + } + }; + Data { + inner, + source, + filters, + } + } +} + +#[cfg(feature = "structured-data")] +fn normalize_value_matches( + actual: &mut serde_json::Value, + expected: &serde_json::Value, + substitutions: &crate::Redactions, +) { + use serde_json::Value::*; + + const KEY_WILDCARD: &str = "..."; + const VALUE_WILDCARD: &str = "{...}"; + + match (actual, expected) { + (act, String(exp)) if exp == VALUE_WILDCARD => { + *act = serde_json::json!(VALUE_WILDCARD); + } + (String(act), String(exp)) => { + *act = normalize_to_pattern(act, exp, substitutions); + } + (Array(act), Array(exp)) => { + let mut sections = exp.split(|e| e == VALUE_WILDCARD).peekable(); + let mut processed = 0; + while let Some(expected_subset) = sections.next() { + // Process all values in the current section + if !expected_subset.is_empty() { + let actual_subset = &mut act[processed..processed + expected_subset.len()]; + for (a, e) in actual_subset.iter_mut().zip(expected_subset) { + normalize_value_matches(a, e, substitutions); + } + processed += expected_subset.len(); + } + + if let Some(next_section) = sections.peek() { + // If the next section has nothing in it, replace from processed to end with + // a single "{...}" + if next_section.is_empty() { + act.splice(processed.., vec![String(VALUE_WILDCARD.to_owned())]); + processed += 1; + } else { + let first = next_section.first().unwrap(); + // Replace everything up until the value we are looking for with + // a single "{...}". + if let Some(index) = act.iter().position(|v| v == first) { + act.splice(processed..index, vec![String(VALUE_WILDCARD.to_owned())]); + processed += 1; + } else { + // If we cannot find the value we are looking for return early + break; + } + } + } + } + } + (Object(act), Object(exp)) => { + let has_key_wildcard = + exp.get(KEY_WILDCARD).and_then(|v| v.as_str()) == Some(VALUE_WILDCARD); + for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { + let actual_key = substitutions.redact(&actual_key); + if let Some(expected_value) = exp.get(&actual_key) { + normalize_value_matches(&mut actual_value, expected_value, substitutions) + } else if has_key_wildcard { + continue; + } + act.insert(actual_key, actual_value); + } + if has_key_wildcard { + act.insert(KEY_WILDCARD.to_owned(), String(VALUE_WILDCARD.to_owned())); + } + } + (_, _) => {} + } +} pub(crate) fn normalize_to_pattern(input: &str, pattern: &str, redactions: &Redactions) -> String { if input == pattern { From adf5bf56368a791662ab99f5cb89042f129bfe78 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 13:14:45 -0500 Subject: [PATCH 04/19] docs(filter): Improve Redactions examples --- crates/snapbox/src/filter/redactions.rs | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/crates/snapbox/src/filter/redactions.rs b/crates/snapbox/src/filter/redactions.rs index ac74cd84..d0564d9d 100644 --- a/crates/snapbox/src/filter/redactions.rs +++ b/crates/snapbox/src/filter/redactions.rs @@ -4,9 +4,22 @@ use std::path::PathBuf; /// Replace data with placeholders /// +/// This can be used for: +/// - Handling test-run dependent data like temp directories or elapsed time +/// - Making special characters more obvious (e.g. redacting a tab a `[TAB]`) +/// - Normalizing platform-specific data like [`std::env::consts::EXE_SUFFIX`] +/// /// Built-in placeholders: /// - `...` on a line of its own: match multiple complete lines /// - `[..]`: match multiple characters within a line +/// +/// # Examples +/// +/// ```rust +/// let mut subst = snapbox::Redactions::new(); +/// subst.insert("[LOCATION]", "World"); +/// assert_eq!(subst.redact("Hello World!"), "Hello [LOCATION]!"); +/// ``` #[derive(Default, Clone, Debug, PartialEq, Eq)] pub struct Redactions { vars: std::collections::BTreeMap>, @@ -43,6 +56,9 @@ impl Redactions { /// # #[cfg(feature = "regex")] { /// let mut subst = snapbox::Redactions::new(); /// subst.insert("[OBJECT]", regex::Regex::new("(?(world|moon))").unwrap()); + /// assert_eq!(subst.redact("Hello world!"), "Hello [OBJECT]!"); + /// assert_eq!(subst.redact("Hello moon!"), "Hello [OBJECT]!"); + /// assert_eq!(subst.redact("Hello other!"), "Hello other!"); /// # } /// ``` pub fn insert( @@ -98,6 +114,15 @@ impl Redactions { } /// Apply redaction only, no pattern-dependent globs + /// + /// # Examples + /// + /// ```rust + /// let mut subst = snapbox::Redactions::new(); + /// subst.insert("[LOCATION]", "World"); + /// let output = subst.redact("Hello World!"); + /// assert_eq!(output, "Hello [LOCATION]!"); + /// ``` pub fn redact(&self, input: &str) -> String { let mut input = input.to_owned(); replace_many( From f132eac12c9b46c9a77f8edf61a7d9dfb459f746 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 13:15:06 -0500 Subject: [PATCH 05/19] fix(filter)!: Remove Redactions::normalize This interface is too simplistic for what we'll be needing --- crates/snapbox/src/filter/mod.rs | 1 - crates/snapbox/src/filter/pattern.rs | 8 +++++++- crates/snapbox/src/filter/redactions.rs | 19 ------------------- 3 files changed, 7 insertions(+), 21 deletions(-) diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index e62a1f07..87fadc0a 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -12,7 +12,6 @@ mod test; use crate::data::DataInner; use crate::Data; -pub(crate) use pattern::normalize_to_pattern; pub use pattern::FilterRedactions; pub use redactions::RedactedValue; pub use redactions::Redactions; diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index eafca5a4..4c1b74ec 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -3,6 +3,12 @@ use super::Redactions; use crate::data::DataInner; use crate::Data; +/// Adjust `actual` based on `expected` +/// +/// As part of this, [`Redactions`] will be used, if any. +/// Additional built-in redactions: +/// - `...` on a line of its own: match multiple complete lines +/// - `[..]`: match multiple characters within a line pub struct FilterRedactions<'a> { substitutions: &'a crate::Redactions, pattern: &'a Data, @@ -138,7 +144,7 @@ fn normalize_value_matches( } } -pub(crate) fn normalize_to_pattern(input: &str, pattern: &str, redactions: &Redactions) -> String { +fn normalize_to_pattern(input: &str, pattern: &str, redactions: &Redactions) -> String { if input == pattern { return input.to_owned(); } diff --git a/crates/snapbox/src/filter/redactions.rs b/crates/snapbox/src/filter/redactions.rs index d0564d9d..7f1798e2 100644 --- a/crates/snapbox/src/filter/redactions.rs +++ b/crates/snapbox/src/filter/redactions.rs @@ -9,10 +9,6 @@ use std::path::PathBuf; /// - Making special characters more obvious (e.g. redacting a tab a `[TAB]`) /// - Normalizing platform-specific data like [`std::env::consts::EXE_SUFFIX`] /// -/// Built-in placeholders: -/// - `...` on a line of its own: match multiple complete lines -/// - `[..]`: match multiple characters within a line -/// /// # Examples /// /// ```rust @@ -98,21 +94,6 @@ impl Redactions { Ok(()) } - /// Apply match pattern to `input` - /// - /// If `pattern` matches `input`, then `pattern` is returned. - /// - /// Otherwise, `input`, with as many patterns replaced as possible, will be returned. - /// - /// ```rust - /// let subst = snapbox::Redactions::new(); - /// let output = subst.normalize("Hello World!", "Hello [..]!"); - /// assert_eq!(output, "Hello [..]!"); - /// ``` - pub fn normalize(&self, input: &str, pattern: &str) -> String { - super::normalize_to_pattern(input, pattern, self) - } - /// Apply redaction only, no pattern-dependent globs /// /// # Examples From 5f99d98d8f10a403b7bf77781d0478d5a3a7040b Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 13:18:46 -0500 Subject: [PATCH 06/19] fix(filter)!: Rename FilterRedactions to NormalizeToExpected --- crates/snapbox/src/assert/mod.rs | 4 ++-- crates/snapbox/src/dir/diff.rs | 4 ++-- crates/snapbox/src/filter/mod.rs | 2 +- crates/snapbox/src/filter/pattern.rs | 8 +++---- crates/snapbox/src/filter/test.rs | 32 +++++++++++++++++----------- crates/trycmd/src/runner.rs | 4 ++-- 6 files changed, 31 insertions(+), 23 deletions(-) diff --git a/crates/snapbox/src/assert/mod.rs b/crates/snapbox/src/assert/mod.rs index d4bc71d5..28654e80 100644 --- a/crates/snapbox/src/assert/mod.rs +++ b/crates/snapbox/src/assert/mod.rs @@ -8,7 +8,7 @@ use anstream::stderr; #[cfg(not(feature = "color"))] use std::io::stderr; -use crate::filter::{Filter as _, FilterNewlines, FilterPaths, FilterRedactions}; +use crate::filter::{Filter as _, FilterNewlines, FilterPaths, NormalizeToExpected}; use crate::IntoData; pub use action::Action; @@ -128,7 +128,7 @@ impl Assert { actual = FilterNewlines.filter(actual); } if expected.filters.is_redaction_set() { - actual = FilterRedactions::new(&self.substitutions, &expected).filter(actual); + actual = NormalizeToExpected::new(&self.substitutions, &expected).filter(actual); } (actual, expected) diff --git a/crates/snapbox/src/dir/diff.rs b/crates/snapbox/src/dir/diff.rs index dafc7860..1e75705b 100644 --- a/crates/snapbox/src/dir/diff.rs +++ b/crates/snapbox/src/dir/diff.rs @@ -1,5 +1,5 @@ #[cfg(feature = "dir")] -use crate::filter::{Filter as _, FilterNewlines, FilterPaths, FilterRedactions}; +use crate::filter::{Filter as _, FilterNewlines, FilterPaths, NormalizeToExpected}; #[derive(Clone, Debug, PartialEq, Eq)] pub enum PathDiff { @@ -160,7 +160,7 @@ impl PathDiff { if normalize_paths { actual = FilterPaths.filter(actual); } - actual = FilterRedactions::new(substitutions, &expected) + actual = NormalizeToExpected::new(substitutions, &expected) .filter(FilterNewlines.filter(actual)); if expected != actual { diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index 87fadc0a..d52cfe99 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -12,7 +12,7 @@ mod test; use crate::data::DataInner; use crate::Data; -pub use pattern::FilterRedactions; +pub use pattern::NormalizeToExpected; pub use redactions::RedactedValue; pub use redactions::Redactions; diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 4c1b74ec..b9812be6 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -9,21 +9,21 @@ use crate::Data; /// Additional built-in redactions: /// - `...` on a line of its own: match multiple complete lines /// - `[..]`: match multiple characters within a line -pub struct FilterRedactions<'a> { +pub struct NormalizeToExpected<'a> { substitutions: &'a crate::Redactions, pattern: &'a Data, } -impl<'a> FilterRedactions<'a> { +impl<'a> NormalizeToExpected<'a> { pub fn new(substitutions: &'a crate::Redactions, pattern: &'a Data) -> Self { - FilterRedactions { + NormalizeToExpected { substitutions, pattern, } } } -impl Filter for FilterRedactions<'_> { +impl Filter for NormalizeToExpected<'_> { fn filter(&self, data: Data) -> Data { let source = data.source; let filters = data.filters; diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index ed83bbe5..b04626b8 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -125,7 +125,8 @@ fn json_normalize_matches_string() { let exp = json!({"name": "{...}"}); let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -144,7 +145,8 @@ fn json_normalize_matches_array() { } ] }); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -161,7 +163,8 @@ fn json_normalize_matches_obj() { "nickname": "John", } }); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -177,7 +180,8 @@ fn json_normalize_matches_diff_order_array() { let actual = json!({ "people": ["Jane", "John"] }); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_ne!(exp, act); } @@ -202,7 +206,7 @@ fn json_obj_redact_keys() { sub.insert("[A]", "key-a").unwrap(); sub.insert("[B]", "key-b").unwrap(); sub.insert("[C]", "key-c").unwrap(); - let actual = FilterRedactions::new(&sub, &expected).filter(actual); + let actual = NormalizeToExpected::new(&sub, &expected).filter(actual); let expected_actual = json!({ "[A]": "value-a", @@ -231,7 +235,7 @@ fn json_obj_redact_with_disparate_keys() { sub.insert("[A]", "value-a").unwrap(); sub.insert("[B]", "value-b").unwrap(); sub.insert("[C]", "value-c").unwrap(); - let actual = FilterRedactions::new(&sub, &expected).filter(actual); + let actual = NormalizeToExpected::new(&sub, &expected).filter(actual); let expected_actual = json!({ "a": "[A]", @@ -256,7 +260,7 @@ fn json_normalize_wildcard_key() { "c": "value-c", }); let actual = Data::json(actual); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(actual); + let actual = NormalizeToExpected::new(&Default::default(), &expected).filter(actual); let expected_actual = json!({ "a": "value-a", @@ -296,7 +300,8 @@ fn json_normalize_wildcard_object_first() { } ] }); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -327,7 +332,8 @@ fn json_normalize_wildcard_array_first() { "nickname": "3", } ]); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -363,7 +369,8 @@ fn json_normalize_wildcard_array_first_last() { "nickname": "4", } ]); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -407,7 +414,8 @@ fn json_normalize_wildcard_array_middle_last() { "nickname": "5", } ]); - let actual = FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual)); + let actual = + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -448,7 +456,7 @@ fn json_normalize_wildcard_array_middle_last_early_return() { } ]); let actual_normalized = - FilterRedactions::new(&Default::default(), &expected).filter(Data::json(actual.clone())); + NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual.clone())); if let DataInner::Json(act) = actual_normalized.inner { assert_eq!(act, actual); } diff --git a/crates/trycmd/src/runner.rs b/crates/trycmd/src/runner.rs index de2e6264..19faff5c 100644 --- a/crates/trycmd/src/runner.rs +++ b/crates/trycmd/src/runner.rs @@ -14,7 +14,7 @@ use std::io::stderr; use rayon::prelude::*; use snapbox::data::DataFormat; use snapbox::dir::FileType; -use snapbox::filter::{Filter as _, FilterNewlines, FilterPaths, FilterRedactions}; +use snapbox::filter::{Filter as _, FilterNewlines, FilterPaths, NormalizeToExpected}; use snapbox::IntoData; #[derive(Debug)] @@ -440,7 +440,7 @@ impl Case { if let Some(expected_content) = expected_content { stream.content = - FilterRedactions::new(substitutions, expected_content).filter(stream.content); + NormalizeToExpected::new(substitutions, expected_content).filter(stream.content); if stream.content != *expected_content { stream.status = StreamStatus::Expected(expected_content.clone()); From 6e8afe17c1db12a5a7c11c7ec3f420fe44d15397 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 13:22:08 -0500 Subject: [PATCH 07/19] fix(filter)!: Replace Filter::filter with NormalizeToExpected::normalize --- crates/snapbox/src/assert/mod.rs | 2 +- crates/snapbox/src/dir/diff.rs | 2 +- crates/snapbox/src/filter/pattern.rs | 11 ++++------- crates/snapbox/src/filter/test.rs | 26 +++++++++++++------------- crates/trycmd/src/runner.rs | 2 +- 5 files changed, 20 insertions(+), 23 deletions(-) diff --git a/crates/snapbox/src/assert/mod.rs b/crates/snapbox/src/assert/mod.rs index 28654e80..cbb38844 100644 --- a/crates/snapbox/src/assert/mod.rs +++ b/crates/snapbox/src/assert/mod.rs @@ -128,7 +128,7 @@ impl Assert { actual = FilterNewlines.filter(actual); } if expected.filters.is_redaction_set() { - actual = NormalizeToExpected::new(&self.substitutions, &expected).filter(actual); + actual = NormalizeToExpected::new(&self.substitutions, &expected).normalize(actual); } (actual, expected) diff --git a/crates/snapbox/src/dir/diff.rs b/crates/snapbox/src/dir/diff.rs index 1e75705b..78c7e404 100644 --- a/crates/snapbox/src/dir/diff.rs +++ b/crates/snapbox/src/dir/diff.rs @@ -161,7 +161,7 @@ impl PathDiff { actual = FilterPaths.filter(actual); } actual = NormalizeToExpected::new(substitutions, &expected) - .filter(FilterNewlines.filter(actual)); + .normalize(FilterNewlines.filter(actual)); if expected != actual { return Err(Self::ContentMismatch { diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index b9812be6..df3592db 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -1,4 +1,3 @@ -use super::Filter; use super::Redactions; use crate::data::DataInner; use crate::Data; @@ -21,13 +20,11 @@ impl<'a> NormalizeToExpected<'a> { pattern, } } -} -impl Filter for NormalizeToExpected<'_> { - fn filter(&self, data: Data) -> Data { - let source = data.source; - let filters = data.filters; - let inner = match data.inner { + pub fn normalize(&self, actual: Data) -> Data { + let source = actual.source; + let filters = actual.filters; + let inner = match actual.inner { DataInner::Error(err) => DataInner::Error(err), DataInner::Binary(bin) => DataInner::Binary(bin), DataInner::Text(text) => { diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index b04626b8..f643a91e 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -126,7 +126,7 @@ fn json_normalize_matches_string() { let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -146,7 +146,7 @@ fn json_normalize_matches_array() { ] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -164,7 +164,7 @@ fn json_normalize_matches_obj() { } }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -181,7 +181,7 @@ fn json_normalize_matches_diff_order_array() { "people": ["Jane", "John"] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_ne!(exp, act); } @@ -206,7 +206,7 @@ fn json_obj_redact_keys() { sub.insert("[A]", "key-a").unwrap(); sub.insert("[B]", "key-b").unwrap(); sub.insert("[C]", "key-c").unwrap(); - let actual = NormalizeToExpected::new(&sub, &expected).filter(actual); + let actual = NormalizeToExpected::new(&sub, &expected).normalize(actual); let expected_actual = json!({ "[A]": "value-a", @@ -235,7 +235,7 @@ fn json_obj_redact_with_disparate_keys() { sub.insert("[A]", "value-a").unwrap(); sub.insert("[B]", "value-b").unwrap(); sub.insert("[C]", "value-c").unwrap(); - let actual = NormalizeToExpected::new(&sub, &expected).filter(actual); + let actual = NormalizeToExpected::new(&sub, &expected).normalize(actual); let expected_actual = json!({ "a": "[A]", @@ -260,7 +260,7 @@ fn json_normalize_wildcard_key() { "c": "value-c", }); let actual = Data::json(actual); - let actual = NormalizeToExpected::new(&Default::default(), &expected).filter(actual); + let actual = NormalizeToExpected::new(&Default::default(), &expected).normalize(actual); let expected_actual = json!({ "a": "value-a", @@ -301,7 +301,7 @@ fn json_normalize_wildcard_object_first() { ] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -333,7 +333,7 @@ fn json_normalize_wildcard_array_first() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -370,7 +370,7 @@ fn json_normalize_wildcard_array_first_last() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -415,7 +415,7 @@ fn json_normalize_wildcard_array_middle_last() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual)); + NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -455,8 +455,8 @@ fn json_normalize_wildcard_array_middle_last_early_return() { "nickname": "5", } ]); - let actual_normalized = - NormalizeToExpected::new(&Default::default(), &expected).filter(Data::json(actual.clone())); + let actual_normalized = NormalizeToExpected::new(&Default::default(), &expected) + .normalize(Data::json(actual.clone())); if let DataInner::Json(act) = actual_normalized.inner { assert_eq!(act, actual); } diff --git a/crates/trycmd/src/runner.rs b/crates/trycmd/src/runner.rs index 19faff5c..e806dd83 100644 --- a/crates/trycmd/src/runner.rs +++ b/crates/trycmd/src/runner.rs @@ -440,7 +440,7 @@ impl Case { if let Some(expected_content) = expected_content { stream.content = - NormalizeToExpected::new(substitutions, expected_content).filter(stream.content); + NormalizeToExpected::new(substitutions, expected_content).normalize(stream.content); if stream.content != *expected_content { stream.status = StreamStatus::Expected(expected_content.clone()); From 5227854a447d9f2d29aa9af13945092e5e5b7b59 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:00:51 -0500 Subject: [PATCH 08/19] fix(filter)!: Clarify normalizing actual needs specific expected --- crates/snapbox/src/assert/mod.rs | 2 +- crates/snapbox/src/dir/diff.rs | 4 ++-- crates/snapbox/src/filter/pattern.rs | 18 +++++++----------- crates/snapbox/src/filter/test.rs | 26 +++++++++++++------------- crates/trycmd/src/runner.rs | 2 +- 5 files changed, 24 insertions(+), 28 deletions(-) diff --git a/crates/snapbox/src/assert/mod.rs b/crates/snapbox/src/assert/mod.rs index cbb38844..7b1030c7 100644 --- a/crates/snapbox/src/assert/mod.rs +++ b/crates/snapbox/src/assert/mod.rs @@ -128,7 +128,7 @@ impl Assert { actual = FilterNewlines.filter(actual); } if expected.filters.is_redaction_set() { - actual = NormalizeToExpected::new(&self.substitutions, &expected).normalize(actual); + actual = NormalizeToExpected::new(&self.substitutions).normalize(actual, &expected); } (actual, expected) diff --git a/crates/snapbox/src/dir/diff.rs b/crates/snapbox/src/dir/diff.rs index 78c7e404..02c18820 100644 --- a/crates/snapbox/src/dir/diff.rs +++ b/crates/snapbox/src/dir/diff.rs @@ -160,8 +160,8 @@ impl PathDiff { if normalize_paths { actual = FilterPaths.filter(actual); } - actual = NormalizeToExpected::new(substitutions, &expected) - .normalize(FilterNewlines.filter(actual)); + actual = NormalizeToExpected::new(substitutions) + .normalize(FilterNewlines.filter(actual), &expected); if expected != actual { return Err(Self::ContentMismatch { diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index df3592db..f02b87cf 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -10,25 +10,21 @@ use crate::Data; /// - `[..]`: match multiple characters within a line pub struct NormalizeToExpected<'a> { substitutions: &'a crate::Redactions, - pattern: &'a Data, } impl<'a> NormalizeToExpected<'a> { - pub fn new(substitutions: &'a crate::Redactions, pattern: &'a Data) -> Self { - NormalizeToExpected { - substitutions, - pattern, - } + pub fn new(substitutions: &'a crate::Redactions) -> Self { + NormalizeToExpected { substitutions } } - pub fn normalize(&self, actual: Data) -> Data { + pub fn normalize(&self, actual: Data, expected: &Data) -> Data { let source = actual.source; let filters = actual.filters; let inner = match actual.inner { DataInner::Error(err) => DataInner::Error(err), DataInner::Binary(bin) => DataInner::Binary(bin), DataInner::Text(text) => { - if let Some(pattern) = self.pattern.render() { + if let Some(pattern) = expected.render() { let lines = normalize_to_pattern(&text, &pattern, self.substitutions); DataInner::Text(lines) } else { @@ -38,7 +34,7 @@ impl<'a> NormalizeToExpected<'a> { #[cfg(feature = "json")] DataInner::Json(value) => { let mut value = value; - if let DataInner::Json(exp) = &self.pattern.inner { + if let DataInner::Json(exp) = &expected.inner { normalize_value_matches(&mut value, exp, self.substitutions); } DataInner::Json(value) @@ -46,14 +42,14 @@ impl<'a> NormalizeToExpected<'a> { #[cfg(feature = "json")] DataInner::JsonLines(value) => { let mut value = value; - if let DataInner::Json(exp) = &self.pattern.inner { + if let DataInner::Json(exp) = &expected.inner { normalize_value_matches(&mut value, exp, self.substitutions); } DataInner::JsonLines(value) } #[cfg(feature = "term-svg")] DataInner::TermSvg(text) => { - if let Some(pattern) = self.pattern.render() { + if let Some(pattern) = expected.render() { let lines = normalize_to_pattern(&text, &pattern, self.substitutions); DataInner::TermSvg(lines) } else { diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index f643a91e..fa824e8b 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -126,7 +126,7 @@ fn json_normalize_matches_string() { let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -146,7 +146,7 @@ fn json_normalize_matches_array() { ] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -164,7 +164,7 @@ fn json_normalize_matches_obj() { } }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -181,7 +181,7 @@ fn json_normalize_matches_diff_order_array() { "people": ["Jane", "John"] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_ne!(exp, act); } @@ -206,7 +206,7 @@ fn json_obj_redact_keys() { sub.insert("[A]", "key-a").unwrap(); sub.insert("[B]", "key-b").unwrap(); sub.insert("[C]", "key-c").unwrap(); - let actual = NormalizeToExpected::new(&sub, &expected).normalize(actual); + let actual = NormalizeToExpected::new(&sub).normalize(actual, &expected); let expected_actual = json!({ "[A]": "value-a", @@ -235,7 +235,7 @@ fn json_obj_redact_with_disparate_keys() { sub.insert("[A]", "value-a").unwrap(); sub.insert("[B]", "value-b").unwrap(); sub.insert("[C]", "value-c").unwrap(); - let actual = NormalizeToExpected::new(&sub, &expected).normalize(actual); + let actual = NormalizeToExpected::new(&sub).normalize(actual, &expected); let expected_actual = json!({ "a": "[A]", @@ -260,7 +260,7 @@ fn json_normalize_wildcard_key() { "c": "value-c", }); let actual = Data::json(actual); - let actual = NormalizeToExpected::new(&Default::default(), &expected).normalize(actual); + let actual = NormalizeToExpected::new(&Default::default()).normalize(actual, &expected); let expected_actual = json!({ "a": "value-a", @@ -301,7 +301,7 @@ fn json_normalize_wildcard_object_first() { ] }); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -333,7 +333,7 @@ fn json_normalize_wildcard_array_first() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -370,7 +370,7 @@ fn json_normalize_wildcard_array_first_last() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -415,7 +415,7 @@ fn json_normalize_wildcard_array_middle_last() { } ]); let actual = - NormalizeToExpected::new(&Default::default(), &expected).normalize(Data::json(actual)); + NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -455,8 +455,8 @@ fn json_normalize_wildcard_array_middle_last_early_return() { "nickname": "5", } ]); - let actual_normalized = NormalizeToExpected::new(&Default::default(), &expected) - .normalize(Data::json(actual.clone())); + let actual_normalized = NormalizeToExpected::new(&Default::default()) + .normalize(Data::json(actual.clone()), &expected); if let DataInner::Json(act) = actual_normalized.inner { assert_eq!(act, actual); } diff --git a/crates/trycmd/src/runner.rs b/crates/trycmd/src/runner.rs index e806dd83..8e3f93fa 100644 --- a/crates/trycmd/src/runner.rs +++ b/crates/trycmd/src/runner.rs @@ -440,7 +440,7 @@ impl Case { if let Some(expected_content) = expected_content { stream.content = - NormalizeToExpected::new(substitutions, expected_content).normalize(stream.content); + NormalizeToExpected::new(substitutions).normalize(stream.content, expected_content); if stream.content != *expected_content { stream.status = StreamStatus::Expected(expected_content.clone()); From ba303b3dfd0485f3e1ec5ac50831368420827af2 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:01:03 -0500 Subject: [PATCH 09/19] feat(filter): Allow const construction of Redactions --- crates/snapbox/src/filter/redactions.rs | 53 +++++++++++++++++-------- 1 file changed, 36 insertions(+), 17 deletions(-) diff --git a/crates/snapbox/src/filter/redactions.rs b/crates/snapbox/src/filter/redactions.rs index 7f1798e2..03f35793 100644 --- a/crates/snapbox/src/filter/redactions.rs +++ b/crates/snapbox/src/filter/redactions.rs @@ -18,13 +18,18 @@ use std::path::PathBuf; /// ``` #[derive(Default, Clone, Debug, PartialEq, Eq)] pub struct Redactions { - vars: std::collections::BTreeMap>, - unused: std::collections::BTreeSet, + vars: Option< + std::collections::BTreeMap>, + >, + unused: Option>, } impl Redactions { - pub fn new() -> Self { - Default::default() + pub const fn new() -> Self { + Self { + vars: None, + unused: None, + } } pub(crate) fn with_exe() -> Self { @@ -65,9 +70,15 @@ impl Redactions { let placeholder = validate_placeholder(placeholder)?; let value = value.into(); if let Some(value) = value.inner { - self.vars.entry(value).or_default().insert(placeholder); + self.vars + .get_or_insert(std::collections::BTreeMap::new()) + .entry(value) + .or_default() + .insert(placeholder); } else { - self.unused.insert(RedactedValueInner::Str(placeholder)); + self.unused + .get_or_insert(std::collections::BTreeSet::new()) + .insert(RedactedValueInner::Str(placeholder)); } Ok(()) } @@ -87,10 +98,12 @@ impl Redactions { pub fn remove(&mut self, placeholder: &'static str) -> crate::assert::Result<()> { let placeholder = validate_placeholder(placeholder)?; - self.vars.retain(|_value, placeholders| { - placeholders.retain(|p| *p != placeholder); - !placeholders.is_empty() - }); + self.vars + .get_or_insert(std::collections::BTreeMap::new()) + .retain(|_value, placeholders| { + placeholders.retain(|p| *p != placeholder); + !placeholders.is_empty() + }); Ok(()) } @@ -108,19 +121,25 @@ impl Redactions { let mut input = input.to_owned(); replace_many( &mut input, - self.vars.iter().flat_map(|(value, placeholders)| { - placeholders - .iter() - .map(move |placeholder| (value, *placeholder)) - }), + self.vars + .iter() + .flatten() + .flat_map(|(value, placeholders)| { + placeholders + .iter() + .map(move |placeholder| (value, *placeholder)) + }), ); input } pub(crate) fn clear<'v>(&self, pattern: &'v str) -> Cow<'v, str> { - if !self.unused.is_empty() && pattern.contains('[') { + if !self.unused.as_ref().map(|s| s.is_empty()).unwrap_or(false) && pattern.contains('[') { let mut pattern = pattern.to_owned(); - replace_many(&mut pattern, self.unused.iter().map(|var| (var, ""))); + replace_many( + &mut pattern, + self.unused.iter().flatten().map(|var| (var, "")), + ); Cow::Owned(pattern) } else { Cow::Borrowed(pattern) From bb11bc3eccaddd7a93be00acbedc9c43f40929d3 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:09:19 -0500 Subject: [PATCH 10/19] fix(filter)!: Split NormalizeToExpected construction --- crates/snapbox/src/assert/mod.rs | 4 ++- crates/snapbox/src/dir/diff.rs | 3 ++- crates/snapbox/src/filter/pattern.rs | 18 +++++++++++-- crates/snapbox/src/filter/test.rs | 38 +++++++++++++--------------- crates/trycmd/src/runner.rs | 5 ++-- 5 files changed, 41 insertions(+), 27 deletions(-) diff --git a/crates/snapbox/src/assert/mod.rs b/crates/snapbox/src/assert/mod.rs index 7b1030c7..42695c32 100644 --- a/crates/snapbox/src/assert/mod.rs +++ b/crates/snapbox/src/assert/mod.rs @@ -128,7 +128,9 @@ impl Assert { actual = FilterNewlines.filter(actual); } if expected.filters.is_redaction_set() { - actual = NormalizeToExpected::new(&self.substitutions).normalize(actual, &expected); + actual = NormalizeToExpected::new() + .redact_with(&self.substitutions) + .normalize(actual, &expected); } (actual, expected) diff --git a/crates/snapbox/src/dir/diff.rs b/crates/snapbox/src/dir/diff.rs index 02c18820..207e43f9 100644 --- a/crates/snapbox/src/dir/diff.rs +++ b/crates/snapbox/src/dir/diff.rs @@ -160,7 +160,8 @@ impl PathDiff { if normalize_paths { actual = FilterPaths.filter(actual); } - actual = NormalizeToExpected::new(substitutions) + actual = NormalizeToExpected::new() + .redact_with(substitutions) .normalize(FilterNewlines.filter(actual), &expected); if expected != actual { diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index f02b87cf..90e12d6b 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -13,8 +13,16 @@ pub struct NormalizeToExpected<'a> { } impl<'a> NormalizeToExpected<'a> { - pub fn new(substitutions: &'a crate::Redactions) -> Self { - NormalizeToExpected { substitutions } + pub fn new() -> Self { + static REDACTIONS: Redactions = Redactions::new(); + Self { + substitutions: &REDACTIONS, + } + } + + pub fn redact_with(mut self, redactions: &'a crate::Redactions) -> Self { + self.substitutions = redactions; + self } pub fn normalize(&self, actual: Data, expected: &Data) -> Data { @@ -65,6 +73,12 @@ impl<'a> NormalizeToExpected<'a> { } } +impl Default for NormalizeToExpected<'_> { + fn default() -> Self { + Self::new() + } +} + #[cfg(feature = "structured-data")] fn normalize_value_matches( actual: &mut serde_json::Value, diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index fa824e8b..07a68e05 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -125,8 +125,7 @@ fn json_normalize_matches_string() { let exp = json!({"name": "{...}"}); let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -145,8 +144,7 @@ fn json_normalize_matches_array() { } ] }); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -163,8 +161,7 @@ fn json_normalize_matches_obj() { "nickname": "John", } }); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -180,8 +177,7 @@ fn json_normalize_matches_diff_order_array() { let actual = json!({ "people": ["Jane", "John"] }); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_ne!(exp, act); } @@ -206,7 +202,9 @@ fn json_obj_redact_keys() { sub.insert("[A]", "key-a").unwrap(); sub.insert("[B]", "key-b").unwrap(); sub.insert("[C]", "key-c").unwrap(); - let actual = NormalizeToExpected::new(&sub).normalize(actual, &expected); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); let expected_actual = json!({ "[A]": "value-a", @@ -235,7 +233,9 @@ fn json_obj_redact_with_disparate_keys() { sub.insert("[A]", "value-a").unwrap(); sub.insert("[B]", "value-b").unwrap(); sub.insert("[C]", "value-c").unwrap(); - let actual = NormalizeToExpected::new(&sub).normalize(actual, &expected); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); let expected_actual = json!({ "a": "[A]", @@ -260,7 +260,7 @@ fn json_normalize_wildcard_key() { "c": "value-c", }); let actual = Data::json(actual); - let actual = NormalizeToExpected::new(&Default::default()).normalize(actual, &expected); + let actual = NormalizeToExpected::new().normalize(actual, &expected); let expected_actual = json!({ "a": "value-a", @@ -300,8 +300,7 @@ fn json_normalize_wildcard_object_first() { } ] }); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -332,8 +331,7 @@ fn json_normalize_wildcard_array_first() { "nickname": "3", } ]); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -369,8 +367,7 @@ fn json_normalize_wildcard_array_first_last() { "nickname": "4", } ]); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -414,8 +411,7 @@ fn json_normalize_wildcard_array_middle_last() { "nickname": "5", } ]); - let actual = - NormalizeToExpected::new(&Default::default()).normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -455,8 +451,8 @@ fn json_normalize_wildcard_array_middle_last_early_return() { "nickname": "5", } ]); - let actual_normalized = NormalizeToExpected::new(&Default::default()) - .normalize(Data::json(actual.clone()), &expected); + let actual_normalized = + NormalizeToExpected::new().normalize(Data::json(actual.clone()), &expected); if let DataInner::Json(act) = actual_normalized.inner { assert_eq!(act, actual); } diff --git a/crates/trycmd/src/runner.rs b/crates/trycmd/src/runner.rs index 8e3f93fa..9033676a 100644 --- a/crates/trycmd/src/runner.rs +++ b/crates/trycmd/src/runner.rs @@ -439,8 +439,9 @@ impl Case { } if let Some(expected_content) = expected_content { - stream.content = - NormalizeToExpected::new(substitutions).normalize(stream.content, expected_content); + stream.content = NormalizeToExpected::new() + .redact_with(substitutions) + .normalize(stream.content, expected_content); if stream.content != *expected_content { stream.status = StreamStatus::Expected(expected_content.clone()); From c765a4caf98133a7abc72b35d8adeeb91c938664 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:14:23 -0500 Subject: [PATCH 11/19] fix(filter)!: Require opt-in for built-in redactions --- crates/snapbox/src/filter/pattern.rs | 43 +++++++++++++++++++--------- crates/snapbox/src/filter/test.rs | 41 +++++++++++++++++++------- 2 files changed, 60 insertions(+), 24 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 90e12d6b..c3e1c95d 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -3,29 +3,46 @@ use crate::data::DataInner; use crate::Data; /// Adjust `actual` based on `expected` -/// -/// As part of this, [`Redactions`] will be used, if any. -/// Additional built-in redactions: -/// - `...` on a line of its own: match multiple complete lines -/// - `[..]`: match multiple characters within a line pub struct NormalizeToExpected<'a> { - substitutions: &'a crate::Redactions, + substitutions: Option<&'a crate::Redactions>, } impl<'a> NormalizeToExpected<'a> { pub fn new() -> Self { - static REDACTIONS: Redactions = Redactions::new(); Self { - substitutions: &REDACTIONS, + substitutions: None, } } + /// Apply built-in redactions. + /// + /// Built-in redactions: + /// - `...` on a line of its own: match multiple complete lines + /// - `[..]`: match multiple characters within a line + /// + /// Built-ins cannot automatically be applied to `actual` but are inferred from `expected` + pub fn redact(mut self) -> Self { + static REDACTIONS: Redactions = Redactions::new(); + self.substitutions = Some(&REDACTIONS); + self + } + + /// Apply built-in and user [`Redactions`] + /// + /// Built-in redactions: + /// - `...` on a line of its own: match multiple complete lines + /// - `[..]`: match multiple characters within a line + /// + /// Built-ins cannot automatically be applied to `actual` but are inferred from `expected` pub fn redact_with(mut self, redactions: &'a crate::Redactions) -> Self { - self.substitutions = redactions; + self.substitutions = Some(redactions); self } pub fn normalize(&self, actual: Data, expected: &Data) -> Data { + let Some(substitutions) = self.substitutions else { + return actual; + }; let source = actual.source; let filters = actual.filters; let inner = match actual.inner { @@ -33,7 +50,7 @@ impl<'a> NormalizeToExpected<'a> { DataInner::Binary(bin) => DataInner::Binary(bin), DataInner::Text(text) => { if let Some(pattern) = expected.render() { - let lines = normalize_to_pattern(&text, &pattern, self.substitutions); + let lines = normalize_to_pattern(&text, &pattern, substitutions); DataInner::Text(lines) } else { DataInner::Text(text) @@ -43,7 +60,7 @@ impl<'a> NormalizeToExpected<'a> { DataInner::Json(value) => { let mut value = value; if let DataInner::Json(exp) = &expected.inner { - normalize_value_matches(&mut value, exp, self.substitutions); + normalize_value_matches(&mut value, exp, substitutions); } DataInner::Json(value) } @@ -51,14 +68,14 @@ impl<'a> NormalizeToExpected<'a> { DataInner::JsonLines(value) => { let mut value = value; if let DataInner::Json(exp) = &expected.inner { - normalize_value_matches(&mut value, exp, self.substitutions); + normalize_value_matches(&mut value, exp, substitutions); } DataInner::JsonLines(value) } #[cfg(feature = "term-svg")] DataInner::TermSvg(text) => { if let Some(pattern) = expected.render() { - let lines = normalize_to_pattern(&text, &pattern, self.substitutions); + let lines = normalize_to_pattern(&text, &pattern, substitutions); DataInner::TermSvg(lines) } else { DataInner::TermSvg(text) diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index 07a68e05..72e407df 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -125,7 +125,9 @@ fn json_normalize_matches_string() { let exp = json!({"name": "{...}"}); let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -144,7 +146,9 @@ fn json_normalize_matches_array() { } ] }); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -161,7 +165,9 @@ fn json_normalize_matches_obj() { "nickname": "John", } }); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -177,7 +183,9 @@ fn json_normalize_matches_diff_order_array() { let actual = json!({ "people": ["Jane", "John"] }); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_ne!(exp, act); } @@ -260,7 +268,9 @@ fn json_normalize_wildcard_key() { "c": "value-c", }); let actual = Data::json(actual); - let actual = NormalizeToExpected::new().normalize(actual, &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(actual, &expected); let expected_actual = json!({ "a": "value-a", @@ -300,7 +310,9 @@ fn json_normalize_wildcard_object_first() { } ] }); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -331,7 +343,9 @@ fn json_normalize_wildcard_array_first() { "nickname": "3", } ]); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -367,7 +381,9 @@ fn json_normalize_wildcard_array_first_last() { "nickname": "4", } ]); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -411,7 +427,9 @@ fn json_normalize_wildcard_array_middle_last() { "nickname": "5", } ]); - let actual = NormalizeToExpected::new().normalize(Data::json(actual), &expected); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { assert_eq!(exp, act); } @@ -451,8 +469,9 @@ fn json_normalize_wildcard_array_middle_last_early_return() { "nickname": "5", } ]); - let actual_normalized = - NormalizeToExpected::new().normalize(Data::json(actual.clone()), &expected); + let actual_normalized = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual.clone()), &expected); if let DataInner::Json(act) = actual_normalized.inner { assert_eq!(act, actual); } From 4e4250bbf5bdf85ae56031983122cb217b614610 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:31:37 -0500 Subject: [PATCH 12/19] refactor(filter): Specialize normalization fn --- crates/snapbox/src/filter/pattern.rs | 56 ++++++++++++++-------------- 1 file changed, 28 insertions(+), 28 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index c3e1c95d..201d8c60 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -50,7 +50,7 @@ impl<'a> NormalizeToExpected<'a> { DataInner::Binary(bin) => DataInner::Binary(bin), DataInner::Text(text) => { if let Some(pattern) = expected.render() { - let lines = normalize_to_pattern(&text, &pattern, substitutions); + let lines = normalize_str_to_redactions(&text, &pattern, substitutions); DataInner::Text(lines) } else { DataInner::Text(text) @@ -60,7 +60,7 @@ impl<'a> NormalizeToExpected<'a> { DataInner::Json(value) => { let mut value = value; if let DataInner::Json(exp) = &expected.inner { - normalize_value_matches(&mut value, exp, substitutions); + normalize_value_to_redactions(&mut value, exp, substitutions); } DataInner::Json(value) } @@ -68,14 +68,14 @@ impl<'a> NormalizeToExpected<'a> { DataInner::JsonLines(value) => { let mut value = value; if let DataInner::Json(exp) = &expected.inner { - normalize_value_matches(&mut value, exp, substitutions); + normalize_value_to_redactions(&mut value, exp, substitutions); } DataInner::JsonLines(value) } #[cfg(feature = "term-svg")] DataInner::TermSvg(text) => { if let Some(pattern) = expected.render() { - let lines = normalize_to_pattern(&text, &pattern, substitutions); + let lines = normalize_str_to_redactions(&text, &pattern, substitutions); DataInner::TermSvg(lines) } else { DataInner::TermSvg(text) @@ -97,7 +97,7 @@ impl Default for NormalizeToExpected<'_> { } #[cfg(feature = "structured-data")] -fn normalize_value_matches( +fn normalize_value_to_redactions( actual: &mut serde_json::Value, expected: &serde_json::Value, substitutions: &crate::Redactions, @@ -112,7 +112,7 @@ fn normalize_value_matches( *act = serde_json::json!(VALUE_WILDCARD); } (String(act), String(exp)) => { - *act = normalize_to_pattern(act, exp, substitutions); + *act = normalize_str_to_redactions(act, exp, substitutions); } (Array(act), Array(exp)) => { let mut sections = exp.split(|e| e == VALUE_WILDCARD).peekable(); @@ -122,7 +122,7 @@ fn normalize_value_matches( if !expected_subset.is_empty() { let actual_subset = &mut act[processed..processed + expected_subset.len()]; for (a, e) in actual_subset.iter_mut().zip(expected_subset) { - normalize_value_matches(a, e, substitutions); + normalize_value_to_redactions(a, e, substitutions); } processed += expected_subset.len(); } @@ -154,7 +154,7 @@ fn normalize_value_matches( for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { let actual_key = substitutions.redact(&actual_key); if let Some(expected_value) = exp.get(&actual_key) { - normalize_value_matches(&mut actual_value, expected_value, substitutions) + normalize_value_to_redactions(&mut actual_value, expected_value, substitutions) } else if has_key_wildcard { continue; } @@ -168,7 +168,7 @@ fn normalize_value_matches( } } -fn normalize_to_pattern(input: &str, pattern: &str, redactions: &Redactions) -> String { +fn normalize_str_to_redactions(input: &str, pattern: &str, redactions: &Redactions) -> String { if input == pattern { return input.to_owned(); } @@ -261,7 +261,7 @@ mod test { let input = ""; let pattern = ""; let expected = ""; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -270,7 +270,7 @@ mod test { let input = "Hello\nWorld"; let pattern = "Hello\nWorld"; let expected = "Hello\nWorld"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -279,7 +279,7 @@ mod test { let input = "Hello\nWorld"; let pattern = "Hello\n"; let expected = "Hello\nWorld"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -288,7 +288,7 @@ mod test { let input = "Hello\n"; let pattern = "Hello\nWorld"; let expected = "Hello\n"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -297,7 +297,7 @@ mod test { let input = "Hello\nWorld"; let pattern = "Goodbye\nMoon"; let expected = "Hello\nWorld"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -306,7 +306,7 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\nMoon\nGoodbye"; let expected = "Hello\nWorld\nGoodbye"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -315,7 +315,7 @@ mod test { let input = "Hello World\nHow are you?\nGoodbye World"; let pattern = "Hello [..]\n...\nGoodbye [..]"; let expected = "Hello [..]\n...\nGoodbye [..]"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -324,7 +324,7 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "...\nGoodbye"; let expected = "...\nGoodbye"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -333,7 +333,7 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n..."; let expected = "Hello\n..."; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -342,7 +342,7 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n...\nGoodbye"; let expected = "Hello\n...\nGoodbye"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -351,7 +351,7 @@ mod test { let input = "Hello\nSun\nAnd\nWorld"; let pattern = "Hello\n...\nMoon"; let expected = "Hello\nSun\nAnd\nWorld"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -360,7 +360,7 @@ mod test { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nMoon\nGoodbye\n..."; let expected = "Hello\nWorld\nGoodbye\nSir"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -369,7 +369,7 @@ mod test { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nW[..]d\nGoodbye\nSir"; let expected = "Hello\nW[..]d\nGoodbye\nSir"; - let actual = normalize_to_pattern(input, pattern, &Redactions::new()); + let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); assert_eq!(expected, actual); } @@ -424,7 +424,7 @@ mod test { let pattern = "Hello [OBJECT]!"; let mut sub = Redactions::new(); sub.insert("[OBJECT]", "world").unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } @@ -436,7 +436,7 @@ mod test { let sep = std::path::MAIN_SEPARATOR.to_string(); let redacted = PathBuf::from(sep).join("home").join("epage"); sub.insert("[HOME]", redacted).unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } @@ -457,7 +457,7 @@ b: [B]"; .join("epage") .join("snapbox"); sub.insert("[B]", redacted).unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } @@ -467,7 +467,7 @@ b: [B]"; let pattern = "cargo[EXE]"; let mut sub = Redactions::new(); sub.insert("[EXE]", "").unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } @@ -479,7 +479,7 @@ b: [B]"; let mut sub = Redactions::new(); sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) .unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } @@ -494,7 +494,7 @@ b: [B]"; regex::Regex::new("(?world)!").unwrap(), ) .unwrap(); - let actual = normalize_to_pattern(input, pattern, &sub); + let actual = normalize_str_to_redactions(input, pattern, &sub); assert_eq!(actual, pattern); } } From ac0acd4482fd83ed9e72d61c2378811f2a77d70c Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 14:35:14 -0500 Subject: [PATCH 13/19] refactor(filter): Pull out normalization fn --- crates/snapbox/src/filter/pattern.rs | 96 +++++++++++++++------------- 1 file changed, 52 insertions(+), 44 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 201d8c60..2e4ed0ae 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -43,50 +43,7 @@ impl<'a> NormalizeToExpected<'a> { let Some(substitutions) = self.substitutions else { return actual; }; - let source = actual.source; - let filters = actual.filters; - let inner = match actual.inner { - DataInner::Error(err) => DataInner::Error(err), - DataInner::Binary(bin) => DataInner::Binary(bin), - DataInner::Text(text) => { - if let Some(pattern) = expected.render() { - let lines = normalize_str_to_redactions(&text, &pattern, substitutions); - DataInner::Text(lines) - } else { - DataInner::Text(text) - } - } - #[cfg(feature = "json")] - DataInner::Json(value) => { - let mut value = value; - if let DataInner::Json(exp) = &expected.inner { - normalize_value_to_redactions(&mut value, exp, substitutions); - } - DataInner::Json(value) - } - #[cfg(feature = "json")] - DataInner::JsonLines(value) => { - let mut value = value; - if let DataInner::Json(exp) = &expected.inner { - normalize_value_to_redactions(&mut value, exp, substitutions); - } - DataInner::JsonLines(value) - } - #[cfg(feature = "term-svg")] - DataInner::TermSvg(text) => { - if let Some(pattern) = expected.render() { - let lines = normalize_str_to_redactions(&text, &pattern, substitutions); - DataInner::TermSvg(lines) - } else { - DataInner::TermSvg(text) - } - } - }; - Data { - inner, - source, - filters, - } + normalize_data_to_redactions(actual, expected, substitutions) } } @@ -96,6 +53,57 @@ impl Default for NormalizeToExpected<'_> { } } +fn normalize_data_to_redactions( + actual: Data, + expected: &Data, + substitutions: &crate::Redactions, +) -> Data { + let source = actual.source; + let filters = actual.filters; + let inner = match actual.inner { + DataInner::Error(err) => DataInner::Error(err), + DataInner::Binary(bin) => DataInner::Binary(bin), + DataInner::Text(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_redactions(&text, &pattern, substitutions); + DataInner::Text(lines) + } else { + DataInner::Text(text) + } + } + #[cfg(feature = "json")] + DataInner::Json(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_redactions(&mut value, exp, substitutions); + } + DataInner::Json(value) + } + #[cfg(feature = "json")] + DataInner::JsonLines(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_redactions(&mut value, exp, substitutions); + } + DataInner::JsonLines(value) + } + #[cfg(feature = "term-svg")] + DataInner::TermSvg(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_redactions(&text, &pattern, substitutions); + DataInner::TermSvg(lines) + } else { + DataInner::TermSvg(text) + } + } + }; + Data { + inner, + source, + filters, + } +} + #[cfg(feature = "structured-data")] fn normalize_value_to_redactions( actual: &mut serde_json::Value, From a83e4d144ac88c133ac88d71618649057b4ed504 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 15:08:27 -0500 Subject: [PATCH 14/19] refactor(filter): Clarify test names --- crates/snapbox/src/filter/pattern.rs | 40 ++-- crates/snapbox/src/filter/test.rs | 273 ++++++++++++--------------- 2 files changed, 140 insertions(+), 173 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 2e4ed0ae..b837038c 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -265,7 +265,7 @@ mod test { use super::*; #[test] - fn empty() { + fn str_normalize_redactions_empty() { let input = ""; let pattern = ""; let expected = ""; @@ -274,7 +274,7 @@ mod test { } #[test] - fn literals_match() { + fn str_normalize_redactions_literals_match() { let input = "Hello\nWorld"; let pattern = "Hello\nWorld"; let expected = "Hello\nWorld"; @@ -283,7 +283,7 @@ mod test { } #[test] - fn pattern_shorter() { + fn str_normalize_redactions_pattern_shorter() { let input = "Hello\nWorld"; let pattern = "Hello\n"; let expected = "Hello\nWorld"; @@ -292,7 +292,7 @@ mod test { } #[test] - fn input_shorter() { + fn str_normalize_redactions_input_shorter() { let input = "Hello\n"; let pattern = "Hello\nWorld"; let expected = "Hello\n"; @@ -301,7 +301,7 @@ mod test { } #[test] - fn all_different() { + fn str_normalize_redactions_all_different() { let input = "Hello\nWorld"; let pattern = "Goodbye\nMoon"; let expected = "Hello\nWorld"; @@ -310,7 +310,7 @@ mod test { } #[test] - fn middles_diverge() { + fn str_normalize_redactions_middles_diverge() { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\nMoon\nGoodbye"; let expected = "Hello\nWorld\nGoodbye"; @@ -319,7 +319,7 @@ mod test { } #[test] - fn elide_delimited_with_sub() { + fn str_normalize_redactions_elide_delimited_with_sub() { let input = "Hello World\nHow are you?\nGoodbye World"; let pattern = "Hello [..]\n...\nGoodbye [..]"; let expected = "Hello [..]\n...\nGoodbye [..]"; @@ -328,7 +328,7 @@ mod test { } #[test] - fn leading_elide() { + fn str_normalize_redactions_leading_elide() { let input = "Hello\nWorld\nGoodbye"; let pattern = "...\nGoodbye"; let expected = "...\nGoodbye"; @@ -337,7 +337,7 @@ mod test { } #[test] - fn trailing_elide() { + fn str_normalize_redactions_trailing_elide() { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n..."; let expected = "Hello\n..."; @@ -346,7 +346,7 @@ mod test { } #[test] - fn middle_elide() { + fn str_normalize_redactions_middle_elide() { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n...\nGoodbye"; let expected = "Hello\n...\nGoodbye"; @@ -355,7 +355,7 @@ mod test { } #[test] - fn post_elide_diverge() { + fn str_normalize_redactions_post_elide_diverge() { let input = "Hello\nSun\nAnd\nWorld"; let pattern = "Hello\n...\nMoon"; let expected = "Hello\nSun\nAnd\nWorld"; @@ -364,7 +364,7 @@ mod test { } #[test] - fn post_diverge_elide() { + fn str_normalize_redactions_post_diverge_elide() { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nMoon\nGoodbye\n..."; let expected = "Hello\nWorld\nGoodbye\nSir"; @@ -373,7 +373,7 @@ mod test { } #[test] - fn inline_elide() { + fn str_normalize_redactions_inline_elide() { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nW[..]d\nGoodbye\nSir"; let expected = "Hello\nW[..]d\nGoodbye\nSir"; @@ -382,7 +382,7 @@ mod test { } #[test] - fn line_matches_cases() { + fn str_normalize_redactions_line_matches_cases() { let cases = [ ("", "", true), ("", "[..]", true), @@ -427,7 +427,7 @@ mod test { } #[test] - fn substitute_literal() { + fn str_normalize_redactions_user_literal() { let input = "Hello world!"; let pattern = "Hello [OBJECT]!"; let mut sub = Redactions::new(); @@ -437,7 +437,7 @@ mod test { } #[test] - fn substitute_path() { + fn str_normalize_redactions_user_path() { let input = "input: /home/epage"; let pattern = "input: [HOME]"; let mut sub = Redactions::new(); @@ -449,7 +449,7 @@ mod test { } #[test] - fn substitute_overlapping_path() { + fn str_normalize_redactions_user_overlapping_path() { let input = "\ a: /home/epage b: /home/epage/snapbox"; @@ -470,7 +470,7 @@ b: [B]"; } #[test] - fn substitute_disabled() { + fn str_normalize_redactions_user_disabled() { let input = "cargo"; let pattern = "cargo[EXE]"; let mut sub = Redactions::new(); @@ -481,7 +481,7 @@ b: [B]"; #[test] #[cfg(feature = "regex")] - fn substitute_regex_unnamed() { + fn str_normalize_redactions_user_regex_unnamed() { let input = "Hello world!"; let pattern = "Hello [OBJECT]!"; let mut sub = Redactions::new(); @@ -493,7 +493,7 @@ b: [B]"; #[test] #[cfg(feature = "regex")] - fn substitute_regex_named() { + fn str_normalize_redactions_user_regex_named() { let input = "Hello world!"; let pattern = "Hello [OBJECT]!"; let mut sub = Redactions::new(); diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index 72e407df..fbb0eb05 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -7,7 +7,7 @@ use super::*; // Tests for normalization on json #[test] #[cfg(feature = "json")] -fn json_normalize_paths_and_lines() { +fn json_normalize_paths_and_lines_string() { let json = json!({"name": "John\\Doe\r\n"}); let data = Data::json(json); let data = FilterPaths.filter(data); @@ -18,7 +18,7 @@ fn json_normalize_paths_and_lines() { #[test] #[cfg(feature = "json")] -fn json_normalize_obj_value_paths_and_lines() { +fn json_normalize_paths_and_lines_nested_string() { let json = json!({ "person": { "name": "John\\Doe\r\n", @@ -46,7 +46,7 @@ fn json_normalize_obj_value_paths_and_lines() { #[test] #[cfg(feature = "json")] -fn json_normalize_obj_key_paths_and_lines() { +fn json_normalize_paths_and_lines_obj_key() { let json = json!({ "person": { "John\\Doe\r\n": "name", @@ -74,7 +74,7 @@ fn json_normalize_obj_key_paths_and_lines() { #[test] #[cfg(feature = "json")] -fn json_normalize_array_paths_and_lines() { +fn json_normalize_paths_and_lines_array() { let json = json!({"people": ["John\\Doe\r\n", "Jo\\hn\r\n"]}); let data = Data::json(json); let data = FilterPaths.filter(data); @@ -87,7 +87,7 @@ fn json_normalize_array_paths_and_lines() { #[test] #[cfg(feature = "json")] -fn json_normalize_array_obj_paths_and_lines() { +fn json_normalize_paths_and_lines_array_obj() { let json = json!({ "people": [ { @@ -121,7 +121,7 @@ fn json_normalize_array_obj_paths_and_lines() { #[test] #[cfg(feature = "json")] -fn json_normalize_matches_string() { +fn json_normalize_redactions_glob_for_string() { let exp = json!({"name": "{...}"}); let expected = Data::json(exp); let actual = json!({"name": "JohnDoe"}); @@ -135,7 +135,7 @@ fn json_normalize_matches_string() { #[test] #[cfg(feature = "json")] -fn json_normalize_matches_array() { +fn json_normalize_redactions_glob_for_array() { let exp = json!({"people": "{...}"}); let expected = Data::json(exp); let actual = json!({ @@ -156,7 +156,7 @@ fn json_normalize_matches_array() { #[test] #[cfg(feature = "json")] -fn json_normalize_matches_obj() { +fn json_normalize_redactions_glob_for_obj() { let exp = json!({"people": "{...}"}); let expected = Data::json(exp); let actual = json!({ @@ -175,115 +175,7 @@ fn json_normalize_matches_obj() { #[test] #[cfg(feature = "json")] -fn json_normalize_matches_diff_order_array() { - let exp = json!({ - "people": ["John", "Jane"] - }); - let expected = Data::json(exp); - let actual = json!({ - "people": ["Jane", "John"] - }); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_ne!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_obj_redact_keys() { - let expected = json!({ - "[A]": "value-a", - "[B]": "value-b", - "[C]": "value-c", - }); - let expected = Data::json(expected); - let actual = json!({ - "key-a": "value-a", - "key-b": "value-b", - "key-c": "value-c", - }); - let actual = Data::json(actual); - let mut sub = Redactions::new(); - sub.insert("[A]", "key-a").unwrap(); - sub.insert("[B]", "key-b").unwrap(); - sub.insert("[C]", "key-c").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(actual, &expected); - - let expected_actual = json!({ - "[A]": "value-a", - "[B]": "value-b", - "[C]": "value-c", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} - -#[test] -#[cfg(feature = "json")] -fn json_obj_redact_with_disparate_keys() { - let expected = json!({ - "a": "[A]", - "b": "[B]", - "c": "[C]", - }); - let expected = Data::json(expected); - let actual = json!({ - "a": "value-a", - "c": "value-c", - }); - let actual = Data::json(actual); - let mut sub = Redactions::new(); - sub.insert("[A]", "value-a").unwrap(); - sub.insert("[B]", "value-b").unwrap(); - sub.insert("[C]", "value-c").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(actual, &expected); - - let expected_actual = json!({ - "a": "[A]", - "c": "[C]", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_wildcard_key() { - let expected = json!({ - "a": "value-a", - "c": "value-c", - "...": "{...}", - }); - let expected = Data::json(expected); - let actual = json!({ - "a": "value-a", - "b": "value-b", - "c": "value-c", - }); - let actual = Data::json(actual); - let actual = NormalizeToExpected::new() - .redact() - .normalize(actual, &expected); - - let expected_actual = json!({ - "a": "value-a", - "c": "value-c", - "...": "{...}", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_wildcard_object_first() { +fn json_normalize_redactions_glob_array_start() { let exp = json!({ "people": [ "{...}", @@ -320,40 +212,7 @@ fn json_normalize_wildcard_object_first() { #[test] #[cfg(feature = "json")] -fn json_normalize_wildcard_array_first() { - let exp = json!([ - "{...}", - { - "name": "three", - "nickname": "3", - } - ]); - let expected = Data::json(exp); - let actual = json!([ - { - "name": "one", - "nickname": "1", - }, - { - "name": "two", - "nickname": "2", - }, - { - "name": "three", - "nickname": "3", - } - ]); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_wildcard_array_first_last() { +fn json_normalize_redactions_glob_for_array_start_end() { let exp = json!([ "{...}", { @@ -391,7 +250,7 @@ fn json_normalize_wildcard_array_first_last() { #[test] #[cfg(feature = "json")] -fn json_normalize_wildcard_array_middle_last() { +fn json_normalize_redactions_glob_for_array_middle_end() { let exp = json!([ { "name": "one", @@ -437,7 +296,7 @@ fn json_normalize_wildcard_array_middle_last() { #[test] #[cfg(feature = "json")] -fn json_normalize_wildcard_array_middle_last_early_return() { +fn json_normalize_redactions_glob_for_array_mismatch() { let exp = json!([ { "name": "one", @@ -476,3 +335,111 @@ fn json_normalize_wildcard_array_middle_last_early_return() { assert_eq!(act, actual); } } + +#[test] +#[cfg(feature = "json")] +fn json_normalize_redactions_bad_order() { + let exp = json!({ + "people": ["John", "Jane"] + }); + let expected = Data::json(exp); + let actual = json!({ + "people": ["Jane", "John"] + }); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_ne!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_redactions_obj_key() { + let expected = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected = Data::json(expected); + let actual = json!({ + "key-a": "value-a", + "key-b": "value-b", + "key-c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "key-a").unwrap(); + sub.insert("[B]", "key-b").unwrap(); + sub.insert("[C]", "key-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); + + let expected_actual = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_redactions_with_missing_obj_key() { + let expected = json!({ + "a": "[A]", + "b": "[B]", + "c": "[C]", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "value-a").unwrap(); + sub.insert("[B]", "value-b").unwrap(); + sub.insert("[C]", "value-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "[A]", + "c": "[C]", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_redactions_glob_obj_key() { + let expected = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "b": "value-b", + "c": "value-c", + }); + let actual = Data::json(actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} From c5c688eb40f2168a611d5c43cff2f14258346cfb Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 15:39:09 -0500 Subject: [PATCH 15/19] test(filter): Transition unit tests to end-to-end tests --- crates/snapbox/src/filter/pattern.rs | 115 ++++++++++++++++++--------- 1 file changed, 77 insertions(+), 38 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index b837038c..6e31d378 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -263,14 +263,17 @@ mod test { use std::path::PathBuf; use super::*; + use crate::prelude::*; #[test] fn str_normalize_redactions_empty() { let input = ""; let pattern = ""; let expected = ""; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -278,8 +281,10 @@ mod test { let input = "Hello\nWorld"; let pattern = "Hello\nWorld"; let expected = "Hello\nWorld"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -287,8 +292,10 @@ mod test { let input = "Hello\nWorld"; let pattern = "Hello\n"; let expected = "Hello\nWorld"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -296,8 +303,10 @@ mod test { let input = "Hello\n"; let pattern = "Hello\nWorld"; let expected = "Hello\n"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -305,8 +314,10 @@ mod test { let input = "Hello\nWorld"; let pattern = "Goodbye\nMoon"; let expected = "Hello\nWorld"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -314,8 +325,10 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\nMoon\nGoodbye"; let expected = "Hello\nWorld\nGoodbye"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -323,8 +336,10 @@ mod test { let input = "Hello World\nHow are you?\nGoodbye World"; let pattern = "Hello [..]\n...\nGoodbye [..]"; let expected = "Hello [..]\n...\nGoodbye [..]"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -332,8 +347,10 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "...\nGoodbye"; let expected = "...\nGoodbye"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -341,8 +358,10 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n..."; let expected = "Hello\n..."; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -350,8 +369,10 @@ mod test { let input = "Hello\nWorld\nGoodbye"; let pattern = "Hello\n...\nGoodbye"; let expected = "Hello\n...\nGoodbye"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -359,8 +380,10 @@ mod test { let input = "Hello\nSun\nAnd\nWorld"; let pattern = "Hello\n...\nMoon"; let expected = "Hello\nSun\nAnd\nWorld"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -368,8 +391,10 @@ mod test { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nMoon\nGoodbye\n..."; let expected = "Hello\nWorld\nGoodbye\nSir"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -377,8 +402,10 @@ mod test { let input = "Hello\nWorld\nGoodbye\nSir"; let pattern = "Hello\nW[..]d\nGoodbye\nSir"; let expected = "Hello\nW[..]d\nGoodbye\nSir"; - let actual = normalize_str_to_redactions(input, pattern, &Redactions::new()); - assert_eq!(expected, actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); } #[test] @@ -432,8 +459,10 @@ mod test { let pattern = "Hello [OBJECT]!"; let mut sub = Redactions::new(); sub.insert("[OBJECT]", "world").unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } #[test] @@ -444,8 +473,10 @@ mod test { let sep = std::path::MAIN_SEPARATOR.to_string(); let redacted = PathBuf::from(sep).join("home").join("epage"); sub.insert("[HOME]", redacted).unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } #[test] @@ -465,8 +496,10 @@ b: [B]"; .join("epage") .join("snapbox"); sub.insert("[B]", redacted).unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } #[test] @@ -475,8 +508,10 @@ b: [B]"; let pattern = "cargo[EXE]"; let mut sub = Redactions::new(); sub.insert("[EXE]", "").unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } #[test] @@ -487,8 +522,10 @@ b: [B]"; let mut sub = Redactions::new(); sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) .unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } #[test] @@ -502,7 +539,9 @@ b: [B]"; regex::Regex::new("(?world)!").unwrap(), ) .unwrap(); - let actual = normalize_str_to_redactions(input, pattern, &sub); - assert_eq!(actual, pattern); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); } } From 8d465cf34930212afb9e9baab72d39d32c110af9 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 15:40:25 -0500 Subject: [PATCH 16/19] refactor(filter): Consolidate end-to-end tests --- crates/snapbox/src/filter/pattern.rs | 238 -------------------------- crates/snapbox/src/filter/test.rs | 239 ++++++++++++++++++++++++++- 2 files changed, 238 insertions(+), 239 deletions(-) diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 6e31d378..857bab20 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -260,153 +260,7 @@ fn line_matches(mut input: &str, pattern: &str, redactions: &Redactions) -> bool #[cfg(test)] mod test { - use std::path::PathBuf; - use super::*; - use crate::prelude::*; - - #[test] - fn str_normalize_redactions_empty() { - let input = ""; - let pattern = ""; - let expected = ""; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_literals_match() { - let input = "Hello\nWorld"; - let pattern = "Hello\nWorld"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_pattern_shorter() { - let input = "Hello\nWorld"; - let pattern = "Hello\n"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_input_shorter() { - let input = "Hello\n"; - let pattern = "Hello\nWorld"; - let expected = "Hello\n"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_all_different() { - let input = "Hello\nWorld"; - let pattern = "Goodbye\nMoon"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_middles_diverge() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\nMoon\nGoodbye"; - let expected = "Hello\nWorld\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_elide_delimited_with_sub() { - let input = "Hello World\nHow are you?\nGoodbye World"; - let pattern = "Hello [..]\n...\nGoodbye [..]"; - let expected = "Hello [..]\n...\nGoodbye [..]"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_leading_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "...\nGoodbye"; - let expected = "...\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_trailing_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n..."; - let expected = "Hello\n..."; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_middle_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n...\nGoodbye"; - let expected = "Hello\n...\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_post_elide_diverge() { - let input = "Hello\nSun\nAnd\nWorld"; - let pattern = "Hello\n...\nMoon"; - let expected = "Hello\nSun\nAnd\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_post_diverge_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nMoon\nGoodbye\n..."; - let expected = "Hello\nWorld\nGoodbye\nSir"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } - - #[test] - fn str_normalize_redactions_inline_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nW[..]d\nGoodbye\nSir"; - let expected = "Hello\nW[..]d\nGoodbye\nSir"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); - } #[test] fn str_normalize_redactions_line_matches_cases() { @@ -452,96 +306,4 @@ mod test { assert_eq!(expected, actual, "line={:?} pattern={:?}", line, pattern); } } - - #[test] - fn str_normalize_redactions_user_literal() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", "world").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } - - #[test] - fn str_normalize_redactions_user_path() { - let input = "input: /home/epage"; - let pattern = "input: [HOME]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(sep).join("home").join("epage"); - sub.insert("[HOME]", redacted).unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } - - #[test] - fn str_normalize_redactions_user_overlapping_path() { - let input = "\ -a: /home/epage -b: /home/epage/snapbox"; - let pattern = "\ -a: [A] -b: [B]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(&sep).join("home").join("epage"); - sub.insert("[A]", redacted).unwrap(); - let redacted = PathBuf::from(sep) - .join("home") - .join("epage") - .join("snapbox"); - sub.insert("[B]", redacted).unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } - - #[test] - fn str_normalize_redactions_user_disabled() { - let input = "cargo"; - let pattern = "cargo[EXE]"; - let mut sub = Redactions::new(); - sub.insert("[EXE]", "").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } - - #[test] - #[cfg(feature = "regex")] - fn str_normalize_redactions_user_regex_unnamed() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) - .unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } - - #[test] - #[cfg(feature = "regex")] - fn str_normalize_redactions_user_regex_named() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert( - "[OBJECT]", - regex::Regex::new("(?world)!").unwrap(), - ) - .unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); - } } diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index fbb0eb05..fe2f526d 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -1,8 +1,245 @@ +use std::path::PathBuf; + #[cfg(feature = "json")] use serde_json::json; -#[cfg(feature = "json")] use super::*; +use crate::prelude::*; + +#[test] +fn str_normalize_redactions_empty() { + let input = ""; + let pattern = ""; + let expected = ""; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_literals_match() { + let input = "Hello\nWorld"; + let pattern = "Hello\nWorld"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_pattern_shorter() { + let input = "Hello\nWorld"; + let pattern = "Hello\n"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_input_shorter() { + let input = "Hello\n"; + let pattern = "Hello\nWorld"; + let expected = "Hello\n"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_all_different() { + let input = "Hello\nWorld"; + let pattern = "Goodbye\nMoon"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_middles_diverge() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\nMoon\nGoodbye"; + let expected = "Hello\nWorld\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_elide_delimited_with_sub() { + let input = "Hello World\nHow are you?\nGoodbye World"; + let pattern = "Hello [..]\n...\nGoodbye [..]"; + let expected = "Hello [..]\n...\nGoodbye [..]"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_leading_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "...\nGoodbye"; + let expected = "...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_trailing_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n..."; + let expected = "Hello\n..."; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_middle_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n...\nGoodbye"; + let expected = "Hello\n...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_post_elide_diverge() { + let input = "Hello\nSun\nAnd\nWorld"; + let pattern = "Hello\n...\nMoon"; + let expected = "Hello\nSun\nAnd\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_post_diverge_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nMoon\nGoodbye\n..."; + let expected = "Hello\nWorld\nGoodbye\nSir"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_inline_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nW[..]d\nGoodbye\nSir"; + let expected = "Hello\nW[..]d\nGoodbye\nSir"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_redactions_user_literal() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", "world").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_redactions_user_path() { + let input = "input: /home/epage"; + let pattern = "input: [HOME]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(sep).join("home").join("epage"); + sub.insert("[HOME]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_redactions_user_overlapping_path() { + let input = "\ +a: /home/epage +b: /home/epage/snapbox"; + let pattern = "\ +a: [A] +b: [B]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(&sep).join("home").join("epage"); + sub.insert("[A]", redacted).unwrap(); + let redacted = PathBuf::from(sep) + .join("home") + .join("epage") + .join("snapbox"); + sub.insert("[B]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_redactions_user_disabled() { + let input = "cargo"; + let pattern = "cargo[EXE]"; + let mut sub = Redactions::new(); + sub.insert("[EXE]", "").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "regex")] +fn str_normalize_redactions_user_regex_unnamed() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) + .unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "regex")] +fn str_normalize_redactions_user_regex_named() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert( + "[OBJECT]", + regex::Regex::new("(?world)!").unwrap(), + ) + .unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} // Tests for normalization on json #[test] From f08fd7a65cbd33eb60c8e87ebecc15af1308792f Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 16:55:50 -0500 Subject: [PATCH 17/19] refactor(filter): Perform a distint redaction pass --- crates/snapbox/src/filter/mod.rs | 50 +++++++++++++++++++++++++--- crates/snapbox/src/filter/pattern.rs | 11 +++--- 2 files changed, 51 insertions(+), 10 deletions(-) diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index d52cfe99..28411d7b 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -35,13 +35,13 @@ impl Filter for FilterNewlines { #[cfg(feature = "json")] DataInner::Json(value) => { let mut value = value; - normalize_json_string(&mut value, normalize_lines); + normalize_json_string(&mut value, &normalize_lines); DataInner::Json(value) } #[cfg(feature = "json")] DataInner::JsonLines(value) => { let mut value = value; - normalize_json_string(&mut value, normalize_lines); + normalize_json_string(&mut value, &normalize_lines); DataInner::JsonLines(value) } #[cfg(feature = "term-svg")] @@ -82,13 +82,13 @@ impl Filter for FilterPaths { #[cfg(feature = "json")] DataInner::Json(value) => { let mut value = value; - normalize_json_string(&mut value, normalize_paths); + normalize_json_string(&mut value, &normalize_paths); DataInner::Json(value) } #[cfg(feature = "json")] DataInner::JsonLines(value) => { let mut value = value; - normalize_json_string(&mut value, normalize_paths); + normalize_json_string(&mut value, &normalize_paths); DataInner::JsonLines(value) } #[cfg(feature = "term-svg")] @@ -119,8 +119,48 @@ fn normalize_paths_chars(data: impl Iterator) -> impl Iterator { + redactions: &'r Redactions, +} +impl Filter for NormalizeRedactions<'_> { + fn filter(&self, data: Data) -> Data { + let source = data.source; + let filters = data.filters; + let inner = match data.inner { + DataInner::Error(err) => DataInner::Error(err), + DataInner::Binary(bin) => DataInner::Binary(bin), + DataInner::Text(text) => { + let lines = self.redactions.redact(&text); + DataInner::Text(lines) + } + #[cfg(feature = "json")] + DataInner::Json(value) => { + let mut value = value; + normalize_json_string(&mut value, &|s| self.redactions.redact(s)); + DataInner::Json(value) + } + #[cfg(feature = "json")] + DataInner::JsonLines(value) => { + let mut value = value; + normalize_json_string(&mut value, &|s| self.redactions.redact(s)); + DataInner::JsonLines(value) + } + #[cfg(feature = "term-svg")] + DataInner::TermSvg(text) => { + let lines = normalize_lines(&text); + DataInner::TermSvg(lines) + } + }; + Data { + inner, + source, + filters, + } + } +} + #[cfg(feature = "structured-data")] -fn normalize_json_string(value: &mut serde_json::Value, op: fn(&str) -> String) { +fn normalize_json_string(value: &mut serde_json::Value, op: &dyn Fn(&str) -> String) { match value { serde_json::Value::String(str) => { *str = op(str); diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index 857bab20..c8fe33fd 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -1,4 +1,4 @@ -use super::Redactions; +use super::{Filter, NormalizeRedactions, Redactions}; use crate::data::DataInner; use crate::Data; @@ -43,6 +43,10 @@ impl<'a> NormalizeToExpected<'a> { let Some(substitutions) = self.substitutions else { return actual; }; + let actual = NormalizeRedactions { + redactions: substitutions, + } + .filter(actual); normalize_data_to_redactions(actual, expected, substitutions) } } @@ -160,7 +164,6 @@ fn normalize_value_to_redactions( let has_key_wildcard = exp.get(KEY_WILDCARD).and_then(|v| v.as_str()) == Some(VALUE_WILDCARD); for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { - let actual_key = substitutions.redact(&actual_key); if let Some(expected_value) = exp.get(&actual_key) { normalize_value_to_redactions(&mut actual_value, expected_value, substitutions) } else if has_key_wildcard { @@ -181,11 +184,9 @@ fn normalize_str_to_redactions(input: &str, pattern: &str, redactions: &Redactio return input.to_owned(); } - let input = redactions.redact(input); - let mut normalized: Vec<&str> = Vec::new(); let mut input_index = 0; - let input_lines: Vec<_> = crate::utils::LinesWithTerminator::new(&input).collect(); + let input_lines: Vec<_> = crate::utils::LinesWithTerminator::new(input).collect(); let mut pattern_lines = crate::utils::LinesWithTerminator::new(pattern).peekable(); 'outer: while let Some(pattern_line) = pattern_lines.next() { if is_line_elide(pattern_line) { From c44c6f4d3169b289e7f177d47e1cb0313efddc5d Mon Sep 17 00:00:00 2001 From: Ed Page Date: Thu, 23 May 2024 13:47:45 -0500 Subject: [PATCH 18/19] refactor(filter): Split up tests --- crates/snapbox/src/filter/mod.rs | 2 + crates/snapbox/src/filter/test.rs | 564 +----------------- crates/snapbox/src/filter/test_redactions.rs | 567 +++++++++++++++++++ 3 files changed, 570 insertions(+), 563 deletions(-) create mode 100644 crates/snapbox/src/filter/test_redactions.rs diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index 28411d7b..6217c40d 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -8,6 +8,8 @@ mod pattern; mod redactions; #[cfg(test)] mod test; +#[cfg(test)] +mod test_redactions; use crate::data::DataInner; use crate::Data; diff --git a/crates/snapbox/src/filter/test.rs b/crates/snapbox/src/filter/test.rs index fe2f526d..46a287b3 100644 --- a/crates/snapbox/src/filter/test.rs +++ b/crates/snapbox/src/filter/test.rs @@ -1,245 +1,8 @@ -use std::path::PathBuf; - #[cfg(feature = "json")] use serde_json::json; +#[cfg(feature = "json")] use super::*; -use crate::prelude::*; - -#[test] -fn str_normalize_redactions_empty() { - let input = ""; - let pattern = ""; - let expected = ""; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_literals_match() { - let input = "Hello\nWorld"; - let pattern = "Hello\nWorld"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_pattern_shorter() { - let input = "Hello\nWorld"; - let pattern = "Hello\n"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_input_shorter() { - let input = "Hello\n"; - let pattern = "Hello\nWorld"; - let expected = "Hello\n"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_all_different() { - let input = "Hello\nWorld"; - let pattern = "Goodbye\nMoon"; - let expected = "Hello\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_middles_diverge() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\nMoon\nGoodbye"; - let expected = "Hello\nWorld\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_elide_delimited_with_sub() { - let input = "Hello World\nHow are you?\nGoodbye World"; - let pattern = "Hello [..]\n...\nGoodbye [..]"; - let expected = "Hello [..]\n...\nGoodbye [..]"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_leading_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "...\nGoodbye"; - let expected = "...\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_trailing_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n..."; - let expected = "Hello\n..."; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_middle_elide() { - let input = "Hello\nWorld\nGoodbye"; - let pattern = "Hello\n...\nGoodbye"; - let expected = "Hello\n...\nGoodbye"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_post_elide_diverge() { - let input = "Hello\nSun\nAnd\nWorld"; - let pattern = "Hello\n...\nMoon"; - let expected = "Hello\nSun\nAnd\nWorld"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_post_diverge_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nMoon\nGoodbye\n..."; - let expected = "Hello\nWorld\nGoodbye\nSir"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_inline_elide() { - let input = "Hello\nWorld\nGoodbye\nSir"; - let pattern = "Hello\nW[..]d\nGoodbye\nSir"; - let expected = "Hello\nW[..]d\nGoodbye\nSir"; - let actual = NormalizeToExpected::new() - .redact() - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, expected.into_data()); -} - -#[test] -fn str_normalize_redactions_user_literal() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", "world").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} - -#[test] -fn str_normalize_redactions_user_path() { - let input = "input: /home/epage"; - let pattern = "input: [HOME]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(sep).join("home").join("epage"); - sub.insert("[HOME]", redacted).unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} - -#[test] -fn str_normalize_redactions_user_overlapping_path() { - let input = "\ -a: /home/epage -b: /home/epage/snapbox"; - let pattern = "\ -a: [A] -b: [B]"; - let mut sub = Redactions::new(); - let sep = std::path::MAIN_SEPARATOR.to_string(); - let redacted = PathBuf::from(&sep).join("home").join("epage"); - sub.insert("[A]", redacted).unwrap(); - let redacted = PathBuf::from(sep) - .join("home") - .join("epage") - .join("snapbox"); - sub.insert("[B]", redacted).unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} - -#[test] -fn str_normalize_redactions_user_disabled() { - let input = "cargo"; - let pattern = "cargo[EXE]"; - let mut sub = Redactions::new(); - sub.insert("[EXE]", "").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} - -#[test] -#[cfg(feature = "regex")] -fn str_normalize_redactions_user_regex_unnamed() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) - .unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} - -#[test] -#[cfg(feature = "regex")] -fn str_normalize_redactions_user_regex_named() { - let input = "Hello world!"; - let pattern = "Hello [OBJECT]!"; - let mut sub = Redactions::new(); - sub.insert( - "[OBJECT]", - regex::Regex::new("(?world)!").unwrap(), - ) - .unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(input.into(), &pattern.into()); - assert_eq!(actual, pattern.into_data()); -} // Tests for normalization on json #[test] @@ -355,328 +118,3 @@ fn json_normalize_paths_and_lines_array_obj() { }); assert_eq!(Data::json(new_lines), data); } - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_string() { - let exp = json!({"name": "{...}"}); - let expected = Data::json(exp); - let actual = json!({"name": "JohnDoe"}); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_array() { - let exp = json!({"people": "{...}"}); - let expected = Data::json(exp); - let actual = json!({ - "people": [ - { - "name": "JohnDoe", - "nickname": "John", - } - ] - }); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_obj() { - let exp = json!({"people": "{...}"}); - let expected = Data::json(exp); - let actual = json!({ - "people": { - "name": "JohnDoe", - "nickname": "John", - } - }); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_array_start() { - let exp = json!({ - "people": [ - "{...}", - { - "name": "three", - "nickname": "3", - } - ] - }); - let expected = Data::json(exp); - let actual = json!({ - "people": [ - { - "name": "one", - "nickname": "1", - }, - { - "name": "two", - "nickname": "2", - }, - { - "name": "three", - "nickname": "3", - } - ] - }); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_array_start_end() { - let exp = json!([ - "{...}", - { - "name": "two", - "nickname": "2", - }, - "{...}" - ]); - let expected = Data::json(exp); - let actual = json!([ - { - "name": "one", - "nickname": "1", - }, - { - "name": "two", - "nickname": "2", - }, - { - "name": "three", - "nickname": "3", - }, - { - "name": "four", - "nickname": "4", - } - ]); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_array_middle_end() { - let exp = json!([ - { - "name": "one", - "nickname": "1", - }, - "{...}", - { - "name": "three", - "nickname": "3", - }, - "{...}" - ]); - let expected = Data::json(exp); - let actual = json!([ - { - "name": "one", - "nickname": "1", - }, - { - "name": "two", - "nickname": "2", - }, - { - "name": "three", - "nickname": "3", - }, - { - "name": "four", - "nickname": "4", - }, - { - "name": "five", - "nickname": "5", - } - ]); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_eq!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_for_array_mismatch() { - let exp = json!([ - { - "name": "one", - "nickname": "1", - }, - "{...}", - { - "name": "three", - "nickname": "3", - }, - "{...}" - ]); - let expected = Data::json(exp); - let actual = json!([ - { - "name": "one", - "nickname": "1", - }, - { - "name": "two", - "nickname": "2", - }, - { - "name": "four", - "nickname": "4", - }, - { - "name": "five", - "nickname": "5", - } - ]); - let actual_normalized = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual.clone()), &expected); - if let DataInner::Json(act) = actual_normalized.inner { - assert_eq!(act, actual); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_bad_order() { - let exp = json!({ - "people": ["John", "Jane"] - }); - let expected = Data::json(exp); - let actual = json!({ - "people": ["Jane", "John"] - }); - let actual = NormalizeToExpected::new() - .redact() - .normalize(Data::json(actual), &expected); - if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { - assert_ne!(exp, act); - } -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_obj_key() { - let expected = json!({ - "[A]": "value-a", - "[B]": "value-b", - "[C]": "value-c", - }); - let expected = Data::json(expected); - let actual = json!({ - "key-a": "value-a", - "key-b": "value-b", - "key-c": "value-c", - }); - let actual = Data::json(actual); - let mut sub = Redactions::new(); - sub.insert("[A]", "key-a").unwrap(); - sub.insert("[B]", "key-b").unwrap(); - sub.insert("[C]", "key-c").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(actual, &expected); - - let expected_actual = json!({ - "[A]": "value-a", - "[B]": "value-b", - "[C]": "value-c", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_with_missing_obj_key() { - let expected = json!({ - "a": "[A]", - "b": "[B]", - "c": "[C]", - }); - let expected = Data::json(expected); - let actual = json!({ - "a": "value-a", - "c": "value-c", - }); - let actual = Data::json(actual); - let mut sub = Redactions::new(); - sub.insert("[A]", "value-a").unwrap(); - sub.insert("[B]", "value-b").unwrap(); - sub.insert("[C]", "value-c").unwrap(); - let actual = NormalizeToExpected::new() - .redact_with(&sub) - .normalize(actual, &expected); - - let expected_actual = json!({ - "a": "[A]", - "c": "[C]", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} - -#[test] -#[cfg(feature = "json")] -fn json_normalize_redactions_glob_obj_key() { - let expected = json!({ - "a": "value-a", - "c": "value-c", - "...": "{...}", - }); - let expected = Data::json(expected); - let actual = json!({ - "a": "value-a", - "b": "value-b", - "c": "value-c", - }); - let actual = Data::json(actual); - let actual = NormalizeToExpected::new() - .redact() - .normalize(actual, &expected); - - let expected_actual = json!({ - "a": "value-a", - "c": "value-c", - "...": "{...}", - }); - let expected_actual = Data::json(expected_actual); - assert_eq!(actual, expected_actual); -} diff --git a/crates/snapbox/src/filter/test_redactions.rs b/crates/snapbox/src/filter/test_redactions.rs new file mode 100644 index 00000000..2f75beca --- /dev/null +++ b/crates/snapbox/src/filter/test_redactions.rs @@ -0,0 +1,567 @@ +use std::path::PathBuf; + +#[cfg(feature = "json")] +use serde_json::json; + +use super::*; +use crate::prelude::*; + +#[test] +fn str_normalize_empty() { + let input = ""; + let pattern = ""; + let expected = ""; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_literals_match() { + let input = "Hello\nWorld"; + let pattern = "Hello\nWorld"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_pattern_shorter() { + let input = "Hello\nWorld"; + let pattern = "Hello\n"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_input_shorter() { + let input = "Hello\n"; + let pattern = "Hello\nWorld"; + let expected = "Hello\n"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_all_different() { + let input = "Hello\nWorld"; + let pattern = "Goodbye\nMoon"; + let expected = "Hello\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_middles_diverge() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\nMoon\nGoodbye"; + let expected = "Hello\nWorld\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_elide_delimited_with_sub() { + let input = "Hello World\nHow are you?\nGoodbye World"; + let pattern = "Hello [..]\n...\nGoodbye [..]"; + let expected = "Hello [..]\n...\nGoodbye [..]"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_leading_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "...\nGoodbye"; + let expected = "...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_trailing_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n..."; + let expected = "Hello\n..."; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_middle_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n...\nGoodbye"; + let expected = "Hello\n...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_post_elide_diverge() { + let input = "Hello\nSun\nAnd\nWorld"; + let pattern = "Hello\n...\nMoon"; + let expected = "Hello\nSun\nAnd\nWorld"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_post_diverge_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nMoon\nGoodbye\n..."; + let expected = "Hello\nWorld\nGoodbye\nSir"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_inline_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nW[..]d\nGoodbye\nSir"; + let expected = "Hello\nW[..]d\nGoodbye\nSir"; + let actual = NormalizeToExpected::new() + .redact() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_user_literal() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", "world").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_path() { + let input = "input: /home/epage"; + let pattern = "input: [HOME]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(sep).join("home").join("epage"); + sub.insert("[HOME]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_overlapping_path() { + let input = "\ +a: /home/epage +b: /home/epage/snapbox"; + let pattern = "\ +a: [A] +b: [B]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(&sep).join("home").join("epage"); + sub.insert("[A]", redacted).unwrap(); + let redacted = PathBuf::from(sep) + .join("home") + .join("epage") + .join("snapbox"); + sub.insert("[B]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_disabled() { + let input = "cargo"; + let pattern = "cargo[EXE]"; + let mut sub = Redactions::new(); + sub.insert("[EXE]", "").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "regex")] +fn str_normalize_user_regex_unnamed() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", regex::Regex::new("world").unwrap()) + .unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "regex")] +fn str_normalize_user_regex_named() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert( + "[OBJECT]", + regex::Regex::new("(?world)!").unwrap(), + ) + .unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_string() { + let exp = json!({"name": "{...}"}); + let expected = Data::json(exp); + let actual = json!({"name": "JohnDoe"}); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array() { + let exp = json!({"people": "{...}"}); + let expected = Data::json(exp); + let actual = json!({ + "people": [ + { + "name": "JohnDoe", + "nickname": "John", + } + ] + }); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_obj() { + let exp = json!({"people": "{...}"}); + let expected = Data::json(exp); + let actual = json!({ + "people": { + "name": "JohnDoe", + "nickname": "John", + } + }); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_array_start() { + let exp = json!({ + "people": [ + "{...}", + { + "name": "three", + "nickname": "3", + } + ] + }); + let expected = Data::json(exp); + let actual = json!({ + "people": [ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "three", + "nickname": "3", + } + ] + }); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array_start_end() { + let exp = json!([ + "{...}", + { + "name": "two", + "nickname": "2", + }, + "{...}" + ]); + let expected = Data::json(exp); + let actual = json!([ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "three", + "nickname": "3", + }, + { + "name": "four", + "nickname": "4", + } + ]); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array_middle_end() { + let exp = json!([ + { + "name": "one", + "nickname": "1", + }, + "{...}", + { + "name": "three", + "nickname": "3", + }, + "{...}" + ]); + let expected = Data::json(exp); + let actual = json!([ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "three", + "nickname": "3", + }, + { + "name": "four", + "nickname": "4", + }, + { + "name": "five", + "nickname": "5", + } + ]); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array_mismatch() { + let exp = json!([ + { + "name": "one", + "nickname": "1", + }, + "{...}", + { + "name": "three", + "nickname": "3", + }, + "{...}" + ]); + let expected = Data::json(exp); + let actual = json!([ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "four", + "nickname": "4", + }, + { + "name": "five", + "nickname": "5", + } + ]); + let actual_normalized = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual.clone()), &expected); + if let DataInner::Json(act) = actual_normalized.inner { + assert_eq!(act, actual); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_bad_order() { + let exp = json!({ + "people": ["John", "Jane"] + }); + let expected = Data::json(exp); + let actual = json!({ + "people": ["Jane", "John"] + }); + let actual = NormalizeToExpected::new() + .redact() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_ne!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_obj_key() { + let expected = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected = Data::json(expected); + let actual = json!({ + "key-a": "value-a", + "key-b": "value-b", + "key-c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "key-a").unwrap(); + sub.insert("[B]", "key-b").unwrap(); + sub.insert("[C]", "key-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); + + let expected_actual = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_with_missing_obj_key() { + let expected = json!({ + "a": "[A]", + "b": "[B]", + "c": "[C]", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "value-a").unwrap(); + sub.insert("[B]", "value-b").unwrap(); + sub.insert("[C]", "value-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "[A]", + "c": "[C]", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_obj_key() { + let expected = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "b": "value-b", + "c": "value-c", + }); + let actual = Data::json(actual); + let actual = NormalizeToExpected::new() + .redact() + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} From 4134e1a9260e81934116db909881cf7f24750701 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Wed, 22 May 2024 16:21:18 -0500 Subject: [PATCH 19/19] feat(filter): Unordered match support Fixes #151 --- crates/snapbox/src/assert/mod.rs | 10 +- crates/snapbox/src/data/filters.rs | 10 + crates/snapbox/src/data/mod.rs | 29 + crates/snapbox/src/filter/mod.rs | 2 + crates/snapbox/src/filter/pattern.rs | 320 ++++++++- crates/snapbox/src/filter/test_unordered.rs | 222 +++++++ .../src/filter/test_unordered_redactions.rs | 622 ++++++++++++++++++ 7 files changed, 1203 insertions(+), 12 deletions(-) create mode 100644 crates/snapbox/src/filter/test_unordered.rs create mode 100644 crates/snapbox/src/filter/test_unordered_redactions.rs diff --git a/crates/snapbox/src/assert/mod.rs b/crates/snapbox/src/assert/mod.rs index 42695c32..ce04d95f 100644 --- a/crates/snapbox/src/assert/mod.rs +++ b/crates/snapbox/src/assert/mod.rs @@ -127,11 +127,15 @@ impl Assert { if expected.filters.is_newlines_set() { actual = FilterNewlines.filter(actual); } + + let mut normalize = NormalizeToExpected::new(); if expected.filters.is_redaction_set() { - actual = NormalizeToExpected::new() - .redact_with(&self.substitutions) - .normalize(actual, &expected); + normalize = normalize.redact_with(&self.substitutions); + } + if expected.filters.is_unordered_set() { + normalize = normalize.unordered(); } + actual = normalize.normalize(actual, &expected); (actual, expected) } diff --git a/crates/snapbox/src/data/filters.rs b/crates/snapbox/src/data/filters.rs index 8b1e4ce1..5b0103b3 100644 --- a/crates/snapbox/src/data/filters.rs +++ b/crates/snapbox/src/data/filters.rs @@ -27,6 +27,11 @@ impl FilterSet { self } + pub(crate) fn unordered(mut self) -> Self { + self.set(Self::UNORDERED); + self + } + pub(crate) const fn is_redaction_set(&self) -> bool { self.is_set(Self::REDACTIONS) } @@ -38,12 +43,17 @@ impl FilterSet { pub(crate) const fn is_paths_set(&self) -> bool { self.is_set(Self::PATHS) } + + pub(crate) const fn is_unordered_set(&self) -> bool { + self.is_set(Self::UNORDERED) + } } impl FilterSet { const REDACTIONS: usize = 1 << 0; const NEWLINES: usize = 1 << 1; const PATHS: usize = 1 << 2; + const UNORDERED: usize = 1 << 3; fn set(&mut self, flag: usize) -> &mut Self { self.flags |= flag; diff --git a/crates/snapbox/src/data/mod.rs b/crates/snapbox/src/data/mod.rs index db8e5026..180c2c56 100644 --- a/crates/snapbox/src/data/mod.rs +++ b/crates/snapbox/src/data/mod.rs @@ -80,6 +80,29 @@ pub trait IntoData: Sized { self.into_data().raw() } + /// Treat lines and json arrays as unordered + /// + /// # Examples + /// + /// ```rust + /// # #[cfg(feature = "json")] { + /// use snapbox::prelude::*; + /// use snapbox::str; + /// use snapbox::assert_data_eq; + /// + /// let actual = str![[r#"["world", "hello"]"#]] + /// .is(snapbox::data::DataFormat::Json) + /// .unordered(); + /// let expected = str![[r#"["hello", "world"]"#]] + /// .is(snapbox::data::DataFormat::Json) + /// .unordered(); + /// assert_data_eq!(actual, expected); + /// # } + /// ``` + fn unordered(self) -> Data { + self.into_data().unordered() + } + /// Initialize as [`format`][DataFormat] or [`Error`][DataFormat::Error] /// /// This is generally used for `expected` data @@ -301,6 +324,12 @@ impl Data { self.filters = FilterSet::empty().newlines(); self } + + /// Treat lines and json arrays as unordered + pub fn unordered(mut self) -> Self { + self.filters = FilterSet::empty().unordered(); + self + } } /// # Assertion frameworks operations diff --git a/crates/snapbox/src/filter/mod.rs b/crates/snapbox/src/filter/mod.rs index 6217c40d..0d4f91d3 100644 --- a/crates/snapbox/src/filter/mod.rs +++ b/crates/snapbox/src/filter/mod.rs @@ -10,6 +10,8 @@ mod redactions; mod test; #[cfg(test)] mod test_redactions; +#[cfg(test)] +mod test_unordered_redactions; use crate::data::DataInner; use crate::Data; diff --git a/crates/snapbox/src/filter/pattern.rs b/crates/snapbox/src/filter/pattern.rs index c8fe33fd..17698336 100644 --- a/crates/snapbox/src/filter/pattern.rs +++ b/crates/snapbox/src/filter/pattern.rs @@ -5,15 +5,25 @@ use crate::Data; /// Adjust `actual` based on `expected` pub struct NormalizeToExpected<'a> { substitutions: Option<&'a crate::Redactions>, + unordered: bool, } impl<'a> NormalizeToExpected<'a> { pub fn new() -> Self { Self { substitutions: None, + unordered: false, } } + /// Make unordered content comparable + /// + /// This is done by re-ordering `actual` according to `expected`. + pub fn unordered(mut self) -> Self { + self.unordered = true; + self + } + /// Apply built-in redactions. /// /// Built-in redactions: @@ -40,14 +50,24 @@ impl<'a> NormalizeToExpected<'a> { } pub fn normalize(&self, actual: Data, expected: &Data) -> Data { - let Some(substitutions) = self.substitutions else { - return actual; + let actual = if let Some(substitutions) = self.substitutions { + NormalizeRedactions { + redactions: substitutions, + } + .filter(actual) + } else { + actual }; - let actual = NormalizeRedactions { - redactions: substitutions, + match (self.substitutions, self.unordered) { + (None, false) => actual, + (Some(substitutions), false) => { + normalize_data_to_redactions(actual, expected, substitutions) + } + (None, true) => normalize_data_to_unordered(actual, expected), + (Some(substitutions), true) => { + normalize_data_to_unordered_redactions(actual, expected, substitutions) + } } - .filter(actual); - normalize_data_to_redactions(actual, expected, substitutions) } } @@ -57,6 +77,291 @@ impl Default for NormalizeToExpected<'_> { } } +fn normalize_data_to_unordered(actual: Data, expected: &Data) -> Data { + let source = actual.source; + let filters = actual.filters; + let inner = match actual.inner { + DataInner::Error(err) => DataInner::Error(err), + DataInner::Binary(bin) => DataInner::Binary(bin), + DataInner::Text(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_unordered(&text, &pattern); + DataInner::Text(lines) + } else { + DataInner::Text(text) + } + } + #[cfg(feature = "json")] + DataInner::Json(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_unordered(&mut value, exp); + } + DataInner::Json(value) + } + #[cfg(feature = "json")] + DataInner::JsonLines(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_unordered(&mut value, exp); + } + DataInner::JsonLines(value) + } + #[cfg(feature = "term-svg")] + DataInner::TermSvg(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_unordered(&text, &pattern); + DataInner::TermSvg(lines) + } else { + DataInner::TermSvg(text) + } + } + }; + Data { + inner, + source, + filters, + } +} + +#[cfg(feature = "structured-data")] +fn normalize_value_to_unordered(actual: &mut serde_json::Value, expected: &serde_json::Value) { + use serde_json::Value::*; + + match (actual, expected) { + (String(act), String(exp)) => { + *act = normalize_str_to_unordered(act, exp); + } + (Array(act), Array(exp)) => { + let mut actual_values = std::mem::take(act); + let mut expected_values = exp.clone(); + expected_values.retain(|expected_value| { + let mut matched = false; + actual_values.retain(|actual_value| { + if !matched && actual_value == expected_value { + matched = true; + false + } else { + true + } + }); + if matched { + act.push(expected_value.clone()); + } + !matched + }); + for actual_value in actual_values { + act.push(actual_value); + } + } + (Object(act), Object(exp)) => { + for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { + if let Some(expected_value) = exp.get(&actual_key) { + normalize_value_to_unordered(&mut actual_value, expected_value) + } + act.insert(actual_key, actual_value); + } + } + (_, _) => {} + } +} + +fn normalize_str_to_unordered(actual: &str, expected: &str) -> String { + if actual == expected { + return actual.to_owned(); + } + + let mut normalized: Vec<&str> = Vec::new(); + let mut actual_lines: Vec<_> = crate::utils::LinesWithTerminator::new(actual).collect(); + let mut expected_lines: Vec<_> = crate::utils::LinesWithTerminator::new(expected).collect(); + expected_lines.retain(|expected_line| { + let mut matched = false; + actual_lines.retain(|actual_line| { + if !matched && actual_line == expected_line { + matched = true; + false + } else { + true + } + }); + if matched { + normalized.push(expected_line); + } + !matched + }); + for actual_line in &actual_lines { + normalized.push(actual_line); + } + + normalized.join("") +} + +#[cfg(feature = "structured-data")] +const KEY_WILDCARD: &str = "..."; +#[cfg(feature = "structured-data")] +const VALUE_WILDCARD: &str = "{...}"; + +fn normalize_data_to_unordered_redactions( + actual: Data, + expected: &Data, + substitutions: &crate::Redactions, +) -> Data { + let source = actual.source; + let filters = actual.filters; + let inner = match actual.inner { + DataInner::Error(err) => DataInner::Error(err), + DataInner::Binary(bin) => DataInner::Binary(bin), + DataInner::Text(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_unordered_redactions(&text, &pattern, substitutions); + DataInner::Text(lines) + } else { + DataInner::Text(text) + } + } + #[cfg(feature = "json")] + DataInner::Json(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_unordered_redactions(&mut value, exp, substitutions); + } + DataInner::Json(value) + } + #[cfg(feature = "json")] + DataInner::JsonLines(value) => { + let mut value = value; + if let DataInner::Json(exp) = &expected.inner { + normalize_value_to_unordered_redactions(&mut value, exp, substitutions); + } + DataInner::JsonLines(value) + } + #[cfg(feature = "term-svg")] + DataInner::TermSvg(text) => { + if let Some(pattern) = expected.render() { + let lines = normalize_str_to_unordered_redactions(&text, &pattern, substitutions); + DataInner::TermSvg(lines) + } else { + DataInner::TermSvg(text) + } + } + }; + Data { + inner, + source, + filters, + } +} + +#[cfg(feature = "structured-data")] +fn normalize_value_to_unordered_redactions( + actual: &mut serde_json::Value, + expected: &serde_json::Value, + substitutions: &crate::Redactions, +) { + use serde_json::Value::*; + + match (actual, expected) { + (act, String(exp)) if exp == VALUE_WILDCARD => { + *act = serde_json::json!(VALUE_WILDCARD); + } + (String(act), String(exp)) => { + *act = normalize_str_to_unordered_redactions(act, exp, substitutions); + } + (Array(act), Array(exp)) => { + let mut actual_values = std::mem::take(act); + let mut expected_values = exp.clone(); + let mut elided = false; + expected_values.retain(|expected_value| { + let mut matched = false; + if expected_value == VALUE_WILDCARD { + matched = true; + elided = true; + } else { + actual_values.retain(|actual_value| { + if !matched && actual_value == expected_value { + matched = true; + false + } else { + true + } + }); + } + if matched { + act.push(expected_value.clone()); + } + !matched + }); + if !elided { + for actual_value in actual_values { + act.push(actual_value); + } + } + } + (Object(act), Object(exp)) => { + let has_key_wildcard = + exp.get(KEY_WILDCARD).and_then(|v| v.as_str()) == Some(VALUE_WILDCARD); + for (actual_key, mut actual_value) in std::mem::replace(act, serde_json::Map::new()) { + if let Some(expected_value) = exp.get(&actual_key) { + normalize_value_to_unordered_redactions( + &mut actual_value, + expected_value, + substitutions, + ) + } else if has_key_wildcard { + continue; + } + act.insert(actual_key, actual_value); + } + if has_key_wildcard { + act.insert(KEY_WILDCARD.to_owned(), String(VALUE_WILDCARD.to_owned())); + } + } + (_, _) => {} + } +} + +fn normalize_str_to_unordered_redactions( + actual: &str, + expected: &str, + substitutions: &crate::Redactions, +) -> String { + if actual == expected { + return actual.to_owned(); + } + + let mut normalized: Vec<&str> = Vec::new(); + let mut actual_lines: Vec<_> = crate::utils::LinesWithTerminator::new(actual).collect(); + let mut expected_lines: Vec<_> = crate::utils::LinesWithTerminator::new(expected).collect(); + let mut elided = false; + expected_lines.retain(|expected_line| { + let mut matched = false; + if is_line_elide(expected_line) { + matched = true; + elided = true; + } else { + actual_lines.retain(|actual_line| { + if !matched && line_matches(actual_line, expected_line, substitutions) { + matched = true; + false + } else { + true + } + }); + } + if matched { + normalized.push(expected_line); + } + !matched + }); + if !elided { + for actual_line in &actual_lines { + normalized.push(actual_line); + } + } + + normalized.join("") +} + fn normalize_data_to_redactions( actual: Data, expected: &Data, @@ -116,9 +421,6 @@ fn normalize_value_to_redactions( ) { use serde_json::Value::*; - const KEY_WILDCARD: &str = "..."; - const VALUE_WILDCARD: &str = "{...}"; - match (actual, expected) { (act, String(exp)) if exp == VALUE_WILDCARD => { *act = serde_json::json!(VALUE_WILDCARD); diff --git a/crates/snapbox/src/filter/test_unordered.rs b/crates/snapbox/src/filter/test_unordered.rs new file mode 100644 index 00000000..6ecf2894 --- /dev/null +++ b/crates/snapbox/src/filter/test_unordered.rs @@ -0,0 +1,222 @@ +use std::path::PathBuf; + +#[cfg(feature = "json")] +use serde_json::json; + +use super::*; +use crate::prelude::*; + +#[test] +fn str_normalize_empty() { + let input = ""; + let pattern = ""; + let expected = ""; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_same_order() { + let input = "1 +2 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +2 +3 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_reverse_order() { + let input = "1 +2 +3 +"; + let pattern = "3 +2 +1 +"; + let expected = "3 +2 +1 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_actual_missing() { + let input = "1 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +3 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_expected_missing() { + let input = "1 +2 +3 +"; + let pattern = "1 +3 +"; + let expected = "1 +3 +2 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_actual_duplicated() { + let input = "1 +2 +2 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +2 +3 +2 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_expected_duplicated() { + let input = "1 +2 +3 +"; + let pattern = "1 +2 +2 +3 +"; + let expected = "1 +2 +3 +"; + let actual = NormalizeToExpected::new() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_empty() { + let input = json!([]); + let pattern = json!([]); + let expected = json!([]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_same_order() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 2, 3]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_reverse_order() { + let input = json!([1, 2, 3]); + let pattern = json!([3, 2, 1]); + let expected = json!([3, 2, 1]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_actual_missing() { + let input = json!([1, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 3]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_expected_missing() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 3]); + let expected = json!([1, 3, 2]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_actual_duplicated() { + let input = json!([1, 2, 2, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 2, 3, 2]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_expected_duplicated() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 2, 2, 3]); + let expected = json!([1, 2, 3]); + let actual = NormalizeToExpected::new() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} diff --git a/crates/snapbox/src/filter/test_unordered_redactions.rs b/crates/snapbox/src/filter/test_unordered_redactions.rs new file mode 100644 index 00000000..82b8a451 --- /dev/null +++ b/crates/snapbox/src/filter/test_unordered_redactions.rs @@ -0,0 +1,622 @@ +use std::path::PathBuf; + +#[cfg(feature = "json")] +use serde_json::json; + +use super::*; +use crate::prelude::*; + +#[test] +fn str_normalize_empty() { + let input = ""; + let pattern = ""; + let expected = ""; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_same_order() { + let input = "1 +2 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +2 +3 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_reverse_order() { + let input = "1 +2 +3 +"; + let pattern = "3 +2 +1 +"; + let expected = "3 +2 +1 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_actual_missing() { + let input = "1 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +3 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_expected_missing() { + let input = "1 +2 +3 +"; + let pattern = "1 +3 +"; + let expected = "1 +3 +2 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_actual_duplicated() { + let input = "1 +2 +2 +3 +"; + let pattern = "1 +2 +3 +"; + let expected = "1 +2 +3 +2 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_expected_duplicated() { + let input = "1 +2 +3 +"; + let pattern = "1 +2 +2 +3 +"; + let expected = "1 +2 +3 +"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into_data(), &pattern.into_data()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_elide_delimited_with_sub() { + let input = "Hello World\nHow are you?\nGoodbye World"; + let pattern = "Hello [..]\n...\nGoodbye [..]"; + let expected = "Hello [..]\n...\nGoodbye [..]"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_leading_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "...\nGoodbye"; + let expected = "...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_trailing_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n..."; + let expected = "Hello\n..."; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_middle_elide() { + let input = "Hello\nWorld\nGoodbye"; + let pattern = "Hello\n...\nGoodbye"; + let expected = "Hello\n...\nGoodbye"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_post_elide_diverge() { + let input = "Hello\nSun\nAnd\nWorld"; + let pattern = "Hello\n...\nMoon"; + let expected = "Hello\n...\n"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_post_diverge_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nMoon\nGoodbye\n..."; + let expected = "Hello\nGoodbye\n..."; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_inline_elide() { + let input = "Hello\nWorld\nGoodbye\nSir"; + let pattern = "Hello\nW[..]d\nGoodbye\nSir"; + let expected = "Hello\nW[..]d\nGoodbye\nSir"; + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, expected.into_data()); +} + +#[test] +fn str_normalize_user_literal() { + let input = "Hello world!"; + let pattern = "Hello [OBJECT]!"; + let mut sub = Redactions::new(); + sub.insert("[OBJECT]", "world").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_path() { + let input = "input: /home/epage"; + let pattern = "input: [HOME]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(sep).join("home").join("epage"); + sub.insert("[HOME]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_overlapping_path() { + let input = "\ +a: /home/epage +b: /home/epage/snapbox"; + let pattern = "\ +a: [A] +b: [B]"; + let mut sub = Redactions::new(); + let sep = std::path::MAIN_SEPARATOR.to_string(); + let redacted = PathBuf::from(&sep).join("home").join("epage"); + sub.insert("[A]", redacted).unwrap(); + let redacted = PathBuf::from(sep) + .join("home") + .join("epage") + .join("snapbox"); + sub.insert("[B]", redacted).unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +fn str_normalize_user_disabled() { + let input = "cargo"; + let pattern = "cargo[EXE]"; + let mut sub = Redactions::new(); + sub.insert("[EXE]", "").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(input.into(), &pattern.into()); + assert_eq!(actual, pattern.into_data()); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_empty() { + let input = json!([]); + let pattern = json!([]); + let expected = json!([]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_same_order() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 2, 3]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_reverse_order() { + let input = json!([1, 2, 3]); + let pattern = json!([3, 2, 1]); + let expected = json!([3, 2, 1]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_actual_missing() { + let input = json!([1, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 3]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_expected_missing() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 3]); + let expected = json!([1, 3, 2]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_actual_duplicated() { + let input = json!([1, 2, 2, 3]); + let pattern = json!([1, 2, 3]); + let expected = json!([1, 2, 3, 2]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_expected_duplicated() { + let input = json!([1, 2, 3]); + let pattern = json!([1, 2, 2, 3]); + let expected = json!([1, 2, 3]); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(input), &Data::json(pattern)); + assert_eq!(actual, Data::json(expected)); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_string() { + let exp = json!({"name": "{...}"}); + let expected = Data::json(exp); + let actual = json!({"name": "JohnDoe"}); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array() { + let exp = json!({"people": "{...}"}); + let expected = Data::json(exp); + let actual = json!({ + "people": [ + { + "name": "JohnDoe", + "nickname": "John", + } + ] + }); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_obj() { + let exp = json!({"people": "{...}"}); + let expected = Data::json(exp); + let actual = json!({ + "people": { + "name": "JohnDoe", + "nickname": "John", + } + }); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_array_start() { + let exp = json!({ + "people": [ + "{...}", + { + "name": "three", + "nickname": "3", + } + ] + }); + let expected = Data::json(exp); + let actual = json!({ + "people": [ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "three", + "nickname": "3", + } + ] + }); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(actual), &expected); + if let (DataInner::Json(exp), DataInner::Json(act)) = (expected.inner, actual.inner) { + assert_eq!(exp, act); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_for_array_mismatch() { + let exp = json!([ + { + "name": "one", + "nickname": "1", + }, + { + "name": "three", + "nickname": "3", + }, + "{...}" + ]); + let expected = Data::json(exp); + let actual = json!([ + { + "name": "one", + "nickname": "1", + }, + { + "name": "two", + "nickname": "2", + }, + { + "name": "four", + "nickname": "4", + }, + { + "name": "five", + "nickname": "5", + } + ]); + let expected_actual = json!([ + { + "name": "one", + "nickname": "1", + }, + "{...}" + ]); + let actual_normalized = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(Data::json(actual.clone()), &expected); + if let DataInner::Json(act) = actual_normalized.inner { + assert_eq!(act, expected_actual); + } +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_obj_key() { + let expected = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected = Data::json(expected); + let actual = json!({ + "key-a": "value-a", + "key-b": "value-b", + "key-c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "key-a").unwrap(); + sub.insert("[B]", "key-b").unwrap(); + sub.insert("[C]", "key-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(actual, &expected); + + let expected_actual = json!({ + "[A]": "value-a", + "[B]": "value-b", + "[C]": "value-c", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_with_missing_obj_key() { + let expected = json!({ + "a": "[A]", + "b": "[B]", + "c": "[C]", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "c": "value-c", + }); + let actual = Data::json(actual); + let mut sub = Redactions::new(); + sub.insert("[A]", "value-a").unwrap(); + sub.insert("[B]", "value-b").unwrap(); + sub.insert("[C]", "value-c").unwrap(); + let actual = NormalizeToExpected::new() + .redact_with(&sub) + .unordered() + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "[A]", + "c": "[C]", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +} + +#[test] +#[cfg(feature = "json")] +fn json_normalize_glob_obj_key() { + let expected = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected = Data::json(expected); + let actual = json!({ + "a": "value-a", + "b": "value-b", + "c": "value-c", + }); + let actual = Data::json(actual); + let actual = NormalizeToExpected::new() + .redact() + .unordered() + .normalize(actual, &expected); + + let expected_actual = json!({ + "a": "value-a", + "c": "value-c", + "...": "{...}", + }); + let expected_actual = Data::json(expected_actual); + assert_eq!(actual, expected_actual); +}