-
Notifications
You must be signed in to change notification settings - Fork 3
/
errors.go
123 lines (107 loc) · 3 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
package chartmogul
import (
"net"
"net/http"
"strconv"
"strings"
"github.com/parnurzeal/gorequest"
"github.com/pkg/errors"
)
// HTTPError is wrapper to easily handle HTTP states.
type HTTPError interface {
StatusCode() int
Status() string
Response() string
}
// RequestErrors wraps multiple request errors to normal 1 error struct.
type RequestErrors interface {
Errors() []error
}
type httpError struct {
statusCode int
status string
response string
}
func (e httpError) StatusCode() int {
return e.statusCode
}
func (e httpError) Status() string {
return e.status
}
func (e httpError) Response() string {
return e.response
}
func (e httpError) Error() string {
return strings.Join([]string{strconv.Itoa(e.statusCode), e.status, e.response}, ": ")
}
type requestErrors struct {
errors []error
}
func (e requestErrors) Errors() []error {
return e.errors
}
func (e requestErrors) Error() string {
errs := make([]string, len(e.errors))
for i := range errs {
errs[i] = e.errors[i].Error()
}
return strings.Join(errs, "; ")
}
// LogErrors converts any unexpected HTTP statuses on API to errors wrapped in a handy struct.
// If there are multiple errors with request, it returns them as a wrapper struct RequestErrors.
//
// In case of no errors returns nil.
func wrapErrors(response gorequest.Response, body []byte, errs []error) error {
if response != nil && (response.StatusCode < 200 || response.StatusCode >= 300) {
return errors.Wrap(&httpError{
statusCode: response.StatusCode,
status: response.Status,
response: string(body)},
"API error")
}
if len(errs) != 0 {
return errors.Wrap(&requestErrors{errs}, "Request error")
}
return nil
}
// List of HTTP status codes which are retryable.
var retryableHTTPStatusCodes = map[int]struct{}{
http.StatusTooManyRequests: {},
http.StatusInternalServerError: {},
http.StatusBadGateway: {},
http.StatusServiceUnavailable: {},
http.StatusGatewayTimeout: {},
// Add more HTTP status codes here.
}
// isHTTPStatusRetryable return true if error message contains the HTTP statuses that needs to be retried
func isHTTPStatusRetryable(res gorequest.Response) (ok bool) {
if res == nil {
return false
}
_, ok = retryableHTTPStatusCodes[res.StatusCode]
return ok
}
// timeoutError - when reading response body takes too long
// It's inside an internal package, so only way to assert this error is to match with following error string
// https://github.com/golang/go/blob/master/src/internal/poll/fd.go#L40-L45
const timeoutError = "i/o timeout"
// networkError returns true if the underlying error is caused by net.OpError
// or if it's an i/o timeout error
func networkError(err error) bool {
if strings.Contains(err.Error(), timeoutError) {
return true
}
if _, ok := (err).(*net.OpError); ok {
return true
}
return false
}
// networkErrors checks if any of the errors is a Network related error
func networkErrors(errs []error) bool {
for _, err := range errs {
if networkError(err) {
return true
}
}
return false
}