From 3cba698800a324930b3525e1f4818a6017307209 Mon Sep 17 00:00:00 2001 From: Seth Hoenig Date: Fri, 11 Nov 2022 08:27:18 -0600 Subject: [PATCH 1/2] client: avoid unconsumed channel in timer construction This PR fixes a bug introduced in #11983 where a Timer initialized with 0 duration causes an immediate tick, even if Reset is called before reading the channel. The fix is to avoid doing that, instead creating a Timer with a non-zero initial wait time, and then immediately calling Stop. --- .changelog/15215.txt | 3 +++ client/allocrunner/taskrunner/task_runner.go | 4 +++- helper/funcs.go | 12 ++++++++++++ helper/funcs_test.go | 13 ++++++++++++- 4 files changed, 30 insertions(+), 2 deletions(-) create mode 100644 .changelog/15215.txt diff --git a/.changelog/15215.txt b/.changelog/15215.txt new file mode 100644 index 00000000000..4428ce62316 --- /dev/null +++ b/.changelog/15215.txt @@ -0,0 +1,3 @@ +```release-note:bug +client: Fixed a bug where tasks would restart without waiting for interval +``` diff --git a/client/allocrunner/taskrunner/task_runner.go b/client/allocrunner/taskrunner/task_runner.go index 12f1abf26be..a0f581e459e 100644 --- a/client/allocrunner/taskrunner/task_runner.go +++ b/client/allocrunner/taskrunner/task_runner.go @@ -563,7 +563,9 @@ func (tr *TaskRunner) Run() { // Set the initial task state. tr.stateUpdater.TaskStateUpdated() - timer, stop := helper.NewSafeTimer(0) // timer duration calculated JIT + // start with a stopped timer; actual restart delay computed later + timer, stop := helper.NewStoppedTimer() + timer.Stop() defer stop() MAIN: diff --git a/helper/funcs.go b/helper/funcs.go index 0f8209c2df3..b37450f1883 100644 --- a/helper/funcs.go +++ b/helper/funcs.go @@ -3,6 +3,7 @@ package helper import ( "crypto/sha512" "fmt" + "math" "net/http" "path/filepath" "reflect" @@ -369,6 +370,9 @@ type StopFunc func() // // Returns the time.Timer and also a StopFunc, forcing the caller to deal // with stopping the time.Timer to avoid leaking a goroutine. +// +// Note: If creating a Timer that should do nothing until Reset is called, use +// NewStoppedTimer instead for safely creating the timer in a stopped state. func NewSafeTimer(duration time.Duration) (*time.Timer, StopFunc) { if duration <= 0 { // Avoid panic by using the smallest positive value. This is close enough @@ -386,6 +390,14 @@ func NewSafeTimer(duration time.Duration) (*time.Timer, StopFunc) { return t, cancel } +// NewStoppedTimer creates a time.Timer in a stopped state. This is useful when +// the actual wait time will computed and set later via Reset. +func NewStoppedTimer() (*time.Timer, StopFunc) { + t, f := NewSafeTimer(math.MaxInt64) + t.Stop() + return t, f +} + // ConvertSlice takes the input slice and generates a new one using the // supplied conversion function to covert the element. This is useful when // converting a slice of strings to a slice of structs which wraps the string. diff --git a/helper/funcs_test.go b/helper/funcs_test.go index d4d68f72cb8..5e366aeebfb 100644 --- a/helper/funcs_test.go +++ b/helper/funcs_test.go @@ -376,7 +376,7 @@ func TestCheckNamespaceScope(t *testing.T) { } } -func Test_NewSafeTimer(t *testing.T) { +func TestTimer_NewSafeTimer(t *testing.T) { t.Run("zero", func(t *testing.T) { timer, stop := NewSafeTimer(0) defer stop() @@ -390,6 +390,17 @@ func Test_NewSafeTimer(t *testing.T) { }) } +func TestTimer_NewStoppedTimer(t *testing.T) { + timer, stop := NewStoppedTimer() + defer stop() + + select { + case <-timer.C: + must.Unreachable(t) + default: + } +} + func Test_ConvertSlice(t *testing.T) { t.Run("string wrapper", func(t *testing.T) { From fdc2c6b822b2d05259b72757e333f3095203f226 Mon Sep 17 00:00:00 2001 From: Seth Hoenig Date: Fri, 11 Nov 2022 09:25:30 -0600 Subject: [PATCH 2/2] pr: remove redundant stop --- client/allocrunner/taskrunner/task_runner.go | 1 - 1 file changed, 1 deletion(-) diff --git a/client/allocrunner/taskrunner/task_runner.go b/client/allocrunner/taskrunner/task_runner.go index a0f581e459e..6f2291e12a0 100644 --- a/client/allocrunner/taskrunner/task_runner.go +++ b/client/allocrunner/taskrunner/task_runner.go @@ -565,7 +565,6 @@ func (tr *TaskRunner) Run() { // start with a stopped timer; actual restart delay computed later timer, stop := helper.NewStoppedTimer() - timer.Stop() defer stop() MAIN: