forked from TrueCloudLab/frostfs-s3-gw
Denis Kirillov
5d6d5e41d0
Add two strategy for PutBucketSettings request retryer:
* exponential backoff (increasing up to `max_backoff` delays with jitter)
* constant backoff (always the same `max_backoff` delay between requests)
Signed-off-by: Denis Kirillov <d.kirillov@yadro.com>
(cherry picked from commit bb81afc14a
)
Signed-off-by: Alex Vanin <a.vanin@yadro.com>
52 lines
855 B
Go
52 lines
855 B
Go
package retryer
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/aws/aws-sdk-go-v2/aws"
|
|
)
|
|
|
|
func MakeWithRetry(ctx context.Context, fn func() error, retryer aws.RetryerV2) (err error) {
|
|
var attemptNum int
|
|
|
|
maxAttempts := retryer.MaxAttempts()
|
|
|
|
for {
|
|
attemptNum++
|
|
|
|
err = fn()
|
|
|
|
if !retryer.IsErrorRetryable(err) {
|
|
return err
|
|
}
|
|
|
|
if attemptNum >= maxAttempts {
|
|
return fmt.Errorf("max retry attempts exausted, max %d: %w", maxAttempts, err)
|
|
}
|
|
|
|
retryDelay, reqErr := retryer.RetryDelay(attemptNum, err)
|
|
if reqErr != nil {
|
|
return reqErr
|
|
}
|
|
|
|
if reqErr = sleepWithContext(ctx, retryDelay); reqErr != nil {
|
|
return reqErr
|
|
}
|
|
}
|
|
}
|
|
|
|
func sleepWithContext(ctx context.Context, dur time.Duration) error {
|
|
t := time.NewTimer(dur)
|
|
defer t.Stop()
|
|
|
|
select {
|
|
case <-t.C:
|
|
break
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
}
|
|
|
|
return nil
|
|
}
|