2019-04-11 20:42:55 -06:00
|
|
|
package caddyhttp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"bitbucket.org/lightcodelabs/caddy2"
|
|
|
|
)
|
|
|
|
|
2019-05-14 14:14:05 -06:00
|
|
|
// ServerRoute represents a set of matching rules,
|
|
|
|
// middlewares, and a responder for handling HTTP
|
|
|
|
// requests.
|
|
|
|
type ServerRoute struct {
|
2019-05-20 10:59:20 -06:00
|
|
|
Group string `json:"group"`
|
2019-04-11 20:42:55 -06:00
|
|
|
Matchers map[string]json.RawMessage `json:"match"`
|
|
|
|
Apply []json.RawMessage `json:"apply"`
|
|
|
|
Respond json.RawMessage `json:"respond"`
|
|
|
|
|
2019-05-10 21:07:02 -06:00
|
|
|
Terminal bool `json:"terminal"`
|
2019-04-11 20:42:55 -06:00
|
|
|
|
|
|
|
// decoded values
|
2019-05-20 10:59:20 -06:00
|
|
|
matchers []RequestMatcher
|
2019-04-11 20:42:55 -06:00
|
|
|
middleware []MiddlewareHandler
|
|
|
|
responder Handler
|
|
|
|
}
|
|
|
|
|
2019-05-14 14:14:05 -06:00
|
|
|
// RouteList is a list of server routes that can
|
|
|
|
// create a middleware chain.
|
|
|
|
type RouteList []ServerRoute
|
2019-04-11 20:42:55 -06:00
|
|
|
|
2019-05-14 14:14:05 -06:00
|
|
|
// Provision sets up all the routes by loading the modules.
|
2019-05-16 16:05:38 -06:00
|
|
|
func (routes RouteList) Provision(ctx caddy2.Context) error {
|
2019-05-14 14:14:05 -06:00
|
|
|
for i, route := range routes {
|
|
|
|
// matchers
|
|
|
|
for modName, rawMsg := range route.Matchers {
|
2019-05-16 16:05:38 -06:00
|
|
|
val, err := ctx.LoadModule("http.matchers."+modName, rawMsg)
|
2019-05-14 14:14:05 -06:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("loading matcher module '%s': %v", modName, err)
|
|
|
|
}
|
2019-05-20 10:59:20 -06:00
|
|
|
routes[i].matchers = append(routes[i].matchers, val.(RequestMatcher))
|
2019-05-14 14:14:05 -06:00
|
|
|
}
|
|
|
|
routes[i].Matchers = nil // allow GC to deallocate - TODO: Does this help?
|
|
|
|
|
|
|
|
// middleware
|
|
|
|
for j, rawMsg := range route.Apply {
|
2019-05-16 16:05:38 -06:00
|
|
|
mid, err := ctx.LoadModuleInline("middleware", "http.middleware", rawMsg)
|
2019-05-14 14:14:05 -06:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("loading middleware module in position %d: %v", j, err)
|
|
|
|
}
|
|
|
|
routes[i].middleware = append(routes[i].middleware, mid.(MiddlewareHandler))
|
|
|
|
}
|
|
|
|
routes[i].Apply = nil // allow GC to deallocate - TODO: Does this help?
|
|
|
|
|
|
|
|
// responder
|
|
|
|
if route.Respond != nil {
|
2019-05-16 16:05:38 -06:00
|
|
|
resp, err := ctx.LoadModuleInline("responder", "http.responders", route.Respond)
|
2019-05-14 14:14:05 -06:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("loading responder module: %v", err)
|
|
|
|
}
|
|
|
|
routes[i].responder = resp.(Handler)
|
|
|
|
}
|
|
|
|
routes[i].Respond = nil // allow GC to deallocate - TODO: Does this help?
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-20 10:59:20 -06:00
|
|
|
// BuildCompositeRoute creates a chain of handlers by
|
2019-05-14 14:14:05 -06:00
|
|
|
// applying all the matching routes.
|
2019-05-20 10:59:20 -06:00
|
|
|
func (routes RouteList) BuildCompositeRoute(w http.ResponseWriter, r *http.Request) Handler {
|
2019-04-11 20:42:55 -06:00
|
|
|
if len(routes) == 0 {
|
|
|
|
return emptyHandler
|
|
|
|
}
|
|
|
|
|
|
|
|
var mid []Middleware
|
|
|
|
var responder Handler
|
|
|
|
mrw := &middlewareResponseWriter{ResponseWriterWrapper: &ResponseWriterWrapper{w}}
|
2019-05-20 10:59:20 -06:00
|
|
|
groups := make(map[string]struct{})
|
2019-04-11 20:42:55 -06:00
|
|
|
|
2019-04-25 13:54:48 -06:00
|
|
|
routeLoop:
|
2019-04-11 20:42:55 -06:00
|
|
|
for _, route := range routes {
|
2019-05-20 10:59:20 -06:00
|
|
|
// see if route matches
|
2019-04-11 20:42:55 -06:00
|
|
|
for _, m := range route.matchers {
|
2019-04-25 13:54:48 -06:00
|
|
|
if !m.Match(r) {
|
|
|
|
continue routeLoop
|
2019-04-11 20:42:55 -06:00
|
|
|
}
|
|
|
|
}
|
2019-05-20 10:59:20 -06:00
|
|
|
|
|
|
|
// if route is part of a group, ensure only
|
|
|
|
// the first matching route in the group is
|
|
|
|
// applied
|
|
|
|
if route.Group != "" {
|
|
|
|
_, ok := groups[route.Group]
|
|
|
|
if ok {
|
|
|
|
// this group has already been satisfied
|
|
|
|
// by a matching route
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// this matching route satisfies the group
|
|
|
|
groups[route.Group] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply the rest of the route
|
2019-04-11 20:42:55 -06:00
|
|
|
for _, m := range route.middleware {
|
|
|
|
mid = append(mid, func(next HandlerFunc) HandlerFunc {
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
2019-05-20 10:59:20 -06:00
|
|
|
// TODO: This is where request tracing could be implemented; also
|
|
|
|
// see below to trace the responder as well
|
|
|
|
// TODO: Trace a diff of the request, would be cool too! see what changed since the last middleware (host, headers, URI...)
|
|
|
|
// TODO: see what the std lib gives us in terms of stack trracing too
|
2019-04-11 20:42:55 -06:00
|
|
|
return m.ServeHTTP(mrw, r, next)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if responder == nil {
|
|
|
|
responder = route.responder
|
|
|
|
}
|
2019-05-10 21:07:02 -06:00
|
|
|
if route.Terminal {
|
2019-04-11 20:42:55 -06:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// build the middleware stack, with the responder at the end
|
|
|
|
stack := HandlerFunc(func(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
if responder == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
mrw.allowWrites = true
|
|
|
|
return responder.ServeHTTP(w, r)
|
|
|
|
})
|
|
|
|
for i := len(mid) - 1; i >= 0; i-- {
|
|
|
|
stack = mid[i](stack)
|
|
|
|
}
|
|
|
|
|
|
|
|
return stack
|
|
|
|
}
|
2019-05-20 10:59:20 -06:00
|
|
|
|
|
|
|
type middlewareResponseWriter struct {
|
|
|
|
*ResponseWriterWrapper
|
|
|
|
allowWrites bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mrw middlewareResponseWriter) WriteHeader(statusCode int) {
|
|
|
|
if !mrw.allowWrites {
|
|
|
|
panic("WriteHeader: middleware cannot write to the response")
|
|
|
|
}
|
|
|
|
mrw.ResponseWriterWrapper.WriteHeader(statusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mrw middlewareResponseWriter) Write(b []byte) (int, error) {
|
|
|
|
if !mrw.allowWrites {
|
|
|
|
panic("Write: middleware cannot write to the response")
|
|
|
|
}
|
|
|
|
return mrw.ResponseWriterWrapper.Write(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Interface guard
|
|
|
|
var _ HTTPInterfaces = middlewareResponseWriter{}
|