mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-11-03 19:17:29 -05:00 
			
		
		
		
	Merge remote-tracking branch 'upstream/master' into rewrite-patch
This commit is contained in:
		
						commit
						281007c482
					
				@ -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
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -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")
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user