0
Fork 0
mirror of https://github.com/caddyserver/caddy.git synced 2024-12-23 22:27:38 -05:00

Merge remote-tracking branch 'upstream/master' into rewrite-patch

This commit is contained in:
Abiola Ibrahim 2015-12-31 20:34:25 +01:00
commit 281007c482
2 changed files with 93 additions and 73 deletions

View file

@ -86,9 +86,9 @@ func NewReplacer(r *http.Request, rr *responseRecorder, emptyValue string) Repla
rep.replacements["{latency}"] = time.Since(rr.start).String() rep.replacements["{latency}"] = time.Since(rr.start).String()
} }
// Header placeholders // Header placeholders (case-insensitive)
for header, val := range r.Header { for header, values := range r.Header {
rep.replacements[headerReplacer+header+"}"] = strings.Join(val, ",") rep.replacements[headerReplacer+strings.ToLower(header)+"}"] = strings.Join(values, ",")
} }
return rep return rep
@ -97,6 +97,24 @@ func NewReplacer(r *http.Request, rr *responseRecorder, emptyValue string) Repla
// Replace performs a replacement of values on s and returns // Replace performs a replacement of values on s and returns
// the string with the replaced values. // the string with the replaced values.
func (r replacer) Replace(s string) string { func (r replacer) Replace(s string) string {
// Header replacements - these are case-insensitive, so we can't just use strings.Replace()
for strings.Contains(s, headerReplacer) {
idxStart := strings.Index(s, headerReplacer)
endOffset := idxStart + len(headerReplacer)
idxEnd := strings.Index(s[endOffset:], "}")
if idxEnd > -1 {
placeholder := strings.ToLower(s[idxStart : endOffset+idxEnd+1])
replacement := r.replacements[placeholder]
if replacement == "" {
replacement = r.emptyValue
}
s = s[:idxStart] + replacement + s[endOffset+idxEnd+1:]
} else {
break
}
}
// Regular replacements - these are easier because they're case-sensitive
for placeholder, replacement := range r.replacements { for placeholder, replacement := range r.replacements {
if replacement == "" { if replacement == "" {
replacement = r.emptyValue replacement = r.emptyValue
@ -104,17 +122,6 @@ func (r replacer) Replace(s string) string {
s = strings.Replace(s, placeholder, replacement, -1) s = strings.Replace(s, placeholder, replacement, -1)
} }
// Replace any header placeholders that weren't found
for strings.Contains(s, headerReplacer) {
idxStart := strings.Index(s, headerReplacer)
endOffset := idxStart + len(headerReplacer)
idxEnd := strings.Index(s[endOffset:], "}")
if idxEnd > -1 {
s = s[:idxStart] + r.emptyValue + s[endOffset+idxEnd+1:]
} else {
break
}
}
return s return s
} }

View file

@ -10,102 +10,115 @@ import (
func TestNewReplacer(t *testing.T) { func TestNewReplacer(t *testing.T) {
w := httptest.NewRecorder() w := httptest.NewRecorder()
recordRequest := NewResponseRecorder(w) recordRequest := NewResponseRecorder(w)
userJSON := `{"username": "dennis"}` reader := strings.NewReader(`{"username": "dennis"}`)
reader := strings.NewReader(userJSON) //Convert string to reader request, err := http.NewRequest("POST", "http://localhost", reader)
request, err := http.NewRequest("POST", "http://caddyserver.com", reader) //Create request with JSON body
if err != nil { if err != nil {
t.Fatalf("Request Formation Failed \n") t.Fatal("Request Formation Failed\n")
} }
replaceValues := NewReplacer(request, recordRequest, "") replaceValues := NewReplacer(request, recordRequest, "")
switch v := replaceValues.(type) { switch v := replaceValues.(type) {
case replacer: case replacer:
if v.replacements["{host}"] != "caddyserver.com" { if v.replacements["{host}"] != "localhost" {
t.Errorf("Expected host to be caddyserver.com") t.Error("Expected host to be localhost")
} }
if v.replacements["{method}"] != "POST" { if v.replacements["{method}"] != "POST" {
t.Errorf("Expected request method to be POST") t.Error("Expected request method to be POST")
} }
if v.replacements["{status}"] != "200" { if v.replacements["{status}"] != "200" {
t.Errorf("Expected status to be 200") t.Error("Expected status to be 200")
} }
default: default:
t.Fatalf("Return Value from New Replacer expected pass type assertion into a replacer type \n") t.Fatal("Return Value from New Replacer expected pass type assertion into a replacer type\n")
} }
} }
func TestReplace(t *testing.T) { func TestReplace(t *testing.T) {
w := httptest.NewRecorder() w := httptest.NewRecorder()
recordRequest := NewResponseRecorder(w) recordRequest := NewResponseRecorder(w)
userJSON := `{"username": "dennis"}` reader := strings.NewReader(`{"username": "dennis"}`)
reader := strings.NewReader(userJSON) //Convert string to reader request, err := http.NewRequest("POST", "http://localhost", reader)
request, err := http.NewRequest("POST", "http://caddyserver.com", reader) //Create request with JSON body
if err != nil { if err != nil {
t.Fatalf("Request Formation Failed \n") t.Fatal("Request Formation Failed\n")
} }
replaceValues := NewReplacer(request, recordRequest, "") request.Header.Set("Custom", "foobarbaz")
request.Header.Set("ShorterVal", "1")
repl := NewReplacer(request, recordRequest, "-")
switch v := replaceValues.(type) { if expected, actual := "This host is localhost.", repl.Replace("This host is {host}."); expected != actual {
case replacer: t.Errorf("{host} replacement: expected '%s', got '%s'", expected, actual)
if v.Replace("This host is {host}") != "This host is caddyserver.com" {
t.Errorf("Expected host replacement failed")
} }
if v.Replace("This request method is {method}") != "This request method is POST" { if expected, actual := "This request method is POST.", repl.Replace("This request method is {method}."); expected != actual {
t.Errorf("Expected method replacement failed") t.Errorf("{method} replacement: expected '%s', got '%s'", expected, actual)
} }
if v.Replace("The response status is {status}") != "The response status is 200" { if expected, actual := "The response status is 200.", repl.Replace("The response status is {status}."); expected != actual {
t.Errorf("Expected status replacement failed") t.Errorf("{status} replacement: expected '%s', got '%s'", expected, actual)
}
if expected, actual := "The Custom header is foobarbaz.", repl.Replace("The Custom header is {>Custom}."); expected != actual {
t.Errorf("{>Custom} replacement: expected '%s', got '%s'", expected, actual)
} }
default: // Test header case-insensitivity
t.Fatalf("Return Value from New Replacer expected pass type assertion into a replacer type \n") if expected, actual := "The cUsToM header is foobarbaz...", repl.Replace("The cUsToM header is {>cUsToM}..."); expected != actual {
t.Errorf("{>cUsToM} replacement: expected '%s', got '%s'", expected, actual)
} }
// Test non-existent header/value
if expected, actual := "The Non-Existent header is -.", repl.Replace("The Non-Existent header is {>Non-Existent}."); expected != actual {
t.Errorf("{>Non-Existent} replacement: expected '%s', got '%s'", expected, actual)
}
// Test bad placeholder
if expected, actual := "Bad {host placeholder...", repl.Replace("Bad {host placeholder..."); expected != actual {
t.Errorf("bad placeholder: expected '%s', got '%s'", expected, actual)
}
// Test bad header placeholder
if expected, actual := "Bad {>Custom placeholder", repl.Replace("Bad {>Custom placeholder"); expected != actual {
t.Errorf("bad header placeholder: expected '%s', got '%s'", expected, actual)
}
// Test bad header placeholder with valid one later
if expected, actual := "Bad -", repl.Replace("Bad {>Custom placeholder {>ShorterVal}"); expected != actual {
t.Errorf("bad header placeholders: expected '%s', got '%s'", expected, actual)
}
// Test shorter header value with multiple placeholders
if expected, actual := "Short value 1 then foobarbaz.", repl.Replace("Short value {>ShorterVal} then {>Custom}."); expected != actual {
t.Errorf("short value: expected '%s', got '%s'", expected, actual)
}
} }
func TestSet(t *testing.T) { func TestSet(t *testing.T) {
w := httptest.NewRecorder() w := httptest.NewRecorder()
recordRequest := NewResponseRecorder(w) recordRequest := NewResponseRecorder(w)
userJSON := `{"username": "dennis"}` reader := strings.NewReader(`{"username": "dennis"}`)
reader := strings.NewReader(userJSON) //Convert string to reader request, err := http.NewRequest("POST", "http://localhost", reader)
request, err := http.NewRequest("POST", "http://caddyserver.com", reader) //Create request with JSON body
if err != nil { if err != nil {
t.Fatalf("Request Formation Failed \n") t.Fatalf("Request Formation Failed \n")
} }
replaceValues := NewReplacer(request, recordRequest, "") repl := NewReplacer(request, recordRequest, "")
replaceValues.Set("host", "getcaddy.com") repl.Set("host", "getcaddy.com")
replaceValues.Set("method", "GET") repl.Set("method", "GET")
replaceValues.Set("status", "201") repl.Set("status", "201")
replaceValues.Set("variable", "value") repl.Set("variable", "value")
switch v := replaceValues.(type) { if repl.Replace("This host is {host}") != "This host is getcaddy.com" {
case replacer: t.Error("Expected host replacement failed")
if v.Replace("This host is {host}") != "This host is getcaddy.com" {
t.Errorf("Expected host replacement failed")
} }
if v.Replace("This request method is {method}") != "This request method is GET" { if repl.Replace("This request method is {method}") != "This request method is GET" {
t.Errorf("Expected method replacement failed") t.Error("Expected method replacement failed")
} }
if v.Replace("The response status is {status}") != "The response status is 201" { if repl.Replace("The response status is {status}") != "The response status is 201" {
t.Errorf("Expected status replacement failed") t.Error("Expected status replacement failed")
} }
if v.Replace("The value of variable is {variable}") != "The value of variable is value" { if repl.Replace("The value of variable is {variable}") != "The value of variable is value" {
t.Errorf("Expected status replacement failed") t.Error("Expected variable replacement failed")
} }
default:
t.Fatalf("Return Value from New Replacer expected pass type assertion into a replacer type \n")
}
} }