2019-06-30 17:07:58 -05:00
|
|
|
// Copyright 2015 Matthew Holt and The Caddy Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2019-05-23 14:16:34 -05:00
|
|
|
package rewrite
|
2019-05-21 00:48:43 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2019-10-19 20:22:29 -05:00
|
|
|
"strconv"
|
2019-05-21 00:48:43 -05:00
|
|
|
"strings"
|
|
|
|
|
2019-07-02 13:37:06 -05:00
|
|
|
"github.com/caddyserver/caddy/v2"
|
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
2019-10-28 15:39:37 -05:00
|
|
|
"go.uber.org/zap"
|
2019-05-21 00:48:43 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2019-08-21 11:46:35 -05:00
|
|
|
caddy.RegisterModule(Rewrite{})
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rewrite is a middleware which can rewrite HTTP requests.
|
2019-12-23 14:45:35 -05:00
|
|
|
//
|
|
|
|
// These rewrite properties are applied to a request in this order:
|
2020-01-10 18:59:57 -05:00
|
|
|
// Method, URI, StripPrefix, StripSuffix, URISubstring.
|
2019-12-23 14:45:35 -05:00
|
|
|
//
|
|
|
|
// TODO: This module is still a WIP and may experience breaking changes.
|
2019-05-21 00:48:43 -05:00
|
|
|
type Rewrite struct {
|
2019-12-23 14:45:35 -05:00
|
|
|
// Changes the request's HTTP verb.
|
2019-10-19 20:22:29 -05:00
|
|
|
Method string `json:"method,omitempty"`
|
|
|
|
|
2019-12-23 14:45:35 -05:00
|
|
|
// Changes the request's URI (path, query string, and fragment if present).
|
|
|
|
// Only components of the URI that are specified will be changed.
|
|
|
|
URI string `json:"uri,omitempty"`
|
|
|
|
|
|
|
|
// Strips the given prefix from the beginning of the URI path.
|
2020-01-10 18:59:57 -05:00
|
|
|
StripPrefix string `json:"strip_prefix,omitempty"`
|
2019-10-19 20:22:29 -05:00
|
|
|
|
2019-12-23 14:45:35 -05:00
|
|
|
// Strips the given suffix from the end of the URI path.
|
2020-01-10 18:59:57 -05:00
|
|
|
StripSuffix string `json:"strip_suffix,omitempty"`
|
2019-12-23 14:45:35 -05:00
|
|
|
|
|
|
|
// Performs substring replacements on the URI.
|
|
|
|
URISubstring []replacer `json:"uri_substring,omitempty"`
|
|
|
|
|
|
|
|
// If set to a 3xx HTTP status code and if the URI was rewritten (changed),
|
|
|
|
// the handler will issue a simple HTTP redirect to the new URI using the
|
|
|
|
// given status code.
|
2019-10-19 20:22:29 -05:00
|
|
|
HTTPRedirect caddyhttp.WeakString `json:"http_redirect,omitempty"`
|
2019-12-23 14:45:35 -05:00
|
|
|
|
2019-10-28 15:39:37 -05:00
|
|
|
logger *zap.Logger
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
|
|
|
|
2019-08-21 11:46:35 -05:00
|
|
|
// CaddyModule returns the Caddy module information.
|
|
|
|
func (Rewrite) CaddyModule() caddy.ModuleInfo {
|
|
|
|
return caddy.ModuleInfo{
|
2019-12-10 15:36:46 -05:00
|
|
|
ID: "http.handlers.rewrite",
|
|
|
|
New: func() caddy.Module { return new(Rewrite) },
|
2019-08-21 11:46:35 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-28 15:39:37 -05:00
|
|
|
// Provision sets up rewr.
|
|
|
|
func (rewr *Rewrite) Provision(ctx caddy.Context) error {
|
|
|
|
rewr.logger = ctx.Logger(rewr)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-21 00:48:43 -05:00
|
|
|
func (rewr Rewrite) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyhttp.Handler) error {
|
2019-12-29 15:12:52 -05:00
|
|
|
repl := r.Context().Value(caddy.ReplacerCtxKey).(*caddy.Replacer)
|
2019-05-21 00:48:43 -05:00
|
|
|
|
2019-10-28 15:39:37 -05:00
|
|
|
logger := rewr.logger.With(
|
|
|
|
zap.Object("request", caddyhttp.LoggableHTTPRequest{Request: r}),
|
|
|
|
)
|
|
|
|
|
2019-12-12 16:32:35 -05:00
|
|
|
changed := rewr.rewrite(r, repl, logger)
|
|
|
|
|
|
|
|
if changed {
|
|
|
|
logger.Debug("rewrote request",
|
|
|
|
zap.String("method", r.Method),
|
|
|
|
zap.String("uri", r.RequestURI),
|
|
|
|
)
|
|
|
|
if rewr.HTTPRedirect != "" {
|
|
|
|
statusCode, err := strconv.Atoi(repl.ReplaceAll(rewr.HTTPRedirect.String(), ""))
|
|
|
|
if err != nil {
|
|
|
|
return caddyhttp.Error(http.StatusInternalServerError, err)
|
|
|
|
}
|
|
|
|
w.Header().Set("Location", r.RequestURI)
|
|
|
|
w.WriteHeader(statusCode)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return next.ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
|
2020-01-10 18:59:57 -05:00
|
|
|
// rewrite performs the rewrites on r using repl, which should
|
|
|
|
// have been obtained from r, but is passed in for efficiency.
|
|
|
|
// It returns true if any changes were made to r.
|
2019-12-29 15:12:52 -05:00
|
|
|
func (rewr Rewrite) rewrite(r *http.Request, repl *caddy.Replacer, logger *zap.Logger) bool {
|
2019-12-12 16:32:35 -05:00
|
|
|
oldMethod := r.Method
|
|
|
|
oldURI := r.RequestURI
|
|
|
|
|
|
|
|
// method
|
2019-05-21 00:48:43 -05:00
|
|
|
if rewr.Method != "" {
|
|
|
|
r.Method = strings.ToUpper(repl.ReplaceAll(rewr.Method, ""))
|
|
|
|
}
|
|
|
|
|
2020-01-10 18:59:57 -05:00
|
|
|
// uri (path, query string, and fragment just because)
|
|
|
|
if uri := rewr.URI; uri != "" {
|
|
|
|
// find the bounds of each part of the URI that exist
|
|
|
|
pathStart, qsStart, fragStart := -1, -1, -1
|
|
|
|
pathEnd, qsEnd := -1, -1
|
|
|
|
for i, ch := range uri {
|
|
|
|
switch {
|
|
|
|
case ch == '?' && qsStart < 0:
|
|
|
|
pathEnd, qsStart = i, i+1
|
|
|
|
case ch == '#' && fragStart < 0:
|
|
|
|
qsEnd, fragStart = i, i+1
|
|
|
|
case pathStart < 0 && qsStart < 0 && fragStart < 0:
|
|
|
|
pathStart = i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if pathStart >= 0 && pathEnd < 0 {
|
|
|
|
pathEnd = len(uri)
|
|
|
|
}
|
|
|
|
if qsStart >= 0 && qsEnd < 0 {
|
|
|
|
qsEnd = len(uri)
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
|
|
|
|
2020-01-10 18:59:57 -05:00
|
|
|
if pathStart >= 0 {
|
|
|
|
r.URL.Path = repl.ReplaceAll(uri[pathStart:pathEnd], "")
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
2020-01-10 18:59:57 -05:00
|
|
|
if qsStart >= 0 {
|
|
|
|
r.URL.RawQuery = buildQueryString(uri[qsStart:qsEnd], repl)
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
2020-01-10 18:59:57 -05:00
|
|
|
if fragStart >= 0 {
|
|
|
|
r.URL.Fragment = repl.ReplaceAll(uri[fragStart:], "")
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-19 20:22:29 -05:00
|
|
|
// strip path prefix or suffix
|
2020-01-10 18:59:57 -05:00
|
|
|
if rewr.StripPrefix != "" {
|
|
|
|
prefix := repl.ReplaceAll(rewr.StripPrefix, "")
|
2019-10-19 20:22:29 -05:00
|
|
|
r.URL.Path = strings.TrimPrefix(r.URL.Path, prefix)
|
|
|
|
}
|
2020-01-10 18:59:57 -05:00
|
|
|
if rewr.StripSuffix != "" {
|
|
|
|
suffix := repl.ReplaceAll(rewr.StripSuffix, "")
|
2019-10-19 20:22:29 -05:00
|
|
|
r.URL.Path = strings.TrimSuffix(r.URL.Path, suffix)
|
|
|
|
}
|
|
|
|
|
2019-12-12 16:32:35 -05:00
|
|
|
// substring replacements in URI
|
|
|
|
for _, rep := range rewr.URISubstring {
|
|
|
|
rep.do(r, repl)
|
2019-10-19 20:22:29 -05:00
|
|
|
}
|
|
|
|
|
2019-12-12 16:32:35 -05:00
|
|
|
// update the encoded copy of the URI
|
|
|
|
r.RequestURI = r.URL.RequestURI()
|
|
|
|
|
|
|
|
// return true if anything changed
|
|
|
|
return r.Method != oldMethod || r.RequestURI != oldURI
|
|
|
|
}
|
|
|
|
|
2020-01-10 18:59:57 -05:00
|
|
|
// buildQueryString takes an input query string and
|
|
|
|
// performs replacements on each component, returning
|
2020-01-11 13:40:03 -05:00
|
|
|
// the resulting query string. This function appends
|
|
|
|
// duplicate keys rather than replaces.
|
2020-01-10 18:59:57 -05:00
|
|
|
func buildQueryString(qs string, repl *caddy.Replacer) string {
|
|
|
|
var sb strings.Builder
|
2020-01-11 13:40:03 -05:00
|
|
|
|
|
|
|
// first component must be key, which is the same
|
|
|
|
// as if we just wrote a value in previous iteration
|
|
|
|
wroteVal := true
|
2020-01-10 18:59:57 -05:00
|
|
|
|
|
|
|
for len(qs) > 0 {
|
2020-01-11 13:40:03 -05:00
|
|
|
// determine the end of this component, which will be at
|
|
|
|
// the next equal sign or ampersand, whichever comes first
|
2020-01-10 18:59:57 -05:00
|
|
|
nextEq, nextAmp := strings.Index(qs, "="), strings.Index(qs, "&")
|
2020-01-11 13:40:03 -05:00
|
|
|
ampIsNext := nextAmp >= 0 && (nextAmp < nextEq || nextEq < 0)
|
|
|
|
end := len(qs) // assume no delimiter remains...
|
|
|
|
if ampIsNext {
|
|
|
|
end = nextAmp // ...unless ampersand is first...
|
|
|
|
} else if nextEq >= 0 && (nextEq < nextAmp || nextAmp < 0) {
|
|
|
|
end = nextEq // ...or unless equal is first.
|
2020-01-10 18:59:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// consume the component and write the result
|
|
|
|
comp := qs[:end]
|
|
|
|
comp, _ = repl.ReplaceFunc(comp, func(name, val string) (string, error) {
|
2020-01-11 13:40:03 -05:00
|
|
|
if name == "http.request.uri.query" && wroteVal {
|
2020-01-10 18:59:57 -05:00
|
|
|
return val, nil // already escaped
|
|
|
|
}
|
|
|
|
return url.QueryEscape(val), nil
|
|
|
|
})
|
|
|
|
if end < len(qs) {
|
|
|
|
end++ // consume delimiter
|
|
|
|
}
|
|
|
|
qs = qs[end:]
|
|
|
|
|
2020-01-11 13:40:03 -05:00
|
|
|
// if previous iteration wrote a value,
|
|
|
|
// that means we are writing a key
|
|
|
|
if wroteVal {
|
|
|
|
if sb.Len() > 0 {
|
|
|
|
sb.WriteRune('&')
|
|
|
|
}
|
|
|
|
} else {
|
2020-01-10 18:59:57 -05:00
|
|
|
sb.WriteRune('=')
|
|
|
|
}
|
|
|
|
sb.WriteString(comp)
|
2020-01-11 13:40:03 -05:00
|
|
|
|
|
|
|
// remember for the next iteration that we just wrote a value,
|
|
|
|
// which means the next iteration MUST write a key
|
|
|
|
wroteVal = ampIsNext
|
2020-01-10 18:59:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return sb.String()
|
|
|
|
}
|
|
|
|
|
2019-12-12 16:32:35 -05:00
|
|
|
// replacer describes a simple and fast substring replacement.
|
|
|
|
type replacer struct {
|
|
|
|
// The substring to find. Supports placeholders.
|
|
|
|
Find string `json:"find,omitempty"`
|
|
|
|
|
|
|
|
// The substring to replace. Supports placeholders.
|
|
|
|
Replace string `json:"replace,omitempty"`
|
|
|
|
|
|
|
|
// Maximum number of replacements per string.
|
|
|
|
// Set to <= 0 for no limit (default).
|
|
|
|
Limit int `json:"limit,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// do performs the replacement on r and returns true if any changes were made.
|
2019-12-29 15:12:52 -05:00
|
|
|
func (rep replacer) do(r *http.Request, repl *caddy.Replacer) bool {
|
2019-12-12 16:32:35 -05:00
|
|
|
if rep.Find == "" || rep.Replace == "" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
lim := rep.Limit
|
|
|
|
if lim == 0 {
|
|
|
|
lim = -1
|
|
|
|
}
|
|
|
|
|
|
|
|
find := repl.ReplaceAll(rep.Find, "")
|
|
|
|
replace := repl.ReplaceAll(rep.Replace, "")
|
|
|
|
|
|
|
|
oldPath := r.URL.Path
|
|
|
|
oldQuery := r.URL.RawQuery
|
|
|
|
|
|
|
|
r.URL.Path = strings.Replace(oldPath, find, replace, lim)
|
|
|
|
r.URL.RawQuery = strings.Replace(oldQuery, find, replace, lim)
|
|
|
|
|
|
|
|
return r.URL.Path != oldPath && r.URL.RawQuery != oldQuery
|
2019-05-21 00:48:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Interface guard
|
|
|
|
var _ caddyhttp.MiddlewareHandler = (*Rewrite)(nil)
|