You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
88 lines
2.2 KiB
88 lines
2.2 KiB
2 months ago
|
package backoff
|
||
|
|
||
|
import (
|
||
|
"cmp"
|
||
|
"fmt"
|
||
|
"log"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// Default configuration values.
|
||
|
const (
|
||
|
DefaultMaxRetries = 3
|
||
|
DefaultRetryInterval = 2 * time.Second
|
||
|
DefaultMaxInterval = 30 * time.Second
|
||
|
)
|
||
|
|
||
|
// Options is the configuration options for retry operations.
|
||
|
type Options struct {
|
||
|
// MaxRetries is the maximum number of retry attempts.
|
||
|
// Default is 3.
|
||
|
MaxRetries int `json:"gmt_max_retries"`
|
||
|
// Interval is the initial interval between retry attempts.
|
||
|
// Default is 2 seconds.
|
||
|
Interval time.Duration `json:"gmt_retry_interval"`
|
||
|
// MaxInterval is the maximum interval between retry attempts.
|
||
|
// Default is 30 seconds.
|
||
|
MaxInterval time.Duration `json:"gmt_retry_max_interval"`
|
||
|
}
|
||
|
|
||
|
// Option is a function that applies an option to an Options instance.
|
||
|
type Option func(*Options)
|
||
|
|
||
|
func (o *Options) Apply(opts ...Option) {
|
||
|
for _, opt := range opts {
|
||
|
opt(o)
|
||
|
}
|
||
|
|
||
|
o.MaxRetries = cmp.Or(max(o.MaxRetries, 0), DefaultMaxRetries)
|
||
|
o.Interval = cmp.Or(max(o.Interval, 0), DefaultRetryInterval)
|
||
|
o.MaxInterval = cmp.Or(max(o.MaxInterval, 0), DefaultMaxInterval)
|
||
|
}
|
||
|
|
||
|
// WithMaxRetries sets the maximum number of retry attempts.
|
||
|
func WithMaxRetries(n int) Option {
|
||
|
return func(o *Options) {
|
||
|
o.MaxRetries = n
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WithRetryInterval sets the initial interval between retry attempts.
|
||
|
func WithRetryInterval(i time.Duration) Option {
|
||
|
return func(o *Options) {
|
||
|
o.Interval = i
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WithMaxInterval sets the maximum interval between retry attempts.
|
||
|
func WithMaxInterval(i time.Duration) Option {
|
||
|
return func(o *Options) {
|
||
|
o.MaxInterval = i
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Retry executes the provided function with retry logic using exponential backoff.
|
||
|
func Retry(fn func() error, opts ...Option) error {
|
||
|
o := new(Options)
|
||
|
o.Apply(opts...)
|
||
|
|
||
|
interval := o.Interval
|
||
|
for i := 0; i < o.MaxRetries; i++ {
|
||
|
if err := fn(); err != nil {
|
||
|
if i == o.MaxRetries-1 {
|
||
|
return fmt.Errorf("backoff: max retries (%d) exceeded: %w", o.MaxRetries, err)
|
||
|
}
|
||
|
time.Sleep(interval)
|
||
|
// TODO 日志
|
||
|
log.Printf("backoff: retrying after %s (attempt %d of %d) due to error: %v", interval, i+1, o.MaxRetries, err)
|
||
|
interval *= 2
|
||
|
if interval > o.MaxInterval {
|
||
|
interval = o.MaxInterval
|
||
|
}
|
||
|
} else {
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|