2016-06-06 11:51:56 +08:00
|
|
|
package rewrite
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/mholt/caddy"
|
|
|
|
"github.com/mholt/caddy/caddyhttp/httpserver"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2016-06-07 05:31:03 +08:00
|
|
|
caddy.RegisterPlugin("rewrite", caddy.Plugin{
|
2016-06-06 11:51:56 +08:00
|
|
|
ServerType: "http",
|
|
|
|
Action: setup,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// setup configures a new Rewrite middleware instance.
|
|
|
|
func setup(c *caddy.Controller) error {
|
|
|
|
rewrites, err := rewriteParse(c)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-06-21 01:44:20 +08:00
|
|
|
cfg := httpserver.GetConfig(c)
|
2016-06-06 11:51:56 +08:00
|
|
|
|
|
|
|
cfg.AddMiddleware(func(next httpserver.Handler) httpserver.Handler {
|
|
|
|
return Rewrite{
|
|
|
|
Next: next,
|
|
|
|
FileSys: http.Dir(cfg.Root),
|
|
|
|
Rules: rewrites,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func rewriteParse(c *caddy.Controller) ([]Rule, error) {
|
|
|
|
var simpleRules []Rule
|
|
|
|
var regexpRules []Rule
|
|
|
|
|
|
|
|
for c.Next() {
|
|
|
|
var rule Rule
|
|
|
|
var err error
|
|
|
|
var base = "/"
|
|
|
|
var pattern, to string
|
|
|
|
var status int
|
|
|
|
var ext []string
|
|
|
|
|
|
|
|
args := c.RemainingArgs()
|
|
|
|
|
2016-06-21 22:59:29 +08:00
|
|
|
var matcher httpserver.RequestMatcher
|
2016-06-06 11:51:56 +08:00
|
|
|
|
|
|
|
switch len(args) {
|
|
|
|
case 1:
|
|
|
|
base = args[0]
|
|
|
|
fallthrough
|
|
|
|
case 0:
|
2016-06-21 22:59:29 +08:00
|
|
|
// Integrate request matcher for 'if' conditions.
|
|
|
|
matcher, err = httpserver.SetupIfMatcher(c.Dispenser)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
block:
|
2016-06-06 11:51:56 +08:00
|
|
|
for c.NextBlock() {
|
|
|
|
switch c.Val() {
|
|
|
|
case "r", "regexp":
|
|
|
|
if !c.NextArg() {
|
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
|
|
|
pattern = c.Val()
|
|
|
|
case "to":
|
|
|
|
args1 := c.RemainingArgs()
|
|
|
|
if len(args1) == 0 {
|
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
|
|
|
to = strings.Join(args1, " ")
|
|
|
|
case "ext":
|
|
|
|
args1 := c.RemainingArgs()
|
|
|
|
if len(args1) == 0 {
|
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
|
|
|
ext = args1
|
|
|
|
case "status":
|
|
|
|
if !c.NextArg() {
|
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
|
|
|
status, _ = strconv.Atoi(c.Val())
|
|
|
|
if status < 200 || (status > 299 && status < 400) || status > 499 {
|
|
|
|
return nil, c.Err("status must be 2xx or 4xx")
|
|
|
|
}
|
|
|
|
default:
|
2016-06-21 22:59:29 +08:00
|
|
|
if httpserver.IfMatcherKeyword(c.Val()) {
|
2016-07-09 01:11:15 +08:00
|
|
|
c.RemainingArgs()
|
2016-06-21 22:59:29 +08:00
|
|
|
continue block
|
|
|
|
}
|
2016-06-06 11:51:56 +08:00
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// ensure to or status is specified
|
|
|
|
if to == "" && status == 0 {
|
|
|
|
return nil, c.ArgErr()
|
|
|
|
}
|
2016-06-21 22:59:29 +08:00
|
|
|
if rule, err = NewComplexRule(base, pattern, to, status, ext, matcher); err != nil {
|
2016-06-06 11:51:56 +08:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
regexpRules = append(regexpRules, rule)
|
|
|
|
|
|
|
|
// the only unhandled case is 2 and above
|
|
|
|
default:
|
|
|
|
rule = NewSimpleRule(args[0], strings.Join(args[1:], " "))
|
|
|
|
simpleRules = append(simpleRules, rule)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// put simple rules in front to avoid regexp computation for them
|
|
|
|
return append(simpleRules, regexpRules...), nil
|
|
|
|
}
|