v2ray-core/common/retry/retry.go

65 lines
1.4 KiB
Go
Raw Normal View History

2018-04-03 20:34:59 +00:00
package retry // import "v2ray.com/core/common/retry"
2015-10-13 10:27:50 +00:00
2018-09-30 21:08:41 +00:00
//go:generate errorgen
2017-04-08 23:43:25 +00:00
2015-10-13 10:27:50 +00:00
import (
"time"
)
var (
2017-04-26 20:35:19 +00:00
ErrRetryFailed = newError("all retry attempts failed")
2015-10-13 10:27:50 +00:00
)
2015-12-02 08:58:00 +00:00
// Strategy is a way to retry on a specific function.
type Strategy interface {
2015-12-02 11:47:54 +00:00
// On performs a retry on a specific function, until it doesn't return any error.
2015-10-13 10:27:50 +00:00
On(func() error) error
}
type retryer struct {
2016-11-20 20:47:51 +00:00
totalAttempt int
nextDelay func() uint32
2015-10-13 10:27:50 +00:00
}
2015-12-02 08:58:00 +00:00
// On implements Strategy.On.
2015-10-13 10:27:50 +00:00
func (r *retryer) On(method func() error) error {
attempt := 0
2017-02-10 10:41:50 +00:00
accumulatedError := make([]error, 0, r.totalAttempt)
2016-11-20 20:47:51 +00:00
for attempt < r.totalAttempt {
2015-10-13 10:27:50 +00:00
err := method()
if err == nil {
return nil
}
2017-02-10 10:41:50 +00:00
numErrors := len(accumulatedError)
if numErrors == 0 || err.Error() != accumulatedError[numErrors-1].Error() {
accumulatedError = append(accumulatedError, err)
}
2016-11-20 20:47:51 +00:00
delay := r.nextDelay()
2017-05-08 22:01:15 +00:00
time.Sleep(time.Duration(delay) * time.Millisecond)
2015-10-13 22:57:00 +00:00
attempt++
2015-10-13 10:27:50 +00:00
}
2017-04-08 23:43:25 +00:00
return newError(accumulatedError).Base(ErrRetryFailed)
2015-10-13 10:27:50 +00:00
}
2015-12-02 08:58:00 +00:00
// Timed returns a retry strategy with fixed interval.
2016-11-20 20:47:51 +00:00
func Timed(attempts int, delay uint32) Strategy {
2015-10-13 10:27:50 +00:00
return &retryer{
2016-11-20 20:47:51 +00:00
totalAttempt: attempts,
nextDelay: func() uint32 {
2015-10-13 10:27:50 +00:00
return delay
},
}
}
2016-11-20 20:47:51 +00:00
func ExponentialBackoff(attempts int, delay uint32) Strategy {
nextDelay := uint32(0)
return &retryer{
totalAttempt: attempts,
nextDelay: func() uint32 {
r := nextDelay
nextDelay += delay
return r
},
}
}