mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-14 06:46:27 +03:00
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:
parent
bc00d840e8
commit
284fb3a98c
4 changed files with 133 additions and 61 deletions
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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"},
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue