Allow multiple matcher sets in routes (OR'ed together)

Also export MatchRegexp in case other matcher modules find it useful.
Add comments to the exported matchers.
This commit is contained in:
Matthew Holt 2019-05-22 13:13:39 -06:00
parent bc00d840e8
commit 284fb3a98c
4 changed files with 133 additions and 61 deletions

View file

@ -53,13 +53,17 @@ func (app *App) Provision(ctx caddy2.Context) error {
for i := range srv.Listen { for i := range srv.Listen {
srv.Listen[i] = repl.ReplaceAll(srv.Listen[i], "") srv.Listen[i] = repl.ReplaceAll(srv.Listen[i], "")
} }
err := srv.Routes.Provision(ctx) if srv.Routes != nil {
if err != nil { err := srv.Routes.Provision(ctx)
return fmt.Errorf("setting up server routes: %v", err) if err != nil {
return fmt.Errorf("setting up server routes: %v", err)
}
} }
err = srv.Errors.Routes.Provision(ctx) if srv.Errors != nil {
if err != nil { err := srv.Errors.Routes.Provision(ctx)
return fmt.Errorf("setting up server error handling routes: %v", err) if err != nil {
return fmt.Errorf("setting up server error handling routes: %v", err)
}
} }
} }
@ -187,13 +191,15 @@ func (app *App) automaticHTTPS() error {
// find all qualifying domain names, de-duplicated // find all qualifying domain names, de-duplicated
domainSet := make(map[string]struct{}) domainSet := make(map[string]struct{})
for _, route := range srv.Routes { for _, route := range srv.Routes {
for _, m := range route.matchers { for _, matcherSet := range route.matcherSets {
if hm, ok := m.(*MatchHost); ok { for _, m := range matcherSet {
for _, d := range *hm { if hm, ok := m.(*MatchHost); ok {
if !certmagic.HostQualifies(d) { for _, d := range *hm {
continue if !certmagic.HostQualifies(d) {
continue
}
domainSet[d] = struct{}{}
} }
domainSet[d] = struct{}{}
} }
} }
} }
@ -245,9 +251,11 @@ func (app *App) automaticHTTPS() error {
redirTo += "{http.request.uri}" redirTo += "{http.request.uri}"
redirRoutes = append(redirRoutes, ServerRoute{ redirRoutes = append(redirRoutes, ServerRoute{
matchers: []RequestMatcher{ matcherSets: []MatcherSet{
MatchProtocol("http"), {
MatchHost(domains), MatchProtocol("http"),
MatchHost(domains),
},
}, },
responder: Static{ responder: Static{
StatusCode: http.StatusTemporaryRedirect, // TODO: use permanent redirect instead StatusCode: http.StatusTemporaryRedirect, // TODO: use permanent redirect instead

View file

@ -17,16 +17,35 @@ import (
) )
type ( type (
MatchHost []string // MatchHost matches requests by the Host value.
MatchPath []string MatchHost []string
MatchPathRE struct{ matchRegexp }
MatchMethod []string // MatchPath matches requests by the URI's path.
MatchQuery url.Values MatchPath []string
MatchHeader http.Header
MatchHeaderRE map[string]*matchRegexp // MatchPathRE matches requests by a regular expression on the URI's path.
MatchProtocol string MatchPathRE struct{ MatchRegexp }
// MatchMethod matches requests by the method.
MatchMethod []string
// MatchQuery matches requests by URI's query string.
MatchQuery url.Values
// MatchHeader matches requests by header fields.
MatchHeader http.Header
// MatchHeaderRE matches requests by a regular expression on header fields.
MatchHeaderRE map[string]*MatchRegexp
// MatchProtocol matches requests by protocol.
MatchProtocol string
// MatchStarlarkExpr matches requests by evaluating a Starlark expression.
MatchStarlarkExpr string MatchStarlarkExpr string
MatchTable string // TODO: finish implementing
// MatchTable matches requests by values in the table.
MatchTable string // TODO: finish implementing
) )
func init() { func init() {
@ -68,6 +87,7 @@ func init() {
}) })
} }
// Match returns true if r matches m.
func (m MatchHost) Match(r *http.Request) bool { func (m MatchHost) Match(r *http.Request) bool {
outer: outer:
for _, host := range m { for _, host := range m {
@ -93,6 +113,7 @@ outer:
return false return false
} }
// Match returns true if r matches m.
func (m MatchPath) Match(r *http.Request) bool { func (m MatchPath) Match(r *http.Request) bool {
for _, matchPath := range m { for _, matchPath := range m {
compare := r.URL.Path compare := r.URL.Path
@ -111,11 +132,13 @@ func (m MatchPath) Match(r *http.Request) bool {
return false return false
} }
// Match returns true if r matches m.
func (m MatchPathRE) Match(r *http.Request) bool { func (m MatchPathRE) Match(r *http.Request) bool {
repl := r.Context().Value(caddy2.ReplacerCtxKey).(caddy2.Replacer) repl := r.Context().Value(caddy2.ReplacerCtxKey).(caddy2.Replacer)
return m.match(r.URL.Path, repl, "path_regexp") return m.MatchRegexp.Match(r.URL.Path, repl, "path_regexp")
} }
// Match returns true if r matches m.
func (m MatchMethod) Match(r *http.Request) bool { func (m MatchMethod) Match(r *http.Request) bool {
for _, method := range m { for _, method := range m {
if r.Method == method { if r.Method == method {
@ -125,6 +148,7 @@ func (m MatchMethod) Match(r *http.Request) bool {
return false return false
} }
// Match returns true if r matches m.
func (m MatchQuery) Match(r *http.Request) bool { func (m MatchQuery) Match(r *http.Request) bool {
for param, vals := range m { for param, vals := range m {
paramVal := r.URL.Query().Get(param) paramVal := r.URL.Query().Get(param)
@ -137,6 +161,7 @@ func (m MatchQuery) Match(r *http.Request) bool {
return false return false
} }
// Match returns true if r matches m.
func (m MatchHeader) Match(r *http.Request) bool { func (m MatchHeader) Match(r *http.Request) bool {
for field, allowedFieldVals := range m { for field, allowedFieldVals := range m {
var match bool var match bool
@ -157,10 +182,11 @@ func (m MatchHeader) Match(r *http.Request) bool {
return true return true
} }
// Match returns true if r matches m.
func (m MatchHeaderRE) Match(r *http.Request) bool { func (m MatchHeaderRE) Match(r *http.Request) bool {
for field, rm := range m { for field, rm := range m {
repl := r.Context().Value(caddy2.ReplacerCtxKey).(caddy2.Replacer) repl := r.Context().Value(caddy2.ReplacerCtxKey).(caddy2.Replacer)
match := rm.match(r.Header.Get(field), repl, "header_regexp") match := rm.Match(r.Header.Get(field), repl, "header_regexp")
if !match { if !match {
return false return false
} }
@ -168,6 +194,7 @@ func (m MatchHeaderRE) Match(r *http.Request) bool {
return true return true
} }
// Provision compiles m's regular expressions.
func (m MatchHeaderRE) Provision() error { func (m MatchHeaderRE) Provision() error {
for _, rm := range m { for _, rm := range m {
err := rm.Provision() err := rm.Provision()
@ -178,6 +205,7 @@ func (m MatchHeaderRE) Provision() error {
return nil return nil
} }
// Validate validates m's regular expressions.
func (m MatchHeaderRE) Validate() error { func (m MatchHeaderRE) Validate() error {
for _, rm := range m { for _, rm := range m {
err := rm.Validate() err := rm.Validate()
@ -188,6 +216,7 @@ func (m MatchHeaderRE) Validate() error {
return nil return nil
} }
// Match returns true if r matches m.
func (m MatchProtocol) Match(r *http.Request) bool { func (m MatchProtocol) Match(r *http.Request) bool {
switch string(m) { switch string(m) {
case "grpc": case "grpc":
@ -200,6 +229,7 @@ func (m MatchProtocol) Match(r *http.Request) bool {
return false return false
} }
// Match returns true if r matches m.
func (m MatchStarlarkExpr) Match(r *http.Request) bool { func (m MatchStarlarkExpr) Match(r *http.Request) bool {
input := string(m) input := string(m)
thread := new(starlark.Thread) thread := new(starlark.Thread)
@ -213,15 +243,16 @@ func (m MatchStarlarkExpr) Match(r *http.Request) bool {
return val.String() == "True" return val.String() == "True"
} }
// matchRegexp is just the fields common among // MatchRegexp is an embeddable type for matching
// matchers that can use regular expressions. // using regular expressions.
type matchRegexp struct { type MatchRegexp struct {
Name string `json:"name"` Name string `json:"name"`
Pattern string `json:"pattern"` Pattern string `json:"pattern"`
compiled *regexp.Regexp compiled *regexp.Regexp
} }
func (mre *matchRegexp) Provision() error { // Provision compiles the regular expression.
func (mre *MatchRegexp) Provision() error {
re, err := regexp.Compile(mre.Pattern) re, err := regexp.Compile(mre.Pattern)
if err != nil { if err != nil {
return fmt.Errorf("compiling matcher regexp %s: %v", mre.Pattern, err) return fmt.Errorf("compiling matcher regexp %s: %v", mre.Pattern, err)
@ -230,14 +261,21 @@ func (mre *matchRegexp) Provision() error {
return nil return nil
} }
func (mre *matchRegexp) Validate() error { // Validate ensures mre is set up correctly.
func (mre *MatchRegexp) Validate() error {
if mre.Name != "" && !wordRE.MatchString(mre.Name) { if mre.Name != "" && !wordRE.MatchString(mre.Name) {
return fmt.Errorf("invalid regexp name (must contain only word characters): %s", mre.Name) return fmt.Errorf("invalid regexp name (must contain only word characters): %s", mre.Name)
} }
return nil return nil
} }
func (mre *matchRegexp) match(input string, repl caddy2.Replacer, scope string) bool { // Match returns true if input matches the compiled regular
// expression in mre. It sets values on the replacer repl
// associated with capture groups, using the given scope
// (namespace). Capture groups stored to repl will take on
// the name "http.matchers.<scope>.<mre.Name>.<N>" where
// <N> is the name or number of the capture group.
func (mre *MatchRegexp) Match(input string, repl caddy2.Replacer, scope string) bool {
matches := mre.compiled.FindStringSubmatch(input) matches := mre.compiled.FindStringSubmatch(input)
if matches == nil { if matches == nil {
return false return false

View file

@ -176,38 +176,38 @@ func TestPathREMatcher(t *testing.T) {
expect: true, expect: true,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "/"}}, match: MatchPathRE{MatchRegexp{Pattern: "/"}},
input: "/", input: "/",
expect: true, expect: true,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "/foo"}}, match: MatchPathRE{MatchRegexp{Pattern: "/foo"}},
input: "/foo", input: "/foo",
expect: true, expect: true,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "/foo"}}, match: MatchPathRE{MatchRegexp{Pattern: "/foo"}},
input: "/foo/", input: "/foo/",
expect: true, expect: true,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "/bar"}}, match: MatchPathRE{MatchRegexp{Pattern: "/bar"}},
input: "/foo/", input: "/foo/",
expect: false, expect: false,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "^/bar"}}, match: MatchPathRE{MatchRegexp{Pattern: "^/bar"}},
input: "/foo/bar", input: "/foo/bar",
expect: false, expect: false,
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "^/foo/(.*)/baz$", Name: "name"}}, match: MatchPathRE{MatchRegexp{Pattern: "^/foo/(.*)/baz$", Name: "name"}},
input: "/foo/bar/baz", input: "/foo/bar/baz",
expect: true, expect: true,
expectRepl: map[string]string{"name.1": "bar"}, expectRepl: map[string]string{"name.1": "bar"},
}, },
{ {
match: MatchPathRE{matchRegexp{Pattern: "^/foo/(?P<myparam>.*)/baz$", Name: "name"}}, match: MatchPathRE{MatchRegexp{Pattern: "^/foo/(?P<myparam>.*)/baz$", Name: "name"}},
input: "/foo/bar/baz", input: "/foo/bar/baz",
expect: true, expect: true,
expectRepl: map[string]string{"name.myparam": "bar"}, expectRepl: map[string]string{"name.myparam": "bar"},
@ -315,17 +315,17 @@ func TestHeaderREMatcher(t *testing.T) {
expectRepl map[string]string expectRepl map[string]string
}{ }{
{ {
match: MatchHeaderRE{"Field": &matchRegexp{Pattern: "foo"}}, match: MatchHeaderRE{"Field": &MatchRegexp{Pattern: "foo"}},
input: http.Header{"Field": []string{"foo"}}, input: http.Header{"Field": []string{"foo"}},
expect: true, expect: true,
}, },
{ {
match: MatchHeaderRE{"Field": &matchRegexp{Pattern: "$foo^"}}, match: MatchHeaderRE{"Field": &MatchRegexp{Pattern: "$foo^"}},
input: http.Header{"Field": []string{"foobar"}}, input: http.Header{"Field": []string{"foobar"}},
expect: false, expect: false,
}, },
{ {
match: MatchHeaderRE{"Field": &matchRegexp{Pattern: "^foo(.*)$", Name: "name"}}, match: MatchHeaderRE{"Field": &MatchRegexp{Pattern: "^foo(.*)$", Name: "name"}},
input: http.Header{"Field": []string{"foobar"}}, input: http.Header{"Field": []string{"foobar"}},
expect: true, expect: true,
expectRepl: map[string]string{"name.1": "bar"}, expectRepl: map[string]string{"name.1": "bar"},

View file

@ -12,17 +12,42 @@ import (
// middlewares, and a responder for handling HTTP // middlewares, and a responder for handling HTTP
// requests. // requests.
type ServerRoute struct { type ServerRoute struct {
Group string `json:"group,omitempty"` Group string `json:"group,omitempty"`
Matchers map[string]json.RawMessage `json:"match,omitempty"` MatcherSets []map[string]json.RawMessage `json:"match,omitempty"`
Apply []json.RawMessage `json:"apply,omitempty"` Apply []json.RawMessage `json:"apply,omitempty"`
Respond json.RawMessage `json:"respond,omitempty"` Respond json.RawMessage `json:"respond,omitempty"`
Terminal bool `json:"terminal,omitempty"` Terminal bool `json:"terminal,omitempty"`
// decoded values // decoded values
matchers []RequestMatcher matcherSets []MatcherSet
middleware []MiddlewareHandler middleware []MiddlewareHandler
responder Handler responder Handler
}
func (sr ServerRoute) anyMatcherSetMatches(r *http.Request) bool {
for _, ms := range sr.matcherSets {
if ms.Match(r) {
return true
}
}
return false
}
// MatcherSet is a set of matchers which
// must all match in order for the request
// to be matched successfully.
type MatcherSet []RequestMatcher
// Match returns true if the request matches all
// matchers in mset.
func (mset MatcherSet) Match(r *http.Request) bool {
for _, m := range mset {
if !m.Match(r) {
return false
}
}
return true
} }
// RouteList is a list of server routes that can // RouteList is a list of server routes that can
@ -33,14 +58,18 @@ type RouteList []ServerRoute
func (routes RouteList) Provision(ctx caddy2.Context) error { func (routes RouteList) Provision(ctx caddy2.Context) error {
for i, route := range routes { for i, route := range routes {
// matchers // matchers
for modName, rawMsg := range route.Matchers { for _, matcherSet := range route.MatcherSets {
val, err := ctx.LoadModule("http.matchers."+modName, rawMsg) var matchers MatcherSet
if err != nil { for modName, rawMsg := range matcherSet {
return fmt.Errorf("loading matcher module '%s': %v", modName, err) val, err := ctx.LoadModule("http.matchers."+modName, rawMsg)
if err != nil {
return fmt.Errorf("loading matcher module '%s': %v", modName, err)
}
matchers = append(matchers, val.(RequestMatcher))
} }
routes[i].matchers = append(routes[i].matchers, val.(RequestMatcher)) routes[i].matcherSets = append(routes[i].matcherSets, matchers)
} }
routes[i].Matchers = nil // allow GC to deallocate - TODO: Does this help? routes[i].MatcherSets = nil // allow GC to deallocate - TODO: Does this help?
// middleware // middleware
for j, rawMsg := range route.Apply { for j, rawMsg := range route.Apply {
@ -78,13 +107,10 @@ func (routes RouteList) BuildCompositeRoute(rw http.ResponseWriter, req *http.Re
var responder Handler var responder Handler
groups := make(map[string]struct{}) groups := make(map[string]struct{})
routeLoop:
for _, route := range routes { for _, route := range routes {
// see if route matches // route must match at least one of the matcher sets
for _, m := range route.matchers { if !route.anyMatcherSetMatches(req) {
if !m.Match(req) { continue
continue routeLoop
}
} }
// if route is part of a group, ensure only // if route is part of a group, ensure only