2019-05-05 03:21:20 +08:00
|
|
|
package caddyhttp
|
|
|
|
|
|
|
|
import (
|
2019-05-07 23:56:13 +08:00
|
|
|
"net"
|
2019-05-05 03:21:20 +08:00
|
|
|
"net/http"
|
2019-05-07 23:56:13 +08:00
|
|
|
"os"
|
2019-05-05 03:21:20 +08:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2019-05-07 23:56:13 +08:00
|
|
|
// Replacer can replace values in strings based
|
2019-05-11 11:07:02 +08:00
|
|
|
// on a request and/or response writer. The zero
|
|
|
|
// Replacer is not valid; it must be initialized
|
|
|
|
// within this package.
|
2019-05-05 03:21:20 +08:00
|
|
|
type Replacer struct {
|
|
|
|
req *http.Request
|
|
|
|
resp http.ResponseWriter
|
|
|
|
custom map[string]string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Map sets a custom variable mapping to a value.
|
|
|
|
func (r *Replacer) Map(variable, value string) {
|
|
|
|
r.custom[variable] = value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace replaces placeholders in input with the value. If
|
|
|
|
// the value is empty string, the placeholder is substituted
|
|
|
|
// with the value empty.
|
|
|
|
func (r *Replacer) Replace(input, empty string) string {
|
|
|
|
if !strings.Contains(input, phOpen) {
|
|
|
|
return input
|
|
|
|
}
|
|
|
|
|
|
|
|
input = r.replaceAll(input, empty, r.defaults())
|
|
|
|
input = r.replaceAll(input, empty, r.custom)
|
|
|
|
|
|
|
|
return input
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Replacer) replaceAll(input, empty string, mapping map[string]string) string {
|
|
|
|
for key, val := range mapping {
|
|
|
|
if val == "" {
|
|
|
|
val = empty
|
|
|
|
}
|
|
|
|
input = strings.ReplaceAll(input, phOpen+key+phClose, val)
|
|
|
|
}
|
|
|
|
return input
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Replacer) defaults() map[string]string {
|
|
|
|
m := map[string]string{
|
2019-05-07 23:56:13 +08:00
|
|
|
"request.host": func() string {
|
|
|
|
host, _, err := net.SplitHostPort(r.req.Host)
|
|
|
|
if err != nil {
|
|
|
|
return r.req.Host // OK; there probably was no port
|
|
|
|
}
|
|
|
|
return host
|
|
|
|
}(),
|
|
|
|
"request.hostport": r.req.Host, // may include both host and port
|
|
|
|
"request.method": r.req.Method,
|
|
|
|
"request.port": func() string {
|
|
|
|
// if there is no port, there will be an error; in
|
|
|
|
// that case, port is the empty string anyway
|
|
|
|
_, port, _ := net.SplitHostPort(r.req.Host)
|
|
|
|
return port
|
|
|
|
}(),
|
|
|
|
"request.scheme": func() string {
|
2019-05-05 03:21:20 +08:00
|
|
|
if r.req.TLS != nil {
|
|
|
|
return "https"
|
|
|
|
}
|
|
|
|
return "http"
|
|
|
|
}(),
|
2019-05-07 23:56:13 +08:00
|
|
|
"request.uri": r.req.URL.RequestURI(),
|
|
|
|
"system.hostname": func() string {
|
|
|
|
// OK if there is an error; just return empty string
|
|
|
|
name, _ := os.Hostname()
|
|
|
|
return name
|
|
|
|
}(),
|
2019-05-05 03:21:20 +08:00
|
|
|
}
|
|
|
|
|
2019-05-11 11:07:02 +08:00
|
|
|
// TODO: why should header fields, cookies, and query params get special treatment like this?
|
|
|
|
// maybe they should be scoped by words like "request.header." just like everything else.
|
2019-05-05 03:21:20 +08:00
|
|
|
for field, vals := range r.req.Header {
|
|
|
|
m[">"+strings.ToLower(field)] = strings.Join(vals, ",")
|
|
|
|
}
|
|
|
|
for field, vals := range r.resp.Header() {
|
|
|
|
m["<"+strings.ToLower(field)] = strings.Join(vals, ",")
|
|
|
|
}
|
|
|
|
for _, cookie := range r.req.Cookies() {
|
|
|
|
m["~"+cookie.Name] = cookie.Value
|
|
|
|
}
|
|
|
|
for param, vals := range r.req.URL.Query() {
|
|
|
|
m["?"+param] = strings.Join(vals, ",")
|
|
|
|
}
|
|
|
|
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
const phOpen, phClose = "{", "}"
|