From 7788cd713bab64f0307414d4d00e5889ae47485d Mon Sep 17 00:00:00 2001 From: Rob Schleusner Date: Tue, 9 Jan 2024 11:35:45 -0600 Subject: [PATCH] Shuts down cleanly on cancel --- cmd/go-quai/start.go | 5 ++- cmd/utils/cmd.go | 67 ++++++++++++++++++++-------------------- ethdb/leveldb/leveldb.go | 5 --- 3 files changed, 38 insertions(+), 39 deletions(-) diff --git a/cmd/go-quai/start.go b/cmd/go-quai/start.go index 68ef32207c..0a65822c2f 100644 --- a/cmd/go-quai/start.go +++ b/cmd/go-quai/start.go @@ -4,6 +4,7 @@ import ( "context" "os" "os/signal" + "sync" "syscall" "github.com/spf13/cobra" @@ -70,7 +71,8 @@ func runStart(cmd *cobra.Command, args []string) error { logLevel := cmd.Flag(utils.LogLevelFlag.Name).Value.String() // create instance of consensus backend - consensus, err := utils.StartQuaiBackend(logLevel) + var nodeWG sync.WaitGroup + consensus, err := utils.StartQuaiBackend(ctx, logLevel, &nodeWG) if err != nil { log.WithField("error", err).Fatal("error creating consensus backend") } @@ -98,6 +100,7 @@ func runStart(cmd *cobra.Command, args []string) error { <-ch log.Warn("Received 'stop' signal, shutting down gracefully...") cancel() + nodeWG.Wait() if err := node.Stop(); err != nil { panic(err) } diff --git a/cmd/utils/cmd.go b/cmd/utils/cmd.go index bb98b5d309..757be35afa 100644 --- a/cmd/utils/cmd.go +++ b/cmd/utils/cmd.go @@ -1,10 +1,12 @@ package utils import ( + "context" "fmt" "io" "os" "runtime" + "sync" "time" "github.com/dominant-strategies/go-quai/common" @@ -21,42 +23,41 @@ import ( ) // Create a new instance of the QuaiBackend consensus service -func StartQuaiBackend(logLevel string) (*quai.QuaiBackend, error) { - var logger *logrus.Logger - // Make full node - go func() { - // Create the prime logger with the log file path - logger = log.NewLogger("nodelogs/prime.log", logLevel) - logger.Info("Starting Prime") - stackPrime := makeFullNode(nil, logger) - defer stackPrime.Close() - StartNode(stackPrime) - stackPrime.Wait() - }() +func StartQuaiBackend(ctx context.Context, logLevel string, nodeWG *sync.WaitGroup) (*quai.QuaiBackend, error) { + startNode := func(logPath string, location common.Location) { + nodeWG.Add(1) + go func() { + defer nodeWG.Done() + logger := log.NewLogger(logPath, logLevel) + logger.Info("Starting Node at location", "location", location) + stack := makeFullNode(location, logger) + StartNode(stack) + // Create a channel to signal when stack.Wait() is done + done := make(chan struct{}) + go func() { + stack.Wait() + close(done) + }() + + select { + case <-done: + logger.Info("Node stopped normally") + stack.Close() + return + case <-ctx.Done(): + logger.Info("Context cancelled, shutting down node") + stack.Close() + return + } + }() + } + // Start nodes in separate goroutines + startNode("nodelogs/prime.log", nil) time.Sleep(2 * time.Second) - - go func() { - // Create the prime logger with the log file path - logger = log.NewLogger("nodelogs/region-0.log", logLevel) - logger.Info("Starting Region") - stackRegion := makeFullNode(common.Location{0}, logger) - defer stackRegion.Close() - StartNode(stackRegion) - stackRegion.Wait() - }() - + startNode("nodelogs/region-0.log", common.Location{0}) time.Sleep(2 * time.Second) - - go func() { - // Create the prime logger with the log file path - logger = log.NewLogger("nodelogs/zone-0-0.log", logLevel) - log.Info("Starting Zone") - stackZone := makeFullNode(common.Location{0, 0}, logger) - defer stackZone.Close() - StartNode(stackZone) - stackZone.Wait() - }() + startNode("nodelogs/zone-0-0.log", common.Location{0, 0}) return &quai.QuaiBackend{}, nil } diff --git a/ethdb/leveldb/leveldb.go b/ethdb/leveldb/leveldb.go index 452dc07e2e..d7cb36ea08 100644 --- a/ethdb/leveldb/leveldb.go +++ b/ethdb/leveldb/leveldb.go @@ -136,11 +136,6 @@ func (db *Database) Close() error { defer db.quitLock.Unlock() if db.quitChan != nil { - errc := make(chan error) - db.quitChan <- errc - if err := <-errc; err != nil { - db.logger.WithField("err", err).Error("Metrics collection failed") - } db.quitChan = nil } return db.db.Close()