2015-01-13 14:43:45 -05:00
|
|
|
package middleware
|
|
|
|
|
|
|
|
import (
|
2015-03-27 00:39:36 -05:00
|
|
|
"net"
|
2015-01-13 14:43:45 -05:00
|
|
|
"net/http"
|
2015-12-24 03:00:10 -05:00
|
|
|
"net/url"
|
2015-09-20 02:49:55 -05:00
|
|
|
"path"
|
2015-01-13 14:43:45 -05:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2015-05-03 14:43:50 -05:00
|
|
|
// Replacer is a type which can replace placeholder
|
2015-01-13 14:43:45 -05:00
|
|
|
// substrings in a string with actual values from a
|
|
|
|
// http.Request and responseRecorder. Always use
|
2015-01-30 00:06:53 -05:00
|
|
|
// NewReplacer to get one of these.
|
2015-05-03 14:43:50 -05:00
|
|
|
type Replacer interface {
|
|
|
|
Replace(string) string
|
2015-12-30 14:42:03 -05:00
|
|
|
Set(key, value string)
|
2015-05-03 14:43:50 -05:00
|
|
|
}
|
|
|
|
|
2015-07-24 11:11:34 -05:00
|
|
|
type replacer struct {
|
|
|
|
replacements map[string]string
|
|
|
|
emptyValue string
|
|
|
|
}
|
2015-01-13 14:43:45 -05:00
|
|
|
|
2015-03-27 00:39:36 -05:00
|
|
|
// NewReplacer makes a new replacer based on r and rr.
|
|
|
|
// Do not create a new replacer until r and rr have all
|
2015-01-13 14:43:45 -05:00
|
|
|
// the needed values, because this function copies those
|
2015-08-04 19:32:27 -05:00
|
|
|
// values into the replacer. rr may be nil if it is not
|
|
|
|
// available. emptyValue should be the string that is used
|
|
|
|
// in place of empty string (can still be empty string).
|
2015-07-24 11:11:34 -05:00
|
|
|
func NewReplacer(r *http.Request, rr *responseRecorder, emptyValue string) Replacer {
|
2015-01-13 14:43:45 -05:00
|
|
|
rep := replacer{
|
2015-07-24 11:11:34 -05:00
|
|
|
replacements: map[string]string{
|
|
|
|
"{method}": r.Method,
|
|
|
|
"{scheme}": func() string {
|
|
|
|
if r.TLS != nil {
|
|
|
|
return "https"
|
|
|
|
}
|
|
|
|
return "http"
|
|
|
|
}(),
|
2015-12-24 03:00:10 -05:00
|
|
|
"{host}": r.Host,
|
|
|
|
"{path}": r.URL.Path,
|
|
|
|
"{path_escaped}": url.QueryEscape(r.URL.Path),
|
|
|
|
"{query}": r.URL.RawQuery,
|
|
|
|
"{query_escaped}": url.QueryEscape(r.URL.RawQuery),
|
|
|
|
"{fragment}": r.URL.Fragment,
|
|
|
|
"{proto}": r.Proto,
|
2015-07-24 11:11:34 -05:00
|
|
|
"{remote}": func() string {
|
|
|
|
if fwdFor := r.Header.Get("X-Forwarded-For"); fwdFor != "" {
|
|
|
|
return fwdFor
|
|
|
|
}
|
|
|
|
host, _, err := net.SplitHostPort(r.RemoteAddr)
|
|
|
|
if err != nil {
|
|
|
|
return r.RemoteAddr
|
|
|
|
}
|
|
|
|
return host
|
|
|
|
}(),
|
|
|
|
"{port}": func() string {
|
|
|
|
_, port, err := net.SplitHostPort(r.RemoteAddr)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return port
|
|
|
|
}(),
|
2015-12-24 03:00:10 -05:00
|
|
|
"{uri}": r.URL.RequestURI(),
|
|
|
|
"{uri_escaped}": url.QueryEscape(r.URL.RequestURI()),
|
2015-07-24 11:11:34 -05:00
|
|
|
"{when}": func() string {
|
|
|
|
return time.Now().Format(timeFormat)
|
|
|
|
}(),
|
2015-09-20 02:49:55 -05:00
|
|
|
"{file}": func() string {
|
|
|
|
_, file := path.Split(r.URL.Path)
|
|
|
|
return file
|
|
|
|
}(),
|
|
|
|
"{dir}": func() string {
|
|
|
|
dir, _ := path.Split(r.URL.Path)
|
|
|
|
return dir
|
|
|
|
}(),
|
2015-07-24 11:11:34 -05:00
|
|
|
},
|
|
|
|
emptyValue: emptyValue,
|
2015-05-03 14:38:06 -05:00
|
|
|
}
|
|
|
|
if rr != nil {
|
2015-07-24 11:11:34 -05:00
|
|
|
rep.replacements["{status}"] = strconv.Itoa(rr.status)
|
|
|
|
rep.replacements["{size}"] = strconv.Itoa(rr.size)
|
|
|
|
rep.replacements["{latency}"] = time.Since(rr.start).String()
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
|
|
|
|
2015-12-31 14:12:16 -05:00
|
|
|
// Header placeholders (case-insensitive)
|
|
|
|
for header, values := range r.Header {
|
|
|
|
rep.replacements[headerReplacer+strings.ToLower(header)+"}"] = strings.Join(values, ",")
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return rep
|
|
|
|
}
|
|
|
|
|
2015-01-30 00:52:21 -05:00
|
|
|
// Replace performs a replacement of values on s and returns
|
2015-01-13 14:43:45 -05:00
|
|
|
// the string with the replaced values.
|
2015-01-30 00:06:53 -05:00
|
|
|
func (r replacer) Replace(s string) string {
|
2015-12-31 14:12:16 -05:00
|
|
|
// Header replacements - these are case-insensitive, so we can't just use strings.Replace()
|
|
|
|
startPos := strings.Index(s, headerReplacer)
|
|
|
|
for startPos > -1 {
|
|
|
|
// carefully find end of placeholder
|
|
|
|
endOffset := strings.Index(s[startPos+1:], "}")
|
|
|
|
if endOffset == -1 {
|
|
|
|
startPos = strings.Index(s[startPos+len(headerReplacer):], headerReplacer)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
endPos := startPos + len(headerReplacer) + endOffset
|
|
|
|
|
|
|
|
// look for replacement, case-insensitive
|
|
|
|
placeholder := strings.ToLower(s[startPos:endPos])
|
|
|
|
replacement := r.replacements[placeholder]
|
2015-01-13 14:43:45 -05:00
|
|
|
if replacement == "" {
|
2015-07-24 11:11:34 -05:00
|
|
|
replacement = r.emptyValue
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
2015-12-31 14:12:16 -05:00
|
|
|
|
|
|
|
// do the replacement manually
|
|
|
|
s = s[:startPos] + replacement + s[endPos:]
|
|
|
|
|
|
|
|
// move to next one
|
|
|
|
startPos = strings.Index(s[endOffset:], headerReplacer)
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
|
|
|
|
2015-12-31 14:12:16 -05:00
|
|
|
// Regular replacements - these are easier because they're case-sensitive
|
|
|
|
for placeholder, replacement := range r.replacements {
|
|
|
|
if replacement == "" {
|
|
|
|
replacement = r.emptyValue
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
2015-12-31 14:12:16 -05:00
|
|
|
s = strings.Replace(s, placeholder, replacement, -1)
|
2015-01-13 14:43:45 -05:00
|
|
|
}
|
2015-12-31 14:12:16 -05:00
|
|
|
|
2015-01-13 14:43:45 -05:00
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2015-12-30 14:42:03 -05:00
|
|
|
// Set sets key to value in the replacements map.
|
|
|
|
func (r replacer) Set(key, value string) {
|
|
|
|
r.replacements["{"+key+"}"] = value
|
|
|
|
}
|
|
|
|
|
2015-01-13 14:43:45 -05:00
|
|
|
const (
|
2015-07-24 11:11:34 -05:00
|
|
|
timeFormat = "02/Jan/2006:15:04:05 -0700"
|
|
|
|
headerReplacer = "{>"
|
2015-01-13 14:43:45 -05:00
|
|
|
)
|