2020-02-05 22:16:58 +00:00
|
|
|
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package backoff
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"log"
|
|
|
|
"math/rand"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
const MAX_BACKOFF_MSEC = 30000
|
|
|
|
|
|
|
|
type Backoff struct {
|
2020-04-29 09:41:39 +00:00
|
|
|
n int
|
|
|
|
// Name is the name of this backoff timer, for logging purposes.
|
|
|
|
Name string
|
|
|
|
// NewTimer is the function that acts like time.NewTimer().
|
|
|
|
// You can override this in unit tests.
|
2020-02-05 22:16:58 +00:00
|
|
|
NewTimer func(d time.Duration) *time.Timer
|
2020-04-29 09:41:39 +00:00
|
|
|
// LogLongerThan sets the minimum time of a single backoff interval
|
|
|
|
// before we mention it in the log.
|
|
|
|
LogLongerThan time.Duration
|
2020-02-05 22:16:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Backoff) BackOff(ctx context.Context, err error) {
|
|
|
|
if ctx.Err() == nil && err != nil {
|
|
|
|
b.n++
|
|
|
|
// n^2 backoff timer is a little smoother than the
|
|
|
|
// common choice of 2^n.
|
|
|
|
msec := b.n * b.n * 10
|
|
|
|
if msec > MAX_BACKOFF_MSEC {
|
|
|
|
msec = MAX_BACKOFF_MSEC
|
|
|
|
}
|
|
|
|
// Randomize the delay between 0.5-1.5 x msec, in order
|
|
|
|
// to prevent accidental "thundering herd" problems.
|
|
|
|
msec = rand.Intn(msec) + msec/2
|
2020-04-29 09:41:39 +00:00
|
|
|
dur := time.Duration(msec) * time.Millisecond
|
|
|
|
if dur >= b.LogLongerThan {
|
2020-04-29 07:41:41 +00:00
|
|
|
log.Printf("%s: backoff: %d msec\n", b.Name, msec)
|
|
|
|
}
|
2020-02-05 22:16:58 +00:00
|
|
|
newTimer := b.NewTimer
|
|
|
|
if newTimer == nil {
|
|
|
|
newTimer = time.NewTimer
|
|
|
|
}
|
2020-04-29 09:41:39 +00:00
|
|
|
t := newTimer(dur)
|
2020-02-05 22:16:58 +00:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
t.Stop()
|
|
|
|
case <-t.C:
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// not a regular error
|
|
|
|
b.n = 0
|
|
|
|
}
|
|
|
|
}
|