create -kbps to rate limit responses
parent
8903c8793b
commit
42b2344fef
10
config.go
10
config.go
|
|
@ -4,18 +4,22 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"local/args"
|
"local/args"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/time/rate"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
Listen string
|
Listen string
|
||||||
Timeout time.Duration
|
Timeout time.Duration
|
||||||
TLSInsecure bool
|
TLSInsecure bool
|
||||||
|
Limiter *rate.Limiter
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewConfig() *Config {
|
func NewConfig() *Config {
|
||||||
as := args.NewArgSet()
|
as := args.NewArgSet()
|
||||||
|
|
||||||
as.Append(args.INT, "p", "port to listen on", 61113)
|
as.Append(args.INT, "p", "port to listen on", 61113)
|
||||||
|
as.Append(args.INT, "kbps", "port to listen on", 61113)
|
||||||
as.Append(args.BOOL, "tls-insecure", "permit tls insecure", false)
|
as.Append(args.BOOL, "tls-insecure", "permit tls insecure", false)
|
||||||
as.Append(args.DURATION, "t", "timeout", time.Minute)
|
as.Append(args.DURATION, "t", "timeout", time.Minute)
|
||||||
|
|
||||||
|
|
@ -23,9 +27,15 @@ func NewConfig() *Config {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var limiter *rate.Limiter
|
||||||
|
if kbps := as.GetInt("kbps"); kbps > 0 {
|
||||||
|
limiter = rate.NewLimiter(rate.Limit(kbps), 100*1024)
|
||||||
|
}
|
||||||
|
|
||||||
return &Config{
|
return &Config{
|
||||||
Listen: fmt.Sprintf(":%v", as.GetInt("p")),
|
Listen: fmt.Sprintf(":%v", as.GetInt("p")),
|
||||||
Timeout: as.GetDuration("t"),
|
Timeout: as.GetDuration("t"),
|
||||||
TLSInsecure: as.GetBool("tls-insecure"),
|
TLSInsecure: as.GetBool("tls-insecure"),
|
||||||
|
Limiter: limiter,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,12 @@
|
||||||
|
module gogs.inhome.blapointe.com/mfproxy
|
||||||
|
|
||||||
|
go 1.18
|
||||||
|
|
||||||
|
require (
|
||||||
|
golang.org/x/time v0.0.0-20220722155302-e5dcc9cfc0b9
|
||||||
|
local/args v0.0.0-00010101000000-000000000000
|
||||||
|
)
|
||||||
|
|
||||||
|
require gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||||
|
|
||||||
|
replace local/args => ../../local/args
|
||||||
|
|
@ -0,0 +1,6 @@
|
||||||
|
golang.org/x/time v0.0.0-20220722155302-e5dcc9cfc0b9 h1:ftMN5LMiBFjbzleLqtoBZk7KdJwhuybIU+FckUHgoyQ=
|
||||||
|
golang.org/x/time v0.0.0-20220722155302-e5dcc9cfc0b9/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
||||||
|
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
||||||
13
server.go
13
server.go
|
|
@ -8,10 +8,13 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/time/rate"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Server struct {
|
type Server struct {
|
||||||
Transport http.RoundTripper
|
Transport http.RoundTripper
|
||||||
|
limiter *rate.Limiter
|
||||||
Timeout time.Duration
|
Timeout time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -21,6 +24,7 @@ func NewServer(c *Config) *Server {
|
||||||
InsecureSkipVerify: c.TLSInsecure,
|
InsecureSkipVerify: c.TLSInsecure,
|
||||||
}
|
}
|
||||||
return &Server{
|
return &Server{
|
||||||
|
limiter: c.Limiter,
|
||||||
Transport: transport,
|
Transport: transport,
|
||||||
Timeout: c.Timeout,
|
Timeout: c.Timeout,
|
||||||
}
|
}
|
||||||
|
|
@ -86,5 +90,12 @@ func (s *Server) Serve(w http.ResponseWriter, r *http.Request) {
|
||||||
w.Header().Add(k, s)
|
w.Header().Add(k, s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
io.Copy(w, resp.Body)
|
io.Copy(
|
||||||
|
throttledWriter{
|
||||||
|
ctx: r.Context(),
|
||||||
|
w: w,
|
||||||
|
limiter: s.limiter,
|
||||||
|
},
|
||||||
|
resp.Body,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,26 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"golang.org/x/time/rate"
|
||||||
|
)
|
||||||
|
|
||||||
|
type throttledWriter struct {
|
||||||
|
ctx context.Context
|
||||||
|
w io.Writer
|
||||||
|
limiter *rate.Limiter
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tw throttledWriter) Write(b []byte) (int, error) {
|
||||||
|
if tw.limiter != nil {
|
||||||
|
if block := tw.limiter.Burst(); len(b) > block {
|
||||||
|
b = b[:block]
|
||||||
|
}
|
||||||
|
if err := tw.limiter.WaitN(tw.ctx, len(b)); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return tw.w.Write(b)
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue