mirror of
https://github.com/caddyserver/caddy.git
synced 2024-12-23 22:27:38 -05:00
Merge pull request #69 from peterhellberg/headers_test
headers: Initial test for Headers and change of Rule.Url to Rule.Path
This commit is contained in:
commit
e2273ea676
4 changed files with 143 additions and 8 deletions
|
@ -29,17 +29,17 @@ func headersParse(c *Controller) ([]headers.Rule, error) {
|
||||||
}
|
}
|
||||||
pattern := c.Val()
|
pattern := c.Val()
|
||||||
|
|
||||||
// See if we already have a definition for this URL pattern...
|
// See if we already have a definition for this Path pattern...
|
||||||
for _, h := range rules {
|
for _, h := range rules {
|
||||||
if h.Url == pattern {
|
if h.Path == pattern {
|
||||||
head = h
|
head = h
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ...otherwise, this is a new pattern
|
// ...otherwise, this is a new pattern
|
||||||
if head.Url == "" {
|
if head.Path == "" {
|
||||||
head.Url = pattern
|
head.Path = pattern
|
||||||
isNewPattern = true
|
isNewPattern = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -72,7 +72,7 @@ func headersParse(c *Controller) ([]headers.Rule, error) {
|
||||||
rules = append(rules, head)
|
rules = append(rules, head)
|
||||||
} else {
|
} else {
|
||||||
for i := 0; i < len(rules); i++ {
|
for i := 0; i < len(rules); i++ {
|
||||||
if rules[i].Url == pattern {
|
if rules[i].Path == pattern {
|
||||||
rules[i] = head
|
rules[i] = head
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
85
config/setup/headers_test.go
Normal file
85
config/setup/headers_test.go
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
package setup
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/mholt/caddy/middleware/headers"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestHeaders(t *testing.T) {
|
||||||
|
c := newTestController(`header / Foo Bar`)
|
||||||
|
|
||||||
|
mid, err := Headers(c)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Expected no errors, but got: %v", err)
|
||||||
|
}
|
||||||
|
if mid == nil {
|
||||||
|
t.Fatal("Expected middleware, was nil instead")
|
||||||
|
}
|
||||||
|
|
||||||
|
handler := mid(emptyNext)
|
||||||
|
myHandler, ok := handler.(headers.Headers)
|
||||||
|
if !ok {
|
||||||
|
t.Fatalf("Expected handler to be type Headers, got: %#v", handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !sameNext(myHandler.Next, emptyNext) {
|
||||||
|
t.Error("'Next' field of handler was not set properly")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestHeadersParse(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
input string
|
||||||
|
shouldErr bool
|
||||||
|
expected []headers.Rule
|
||||||
|
}{
|
||||||
|
{`header /foo Foo "Bar Baz"`,
|
||||||
|
false, []headers.Rule{
|
||||||
|
{Path: "/foo", Headers: []headers.Header{
|
||||||
|
{"Foo", "Bar Baz"},
|
||||||
|
}},
|
||||||
|
}},
|
||||||
|
{`header /bar { Foo "Bar Baz" Baz Qux }`,
|
||||||
|
false, []headers.Rule{
|
||||||
|
{Path: "/bar", Headers: []headers.Header{
|
||||||
|
{"Foo", "Bar Baz"},
|
||||||
|
{"Baz", "Qux"},
|
||||||
|
}},
|
||||||
|
}},
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, test := range tests {
|
||||||
|
c := newTestController(test.input)
|
||||||
|
actual, err := headersParse(c)
|
||||||
|
|
||||||
|
if err == nil && test.shouldErr {
|
||||||
|
t.Errorf("Test %d didn't error, but it should have", i)
|
||||||
|
} else if err != nil && !test.shouldErr {
|
||||||
|
t.Errorf("Test %d errored, but it shouldn't have; got '%v'", i, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(actual) != len(test.expected) {
|
||||||
|
t.Fatalf("Test %d expected %d rules, but got %d",
|
||||||
|
i, len(test.expected), len(actual))
|
||||||
|
}
|
||||||
|
|
||||||
|
for j, expectedRule := range test.expected {
|
||||||
|
actualRule := actual[j]
|
||||||
|
|
||||||
|
if actualRule.Path != expectedRule.Path {
|
||||||
|
t.Errorf("Test %d, rule %d: Expected path %s, but got %s",
|
||||||
|
i, j, expectedRule.Path, actualRule.Path)
|
||||||
|
}
|
||||||
|
|
||||||
|
expectedHeaders := fmt.Sprintf("%v", expectedRule.Headers)
|
||||||
|
actualHeaders := fmt.Sprintf("%v", actualRule.Headers)
|
||||||
|
|
||||||
|
if actualHeaders != expectedHeaders {
|
||||||
|
t.Errorf("Test %d, rule %d: Expected headers %s, but got %s",
|
||||||
|
i, j, expectedHeaders, actualHeaders)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -18,10 +18,10 @@ type Headers struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// ServeHTTP implements the middleware.Handler interface and serves requests,
|
// ServeHTTP implements the middleware.Handler interface and serves requests,
|
||||||
// adding headers to the response according to the configured rules.
|
// setting headers on the response according to the configured rules.
|
||||||
func (h Headers) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
func (h Headers) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||||
for _, rule := range h.Rules {
|
for _, rule := range h.Rules {
|
||||||
if middleware.Path(r.URL.Path).Matches(rule.Url) {
|
if middleware.Path(r.URL.Path).Matches(rule.Path) {
|
||||||
for _, header := range rule.Headers {
|
for _, header := range rule.Headers {
|
||||||
if strings.HasPrefix(header.Name, "-") {
|
if strings.HasPrefix(header.Name, "-") {
|
||||||
w.Header().Del(strings.TrimLeft(header.Name, "-"))
|
w.Header().Del(strings.TrimLeft(header.Name, "-"))
|
||||||
|
@ -38,7 +38,7 @@ type (
|
||||||
// Rule groups a slice of HTTP headers by a URL pattern.
|
// Rule groups a slice of HTTP headers by a URL pattern.
|
||||||
// TODO: use http.Header type instead?
|
// TODO: use http.Header type instead?
|
||||||
Rule struct {
|
Rule struct {
|
||||||
Url string
|
Path string
|
||||||
Headers []Header
|
Headers []Header
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
50
middleware/headers/headers_test.go
Normal file
50
middleware/headers/headers_test.go
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
package headers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/mholt/caddy/middleware"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestHeaders(t *testing.T) {
|
||||||
|
for i, test := range []struct {
|
||||||
|
from string
|
||||||
|
name string
|
||||||
|
value string
|
||||||
|
}{
|
||||||
|
{"/a", "Foo", "Bar"},
|
||||||
|
{"/a", "Bar", ""},
|
||||||
|
{"/a", "Baz", ""},
|
||||||
|
{"/b", "Foo", ""},
|
||||||
|
{"/b", "Bar", "Removed in /a"},
|
||||||
|
} {
|
||||||
|
he := Headers{
|
||||||
|
Next: middleware.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||||
|
return 0, nil
|
||||||
|
}),
|
||||||
|
Rules: []Rule{
|
||||||
|
{Path: "/a", Headers: []Header{
|
||||||
|
{Name: "Foo", Value: "Bar"},
|
||||||
|
{Name: "-Bar"},
|
||||||
|
}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("GET", test.from, nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Test %d: Could not create HTTP request: %v", i, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
rec.Header().Set("Bar", "Removed in /a")
|
||||||
|
|
||||||
|
he.ServeHTTP(rec, req)
|
||||||
|
|
||||||
|
if got := rec.Header().Get(test.name); got != test.value {
|
||||||
|
t.Errorf("Test %d: Expected %s header to be %q but was %q",
|
||||||
|
i, test.name, test.value, got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue