2017-10-08 11:28:03 -07:00
|
|
|
package limiter
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2017-12-29 12:43:49 +01:00
|
|
|
"net/http"
|
2017-10-08 11:28:03 -07:00
|
|
|
|
|
|
|
"github.com/juju/ratelimit"
|
|
|
|
)
|
|
|
|
|
|
|
|
type staticLimiter struct {
|
|
|
|
upstream *ratelimit.Bucket
|
|
|
|
downstream *ratelimit.Bucket
|
|
|
|
}
|
|
|
|
|
2022-06-22 18:29:58 +02:00
|
|
|
// Limits represents static upload and download limits.
|
|
|
|
// For both, zero means unlimited.
|
|
|
|
type Limits struct {
|
|
|
|
UploadKb int
|
|
|
|
DownloadKb int
|
|
|
|
}
|
|
|
|
|
2017-10-08 11:28:03 -07:00
|
|
|
// NewStaticLimiter constructs a Limiter with a fixed (static) upload and
|
|
|
|
// download rate cap
|
2022-06-22 18:29:58 +02:00
|
|
|
func NewStaticLimiter(l Limits) Limiter {
|
2017-10-08 11:28:03 -07:00
|
|
|
var (
|
|
|
|
upstreamBucket *ratelimit.Bucket
|
|
|
|
downstreamBucket *ratelimit.Bucket
|
|
|
|
)
|
|
|
|
|
2022-06-22 18:29:58 +02:00
|
|
|
if l.UploadKb > 0 {
|
|
|
|
upstreamBucket = ratelimit.NewBucketWithRate(toByteRate(l.UploadKb), int64(toByteRate(l.UploadKb)))
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
2022-06-22 18:29:58 +02:00
|
|
|
if l.DownloadKb > 0 {
|
|
|
|
downstreamBucket = ratelimit.NewBucketWithRate(toByteRate(l.DownloadKb), int64(toByteRate(l.DownloadKb)))
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return staticLimiter{
|
|
|
|
upstream: upstreamBucket,
|
|
|
|
downstream: downstreamBucket,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l staticLimiter) Upstream(r io.Reader) io.Reader {
|
2018-05-22 20:48:17 +02:00
|
|
|
return l.limitReader(r, l.upstream)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l staticLimiter) UpstreamWriter(w io.Writer) io.Writer {
|
|
|
|
return l.limitWriter(w, l.upstream)
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (l staticLimiter) Downstream(r io.Reader) io.Reader {
|
2018-05-22 20:48:17 +02:00
|
|
|
return l.limitReader(r, l.downstream)
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
2021-01-01 12:46:20 +01:00
|
|
|
func (l staticLimiter) DownstreamWriter(w io.Writer) io.Writer {
|
|
|
|
return l.limitWriter(w, l.downstream)
|
|
|
|
}
|
|
|
|
|
2017-12-29 12:43:49 +01:00
|
|
|
type roundTripper func(*http.Request) (*http.Response, error)
|
|
|
|
|
|
|
|
func (rt roundTripper) RoundTrip(req *http.Request) (*http.Response, error) {
|
|
|
|
return rt(req)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l staticLimiter) roundTripper(rt http.RoundTripper, req *http.Request) (*http.Response, error) {
|
2021-04-24 11:46:06 +02:00
|
|
|
type readCloser struct {
|
|
|
|
io.Reader
|
|
|
|
io.Closer
|
|
|
|
}
|
|
|
|
|
2017-12-29 12:43:49 +01:00
|
|
|
if req.Body != nil {
|
2021-04-24 11:46:06 +02:00
|
|
|
req.Body = &readCloser{
|
|
|
|
Reader: l.Upstream(req.Body),
|
|
|
|
Closer: req.Body,
|
2017-12-29 12:43:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res, err := rt.RoundTrip(req)
|
|
|
|
|
|
|
|
if res != nil && res.Body != nil {
|
2021-04-24 11:46:06 +02:00
|
|
|
res.Body = &readCloser{
|
|
|
|
Reader: l.Downstream(res.Body),
|
|
|
|
Closer: res.Body,
|
2017-12-29 12:43:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Transport returns an HTTP transport limited with the limiter l.
|
|
|
|
func (l staticLimiter) Transport(rt http.RoundTripper) http.RoundTripper {
|
|
|
|
return roundTripper(func(req *http.Request) (*http.Response, error) {
|
|
|
|
return l.roundTripper(rt, req)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-05-22 20:48:17 +02:00
|
|
|
func (l staticLimiter) limitReader(r io.Reader, b *ratelimit.Bucket) io.Reader {
|
2017-10-08 11:28:03 -07:00
|
|
|
if b == nil {
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
return ratelimit.Reader(r, b)
|
|
|
|
}
|
|
|
|
|
2018-05-22 20:48:17 +02:00
|
|
|
func (l staticLimiter) limitWriter(w io.Writer, b *ratelimit.Bucket) io.Writer {
|
|
|
|
if b == nil {
|
|
|
|
return w
|
|
|
|
}
|
|
|
|
return ratelimit.Writer(w, b)
|
|
|
|
}
|
|
|
|
|
2017-10-08 11:28:03 -07:00
|
|
|
func toByteRate(val int) float64 {
|
|
|
|
return float64(val) * 1024.
|
|
|
|
}
|