-
Notifications
You must be signed in to change notification settings - Fork 1.4k
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
Snapshot recap #1275
Comments
@novemberborn Thanks for a detailed writeup, very helpful! I think I'd want to go with a new solution to snapshot testing and try to find a better way to serialize snapshot values. We need to look for a library that handles serializing/deserializing of data and which also includes types that JSON ignores. Any tips are welcome! I think I've seen some time ago something like an extended JSON serializer, which handled most of its limitations. Need to look for it. The reasons why I'd go without
I don't think there's any possibility for that to work with any serialization/deserialization method, is there? We just can't "recreate" non-native (Map, Set, etc - language features) instances. As for key order, I'm not even sure we'd want to check that. In what cases key order of objects/props (in React components) affects the output/functionality? I can't think of any at this moment, because I've never encountered such a problem. |
According to
So yes there are dependencies but it's not our worst offender.
Yes, but what should be the short-term priority?
Nah, you'd just go "the snapshot says it was a Foo, and this object also says it's a Foo, so they match".
|
This isn't entirely accurate. The diffs are done using formatted objects. I have a longer term plan that harmonizes |
Well, we've built our own, which has landed in #1341. A new release will be out soon. |
A summary of snapshot related issues. See #1218, #1220, #1254 and #1223.
Fundamentally,
jest-snapshot
is string based. The snapshots contain human-readable representations of expected values, which are then compared against another representation of the actual value when tests are run.Snapshot testing is useful when code generates complex tree structures that are tedious to verify by writing assertions, and that may change significantly as part of day-to-day development. Humans manually verify that the snapshot is as expected the first time it's generated. To do this, the snapshot needs to be readable. Then, when the actual value changes and the snapshot is regenerated, source control diffing makes it easy to see what's actually changed and if that is as expected.
The original snapshot PR (#1113) used
jest-snapshot
as-is (thanks @lithin!). Here,jest-snapshot
was also responsible for the error message:Soon after this landed @vadimdemedes started working on improving our error output, specifically showing diffs for
t.deepEqual()
andt.is()
assertions (#1154). This work also includedt.snapshot()
. However, to generate these diffs we need actual objects, not the error message thatjest-snapshot
returns. (Plus, thatexpect(value).toMatchSnapshot()
line is a bit jarring.)The solution was to send a JSON string into
jest-snapshot
, with a workaround for React trees. This means though that the snapshot contains a fairly unreadable JSON string (#1254). Also, whereasjest-snapshot
normalizes key order inObject
objects, AVA can no longer rely on that because we're now just comparing JSON strings. Then when we generate a diff it comes up empty since our diffing logic does not care about key order (#1220).@vadimdemedes started a PR (#1223) to replace
jest-snapshot
with our own implementation, usingt.deepEqual()
for the comparisons. The advantage of this approach is that diffs look the same for all our (diffing) assertions. Unfortunately, because it uses JSON serialization it limits the kinds of object structures we can safely snapshot:undefined
,Symbol()
and function property values are droppedundefined
,Symbol()
and function array items are replaced bynull
{}
Map
andSet
toJSON()
for comparisons, which we don't do fort.deepEqual()
We need to decide whether we want error output for
t.snapshot()
to be the same as that for an equivalent error fromt.deepEqual()
. I don't think we can do this while still usingjest-snapshot
without hurting snapshot readability.The current JSON proposal severely limits the kinds of snapshots that can be created. This should only be a short-term solution, but as this issue establishes we have some critical issues with our current snapshot implementation right now, so I'd rather we ship something that works real soon. We can then expand the functionality going forward.
If we go this route we should enforce that snapshots cannot contain any values incompatible with JSON serialization. We can do this when adding a snapshot by comparing the snapshot against the value being added.
On the other hand using
jest-snapshot
provides more immediate user value. We could go that route and try and improve the diff output later. At that point though a reduction in functionality would not be acceptable.If you've made is this far I thank you for your dedication. Please respond with your feedback 😄
The text was updated successfully, but these errors were encountered: