-
Notifications
You must be signed in to change notification settings - Fork 819
/
Copy pathretry.go
80 lines (72 loc) · 2.63 KB
/
retry.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
// Copyright 2018 The Go Cloud Development Kit Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Package retry provides retry logic.
package retry // import "gocloud.dev/internal/retry"
import (
"context"
"fmt"
"time"
"github.com/googleapis/gax-go/v2"
)
// Call calls the supplied function f repeatedly, using the isRetryable function and
// the provided backoff parameters to control the repetition.
//
// When f returns nil, Call immediately returns nil.
//
// When f returns an error for which isRetryable returns false, Call immediately
// returns that error.
//
// When f returns an error for which isRetryable returns true, Call sleeps for the
// provided backoff value and invokes f again.
//
// When the provided context is done, Retry returns a ContextError that includes both
// ctx.Error() and the last error returned by f, or nil if there isn't one.
func Call(ctx context.Context, bo gax.Backoff, isRetryable func(error) bool, f func() error) error {
return call(ctx, bo, isRetryable, f, gax.Sleep)
}
// Split out for testing.
func call(ctx context.Context, bo gax.Backoff, isRetryable func(error) bool, f func() error,
sleep func(context.Context, time.Duration) error,
) error {
// Do nothing if context is done on entry.
if err := ctx.Err(); err != nil {
return &ContextError{CtxErr: err}
}
for {
err := f()
if err == nil {
return nil
}
if !isRetryable(err) {
return err
}
if cerr := sleep(ctx, bo.Pause()); cerr != nil {
return &ContextError{CtxErr: cerr, FuncErr: err}
}
}
}
// A ContextError contains both a context error (either context.Canceled or
// context.DeadlineExceeded), and the last error from the function being retried,
// or nil if the function was never called.
type ContextError struct {
CtxErr error // The error obtained from ctx.Err()
FuncErr error // The error obtained from the function being retried, or nil
}
func (e *ContextError) Error() string {
return fmt.Sprintf("%v; last error: %v", e.CtxErr, e.FuncErr)
}
// Is returns true iff one of the two errors held in e is equal to target.
func (e *ContextError) Is(target error) bool {
return e.CtxErr == target || e.FuncErr == target
}