forked from atlassian/jec
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathretryer.go
100 lines (83 loc) · 2.13 KB
/
retryer.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
package retryer
import (
"fmt"
"github.com/pkg/errors"
"github.com/sirupsen/logrus"
"io"
"io/ioutil"
"math"
"net"
"net/http"
"time"
)
const maxRetryCount = 5
const timeout = 40 * time.Second
var DefaultClient = &http.Client{Timeout: timeout}
var retryStatusCodes = map[int]struct{}{
429: {},
}
type Retryer struct {
DoFunc func(retryer *Retryer, request *Request) (*http.Response, error)
client *http.Client
}
func (r *Retryer) Do(request *Request) (*http.Response, error) {
if r.DoFunc != nil {
return r.DoFunc(r, request)
}
return DoWithExponentialBackoff(r, request)
}
func shouldRetry(statusCode int) bool {
_, shouldRetry := retryStatusCodes[statusCode]
if (statusCode >= 500 && statusCode <= 599) || shouldRetry {
return true
}
return false
}
func getWaitTime(retryCount int) time.Duration {
waitTime := math.Pow(2, float64(retryCount)) * 100
return time.Duration(waitTime) * time.Millisecond
}
func DoWithExponentialBackoff(retryer *Retryer, request *Request) (*http.Response, error) {
client := DefaultClient
if retryer.client != nil {
client = retryer.client
}
retryCount := 0
errMessage := ""
for {
if request.body != nil {
_, err := request.body.Seek(0, 0)
if err != nil {
return nil, err
}
}
response, err := client.Do(request.Request)
if err, ok := err.(net.Error); ok {
// On error, any Response can be ignored.
if err.Timeout() {
logrus.Warn(err)
} else {
return nil, err
}
} else if shouldRetry(response.StatusCode) {
// If the returned error is nil, the Response will contain a non-nil
// Body which the user is expected to close.
io.Copy(ioutil.Discard, response.Body)
response.Body.Close()
} else {
return response, err
}
retryCount++
if retryCount == maxRetryCount {
if err != nil {
errMessage = fmt.Sprintf("last error: %s", err)
} else {
errMessage = fmt.Sprintf("status code: %d", response.StatusCode)
}
break
}
waitDuration := getWaitTime(retryCount - 1)
time.Sleep(waitDuration)
}
return nil, errors.Errorf("Couldn't get a success response, maximum retry count[%d] is exceeded, %s", maxRetryCount, errMessage)
}