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

Improve error information for serde errors #394

Merged
merged 5 commits into from
Aug 15, 2022

Conversation

juntyr
Copy link
Member

@juntyr juntyr commented Jul 24, 2022

Followup to #393

Adds explicit variants in ron::Error for serde errors from serde::de::Error (serde::ser::Error only has the Error::custom method).

  • I've included my change in CHANGELOG.md

@juntyr juntyr requested a review from torkleyy July 24, 2022 18:27
@juntyr
Copy link
Member Author

juntyr commented Jul 24, 2022

?r @torkleyy

@juntyr
Copy link
Member Author

juntyr commented Jul 24, 2022

This is the first proof-of-concept. Notably, I've also ported serde's Unexpected enum, which I'm not quite sure about. It does give us slightly nicer error messages and would allow users to match on the unexpected value (though I'm not quite sure if anyone would need that). However, it also adds some very serde-specific terms to our codebase, which don't map quite so well to ron. We could decide to just remove that part and stringify the unexpected values as serde does, or we could limit the enum to entries that make sense for us (we'd still adopt serde's decision to only store [u|i]64).

@codecov-commenter
Copy link

codecov-commenter commented Jul 24, 2022

Codecov Report

Merging #394 (021eb4c) into master (f7e8417) will decrease coverage by 3.09%.
The diff coverage is 76.69%.

@@            Coverage Diff             @@
##           master     #394      +/-   ##
==========================================
- Coverage   89.65%   86.55%   -3.10%     
==========================================
  Files          48       49       +1     
  Lines        5228     5803     +575     
==========================================
+ Hits         4687     5023     +336     
- Misses        541      780     +239     
Impacted Files Coverage Δ
src/error.rs 30.46% <43.35%> (+22.57%) ⬆️
tests/203_error_positions.rs 86.02% <86.44%> (-13.98%) ⬇️
src/de/mod.rs 74.46% <93.22%> (-15.40%) ⬇️
src/parse.rs 93.76% <100.00%> (ø)
tests/359_deserialize_seed.rs 100.00% <100.00%> (ø)
tests/393_serde_errors.rs 100.00% <100.00%> (ø)
tests/roundtrip.rs 100.00% <100.00%> (ø)

Help us with your feedback. Take ten seconds to tell us how you rate us. Have a feature suggestion? Share it here.

@juntyr
Copy link
Member Author

juntyr commented Jul 24, 2022

What I haven't touched yet is the area of adding more contextual information to those new errors. Something we might be able to do is to decorate the deserialize_enum and deserialize_struct-like methods to check for the new errors and insert struct/enum name information.

We could probably also try to find the name of the current field, though in my opinion, that would be overkill and is already handled by crates like serde_path_to_error.

@juntyr
Copy link
Member Author

juntyr commented Jul 24, 2022

I've tried to do the struct/enum/variant name extraction for even better error messages. Just extracting struct/enum names is easy, and we can store them as &'static str. For variant names, however, which are important for struct variants which to the ron-writing user look just like structs, I had to add a small hack. The Deserializer now remembers what the last deserialized identifier was. Since enums in ron use identifiers for the variants, I can check for that last one and assume it was the variant id. This does provide better errors (for externally tagged enums, internally tagged ones cannot get any context since they are based on deserialize_any, adjacently tagged and untagged enums get ok-ish errors), but means that the struct/enum/variant name is stored as a String.

@juntyr
Copy link
Member Author

juntyr commented Jul 24, 2022

@torkleyy @d86leader What are your thoughts?

src/error.rs Outdated Show resolved Hide resolved
Copy link
Contributor

@torkleyy torkleyy left a comment

Choose a reason for hiding this comment

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

Looks good to me! I'm not entirely sure if we should expose provide api guarantees for all of our erroor types. serde_json doesn't expose much in that regard.

@torkleyy
Copy link
Contributor

torkleyy commented Jul 27, 2022

I will have to give your comments a closer look later this week, I'm busy right now, sorry. But feel free to proceed without me.
I think workarounds are fine if they don't overly complicate the codebase and provide useful information, but I'd be cautious of exposing those details via an API because there's a higher chance we'll change them in the future. In general I think we shouldn't expose the full error enum as we do know.

@torkleyy torkleyy mentioned this pull request Aug 14, 2022
7 tasks
@juntyr
Copy link
Member Author

juntyr commented Aug 15, 2022

@torkleyy I've opted to simplify the error enum a bit by merging serde's invalid_type and invalid_value into the same InvalidValueForType type in ron. Furthermore, this error now contains the stringified version of found field instead of replicating serde's de::Unexpected type (however, I still implement some custom formatting for the conversion to get slightly nicer error messages). This now seems like a good compromise to me where we get some extra info from serde's useful errors like the missing field or unknown variant ones without having to also include its expected vs unexpected system.

@juntyr juntyr force-pushed the 393-better-serde-errors branch from 8838cad to 6668336 Compare August 15, 2022 05:29
@juntyr juntyr marked this pull request as ready for review August 15, 2022 05:32
@juntyr
Copy link
Member Author

juntyr commented Aug 15, 2022

CI is blocked on serde-rs/serde#2255

@torkleyy
Copy link
Contributor

Since we're doing a API-breaking release, I think we should use the opportunity to bump MSRV and get rid of the CI problem at the same time :)

@juntyr
Copy link
Member Author

juntyr commented Aug 15, 2022

Since we're doing a API-breaking release, I think we should use the opportunity to bump MSRV and get rid of the CI problem at the same time :)

@torkleyy I've opened #396 to bump the MSRV to 1.56. Once that PR is merged I'll rebase this one off it.

@juntyr juntyr force-pushed the 393-better-serde-errors branch from 6668336 to 021eb4c Compare August 15, 2022 14:53
@juntyr
Copy link
Member Author

juntyr commented Aug 15, 2022

@torkleyy This PR is now ready to be merged if you're happy with the changes :)

Copy link
Contributor

@torkleyy torkleyy left a comment

Choose a reason for hiding this comment

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

Looks great, can't wait to use ron with such nice errors :)

@torkleyy torkleyy merged commit 172bdd6 into ron-rs:master Aug 15, 2022
@juntyr juntyr deleted the 393-better-serde-errors branch August 15, 2022 15:51
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.

4 participants