-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patherrors.go
343 lines (297 loc) · 7.11 KB
/
errors.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
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
package core
import (
"errors"
"fmt"
)
var (
// ErrNotImplemented indicates something hasn't been implemented yet
ErrNotImplemented = errors.New("not implemented")
// ErrTODO is like ErrNotImplemented but used especially to
// indicate something needs to be implemented
ErrTODO = Wrap(ErrNotImplemented, "TODO")
// ErrExists indicates something already exists
ErrExists = errors.New("already exists")
// ErrNotExists indicates something doesn't exist
ErrNotExists = errors.New("does not exist")
// ErrInvalid indicates an argument isn't valid
ErrInvalid = errors.New("invalid argument")
// ErrUnknown indicates something isn't recognized
ErrUnknown = errors.New("unknown")
// ErrNilReceiver indicates a method was called over a nil instance
ErrNilReceiver = errors.New("nil receiver")
// ErrUnreachable indicates something impossible happened
ErrUnreachable = errors.New("unreachable")
)
var (
_ Unwrappable = (*WrappedError)(nil)
)
// Unwrappable represents an error that can be Unwrap() to get the cause
type Unwrappable interface {
Error() string
Unwrap() error
}
// Wrap annotates an error with a single string.
func Wrap(err error, msg string) error {
return doWrap(err, false, "%s", msg)
}
// Wrapf annotates an error with a formatted string.
func Wrapf(err error, format string, args ...any) error {
return doWrap(err, false, format, args...)
}
// QuietWrap replaces the text of the error it's wrapping.
func QuietWrap(err error, format string, args ...any) error {
return doWrap(err, true, format, args...)
}
func doWrap(err error, quiet bool, format string, args ...any) error {
var note string
switch {
case err == nil:
return nil
case len(args) > 0:
note = fmt.Sprintf(format, args...)
default:
note = format
}
if len(note) == 0 {
return err
}
return &WrappedError{
cause: err,
note: note,
quiet: quiet,
}
}
// WrappedError is an annotated error that can be Unwrapped
type WrappedError struct {
cause error
note string
quiet bool
}
func (w *WrappedError) Error() string {
switch {
case w == nil:
return ""
case w.cause == nil, w.quiet:
return w.note
}
s := w.cause.Error()
if len(s) == 0 {
return w.note
}
return fmt.Sprintf("%s: %s", w.note, s)
}
func (w *WrappedError) Unwrap() error {
if w == nil {
return nil
}
return w.cause
}
// TemporaryError is an error wrapper that satisfies IsTimeout()
// and IsTemporary()
type TemporaryError struct {
cause error
timeout bool
}
func (w *TemporaryError) Error() string {
var cause string
switch {
case w == nil:
return ""
case w.cause != nil:
cause = w.cause.Error()
}
switch {
case !w.timeout:
return cause
case cause == "":
return "time-out"
default:
return fmt.Sprintf("%s: %s", "time-out", cause)
}
}
// IsTemporary tells this error is temporary.
func (*TemporaryError) IsTemporary() bool { return true }
// IsTimeout tells if this error is a time-out or not.
func (w *TemporaryError) IsTimeout() bool {
if w != nil {
return w.timeout
}
return false
}
// Temporary tells this error is temporary.
func (*TemporaryError) Temporary() bool { return true }
// Timeout tells if this error is a time-out or not.
func (w *TemporaryError) Timeout() bool { return w.IsTimeout() }
// NewTimeoutError returns an error that returns true
// to IsTimeout() and IsTemporary()
func NewTimeoutError(err error) error {
return &TemporaryError{
cause: err,
timeout: true,
}
}
// NewTemporaryError returns an error that returns false
// to IsTimeout() and true to IsTemporary()
func NewTemporaryError(err error) error {
return &TemporaryError{
cause: err,
}
}
// CoalesceError returns the first non-nil error argument.
// error isn't compatible with Coalesce's comparable generic
// type.
func CoalesceError(errs ...error) error {
for _, err := range errs {
if err != nil {
return err
}
}
return nil
}
// Unwrap unwraps one layer of a compound error,
// ensuring there are no nil entries.
func Unwrap(err error) []error {
var errs []error
if err == nil {
return nil
}
switch w := err.(type) {
case interface {
Unwrap() []error
}:
errs = w.Unwrap()
case interface {
Errors() []error
}:
errs = w.Errors()
case interface {
Unwrap() error
}:
errs = append(errs, w.Unwrap())
}
return SliceReplaceFn(errs, func(_ []error, err error) (error, bool) {
return err, err != nil
})
}
// IsError recursively check if the given error is in in the given list,
// or just non-nil if no options to check are given.
func IsError(err error, errs ...error) bool {
switch {
case err == nil:
return false
case len(errs) == 0:
return true
}
fn := func(err error) bool {
for _, e := range errs {
if err == e {
return true
}
}
return false
}
return IsErrorFn(fn, err)
}
// IsErrorFn recursively checks if any of the given errors satisfies
// the specified check function.
//
// revive:disable:cognitive-complexity
func IsErrorFn(check func(error) bool, errs ...error) bool {
// revive:enable:cognitive-complexity
if check == nil || len(errs) == 0 {
return false
}
// direct match first
for _, e := range errs {
if e != nil && check(e) {
return true
}
}
// and unwrapping
for _, e := range errs {
if errs := Unwrap(e); len(errs) > 0 {
if IsErrorFn(check, errs...) {
return true
}
}
}
return false
}
// IsErrorFn2 recursively checks if any of the given errors gets a
// certain answer from the check function.
// As opposed to IsErrorFn, IsErrorFn2 will stop when it has certainty
// of a false result.
//
// revive:disable:cognitive-complexity
func IsErrorFn2(check func(error) (bool, bool), errs ...error) (is bool, known bool) {
// revive:enable:cognitive-complexity
if check == nil || len(errs) == 0 {
return false, true
}
// direct match first
for _, e := range errs {
if e != nil {
if is, known = check(e); known {
return is, true
}
}
}
// and unwrapping
for _, e := range errs {
if errs := Unwrap(e); len(errs) > 0 {
if is, known = IsErrorFn2(check, errs...); known {
return is, true
}
}
}
// unknown
return false, false
}
// CheckIsTemporary tests an error for Temporary(), IsTemporary(),
// Timeout() and IsTimeout() without unwrapping.
func CheckIsTemporary(err error) (is, known bool) {
switch e := err.(type) {
case nil:
return false, true
case interface {
Temporary() bool
}:
return e.Temporary(), true
case interface {
IsTemporary() bool
}:
return e.IsTemporary(), true
default:
return CheckIsTimeout(err)
}
}
// IsTemporary tests an error for Temporary(), IsTemporary(),
// Timeout() and IsTimeout() recursively.
func IsTemporary(err error) bool {
is, _ := IsErrorFn2(CheckIsTemporary, err)
return is
}
// CheckIsTimeout tests an error for Timeout() and IsTimeout()
// without unwrapping.
func CheckIsTimeout(err error) (is, known bool) {
switch e := err.(type) {
case nil:
return false, true
case interface {
Timeout() bool
}:
return e.Timeout(), true
case interface {
IsTimeout() bool
}:
return e.IsTimeout(), true
default:
return false, false
}
}
// IsTimeout tests an error for Timeout() and IsTimeout()
// recursively.
func IsTimeout(err error) bool {
is, _ := IsErrorFn2(CheckIsTimeout, err)
return is
}