-
-
Notifications
You must be signed in to change notification settings - Fork 38
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
should Jiff use something other than a "One True God" error type? #8
Comments
I ended up going with the one god error type pattern. Once I did, I found that an enormous mental burden had been lifted, and composing high level APIs had a lot less friction. I was constant chaffing every time I composed APIs before. I find this to be an immense benefit, and it will probably take a lot of convincing for me to roll this back. |
Another benefit I found to have a god error type is that my error messages have substantially improved. A Jiff |
I'm pretty solidly in agreement with you here. As long as there's a straightforward way to check for specific errors for special handling in certain cases, the overwhelming majority of the time people are just going to use What I haven't seen a good solution for is ease of documenting specific errors that can be returned in these situations. For things like kernel and libc APIs it is often useful to know the full set of errors, and which specific errors map to specific failure modes. I haven't looked through the jiff APIs yet so I don't know if that is actually a problem that is relevant here. |
Yeah, I don't have a good sense of whether folks will even care about the class of error. If I had to guess, it might be useful to distinguish between "overflow" and "something in the input is not supported." In any case, I'm happy to add more introspection capabilities to |
A possible use case: In rust-postgres #1164, I handled overflow here. With the current variants, I only encounter If |
@allan2 Can you say more? Why not return the jiff |
Is there a reason why you don't just use |
Because I didn't think it was worth the dependency in this context. |
I ended up making Jiff work in a no-alloc context by almost completely sacrificing the quality of error messages. Serde itself does the same. In any case, I've overall been extremely happy with Jiff's "God" |
When I initially started building this crate, my intent was to try and avoid the "one true god error type" pattern that I've seen a lot of folks complain about. The "one true god error type" pattern occurs when a crate defines approximately one main
Error
type, and then uses that error type for approximately every fallible operation in the crate. This is done even when, internally, only a subset of error cases are possible for any given operation.One need not look far to find canonical examples of the "one true god error type" pattern. For example,
std::io::Error
is one such manifestation.As hinted at, the main drawback of this pattern is that you get less information in the type system about what kinds of errors can occur for any given operation. I basically agree that this is a drawback, but I don't think it exists in a vacuum. Namely, there are benefits to omitting specificity:
RoundInputError
, the routine might now also want to return aRangeError
. If you're using a god error type that subsumes all error types, it is not definitively a breaking change. But if your rounding function was previously returningRoundInputError
because you tried to be really careful about your error types, it is definitively a breaking change to change the error type to something else. You're only way out is with a semver incompatible release, or somehow hackRoundInputError
to be capable of containing aRangeError
. (Which is probably possible because error types tend to be opaque.)A
,B
andC
. And now you want to write a fallible operation that can only fail viaA
andB
. And maybe another fallible op can only fail viaA
andC
. And yet another fallible op can only fail viaB
andC
. In other words, in the worst and more precise case, you need a distinct error type for every possible combination of error types. Since Rust lacks anonymous union types, this is a pretty difficult pill to swallow.I also think it's worth questioning just how useful it is to know the universe of possible error cases for any given fallible operation. Usually all you need is the ability to check whether a specific error case occurred, as opposed to needing knowledge about what's possible. For example,
std::io::Error
enables this viastd::io::ErrorKind
. I think I can do the same in Jiff, but might start without it to see how far we can get. With that said, I have tried very precise errors in the past, and my sense of things is that nobody really cares. Most of the time, for errors, you just want to print them and/or combine them with other errors, and maybe in some rare cases, inspect what "class" of error it is.So when I initially started building this crate, I tried hard to assign specific error types to fallible operations. A lot of the lower level date arithmetic routines, for example, are either infallible or can only fail as a result of a
RangeError
. But as I started building higher level abstractions, the error types got more annoying, as it often felt entirely arbitrary whether a routine returned anError
(that is, any possible error type) or a more specific error type. It felt like I had implementation details leaking out into the API. And in particular, I had started convertingRangeError
s intoString
values in order to combine them withParseError
s because my parsing functions all returnedParseError
and not a god error type.On top of that, I often want to combine multiple error together.
So I think because of that, where I'm leaning is creating my own simplified
anyhow::Error
, but limited to the error types in Jiff. Internally it will create a chain, but will only expose a single error in order to make formatting ofBox<dyn std::error::Error>
work like one expects. And then I'll basically move all routines over to returning just ajiff::Error
. And this simplifies a lot.(Another reason why I didn't initially start with a chainable god error type was because I wanted Jiff to be usable in a
no-{alloc,std}
context, butno-alloc
is an incredibly annoying constraint. I'm not sure that it's worth it. I think that if folks need no-alloc time handling, we should handle that on a case-by-case basis and possible build some newjiff-core
crate.)The text was updated successfully, but these errors were encountered: