forked from hashicorp/go-multierror
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmultierror.go
93 lines (79 loc) · 2.16 KB
/
multierror.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
package multierror
import (
"encoding/json"
"errors"
"fmt"
"sync"
)
// Error is an error type to track multiple errors. This is used to
// accumulate errors in cases and return them as a single "error".
type Error struct {
m sync.Mutex
Errors []error
ErrorFormat ErrorFormatFunc
}
func (e *Error) Error() string {
fn := e.ErrorFormat
if fn == nil {
fn = ListFormatFunc
}
return fn(e.Errors)
}
// MarshalJSON returns a valid json representation of a multierror,
// as an object with an array of error strings.
func (e *Error) MarshalJSON() ([]byte, error) {
j := map[string][]string{
"errors": []string{},
}
for _, err := range e.Errors {
j["errors"] = append(j["errors"], err.Error())
}
return json.Marshal(j)
}
// UnmarshalJSON parses the output of Marshal json.
func (e *Error) UnmarshalJSON(b []byte) error {
j := make(map[string][]string)
err := json.Unmarshal(b, &j)
if err != nil {
return err
}
if _, ok := j["errors"]; ok {
for _, msg := range j["errors"] {
e.Errors = append(e.Errors, errors.New(msg))
}
}
return nil
}
// ErrorOrNil returns an error interface if this Error represents
// a list of errors, or returns nil if the list of errors is empty. This
// function is useful at the end of accumulation to make sure that the value
// returned represents the existence of errors.
func (e *Error) ErrorOrNil() error {
if e == nil {
return nil
}
if len(e.Errors) == 0 {
return nil
}
return e
}
func (e *Error) GoString() string {
return fmt.Sprintf("*%#v", *e)
}
// WrappedErrors returns the list of errors that this Error is wrapping.
// It is an implementation of the errwrap.Wrapper interface so that
// multierror.Error can be used with that library.
//
// This method is not safe to be called concurrently and is no different
// than accessing the Errors field directly. It is implemented only to
// satisfy the errwrap.Wrapper interface.
func (e *Error) WrappedErrors() []error {
return e.Errors
}
// AppendSafe appends multiple errors in the current Error instance
// It is thread-safe and can be used in concurrent environments
func (e *Error) AppendSafe(errs ...error) {
e.m.Lock()
defer e.m.Unlock()
e = Append(e, errs...)
}