From 5af1f3ebd472b62a4a708ad3aa2d252489b91b27 Mon Sep 17 00:00:00 2001 From: Chris Bandy Date: Sat, 30 Mar 2024 11:24:54 -0500 Subject: [PATCH] Quiet context.Canceled errors during shutdown Runnable implementations that return ctx.Err() cause a spurious "error received after stop" log message. --- pkg/manager/internal.go | 4 +-- pkg/manager/manager_test.go | 52 +++++++++++++++++++++++++++++++++++++ 2 files changed, 54 insertions(+), 2 deletions(-) diff --git a/pkg/manager/internal.go b/pkg/manager/internal.go index 040b3c5b82..e5204a7506 100644 --- a/pkg/manager/internal.go +++ b/pkg/manager/internal.go @@ -507,8 +507,8 @@ func (cm *controllerManager) engageStopProcedure(stopComplete <-chan struct{}) e cm.internalCancel() }) select { - case err, ok := <-cm.errChan: - if ok { + case err := <-cm.errChan: + if !errors.Is(err, context.Canceled) { cm.logger.Error(err, "error received after stop sequence was engaged") } case <-stopComplete: diff --git a/pkg/manager/manager_test.go b/pkg/manager/manager_test.go index b3383edf88..c42d2f2ae7 100644 --- a/pkg/manager/manager_test.go +++ b/pkg/manager/manager_test.go @@ -29,6 +29,7 @@ import ( "time" "github.com/go-logr/logr" + "github.com/go-logr/logr/funcr" . "github.com/onsi/ginkgo/v2" . "github.com/onsi/gomega" "github.com/prometheus/client_golang/prometheus" @@ -982,6 +983,57 @@ var _ = Describe("manger.Manager", func() { }))).NotTo(Succeed()) }) + It("should not return runnables context.Canceled errors", func() { + Expect(options.Logger).To(BeZero(), "this test overrides Logger") + + var log struct { + sync.Mutex + messages []string + } + options.Logger = funcr.NewJSON(func(object string) { + log.Lock() + log.messages = append(log.messages, object) + log.Unlock() + }, funcr.Options{}) + + m, err := New(cfg, options) + Expect(err).NotTo(HaveOccurred()) + for _, cb := range callbacks { + cb(m) + } + + // Runnables may return ctx.Err() as shown in some [context.Context] examples. + started := make(chan struct{}) + Expect(m.Add(RunnableFunc(func(ctx context.Context) error { + close(started) + <-ctx.Done() + return ctx.Err() + }))).To(Succeed()) + + stopped := make(chan error) + ctx, cancel := context.WithCancel(context.Background()) + go func() { + stopped <- m.Start(ctx) + }() + + // Wait for runnables to start, signal the manager, and wait for it to return. + <-started + cancel() + Expect(<-stopped).To(Succeed()) + + // The leader election goroutine emits one more log message after Start() returns. + // Take the lock here to avoid a race between it writing to log.messages and the + // following read from log.messages. + if options.LeaderElection { + log.Lock() + defer log.Unlock() + } + + Expect(log.messages).To(Not(ContainElement( + ContainSubstring(context.Canceled.Error()), + ))) + }) + It("should return both runnables and stop errors when both error", func() { m, err := New(cfg, options) Expect(err).NotTo(HaveOccurred())