-
-
Notifications
You must be signed in to change notification settings - Fork 3.1k
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
Logging non-fatal async errors #66
Comments
@perfmode i like it. a lot. that way the callee can decide too whether/when to cancel via ctx. Can we implement the retrieval of |
Great idea. I think we can use the setters and getters on the default context to accomplish this! |
I'm going to hack something together real quick and report back with progress. |
I really like the idea of tying logging to a context. If we can add different levels of logging to this interface as well I would be quite happy |
What levels do you have in mind? |
Traditionally I like trace, info warn and error. |
(I know that's slightly out of the range of logging errors though) |
Yeah, maybe we can have both errors and broader logging on it. // given some Logger we decide to use, add it to the context and then:
type IpfsContext interface {
context.Context
Errs() chan error
Logger() *Logger
}
// send an error:
ctx.Errs()<- err
// or, as btc suggested above
ctx.Error(err)
// get the logger:
ctx.Logger().Errorf("There was an error: %v", err)
ctx.Logger().Warning("Just what do you think you're doing, Dave?") |
We should use a logger that has both:
where |
@jbenet definitely prefer a synchronous interface (even if implemented with buffered channel) Re: logger... The context can embed a logger like so:
Because the caller may use the error to modify control flow, it's important to keep the error reporting and general logging separate (even if the error is also logged). Mmm. How about this interface then.
|
👍
Don't think you can add variables into the interface-- oh the Logger is an interface itself.
👎 on |
Agreed on explicit functions. |
SGTM. I'm going to punt on logging (at first). Will implement error |
As a side note to this. I really want to start using contexts everywhere. |
use case: parent neglects childParent (caller) is not interested in errors. so: Errors reported by child (async actor) should go to /dev/null. use case: parent is lateParent (caller) starts listening to errors some time after dispatching the goroutine. so: Errors received up until that point should be buffered. challenge: never block the childIn both cases, it is desirable to uphold a contract that the child will never be blocked on error logging. Violation of this contract introduces potentially tricky deadlock issues. implementation non-optionsDue to constraints imposed by the neglected use case, at the time of instantiation of the context, the context must be informed that the parent never intends to retrieve errors. Otherwise we are forced to choose between: If the parent has not yet decided to start listening on the channel, we can do a few different things once the child starts logging errors: a. let errors accumulate in a buffer, blocking the child (violates contract) edit: A finalizer can be used to perform garbage collection in the case of option c. http://golang.org/pkg/runtime/#SetFinalizer potential solutionThis can be solved by having default behavior where the client can log errors but they go to /dev/null. If the parent is actually interested in receiving the errors, it can instantiate a context with error reporting like so: ctx, errs := context.WithErrorLog(ctx) This is in the same style as the golang Context factory methods:
RFC@jbenet @whyrusleeping @mappum I believe this is a sensible solution to the problem. If you have any thoughts on the matter, I would love to hear them. |
I don't believe that errors should ever be sent to /dev/null (or equivalent). All information is useful. If an error is expected, it should be checked and handled explicitly. |
Perhaps the term Perhaps there's a way to force parents to acknowledge errors, but it's dangerous to block children when their reported errors are ignored. |
@perfmode understand the constraints. I think though that the intuition is right. Should by default silence errors that the child goroutine thinks may be being heard. Silencing should be explicit. Parent should only be able to neglect by doing so explicitly like: func silenceErrors(ctx) {
for {
select {
case <-ctx.Errs(): // discard error.
case <-ctx.Done():
return
}
}
}
go doSomething(ctx)
go silenceErrors(ctx) I think it's fine if the child ends up blocking if the parent doesn't consume the errors. |
@perfmode btw there is some complexity to be considered when it's not just parent -> child, but there's a whole tree in between, with potentially many children at different levels logging. |
on that last point, i think it should work symmetrical to Done. i.e. calling: ctx2, errs := context.WithErrorLog(ctx1) captures all errors from |
Thanks for the feedback guys! Contexts are about the go from good to great! Looking forward to integrating logging functionality.
@jbenet re: error notifications not leaking up tree. I agree. Implemented and tested. https://github.com/jbenet/go-ipfs/blob/ipfs-context/util/context/context_test.go#L34 @whyrusleeping I agree it's best to explicitly handle errors generated by children. Implemented
@jbenet It's not clear to me what you mean here. Despite not being able to parse these statements, I think we're actually on the same page. If, in the implementation, I missed the point you're making, please let me know. |
This commit... is a second attempt at: #68 partially addresses: #66 is the result of discussion at: https://gist.github.com/perfmode/f2951c1ed3a02c484d0b
This commit... is a second attempt at: #68 partially addresses: #66 is the result of discussion at: https://gist.github.com/perfmode/f2951c1ed3a02c484d0b
This commit... is a second attempt at: #68 partially addresses: #66 is the result of discussion at: https://gist.github.com/perfmode/f2951c1ed3a02c484d0b
@jbenet @perfmode Would you two say this one is closed? |
I think we still need to go through and replace some output with calling logger.— On Sun, Oct 5, 2014 at 7:58 PM, Jeromy Johnson notifications@github.com
|
gx publish 2.5.5
I wish to propose a pattern for logging the non-fatal errors that occur in asynchronous operations.
Context exposes an error logging method:
The interface extends the golang Context:
Now async actors can get information back out to parent without modifying control flow or confusing the caller/callee contract!
the context creator can listen for errors:
The pattern is analogous to the way children listen on Done()
Thoughts? @jbenet @whyrusleeping
The text was updated successfully, but these errors were encountered: