diff --git a/pkg/poll/poll.go b/pkg/poll/poll.go index 398513b9ee3..1a7269c115d 100644 --- a/pkg/poll/poll.go +++ b/pkg/poll/poll.go @@ -91,7 +91,10 @@ func WaitWithBackoffWithRetries(ctx context.Context, b backoff.Backoff, numRetri sleep := b.Duration() if deadline, ok := ctx.Deadline(); ok { ctxSleep := time.Until(deadline) - sleep = minDuration(sleep, ctxSleep) + // We want to wait for smaller of backoff sleep and context sleep + // but it has to be over os scheduling interval + // otherwise context doesn't have time to finish + sleep = max(min(sleep, ctxSleep), 5*time.Millisecond) } t.Reset(sleep) select { @@ -101,10 +104,3 @@ func WaitWithBackoffWithRetries(ctx context.Context, b backoff.Backoff, numRetri } } } - -func minDuration(a, b time.Duration) time.Duration { - if a < b { - return a - } - return b -} diff --git a/pkg/poll/poll_test.go b/pkg/poll/poll_test.go index c7de2056754..4a4983e5709 100644 --- a/pkg/poll/poll_test.go +++ b/pkg/poll/poll_test.go @@ -16,7 +16,9 @@ package poll import ( "context" + "errors" "fmt" + "runtime" "testing" "time" @@ -130,6 +132,30 @@ func (s *PollSuite) TestWaitWithBackoffCancellation(c *C) { c.Check(err, NotNil) } +func (s *PollSuite) TestWaitWithRetriesTimeout(c *C) { + // There's a better chance of catching a race condition + // if there is only one thread + maxprocs := runtime.GOMAXPROCS(1) + defer runtime.GOMAXPROCS(maxprocs) + + f := func(context.Context) (bool, error) { + return false, errors.New("retryable") + } + errf := func(err error) bool { + return err.Error() == "retryable" + } + ctx := context.Background() + var cancel context.CancelFunc + ctx, cancel = context.WithTimeout(ctx, time.Millisecond) + defer cancel() + + backoff := backoff.Backoff{} + backoff.Min = 2 * time.Millisecond + err := WaitWithBackoffWithRetries(ctx, backoff, 10, errf, f) + c.Check(err, NotNil) + c.Assert(err.Error(), Matches, ".*context deadline exceeded*") +} + func (s *PollSuite) TestWaitWithBackoffBackoff(c *C) { const numIterations = 10 i := 0