mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-22 02:15:45 +03:00
Proxy performance (#946)
* proxy: add benchmark Signed-off-by: Tw <tw19881113@gmail.com> * replacer: prepare lazily update issue#939 benchmark old ns/op new ns/op delta BenchmarkProxy-4 83865 72544 -13.50% Signed-off-by: Tw <tw19881113@gmail.com> * proxy: use buffer pool to avoid temporary allocation Signed-off-by: Tw <tw19881113@gmail.com>
This commit is contained in:
parent
1240690973
commit
beae16f07c
4 changed files with 99 additions and 37 deletions
|
@ -37,8 +37,8 @@ type Replacer interface {
|
||||||
// they will be used to overwrite other replacements
|
// they will be used to overwrite other replacements
|
||||||
// if there is a name conflict.
|
// if there is a name conflict.
|
||||||
type replacer struct {
|
type replacer struct {
|
||||||
replacements map[string]string
|
replacements map[string]func() string
|
||||||
customReplacements map[string]string
|
customReplacements map[string]func() string
|
||||||
emptyValue string
|
emptyValue string
|
||||||
responseRecorder *ResponseRecorder
|
responseRecorder *ResponseRecorder
|
||||||
}
|
}
|
||||||
|
@ -53,36 +53,36 @@ type replacer struct {
|
||||||
func NewReplacer(r *http.Request, rr *ResponseRecorder, emptyValue string) Replacer {
|
func NewReplacer(r *http.Request, rr *ResponseRecorder, emptyValue string) Replacer {
|
||||||
rep := &replacer{
|
rep := &replacer{
|
||||||
responseRecorder: rr,
|
responseRecorder: rr,
|
||||||
customReplacements: make(map[string]string),
|
customReplacements: make(map[string]func() string),
|
||||||
replacements: map[string]string{
|
replacements: map[string]func() string{
|
||||||
"{method}": r.Method,
|
"{method}": func() string { return r.Method },
|
||||||
"{scheme}": func() string {
|
"{scheme}": func() string {
|
||||||
if r.TLS != nil {
|
if r.TLS != nil {
|
||||||
return "https"
|
return "https"
|
||||||
}
|
}
|
||||||
return "http"
|
return "http"
|
||||||
}(),
|
},
|
||||||
"{hostname}": func() string {
|
"{hostname}": func() string {
|
||||||
name, err := os.Hostname()
|
name, err := os.Hostname()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return name
|
return name
|
||||||
}(),
|
},
|
||||||
"{host}": r.Host,
|
"{host}": func() string { return r.Host },
|
||||||
"{hostonly}": func() string {
|
"{hostonly}": func() string {
|
||||||
host, _, err := net.SplitHostPort(r.Host)
|
host, _, err := net.SplitHostPort(r.Host)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return r.Host
|
return r.Host
|
||||||
}
|
}
|
||||||
return host
|
return host
|
||||||
}(),
|
},
|
||||||
"{path}": r.URL.Path,
|
"{path}": func() string { return r.URL.Path },
|
||||||
"{path_escaped}": url.QueryEscape(r.URL.Path),
|
"{path_escaped}": func() string { return url.QueryEscape(r.URL.Path) },
|
||||||
"{query}": r.URL.RawQuery,
|
"{query}": func() string { return r.URL.RawQuery },
|
||||||
"{query_escaped}": url.QueryEscape(r.URL.RawQuery),
|
"{query_escaped}": func() string { return url.QueryEscape(r.URL.RawQuery) },
|
||||||
"{fragment}": r.URL.Fragment,
|
"{fragment}": func() string { return r.URL.Fragment },
|
||||||
"{proto}": r.Proto,
|
"{proto}": func() string { return r.Proto },
|
||||||
"{remote}": func() string {
|
"{remote}": func() string {
|
||||||
if fwdFor := r.Header.Get("X-Forwarded-For"); fwdFor != "" {
|
if fwdFor := r.Header.Get("X-Forwarded-For"); fwdFor != "" {
|
||||||
return fwdFor
|
return fwdFor
|
||||||
|
@ -92,25 +92,25 @@ func NewReplacer(r *http.Request, rr *ResponseRecorder, emptyValue string) Repla
|
||||||
return r.RemoteAddr
|
return r.RemoteAddr
|
||||||
}
|
}
|
||||||
return host
|
return host
|
||||||
}(),
|
},
|
||||||
"{port}": func() string {
|
"{port}": func() string {
|
||||||
_, port, err := net.SplitHostPort(r.RemoteAddr)
|
_, port, err := net.SplitHostPort(r.RemoteAddr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return port
|
return port
|
||||||
}(),
|
},
|
||||||
"{uri}": r.URL.RequestURI(),
|
"{uri}": func() string { return r.URL.RequestURI() },
|
||||||
"{uri_escaped}": url.QueryEscape(r.URL.RequestURI()),
|
"{uri_escaped}": func() string { return url.QueryEscape(r.URL.RequestURI()) },
|
||||||
"{when}": time.Now().Format(timeFormat),
|
"{when}": func() string { return time.Now().Format(timeFormat) },
|
||||||
"{file}": func() string {
|
"{file}": func() string {
|
||||||
_, file := path.Split(r.URL.Path)
|
_, file := path.Split(r.URL.Path)
|
||||||
return file
|
return file
|
||||||
}(),
|
},
|
||||||
"{dir}": func() string {
|
"{dir}": func() string {
|
||||||
dir, _ := path.Split(r.URL.Path)
|
dir, _ := path.Split(r.URL.Path)
|
||||||
return dir
|
return dir
|
||||||
}(),
|
},
|
||||||
"{request}": func() string {
|
"{request}": func() string {
|
||||||
dump, err := httputil.DumpRequest(r, false)
|
dump, err := httputil.DumpRequest(r, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -118,14 +118,15 @@ func NewReplacer(r *http.Request, rr *ResponseRecorder, emptyValue string) Repla
|
||||||
}
|
}
|
||||||
|
|
||||||
return requestReplacer.Replace(string(dump))
|
return requestReplacer.Replace(string(dump))
|
||||||
}(),
|
},
|
||||||
},
|
},
|
||||||
emptyValue: emptyValue,
|
emptyValue: emptyValue,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Header placeholders (case-insensitive)
|
// Header placeholders (case-insensitive)
|
||||||
for header, values := range r.Header {
|
for header, values := range r.Header {
|
||||||
rep.replacements[headerReplacer+strings.ToLower(header)+"}"] = strings.Join(values, ",")
|
values := values
|
||||||
|
rep.replacements[headerReplacer+strings.ToLower(header)+"}"] = func() string { return strings.Join(values, ",") }
|
||||||
}
|
}
|
||||||
|
|
||||||
return rep
|
return rep
|
||||||
|
@ -141,9 +142,9 @@ func (r *replacer) Replace(s string) string {
|
||||||
|
|
||||||
// Make response placeholders now
|
// Make response placeholders now
|
||||||
if r.responseRecorder != nil {
|
if r.responseRecorder != nil {
|
||||||
r.replacements["{status}"] = strconv.Itoa(r.responseRecorder.status)
|
r.replacements["{status}"] = func() string { return strconv.Itoa(r.responseRecorder.status) }
|
||||||
r.replacements["{size}"] = strconv.Itoa(r.responseRecorder.size)
|
r.replacements["{size}"] = func() string { return strconv.Itoa(r.responseRecorder.size) }
|
||||||
r.replacements["{latency}"] = time.Since(r.responseRecorder.start).String()
|
r.replacements["{latency}"] = func() string { return time.Since(r.responseRecorder.start).String() }
|
||||||
}
|
}
|
||||||
|
|
||||||
// Include custom placeholders, overwriting existing ones if necessary
|
// Include custom placeholders, overwriting existing ones if necessary
|
||||||
|
@ -158,7 +159,10 @@ func (r *replacer) Replace(s string) string {
|
||||||
idxEnd := strings.Index(s[endOffset:], "}")
|
idxEnd := strings.Index(s[endOffset:], "}")
|
||||||
if idxEnd > -1 {
|
if idxEnd > -1 {
|
||||||
placeholder := strings.ToLower(s[idxStart : endOffset+idxEnd+1])
|
placeholder := strings.ToLower(s[idxStart : endOffset+idxEnd+1])
|
||||||
replacement := r.replacements[placeholder]
|
replacement := ""
|
||||||
|
if getReplacement, ok := r.replacements[placeholder]; ok {
|
||||||
|
replacement = getReplacement()
|
||||||
|
}
|
||||||
if replacement == "" {
|
if replacement == "" {
|
||||||
replacement = r.emptyValue
|
replacement = r.emptyValue
|
||||||
}
|
}
|
||||||
|
@ -169,7 +173,11 @@ func (r *replacer) Replace(s string) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Regular replacements - these are easier because they're case-sensitive
|
// Regular replacements - these are easier because they're case-sensitive
|
||||||
for placeholder, replacement := range r.replacements {
|
for placeholder, getReplacement := range r.replacements {
|
||||||
|
if !strings.Contains(s, placeholder) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
replacement := getReplacement()
|
||||||
if replacement == "" {
|
if replacement == "" {
|
||||||
replacement = r.emptyValue
|
replacement = r.emptyValue
|
||||||
}
|
}
|
||||||
|
@ -181,7 +189,7 @@ func (r *replacer) Replace(s string) string {
|
||||||
|
|
||||||
// Set sets key to value in the r.customReplacements map.
|
// Set sets key to value in the r.customReplacements map.
|
||||||
func (r *replacer) Set(key, value string) {
|
func (r *replacer) Set(key, value string) {
|
||||||
r.customReplacements["{"+key+"}"] = value
|
r.customReplacements["{"+key+"}"] = func() string { return value }
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|
|
@ -21,19 +21,26 @@ func TestNewReplacer(t *testing.T) {
|
||||||
|
|
||||||
switch v := rep.(type) {
|
switch v := rep.(type) {
|
||||||
case *replacer:
|
case *replacer:
|
||||||
if v.replacements["{host}"] != "localhost" {
|
if v.replacements["{host}"]() != "localhost" {
|
||||||
t.Error("Expected host to be localhost")
|
t.Error("Expected host to be localhost")
|
||||||
}
|
}
|
||||||
if v.replacements["{method}"] != "POST" {
|
if v.replacements["{method}"]() != "POST" {
|
||||||
t.Error("Expected request method to be POST")
|
t.Error("Expected request method to be POST")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Response placeholders should only be set after call to Replace()
|
// Response placeholders should only be set after call to Replace()
|
||||||
if got, want := v.replacements["{status}"], ""; got != want {
|
got, want := "", ""
|
||||||
|
if getReplacement, ok := v.replacements["{status}"]; ok {
|
||||||
|
got = getReplacement()
|
||||||
|
}
|
||||||
|
if want := ""; got != want {
|
||||||
t.Errorf("Expected status to NOT be set before Replace() is called; was: %s", got)
|
t.Errorf("Expected status to NOT be set before Replace() is called; was: %s", got)
|
||||||
}
|
}
|
||||||
rep.Replace("{foobar}")
|
rep.Replace("{foobar}")
|
||||||
if got, want := v.replacements["{status}"], "200"; got != want {
|
if getReplacement, ok := v.replacements["{status}"]; ok {
|
||||||
|
got = getReplacement()
|
||||||
|
}
|
||||||
|
if want = "200"; got != want {
|
||||||
t.Errorf("Expected status to be %s, was: %s", want, got)
|
t.Errorf("Expected status to be %s, was: %s", want, got)
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
@ -84,10 +91,14 @@ func TestReplace(t *testing.T) {
|
||||||
|
|
||||||
complexCases := []struct {
|
complexCases := []struct {
|
||||||
template string
|
template string
|
||||||
replacements map[string]string
|
replacements map[string]func() string
|
||||||
expect string
|
expect string
|
||||||
}{
|
}{
|
||||||
{"/a{1}/{2}", map[string]string{"{1}": "12", "{2}": ""}, "/a12/"},
|
{"/a{1}/{2}",
|
||||||
|
map[string]func() string{
|
||||||
|
"{1}": func() string { return "12" },
|
||||||
|
"{2}": func() string { return "" }},
|
||||||
|
"/a12/"},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, c := range complexCases {
|
for _, c := range complexCases {
|
||||||
|
|
|
@ -761,3 +761,37 @@ func (c *fakeConn) SetWriteDeadline(t time.Time) error { return nil }
|
||||||
func (c *fakeConn) Close() error { return nil }
|
func (c *fakeConn) Close() error { return nil }
|
||||||
func (c *fakeConn) Read(b []byte) (int, error) { return c.readBuf.Read(b) }
|
func (c *fakeConn) Read(b []byte) (int, error) { return c.readBuf.Read(b) }
|
||||||
func (c *fakeConn) Write(b []byte) (int, error) { return c.writeBuf.Write(b) }
|
func (c *fakeConn) Write(b []byte) (int, error) { return c.writeBuf.Write(b) }
|
||||||
|
|
||||||
|
func BenchmarkProxy(b *testing.B) {
|
||||||
|
backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.Write([]byte("Hello, client"))
|
||||||
|
}))
|
||||||
|
defer backend.Close()
|
||||||
|
|
||||||
|
upstream := newFakeUpstream(backend.URL, false)
|
||||||
|
upstream.host.UpstreamHeaders = http.Header{
|
||||||
|
"Hostname": {"{hostname}"},
|
||||||
|
"Host": {"{host}"},
|
||||||
|
"X-Real-IP": {"{remote}"},
|
||||||
|
"X-Forwarded-Proto": {"{scheme}"},
|
||||||
|
}
|
||||||
|
// set up proxy
|
||||||
|
p := &Proxy{
|
||||||
|
Next: httpserver.EmptyNext, // prevents panic in some cases when test fails
|
||||||
|
Upstreams: []Upstream{upstream},
|
||||||
|
}
|
||||||
|
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
b.ResetTimer()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
b.StopTimer()
|
||||||
|
// create request and response recorder
|
||||||
|
r, err := http.NewRequest("GET", "/", nil)
|
||||||
|
if err != nil {
|
||||||
|
b.Fatalf("Failed to create request: %v", err)
|
||||||
|
}
|
||||||
|
b.StartTimer()
|
||||||
|
p.ServeHTTP(w, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -22,6 +22,12 @@ import (
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var bufferPool = sync.Pool{New: createBuffer}
|
||||||
|
|
||||||
|
func createBuffer() interface{} {
|
||||||
|
return make([]byte, 32*1024)
|
||||||
|
}
|
||||||
|
|
||||||
// onExitFlushLoop is a callback set by tests to detect the state of the
|
// onExitFlushLoop is a callback set by tests to detect the state of the
|
||||||
// flushLoop() goroutine.
|
// flushLoop() goroutine.
|
||||||
var onExitFlushLoop func()
|
var onExitFlushLoop func()
|
||||||
|
@ -214,6 +220,9 @@ func (p *ReverseProxy) ServeHTTP(rw http.ResponseWriter, outreq *http.Request, r
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ReverseProxy) copyResponse(dst io.Writer, src io.Reader) {
|
func (p *ReverseProxy) copyResponse(dst io.Writer, src io.Reader) {
|
||||||
|
buf := bufferPool.Get()
|
||||||
|
defer bufferPool.Put(buf)
|
||||||
|
|
||||||
if p.FlushInterval != 0 {
|
if p.FlushInterval != 0 {
|
||||||
if wf, ok := dst.(writeFlusher); ok {
|
if wf, ok := dst.(writeFlusher); ok {
|
||||||
mlw := &maxLatencyWriter{
|
mlw := &maxLatencyWriter{
|
||||||
|
@ -226,7 +235,7 @@ func (p *ReverseProxy) copyResponse(dst io.Writer, src io.Reader) {
|
||||||
dst = mlw
|
dst = mlw
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
io.Copy(dst, src)
|
io.CopyBuffer(dst, src, buf.([]byte))
|
||||||
}
|
}
|
||||||
|
|
||||||
type writeFlusher interface {
|
type writeFlusher interface {
|
||||||
|
|
Loading…
Reference in a new issue