caddy/middleware/replacer.go

104 lines
2.4 KiB
Go
Raw Normal View History

2015-01-14 03:43:45 +08:00
package middleware
import (
"net"
2015-01-14 03:43:45 +08:00
"net/http"
"strconv"
"strings"
"time"
)
2015-05-04 03:43:50 +08:00
// Replacer is a type which can replace placeholder
2015-01-14 03:43:45 +08:00
// substrings in a string with actual values from a
// http.Request and responseRecorder. Always use
2015-01-30 13:06:53 +08:00
// NewReplacer to get one of these.
2015-05-04 03:43:50 +08:00
type Replacer interface {
Replace(string) string
}
2015-01-14 03:43:45 +08:00
type replacer map[string]string
// NewReplacer makes a new replacer based on r and rr.
// Do not create a new replacer until r and rr have all
2015-01-14 03:43:45 +08:00
// the needed values, because this function copies those
// values into the replacer.
2015-05-04 03:43:50 +08:00
func NewReplacer(r *http.Request, rr *responseRecorder) Replacer {
2015-01-14 03:43:45 +08:00
rep := replacer{
"{method}": r.Method,
"{scheme}": func() string {
if r.TLS != nil {
return "https"
}
return "http"
}(),
"{host}": r.Host,
"{path}": r.URL.Path,
"{query}": r.URL.RawQuery,
"{fragment}": r.URL.Fragment,
"{proto}": r.Proto,
"{remote}": func() string {
if fwdFor := r.Header.Get("X-Forwarded-For"); fwdFor != "" {
return fwdFor
}
host, _, err := net.SplitHostPort(r.RemoteAddr)
if err != nil {
2015-01-14 03:43:45 +08:00
return r.RemoteAddr
}
return host
2015-01-14 03:43:45 +08:00
}(),
"{port}": func() string {
_, port, err := net.SplitHostPort(r.RemoteAddr)
if err != nil {
return ""
2015-01-14 03:43:45 +08:00
}
return port
2015-01-14 03:43:45 +08:00
}(),
"{uri}": r.RequestURI,
2015-01-20 08:12:38 +08:00
"{when}": func() string {
2015-01-14 03:43:45 +08:00
return time.Now().Format(timeFormat)
}(),
2015-05-04 03:38:06 +08:00
}
if rr != nil {
rep["{status}"] = strconv.Itoa(rr.status)
rep["{size}"] = strconv.Itoa(rr.size)
rep["{latency}"] = time.Since(rr.start).String()
2015-01-14 03:43:45 +08:00
}
// Header placeholders
for header, val := range r.Header {
rep[headerReplacer+header+"}"] = strings.Join(val, ",")
}
return rep
}
// Replace performs a replacement of values on s and returns
2015-01-14 03:43:45 +08:00
// the string with the replaced values.
2015-01-30 13:06:53 +08:00
func (r replacer) Replace(s string) string {
2015-01-14 03:43:45 +08:00
for placeholder, replacement := range r {
if replacement == "" {
2015-01-30 13:06:53 +08:00
replacement = EmptyStringReplacer
2015-01-14 03:43:45 +08:00
}
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 {
2015-01-30 13:06:53 +08:00
s = s[:idxStart] + EmptyStringReplacer + s[endOffset+idxEnd+1:]
2015-01-14 03:43:45 +08:00
} else {
break
}
}
return s
}
const (
timeFormat = "02/Jan/2006:15:04:05 -0700"
headerReplacer = "{>"
2015-01-30 13:06:53 +08:00
EmptyStringReplacer = "-"
2015-01-14 03:43:45 +08:00
)