Commit efdb8db2 authored by Steven Allen's avatar Steven Allen

fix: get rid of shutdown errors

Instead of feeding through the top-level context, feed through a cancel-free
context (that still carries the same context values). Then, when the top-level
context is canceled, call `stop` to shut everything down in-order. Finally,
cancel the inner context to make sure everything has been cleaned up.

Ideally, we just wouldn't use contexts for this. But this is strictly better
than what we have.
parent 474ade1f
...@@ -3,6 +3,7 @@ package core ...@@ -3,6 +3,7 @@ package core
import ( import (
"context" "context"
"sync" "sync"
"time"
"github.com/ipfs/go-ipfs/core/bootstrap" "github.com/ipfs/go-ipfs/core/bootstrap"
"github.com/ipfs/go-ipfs/core/node" "github.com/ipfs/go-ipfs/core/node"
...@@ -11,10 +12,26 @@ import ( ...@@ -11,10 +12,26 @@ import (
"go.uber.org/fx" "go.uber.org/fx"
) )
// from https://stackoverflow.com/a/59348871
type valueContext struct {
context.Context
}
func (valueContext) Deadline() (deadline time.Time, ok bool) { return }
func (valueContext) Done() <-chan struct{} { return nil }
func (valueContext) Err() error { return nil }
type BuildCfg = node.BuildCfg // Alias for compatibility until we properly refactor the constructor interface type BuildCfg = node.BuildCfg // Alias for compatibility until we properly refactor the constructor interface
// NewNode constructs and returns an IpfsNode using the given cfg. // NewNode constructs and returns an IpfsNode using the given cfg.
func NewNode(ctx context.Context, cfg *BuildCfg) (*IpfsNode, error) { func NewNode(ctx context.Context, cfg *BuildCfg) (*IpfsNode, error) {
// save this context as the "lifetime" ctx.
lctx := ctx
// derive a new context that ignores cancellations from the lifetime ctx.
ctx, cancel := context.WithCancel(valueContext{ctx})
// add a metrics scope.
ctx = metrics.CtxScope(ctx, "ipfs") ctx = metrics.CtxScope(ctx, "ipfs")
n := &IpfsNode{ n := &IpfsNode{
...@@ -33,18 +50,27 @@ func NewNode(ctx context.Context, cfg *BuildCfg) (*IpfsNode, error) { ...@@ -33,18 +50,27 @@ func NewNode(ctx context.Context, cfg *BuildCfg) (*IpfsNode, error) {
n.stop = func() error { n.stop = func() error {
once.Do(func() { once.Do(func() {
stopErr = app.Stop(context.Background()) stopErr = app.Stop(context.Background())
if stopErr != nil {
log.Error("failure on stop: ", stopErr)
}
// Cancel the context _after_ the app has stopped.
cancel()
}) })
return stopErr return stopErr
} }
n.IsOnline = cfg.Online n.IsOnline = cfg.Online
go func() { go func() {
// Note that some services use contexts to signal shutting down, which is // Shut down the application if the lifetime context is canceled.
// very suboptimal. This needs to be here until that's addressed somehow // NOTE: we _should_ stop the application by calling `Close()`
<-ctx.Done() // on the process. But we currently manage everything with contexts.
err := n.stop() select {
if err != nil { case <-lctx.Done():
log.Error("failure on stop: ", err) err := n.stop()
if err != nil {
log.Error("failure on stop: ", err)
}
case <-ctx.Done():
} }
}() }()
......
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment