@@ -19,12 +19,21 @@ import (
19
19
"io"
20
20
"net/http"
21
21
"regexp"
22
+ "strconv"
22
23
"time"
23
24
24
25
"github.com/gregjones/httpcache"
25
26
"github.com/rs/zerolog"
26
27
)
27
28
29
+ const (
30
+ httpHeaderRateLimit = "X-Ratelimit-Limit"
31
+ httpHeaderRateRemaining = "X-Ratelimit-Remaining"
32
+ httpHeaderRateUsed = "X-Ratelimit-Used"
33
+ httpHeaderRateReset = "X-Ratelimit-Reset"
34
+ httpHeaderRateResource = "X-Ratelimit-Resource"
35
+ )
36
+
28
37
// ClientLogging creates client middleware that logs request and response
29
38
// information at the given level. If the request fails without creating a
30
39
// response, it is logged with a status code of -1. The middleware uses a
@@ -83,6 +92,7 @@ func ClientLogging(lvl zerolog.Level, opts ...ClientLoggingOption) ClientMiddlew
83
92
Int64 ("size" , - 1 )
84
93
}
85
94
95
+ addRateLimitInformationToLog (options .LogRateLimitInformation , evt , res )
86
96
evt .Msg ("github_request" )
87
97
return res , err
88
98
})
@@ -95,6 +105,18 @@ type ClientLoggingOption func(*clientLoggingOptions)
95
105
type clientLoggingOptions struct {
96
106
RequestBodyPatterns []* regexp.Regexp
97
107
ResponseBodyPatterns []* regexp.Regexp
108
+
109
+ // Output control
110
+ LogRateLimitInformation * RateLimitLoggingOption
111
+ }
112
+
113
+ // RateLimitLoggingOption controls which rate limit information is logged.
114
+ type RateLimitLoggingOption struct {
115
+ Limit bool
116
+ Remaining bool
117
+ Used bool
118
+ Reset bool
119
+ Resource bool
98
120
}
99
121
100
122
// LogRequestBody enables request body logging for requests to paths matching
@@ -117,6 +139,14 @@ func LogResponseBody(patterns ...string) ClientLoggingOption {
117
139
}
118
140
}
119
141
142
+ // LogRateLimitInformation defines which rate limit information like
143
+ // the number of requests remaining in the current rate limit window is getting logged.
144
+ func LogRateLimitInformation (options * RateLimitLoggingOption ) ClientLoggingOption {
145
+ return func (opts * clientLoggingOptions ) {
146
+ opts .LogRateLimitInformation = options
147
+ }
148
+ }
149
+
120
150
func mirrorRequestBody (r * http.Request ) (* http.Request , []byte , error ) {
121
151
switch {
122
152
case r .Body == nil || r .Body == http .NoBody :
@@ -174,3 +204,34 @@ func requestMatches(r *http.Request, pats []*regexp.Regexp) bool {
174
204
func closeBody (b io.ReadCloser ) {
175
205
_ = b .Close () // per http.Transport impl, ignoring close errors is fine
176
206
}
207
+
208
+ func addRateLimitInformationToLog (loggingOptions * RateLimitLoggingOption , evt * zerolog.Event , res * http.Response ) {
209
+ // Exit early if no rate limit information is requested
210
+ if loggingOptions == nil {
211
+ return
212
+ }
213
+
214
+ rateLimitDict := zerolog .Dict ()
215
+ if limitHeader := res .Header .Get (httpHeaderRateLimit ); loggingOptions .Limit && limitHeader != "" {
216
+ limit , _ := strconv .Atoi (limitHeader )
217
+ rateLimitDict .Int ("limit" , limit )
218
+ }
219
+ if remainingHeader := res .Header .Get (httpHeaderRateRemaining ); loggingOptions .Remaining && remainingHeader != "" {
220
+ remaining , _ := strconv .Atoi (remainingHeader )
221
+ rateLimitDict .Int ("remaining" , remaining )
222
+ }
223
+ if usedHeader := res .Header .Get (httpHeaderRateUsed ); loggingOptions .Used && usedHeader != "" {
224
+ used , _ := strconv .Atoi (usedHeader )
225
+ rateLimitDict .Int ("used" , used )
226
+ }
227
+ if resetHeader := res .Header .Get (httpHeaderRateReset ); loggingOptions .Reset && resetHeader != "" {
228
+ if v , _ := strconv .ParseInt (resetHeader , 10 , 64 ); v != 0 {
229
+ rateLimitDict .Time ("reset" , time .Unix (v , 0 ))
230
+ }
231
+ }
232
+ if resourceHeader := res .Header .Get (httpHeaderRateResource ); loggingOptions .Resource && resourceHeader != "" {
233
+ rateLimitDict .Str ("resource" , resourceHeader )
234
+ }
235
+
236
+ evt .Dict ("ratelimit" , rateLimitDict )
237
+ }
0 commit comments