Overview
It'd be nice if this library supported a way for http.Client outgoing requests to utilizing limiter.Limit to rate limit based on configured API rate limits useful for things like:
Features
- Filter and intercept
http.Client requests and map to a limit.Limiter which could then have some policy callback to let the client
- block and wait until reset time
- abort request and send error back
- maybe fake an HTTP rate-limit response (but avoid hitting actual servers)
- HTTP API's usually return with headers like
X-RateLimit-* it'd be useful if there was a way to call limiter.Limiter.Set(key string, c *Context) to attempt to keep in sync with server state
I think you'd want to be able to configure a limit.Limiter per URL request path e.g.:
/1.1/users/user_timeline.json
Similar to how server-side HTTP handlers are setup:
mux.HandleFunc("/1.1/statuses/user_timeline.json", func(r *http.Request) error {
c, err := userTimelineLimiter.Get("user_timeline.json")
if err != nil {
return err
}
if c.Reached {
// client policy:
// 1. wait and retry
// 2. error rate limit
// 3. simulate http response error with rate limit
err := handleRatelimitReached(r)
}
})
Maybe even use http.NewServeMux to client-side proxy rate limit responses?
There's some example of it here:
// RateLimitedTransport is used to conform to rate limits that are
// communicated through "X-RateLimit-" headers, like GitHub's API. It
// implements http.RoundTripper and can be used for configuring a http.Client.
type RateLimitedTransport struct {
Base http.RoundTripper
}
func (t *RateLimitedTransport) RoundTrip(req *http.Request) (*http.Response, error) {
res, err := t.base().RoundTrip(req)
if err != nil {
return res, err
}
// Fetch headers
remStr := res.Header.Get("X-RateLimit-Remaining")
if remStr == "" {
return res, err
}
resetStr := res.Header.Get("X-RateLimit-Reset")
if resetStr == "" {
return res, err
}
rem, err := strconv.Atoi(remStr)
if err != nil {
return res, err
}
epoch, err := strconv.ParseInt(resetStr, 10, 64)
if err != nil {
return res, err
}
reset := time.Unix(epoch, 0)
// Determine sleep time
untilReset := reset.Sub(time.Now())
delay := time.Duration(float64(untilReset) / (float64(rem) + 1))
time.Sleep(delay)
return res, err
}
Overview
It'd be nice if this library supported a way for
http.Clientoutgoing requests to utilizinglimiter.Limitto rate limit based on configured API rate limits useful for things like:Features
http.Clientrequests and map to alimit.Limiterwhich could then have some policy callback to let the clientX-RateLimit-*it'd be useful if there was a way to calllimiter.Limiter.Set(key string, c *Context)to attempt to keep in sync with server stateI think you'd want to be able to configure a
limit.Limiterper URL request path e.g.:/1.1/users/user_timeline.jsonSimilar to how server-side HTTP handlers are setup:
Maybe even use
http.NewServeMuxto client-side proxy rate limit responses?There's some example of it here: