Commit d01bcd59 authored by Matt Holt's avatar Matt Holt Committed by GitHub

Merge pull request #1112 from tw4452852/proxy_header

proxy: don't append predefined headers
parents 9077cce1 c9b022b5
...@@ -13,6 +13,7 @@ import ( ...@@ -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) { ...@@ -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) { ...@@ -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) { ...@@ -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) { ...@@ -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) { ...@@ -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) { ...@@ -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) { ...@@ -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)
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment