mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-13 22:51:08 -05:00
Merge pull request #1112 from tw4452852/proxy_header
proxy: don't append predefined headers
This commit is contained in:
commit
d01bcd591c
2 changed files with 72 additions and 26 deletions
|
@ -13,6 +13,7 @@ import (
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"reflect"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
|
@ -407,16 +408,19 @@ func TestUpstreamHeadersUpdate(t *testing.T) {
|
||||||
replacer := httpserver.NewReplacer(r, nil, "")
|
replacer := httpserver.NewReplacer(r, nil, "")
|
||||||
|
|
||||||
headerKey := "Merge-Me"
|
headerKey := "Merge-Me"
|
||||||
values, ok := actualHeaders[headerKey]
|
got := actualHeaders[headerKey]
|
||||||
if !ok {
|
expect := []string{"Initial", "Merge-Value"}
|
||||||
t.Errorf("Request sent to upstream backend does not contain expected %v header. Expected header to be added", headerKey)
|
if !reflect.DeepEqual(got, expect) {
|
||||||
} else if len(values) < 2 && (values[0] != "Initial" || values[1] != replacer.Replace("{hostname}")) {
|
t.Errorf("Request sent to upstream backend does not contain expected %v header: expect %v, but got %v",
|
||||||
t.Errorf("Values for proxy header `+Merge-Me` should be merged. Got %v", values)
|
headerKey, expect, got)
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Add-Me"
|
headerKey = "Add-Me"
|
||||||
if _, ok := actualHeaders[headerKey]; !ok {
|
got = actualHeaders[headerKey]
|
||||||
t.Errorf("Request sent to upstream backend does not contain expected %v header", headerKey)
|
expect = []string{"Add-Value"}
|
||||||
|
if !reflect.DeepEqual(got, expect) {
|
||||||
|
t.Errorf("Request sent to upstream backend does not contain expected %v header: expect %v, but got %v",
|
||||||
|
headerKey, expect, got)
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Remove-Me"
|
headerKey = "Remove-Me"
|
||||||
|
@ -425,12 +429,11 @@ func TestUpstreamHeadersUpdate(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Replace-Me"
|
headerKey = "Replace-Me"
|
||||||
headerValue := replacer.Replace("{hostname}")
|
got = actualHeaders[headerKey]
|
||||||
value, ok := actualHeaders[headerKey]
|
expect = []string{replacer.Replace("{hostname}")}
|
||||||
if !ok {
|
if !reflect.DeepEqual(got, expect) {
|
||||||
t.Errorf("Request sent to upstream backend should not remove %v header", headerKey)
|
t.Errorf("Request sent to upstream backend does not contain expected %v header: expect %v, but got %v",
|
||||||
} else if len(value) > 0 && headerValue != value[0] {
|
headerKey, expect, got)
|
||||||
t.Errorf("Request sent to upstream backend should replace value of %v header with %v. Instead value was %v", headerKey, headerValue, value)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if actualHost != expectHost {
|
if actualHost != expectHost {
|
||||||
|
@ -447,6 +450,8 @@ func TestDownstreamHeadersUpdate(t *testing.T) {
|
||||||
w.Header().Add("Merge-Me", "Initial")
|
w.Header().Add("Merge-Me", "Initial")
|
||||||
w.Header().Add("Remove-Me", "Remove-Value")
|
w.Header().Add("Remove-Me", "Remove-Value")
|
||||||
w.Header().Add("Replace-Me", "Replace-Value")
|
w.Header().Add("Replace-Me", "Replace-Value")
|
||||||
|
w.Header().Add("Content-Type", "text/html")
|
||||||
|
w.Header().Add("Overwrite-Me", "Overwrite-Value")
|
||||||
w.Write([]byte("Hello, client"))
|
w.Write([]byte("Hello, client"))
|
||||||
}))
|
}))
|
||||||
defer backend.Close()
|
defer backend.Close()
|
||||||
|
@ -470,6 +475,10 @@ func TestDownstreamHeadersUpdate(t *testing.T) {
|
||||||
t.Fatalf("Failed to create request: %v", err)
|
t.Fatalf("Failed to create request: %v", err)
|
||||||
}
|
}
|
||||||
w := httptest.NewRecorder()
|
w := httptest.NewRecorder()
|
||||||
|
// set a predefined skip header
|
||||||
|
w.Header().Set("Content-Type", "text/css")
|
||||||
|
// set a predefined overwritten header
|
||||||
|
w.Header().Set("Overwrite-Me", "Initial")
|
||||||
|
|
||||||
p.ServeHTTP(w, r)
|
p.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
@ -477,16 +486,19 @@ func TestDownstreamHeadersUpdate(t *testing.T) {
|
||||||
actualHeaders := w.Header()
|
actualHeaders := w.Header()
|
||||||
|
|
||||||
headerKey := "Merge-Me"
|
headerKey := "Merge-Me"
|
||||||
values, ok := actualHeaders[headerKey]
|
got := actualHeaders[headerKey]
|
||||||
if !ok {
|
expect := []string{"Initial", "Merge-Value"}
|
||||||
t.Errorf("Downstream response does not contain expected %v header. Expected header should be added", headerKey)
|
if !reflect.DeepEqual(got, expect) {
|
||||||
} else if len(values) < 2 && (values[0] != "Initial" || values[1] != replacer.Replace("{hostname}")) {
|
t.Errorf("Downstream response does not contain expected %s header: expect %v, but got %v",
|
||||||
t.Errorf("Values for header `+Merge-Me` should be merged. Got %v", values)
|
headerKey, expect, got)
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Add-Me"
|
headerKey = "Add-Me"
|
||||||
if _, ok := actualHeaders[headerKey]; !ok {
|
got = actualHeaders[headerKey]
|
||||||
t.Errorf("Downstream response does not contain expected %v header", headerKey)
|
expect = []string{"Add-Value"}
|
||||||
|
if !reflect.DeepEqual(got, expect) {
|
||||||
|
t.Errorf("Downstream response does not contain expected %s header: expect %v, but got %v",
|
||||||
|
headerKey, expect, got)
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Remove-Me"
|
headerKey = "Remove-Me"
|
||||||
|
@ -495,14 +507,28 @@ func TestDownstreamHeadersUpdate(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
headerKey = "Replace-Me"
|
headerKey = "Replace-Me"
|
||||||
headerValue := replacer.Replace("{hostname}")
|
got = actualHeaders[headerKey]
|
||||||
value, ok := actualHeaders[headerKey]
|
expect = []string{replacer.Replace("{hostname}")}
|
||||||
if !ok {
|
if !reflect.DeepEqual(got, expect) {
|
||||||
t.Errorf("Downstream response should contain %v header and not remove it", headerKey)
|
t.Errorf("Downstream response does not contain expected %s header: expect %v, but got %v",
|
||||||
} else if len(value) > 0 && headerValue != value[0] {
|
headerKey, expect, got)
|
||||||
t.Errorf("Downstream response should have header %v with value %v. Instead value was %v", headerKey, headerValue, value)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
headerKey = "Content-Type"
|
||||||
|
got = actualHeaders[headerKey]
|
||||||
|
expect = []string{"text/css"}
|
||||||
|
if !reflect.DeepEqual(got, expect) {
|
||||||
|
t.Errorf("Downstream response does not contain expected %s header: expect %v, but got %v",
|
||||||
|
headerKey, expect, got)
|
||||||
|
}
|
||||||
|
|
||||||
|
headerKey = "Overwrite-Me"
|
||||||
|
got = actualHeaders[headerKey]
|
||||||
|
expect = []string{"Overwrite-Value"}
|
||||||
|
if !reflect.DeepEqual(got, expect) {
|
||||||
|
t.Errorf("Downstream response does not contain expected %s header: expect %v, but got %v",
|
||||||
|
headerKey, expect, got)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
|
|
@ -252,8 +252,28 @@ func (rp *ReverseProxy) copyResponse(dst io.Writer, src io.Reader) {
|
||||||
io.CopyBuffer(dst, src, buf.([]byte))
|
io.CopyBuffer(dst, src, buf.([]byte))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// skip these headers if they already exist.
|
||||||
|
// see https://github.com/mholt/caddy/pull/1112#discussion_r80092582
|
||||||
|
var skipHeaders = map[string]struct{}{
|
||||||
|
"Content-Type": {},
|
||||||
|
"Content-Disposition": {},
|
||||||
|
"Accept-Ranges": {},
|
||||||
|
"Set-Cookie": {},
|
||||||
|
"Cache-Control": {},
|
||||||
|
"Expires": {},
|
||||||
|
}
|
||||||
|
|
||||||
func copyHeader(dst, src http.Header) {
|
func copyHeader(dst, src http.Header) {
|
||||||
for k, vv := range src {
|
for k, vv := range src {
|
||||||
|
if _, ok := dst[k]; ok {
|
||||||
|
// skip some predefined headers
|
||||||
|
// see https://github.com/mholt/caddy/issues/1086
|
||||||
|
if _, shouldSkip := skipHeaders[k]; shouldSkip {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// otherwise, overwrite
|
||||||
|
dst.Del(k)
|
||||||
|
}
|
||||||
for _, v := range vv {
|
for _, v := range vv {
|
||||||
dst.Add(k, v)
|
dst.Add(k, v)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue