123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- package fx
- import (
- "context"
- "time"
- "github.com/wuntsong-org/go-zero-plus/core/errorx"
- )
- const defaultRetryTimes = 3
- type (
- // RetryOption defines the method to customize DoWithRetry.
- RetryOption func(*retryOptions)
- retryOptions struct {
- times int
- interval time.Duration
- timeout time.Duration
- }
- )
- // DoWithRetry runs fn, and retries if failed. Default to retry 3 times.
- // Note that if the fn function accesses global variables outside the function
- // and performs modification operations, it is best to lock them,
- // otherwise there may be data race issues
- func DoWithRetry(fn func() error, opts ...RetryOption) error {
- return retry(context.Background(), func(errChan chan error, retryCount int) {
- errChan <- fn()
- }, opts...)
- }
- // DoWithRetryCtx runs fn, and retries if failed. Default to retry 3 times.
- // fn retryCount indicates the current number of retries, starting from 0
- // Note that if the fn function accesses global variables outside the function
- // and performs modification operations, it is best to lock them,
- // otherwise there may be data race issues
- func DoWithRetryCtx(ctx context.Context, fn func(ctx context.Context, retryCount int) error,
- opts ...RetryOption) error {
- return retry(ctx, func(errChan chan error, retryCount int) {
- errChan <- fn(ctx, retryCount)
- }, opts...)
- }
- func retry(ctx context.Context, fn func(errChan chan error, retryCount int), opts ...RetryOption) error {
- options := newRetryOptions()
- for _, opt := range opts {
- opt(options)
- }
- var berr errorx.BatchError
- var cancelFunc context.CancelFunc
- if options.timeout > 0 {
- ctx, cancelFunc = context.WithTimeout(ctx, options.timeout)
- defer cancelFunc()
- }
- errChan := make(chan error, 1)
- for i := 0; i < options.times; i++ {
- go fn(errChan, i)
- select {
- case err := <-errChan:
- if err != nil {
- berr.Add(err)
- } else {
- return nil
- }
- case <-ctx.Done():
- berr.Add(ctx.Err())
- return berr.Err()
- }
- if options.interval > 0 {
- select {
- case <-ctx.Done():
- berr.Add(ctx.Err())
- return berr.Err()
- case <-time.After(options.interval):
- }
- }
- }
- return berr.Err()
- }
- // WithRetry customize a DoWithRetry call with given retry times.
- func WithRetry(times int) RetryOption {
- return func(options *retryOptions) {
- options.times = times
- }
- }
- func WithInterval(interval time.Duration) RetryOption {
- return func(options *retryOptions) {
- options.interval = interval
- }
- }
- func WithTimeout(timeout time.Duration) RetryOption {
- return func(options *retryOptions) {
- options.timeout = timeout
- }
- }
- func newRetryOptions() *retryOptions {
- return &retryOptions{
- times: defaultRetryTimes,
- }
- }
|