-
Notifications
You must be signed in to change notification settings - Fork 65
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
How should Aff users use exceptions and "error channel"? #136
Comments
The primary reason For control-flow exceptions, I would currently advise one to use I do appreciate the desire for a one-stop-shop sort of thing for control-flow. I would love to make One possibility is to not allow you to recover from underlying We could also potentially use something like |
Ooooh, I totally forgot that try-catch doesn't work for JavaScript's callback-style async. That's a missing piece to the puzzle. https://www.joyent.com/node-js/production/design/errors So the reason an Eff turned into an Aff must be wrapped in try-catch is to convert from synchronous exception-handling (try-catch) to asynchronous exception-handling (error in callback). Thanks for the great answer, @natefaubion - it's most of the info I was missing. I was a bit sad that I'll just handle that error at the lowest level, when the Aff enters my app, and change it into a better-typed error, if applicable. Can do that using |
btw if |
Here's how I am solving these problems in Scalaz 8 IO, but translated into PureScript and patterned after data IO e a -- error type and value
-- Import async code:
makeIO :: forall a. (((Either Error a) -> Eff Unit) -> Eff Unit) -> IO Error a
-- Import sync code:
liftEff :: forall a. Eff a -> IO Error a
-- Catch exception:
attempt :: forall e1 e2 a. IO e1 a -> IO e2 (Either e1 a)
-- Throw exception:
fail :: forall e a. e -> IO e a A few notes:
|
So, it seems to me that idiomatic async JavaScript libs and PureScript libs use the "error channel" differently:
As-it-is, I think PureScript apps best use In light of that, I feel like this Aff library needs a function for converting from JS-idiomatic async errors to PS-idiomatic async errors. Something like these: -- Similar to `makeAff`, but converts from JS-idiomatic error handling
-- to PS-idiomatic error handling.
makeAffFromJS ::
forall eff a.
((Either Error a -> Eff eff Unit) -> Eff eff (Canceler eff))
-> Aff eff (Either Error a)
makeAffFromJS = attempt <<< makeAff
makeExceptTAffFromJS ::
forall eff a.
((Either Error a -> Eff eff Unit) -> Eff eff (Canceler eff))
-> ExceptT Aff eff (Either Error a)
makeExceptTAffFromJS = ExceptT <<< attempt <<< makeAff With these available and the Aff lib educating its users to rarely use the error channel and be sure to convert from JS-idiomatic style to PS-idiomatic style when wrapping JS libs, I will be more likely to use Aff in a "safe" way, that is, not letting me forget to catch JS-thrown errors in my app. |
Initially I wanted something like Is it useful to recover from I definitely think this is something we should plan for given that we are going to have a breaking release anyway when we lose the effect indexing. |
In the same way, if an exception occurs in |
Have you considered this? liftEff:: (Error -> e) -> Eff a -> Aff e a So the first function will be used when error is raised in eff |
That’s no different than lmap. Why not just use catchException to lift it If you want it to be caught? |
A mapping from Error to e isn’t that useful in general because it doesn’t carry much useful information. I can’t think of a case in PS where I’ve tried to introspect a native Error type. |
Although much of this post follows from GHC’s implementation of IO and exceptions and therefore isn’t really relevant to us, I think there’s also a decent amount which is: https://www.fpcomplete.com/blog/2016/11/exceptions-best-practices-haskell. In particular, this post argues (and I agree) that GHC’s point in the design space — similar to the current Aff — where there is one fixed exception type (kinda) is actually a good one, since pretty much any IO action can fail for one reason or another, and that having to find a way of unifying all the different types of errors from all the libraries you’re using would quickly become very tedious. Personally I like how it is now - I think the current design makes a lot of sense from both the perspectives of an application using the FFI a lot, ie in terms of interop, as well as that of a pure PS application which doesn’t take much if any notice of the JS ecosystem. As a data point Pulp does make quite a bit of use of Aff’s MonadError instance, both to handle exceptions thrown by JS code, eg reading files which might turn out not to exist (for which I needed to introspect an |
After reading your answer and discussion in Slack, it seems I still don't understand how the "error channel" is meant to be used and why you're leaning towards making it the right place to put "business decided" error values rather than only truly exceptional events (interrupts?). Still unclear to me why we can't separate the "business errors" channel out of Aff and into something like ExceptT. Anyways, this issue was presuming there'd be a simple answer/recommendation, but it's not the case. I hope someone can figure it out and contribute docs to better specify these things and recommend how to do it. Until then, I'll do my best to never need to use the error channel, or just work on a "SimpleAff" :). I heard that you'd like to make a proposal related to changing/better specifying Aff's error channel, so I'll wait for that to happen to discuss further. Thanks for your time, @natefaubion! |
I see that
Aff
will implicitly try-catch some exceptions, later surfacing them inrunAff
asError
, but it's not clear to me the cases in which this should happen and the Aff-related functions and combinators that should do this.Also, it's not clear to me what will appear in the
Error
value inrunAff
. Anything thrown from PS usingthrowError
? Anything thrown from JS usingthrow
? Anything that is passed to the error channel, for example, from JS FFI using an error callback?Is Aff's "error channel" and thrown exceptions one-and-the-same?
Also, it's not clear to me what kind of things should be thrown and which I should manage in my app, for example by using
ExceptT SomeError Aff a
.If I use
ExceptT
to manage them on my own, should I still be usingError
, or should I use a regular PS data type, likeNonEmptyList String
?If I use
ExceptT
to manage them on my own, I need to handle errors in two spots,runAff
and when handling theExceptT
value which is async-returned from Aff: which one should I send these example errors to? OOM OS error, invalid user when connecting to SQL database, HTTP 500 status response, insufficient user permissions to access requested data, requested 1 record but 0 record found.I found some prior discussion on this [1], but I couldn't figure out an answer to my question from that.
[1] #65
The text was updated successfully, but these errors were encountered: