Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Automatically use StrComparison for comparing strings #92

Merged
merged 3 commits into from
Mar 11, 2022

Conversation

dtolnay
Copy link
Contributor

@dtolnay dtolnay commented Mar 9, 2022

This PR makes assert_eq!(left, right) automatically behave like assert_str_eq!(left, right) when both sides are str/String.

We could alternatively do this only if one or both strings contains \n if you prefer, as that's when the old behavior of assert_eq! is definitely worse; #24.

Copy link
Collaborator

@tommilligan tommilligan left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi, thanks for using and contributing to the project!

This certainly makes the multiline string diffing case more ergonomic for the user. I have a couple of thoughts surrounding this though:

  • Could you expand on your use case a bit? i.e. where it's not suitable to simply use pretty_assertions::assert_str_eq? I can think of a couple, but interested to know what you're actually running into in the wild.
  • In this PR there's no way to recover the existing behaviour for the user, if they desire it - the new format can't be opted out of. We could fix this by introducing a new macro assert_debug_eq, which is an exact copy of the existing assert_eq. It would then be the job of assert_eq to pick which representation to use, by some internal heuristic.
  • This is a breaking change in the output format, so it's questionable whether this necessitates a v2 release (open to opinions about this)
  • This moves the crate away from the current behaviour of using the Debug representation in all cases. This isn't something I'm opposed to, but I'd like a semver strategy so we can say, add a specific output for Vec<u8> in future, without needing major version bumps each time.
  • (edit): You mentioned possibly introspecting the strings to see if they contain \n chars - I don't think this is a good idea, so very happy without it

I think overall, I need to write up some guarantees for the crate around semver, and decide whether that includes exact diff output. I'll take a look around at similar dev-tooling style crates and see what guarantees they provide.

pretty_assertions/src/lib.rs Show resolved Hide resolved
"#)]
fn fails_str() {
::pretty_assertions::assert_eq!("foo\nbar", "foo\nbaz");
}
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for adding these - I also added a couple more tests to cover the String/AsRef cases. Could you push these onto your branch, or give me access to do so?

commit 872b7233da10a4c69ca5cc1fd744e95fbb5af3aa
Author: Tom Milligan <tom@reinfer.io>
Date:   Wed Mar 9 07:55:13 2022 +0000

    [review] add more macro tests

diff --git a/pretty_assertions/tests/macros.rs b/pretty_assertions/tests/macros.rs
index bbbe8de..fbb5e84 100644
--- a/pretty_assertions/tests/macros.rs
+++ b/pretty_assertions/tests/macros.rs
@@ -32,23 +32,23 @@ mod assert_str_eq {
         ::pretty_assertions::assert_str_eq!(s0, s1);
     }
 
-    #[test]
-    fn passes_as_ref_types() {
-        #[derive(PartialEq)]
-        struct MyString(String);
+    #[derive(PartialEq)]
+    struct MyString(String);
 
-        impl AsRef<str> for MyString {
-            fn as_ref(&self) -> &str {
-                &self.0
-            }
+    impl AsRef<str> for MyString {
+        fn as_ref(&self) -> &str {
+            &self.0
         }
+    }
 
-        impl PartialEq<String> for MyString {
-            fn eq(&self, other: &String) -> bool {
-                &self.0 == other
-            }
+    impl PartialEq<String> for MyString {
+        fn eq(&self, other: &String) -> bool {
+            &self.0 == other
         }
+    }
 
+    #[test]
+    fn passes_as_ref_types() {
         let s0 = MyString("foo".to_string());
         let s1 = "foo".to_string();
         ::pretty_assertions::assert_str_eq!(s0, s1);
@@ -62,10 +62,38 @@ mod assert_str_eq {
 �[31m<ba�[0m�[1;48;5;52;31mr�[0m
 �[32m>ba�[0m�[1;48;5;22;32mz�[0m
 
+"#)]
+    fn fails_as_ref_types() {
+        let s0 = MyString("foo\nbar".to_string());
+        let s1 = "foo\nbaz".to_string();
+        ::pretty_assertions::assert_str_eq!(s0, s1);
+    }
+
+    #[test]
+    #[should_panic(expected = r#"assertion failed: `(left == right)`
+
+�[1mDiff�[0m �[31m< left�[0m / �[32mright >�[0m :
+ foo
+�[31m<ba�[0m�[1;48;5;52;31mr�[0m
+�[32m>ba�[0m�[1;48;5;22;32mz�[0m
+
 "#)]
     fn fails_foo() {
         ::pretty_assertions::assert_str_eq!("foo\nbar", "foo\nbaz");
     }
+
+    #[test]
+    #[should_panic(expected = r#"assertion failed: `(left == right)`
+
+�[1mDiff�[0m �[31m< left�[0m / �[32mright >�[0m :
+ foo
+�[31m<ba�[0m�[1;48;5;52;31mr�[0m
+�[32m>ba�[0m�[1;48;5;22;32mz�[0m
+
+"#)]
+    fn fails_string() {
+        ::pretty_assertions::assert_eq!("foo\nbar".to_string(), "foo\nbaz".to_string());
+    }
 }
 
 #[allow(clippy::eq_op)]

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great—I've committed this change (except I moved your new fails_string from mod assert_str_eq to mod assert_eq, since it is testing assert_eq!).

@dtolnay
Copy link
Contributor Author

dtolnay commented Mar 9, 2022

Could you expand on your use case a bit? i.e. where it's not suitable to simply use pretty_assertions::assert_str_eq? I can think of a couple, but interested to know what you're actually running into in the wild.

I don't use this crate, but I was reviewing someone else's code where they started replacing pretty_assertions::assert_eq -> assert_str_eq and it seemed like a waste of time, compared to producing the most helpful message possible whenever any of this crate's macros is used.

In this PR there's no way to recover the existing behaviour for the user, if they desire it - the new format can't be opted out of. We could fix this by introducing a new macro assert_debug_eq, which is an exact copy of the existing assert_eq. It would then be the job of assert_eq to pick which representation to use, by some internal heuristic.

That's up to whether you feel enough of your users would want that to justify building it into this crate's API. From the use case that I am familiar with, I am skeptical that it would be useful.

This is a breaking change in the output format, so it's questionable whether this necessitates a v2 release (open to opinions about this)

This sounds like an extreme stance. I don't understand why it would be valuable to your users that the output remain bitwise identical. Rustc definitely doesn't guarantee that for its diagnostics, and the standard library does not guarantee that for its Debug impls, so it's already not the case for uses of pretty-assertions in general.

@dtolnay
Copy link
Contributor Author

dtolnay commented Mar 9, 2022

@jkeljo
Copy link

jkeljo commented Mar 10, 2022

Could you expand on your use case a bit? i.e. where it's not suitable to simply use pretty_assertions::assert_str_eq? I can think of a couple, but interested to know what you're actually running into in the wild.

I don't use this crate, but I was reviewing someone else's code where they started replacing pretty_assertions::assert_eq -> assert_str_eq and it seemed like a waste of time, compared to producing the most helpful message possible whenever any of this crate's macros is used.

I'm the one whose code he was reviewing ;), and I agree that the assert_eq/assert_str_eq split is problematic. If I'm looking at an assert_eq! invocation, unless there is a string literal or a format! call in one of its arguments, it's not immediately obvious whether assert_str_eq! is allowed. So I migrated as much as I could quickly ascertain was possible, but I expected that I'd probably end up migrating more later when I encountered less-than-readable test failures. To be honest, as a user I was surprised when I started using assert_eq! with multiline strings and it didn't already behave this way, given how it behaves for other types.

In this PR there's no way to recover the existing behaviour for the user, if they desire it - the new format can't be opted out of. We could fix this by introducing a new macro assert_debug_eq, which is an exact copy of the existing assert_eq. It would then be the job of assert_eq to pick which representation to use, by some internal heuristic.

That's up to whether you feel enough of your users would want that to justify building it into this crate's API. From the use case that I am familiar with, I am skeptical that it would be useful.

I, too, have a hard time imagining a case in which I would rather see a bunch of \ns in the output, for the same reason that the default comparison in this crate uses {:#?} instead of {:?}; it's just way easier to read and understand a multi-line diff.

This is a breaking change in the output format, so it's questionable whether this necessitates a v2 release (open to opinions about this)

This sounds like an extreme stance. I don't understand why it would be valuable to your users that the output remain bitwise identical. Rustc definitely doesn't guarantee that for its diagnostics, and the standard library does not guarantee that for its Debug impls, so it's already not the case for uses of pretty-assertions in general.

I concur. From what I understand, the primary (only?) consumer of this output is the human who is running the tests, so this is more like a UI tweak than a breaking change. A breaking change would be if a given usage of assert_eq! would panic in one version and not in the other, or would no longer compile, or something like that.

@tommilligan
Copy link
Collaborator

Thanks both for your comments - I agree that I can't think of a good case for preserving the existing behaviour over a more readable one. I find @jkeljo's point about {:#?} and @dtolnay's point about Debug impls not being guaranteed persuasive, so I'm happy to merge this and cut a minor release.

I will also update the readme with some notes about the guarantees this crate provides about the output changing moving forwards between versions.

@tommilligan tommilligan merged commit 8a223f3 into rust-pretty-assertions:main Mar 11, 2022
@tommilligan
Copy link
Collaborator

Released in v1.2.0

@dtolnay dtolnay deleted the str branch March 11, 2022 12:57
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants