/
retrying_client.go
149 lines (118 loc) · 3.65 KB
/
retrying_client.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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
package envtest
import (
"context"
"strings"
"time"
"github.com/pkg/errors"
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/gardener/landscaper/hack/testcluster/pkg/utils"
)
type retryingClient struct {
client.Client
log utils.Logger
}
func NewRetryingClient(innerClient client.Client, log utils.Logger) client.Client {
if log == nil {
log = utils.NewDiscardLogger()
}
return &retryingClient{
Client: innerClient,
log: log,
}
}
func (r *retryingClient) Get(ctx context.Context, key client.ObjectKey, obj client.Object, opts ...client.GetOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.Client.Get(ctx, key, obj, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingClient: "+err.Error())
}
return err
}
func (r *retryingClient) List(ctx context.Context, list client.ObjectList, opts ...client.ListOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.Client.List(ctx, list, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingClient: "+err.Error())
}
return err
}
func (r *retryingClient) Create(ctx context.Context, obj client.Object, opts ...client.CreateOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.Client.Create(ctx, obj, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingClient: "+err.Error())
}
return err
}
func (r *retryingClient) Update(ctx context.Context, obj client.Object, opts ...client.UpdateOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.Client.Update(ctx, obj, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingClient: "+err.Error())
}
return err
}
func (r *retryingClient) Patch(ctx context.Context, obj client.Object, patch client.Patch, opts ...client.PatchOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.Client.Patch(ctx, obj, patch, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingClient: "+err.Error())
}
return err
}
func (r *retryingClient) Status() client.SubResourceWriter {
return &retryingSubResourceWriter{
SubResourceWriter: r.Client.Status(),
log: r.log,
}
}
type retryingSubResourceWriter struct {
client.SubResourceWriter
log utils.Logger
}
func (r *retryingSubResourceWriter) Update(ctx context.Context, obj client.Object, opts ...client.SubResourceUpdateOption) error {
err := retrySporadic(ctx, r.log, func() error {
return r.SubResourceWriter.Update(ctx, obj, opts...)
})
if err != nil {
err = errors.Wrap(err, "retryingSubResourceWriter: "+err.Error())
}
return err
}
func retrySporadic(ctx context.Context, log utils.Logger, fn func() error) error {
retries := 10
for i := 0; i < retries; i++ {
if err := ctx.Err(); err != nil {
log.Logfln("retrying client: context is closed: %w", err)
return err
}
err := fn()
if err == nil {
return nil
} else if i == retries-1 {
log.Logfln("retrying client: all attempts failed: %w", err)
return err
} else if !isSporadicError(err) {
log.Logfln("retrying client: stop retrying after non-sporadic error: %w", err)
return err
} else if ctxError := ctx.Err(); ctxError != nil {
log.Logfln("retrying client: stop retrying because context is closed: %w", ctxError)
return ctxError
} else {
log.Logfln("retrying client: continue retrying after sporadic error: %w", err)
time.Sleep(3 * time.Second)
}
}
return nil
}
func isSporadicError(err error) bool {
return strings.Contains(err.Error(), "connection refused") ||
strings.Contains(err.Error(), "context deadline exceeded") ||
strings.Contains(err.Error(), "failed to call webhook") ||
strings.Contains(err.Error(), "connection reset by peer")
}