blob: c17468896a59440a9a8948982a998da3b4867d27 [file] [log] [blame]
// Copyright 2015 The Vanadium 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 retry provides a facility for retrying function
// invocations.
package retry
import (
"fmt"
"math"
"math/rand"
"time"
"go.fuchsia.dev/jiri"
)
type RetryOpt interface {
retryOpt()
}
type AttemptsOpt int
func (a AttemptsOpt) retryOpt() {}
type IntervalOpt time.Duration
func (i IntervalOpt) retryOpt() {}
const (
defaultAttempts = 3
defaultInterval = 5 * time.Second
)
type exponentialBackoff struct {
InitialInterval float64
MaxInterval float64
Multiplier float64
Iteration int
Rand *rand.Rand
}
func newExponentialBackoff(initialInterval float64, maxInterval float64, multiplier float64) *exponentialBackoff {
e := &exponentialBackoff{
InitialInterval: initialInterval,
MaxInterval: maxInterval,
Multiplier: multiplier,
Iteration: 0,
Rand: rand.New(rand.NewSource(time.Now().UnixNano())),
}
return e
}
func (e *exponentialBackoff) nextBackoff() time.Duration {
next := e.InitialInterval*math.Pow(e.Multiplier, float64(e.Iteration)) + 10*e.Rand.Float64()
e.Iteration++
if next > e.MaxInterval {
next = e.MaxInterval
}
return time.Duration(float64(time.Second) * next)
}
// Function retries the given function for the given number of
// attempts at the given interval.
func Function(jirix *jiri.X, fn func() error, task string, opts ...RetryOpt) error {
attempts, interval := defaultAttempts, defaultInterval
for _, opt := range opts {
switch typedOpt := opt.(type) {
case AttemptsOpt:
attempts = int(typedOpt)
case IntervalOpt:
interval = time.Duration(typedOpt)
}
}
backoff := newExponentialBackoff(float64(interval), 64, 2)
var err error
for i := 1; i <= attempts; i++ {
if i > 1 {
jirix.Logger.Infof("Attempt %d/%d: %s\n\n", i, attempts, task)
}
if err = fn(); err == nil {
return nil
}
if i < attempts {
jirix.Logger.Errorf("%s\n\n", err)
backoffInterval := backoff.nextBackoff()
jirix.Logger.Infof("Wait for %s before next attempt...: %s\n\n", backoffInterval, task)
time.Sleep(backoffInterval)
}
}
if attempts > 1 {
return fmt.Errorf("%q failed %d times in a row, Last error: %s", task, attempts, err)
}
return err
}