2023-02-22 12:06:55 +00:00
|
|
|
package backoff
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
backoffRetries = 5
|
|
|
|
backoffFactor = 1.7
|
|
|
|
backoffMinDuration = time.Second
|
|
|
|
)
|
|
|
|
|
|
|
|
// retryableFunc describes call back which will repeat on errors
|
|
|
|
type retryableFunc func() error
|
|
|
|
|
2023-02-26 20:18:59 +00:00
|
|
|
// ErrBadRequest is an error returned on bad request
|
2023-02-22 12:06:55 +00:00
|
|
|
var ErrBadRequest = errors.New("bad request")
|
|
|
|
|
|
|
|
// Backoff describes object with backoff policy params
|
|
|
|
type Backoff struct {
|
|
|
|
retries int
|
|
|
|
factor float64
|
|
|
|
minDuration time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
// New initialize backoff object
|
|
|
|
func New() *Backoff {
|
|
|
|
return &Backoff{
|
|
|
|
retries: backoffRetries,
|
|
|
|
factor: backoffFactor,
|
|
|
|
minDuration: backoffMinDuration,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retry process retries until all attempts are completed
|
|
|
|
func (b *Backoff) Retry(ctx context.Context, cb retryableFunc) (uint64, error) {
|
|
|
|
var attempt uint64
|
|
|
|
for i := 0; i < b.retries; i++ {
|
|
|
|
// @TODO we should use context to cancel retries
|
|
|
|
err := cb()
|
|
|
|
if err == nil {
|
|
|
|
return attempt, nil
|
|
|
|
}
|
|
|
|
if errors.Is(err, ErrBadRequest) {
|
|
|
|
logger.Errorf("unrecoverable error: %s", err)
|
|
|
|
return attempt, err // fail fast if not recoverable
|
|
|
|
}
|
|
|
|
attempt++
|
|
|
|
backoff := float64(b.minDuration) * math.Pow(b.factor, float64(i))
|
|
|
|
dur := time.Duration(backoff)
|
|
|
|
logger.Errorf("got error: %s on attempt: %d; will retry in %v", err, attempt, dur)
|
|
|
|
time.Sleep(time.Duration(backoff))
|
|
|
|
}
|
|
|
|
return attempt, fmt.Errorf("execution failed after %d retry attempts", b.retries)
|
|
|
|
}
|