mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-07 00:22:16 +01:00
61 lines
1.4 KiB
Go
61 lines
1.4 KiB
Go
|
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
|
||
|
|
||
|
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)
|
||
|
}
|