mirror of
https://github.com/tailscale/tailscale.git
synced 2024-11-29 04:55:31 +00:00
backoff: update to Go style, document a bit, make 30s explicit
Also, bit of behavior change: on non-nil err but expired context, don't reset the consecutive failure count. I don't think the old behavior was intentional. Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
parent
696020227c
commit
dd97111d06
@ -241,7 +241,7 @@ func (c *Client) cancelMapSafely() {
|
|||||||
|
|
||||||
func (c *Client) authRoutine() {
|
func (c *Client) authRoutine() {
|
||||||
defer close(c.authDone)
|
defer close(c.authDone)
|
||||||
bo := backoff.NewBackoff("authRoutine", c.logf)
|
bo := backoff.NewBackoff("authRoutine", c.logf, 30*time.Second)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
@ -401,7 +401,7 @@ func (c *Client) Direct() *Direct {
|
|||||||
|
|
||||||
func (c *Client) mapRoutine() {
|
func (c *Client) mapRoutine() {
|
||||||
defer close(c.mapDone)
|
defer close(c.mapDone)
|
||||||
bo := backoff.NewBackoff("mapRoutine", c.logf)
|
bo := backoff.NewBackoff("mapRoutine", c.logf, 30*time.Second)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
|
@ -175,7 +175,7 @@ func Run(ctx context.Context, logf logger.Logf, logid string, getEngine func() (
|
|||||||
}()
|
}()
|
||||||
logf("Listening on %v", listen.Addr())
|
logf("Listening on %v", listen.Addr())
|
||||||
|
|
||||||
bo := backoff.NewBackoff("ipnserver", logf)
|
bo := backoff.NewBackoff("ipnserver", logf, 30*time.Second)
|
||||||
|
|
||||||
var unservedConn net.Conn // if non-nil, accepted, but hasn't served yet
|
var unservedConn net.Conn // if non-nil, accepted, but hasn't served yet
|
||||||
|
|
||||||
@ -306,7 +306,7 @@ func BabysitProc(ctx context.Context, args []string, logf logger.Logf) {
|
|||||||
proc.mu.Unlock()
|
proc.mu.Unlock()
|
||||||
}()
|
}()
|
||||||
|
|
||||||
bo := backoff.NewBackoff("BabysitProc", logf)
|
bo := backoff.NewBackoff("BabysitProc", logf, 30*time.Second)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
startTime := time.Now()
|
startTime := time.Now()
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Package backoff provides a back-off timer type.
|
||||||
package backoff
|
package backoff
|
||||||
|
|
||||||
import (
|
import (
|
||||||
@ -12,54 +13,70 @@
|
|||||||
"tailscale.com/types/logger"
|
"tailscale.com/types/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
const MAX_BACKOFF_MSEC = 30000
|
// Backoff tracks state the history of consecutive failures and sleeps
|
||||||
|
// an increasing amount of time, up to a provided limit.
|
||||||
type Backoff struct {
|
type Backoff struct {
|
||||||
n int
|
n int // number of consecutive failures
|
||||||
|
maxBackoff time.Duration
|
||||||
|
|
||||||
// Name is the name of this backoff timer, for logging purposes.
|
// Name is the name of this backoff timer, for logging purposes.
|
||||||
name string
|
name string
|
||||||
// logf is the function used for log messages when backing off.
|
// logf is the function used for log messages when backing off.
|
||||||
logf logger.Logf
|
logf logger.Logf
|
||||||
// NewTimer is the function that acts like time.NewTimer().
|
|
||||||
// You can override this in unit tests.
|
// NewTimer is the function that acts like time.NewTimer.
|
||||||
NewTimer func(d time.Duration) *time.Timer
|
// It's for use in unit tests.
|
||||||
|
NewTimer func(time.Duration) *time.Timer
|
||||||
|
|
||||||
// LogLongerThan sets the minimum time of a single backoff interval
|
// LogLongerThan sets the minimum time of a single backoff interval
|
||||||
// before we mention it in the log.
|
// before we mention it in the log.
|
||||||
LogLongerThan time.Duration
|
LogLongerThan time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewBackoff(name string, logf logger.Logf) Backoff {
|
// NewBackoff returns a new Backoff timer with the provided name (for logging), logger,
|
||||||
return Backoff{
|
// and max backoff time. By default, all failures (calls to BackOff with a non-nil err)
|
||||||
name: name,
|
// are logged unless the returned Backoff.LogLongerThan is adjusted.
|
||||||
logf: logf,
|
func NewBackoff(name string, logf logger.Logf, maxBackoff time.Duration) *Backoff {
|
||||||
NewTimer: time.NewTimer,
|
return &Backoff{
|
||||||
|
name: name,
|
||||||
|
logf: logf,
|
||||||
|
maxBackoff: maxBackoff,
|
||||||
|
NewTimer: time.NewTimer,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Backoff sleeps an increasing amount of time if err is non-nil.
|
||||||
|
// and the context is not a
|
||||||
|
// It resets the backoff schedule once err is nil.
|
||||||
func (b *Backoff) BackOff(ctx context.Context, err error) {
|
func (b *Backoff) BackOff(ctx context.Context, err error) {
|
||||||
if ctx.Err() == nil && err != nil {
|
if err == nil {
|
||||||
b.n++
|
// No error. Reset number of consecutive failures.
|
||||||
// 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
|
|
||||||
dur := time.Duration(msec) * time.Millisecond
|
|
||||||
if dur >= b.LogLongerThan {
|
|
||||||
b.logf("%s: backoff: %d msec\n", b.name, msec)
|
|
||||||
}
|
|
||||||
t := b.NewTimer(dur)
|
|
||||||
select {
|
|
||||||
case <-ctx.Done():
|
|
||||||
t.Stop()
|
|
||||||
case <-t.C:
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// not a regular error
|
|
||||||
b.n = 0
|
b.n = 0
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if ctx.Err() != nil {
|
||||||
|
// Fast path.
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
b.n++
|
||||||
|
// n^2 backoff timer is a little smoother than the
|
||||||
|
// common choice of 2^n.
|
||||||
|
d := time.Duration(b.n*b.n) * 10 * time.Millisecond
|
||||||
|
if d > b.maxBackoff {
|
||||||
|
d = b.maxBackoff
|
||||||
|
}
|
||||||
|
// Randomize the delay between 0.5-1.5 x msec, in order
|
||||||
|
// to prevent accidental "thundering herd" problems.
|
||||||
|
d = time.Duration(float64(d) * (rand.Float64() + 0.5))
|
||||||
|
|
||||||
|
if d >= b.LogLongerThan {
|
||||||
|
b.logf("%s: backoff: %d msec", b.name, d.Milliseconds())
|
||||||
|
}
|
||||||
|
t := b.NewTimer(d)
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
t.Stop()
|
||||||
|
case <-t.C:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -105,7 +105,7 @@ func Log(cfg Config, logf tslogger.Logf) Logger {
|
|||||||
sentinel: make(chan int32, 16),
|
sentinel: make(chan int32, 16),
|
||||||
drainLogs: cfg.DrainLogs,
|
drainLogs: cfg.DrainLogs,
|
||||||
timeNow: cfg.TimeNow,
|
timeNow: cfg.TimeNow,
|
||||||
bo: backoff.NewBackoff("logtail", logf),
|
bo: backoff.NewBackoff("logtail", logf, 30*time.Second),
|
||||||
|
|
||||||
shutdownStart: make(chan struct{}),
|
shutdownStart: make(chan struct{}),
|
||||||
shutdownDone: make(chan struct{}),
|
shutdownDone: make(chan struct{}),
|
||||||
@ -133,7 +133,7 @@ type logger struct {
|
|||||||
drainLogs <-chan struct{} // if non-nil, external signal to attempt a drain
|
drainLogs <-chan struct{} // if non-nil, external signal to attempt a drain
|
||||||
sentinel chan int32
|
sentinel chan int32
|
||||||
timeNow func() time.Time
|
timeNow func() time.Time
|
||||||
bo backoff.Backoff
|
bo *backoff.Backoff
|
||||||
zstdEncoder Encoder
|
zstdEncoder Encoder
|
||||||
uploadCancel func()
|
uploadCancel func()
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user