| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588 | // Copyright 2012 The Gorilla Authors. All rights reserved.// Use of this source code is governed by a BSD-style// license that can be found in the LICENSE file.package muximport (	"errors"	"fmt"	"net/http"	"path"	"regexp")var (	// ErrMethodMismatch is returned when the method in the request does not match	// the method defined against the route.	ErrMethodMismatch = errors.New("method is not allowed")	// ErrNotFound is returned when no route match is found.	ErrNotFound = errors.New("no matching route was found"))// NewRouter returns a new router instance.func NewRouter() *Router {	return &Router{namedRoutes: make(map[string]*Route), KeepContext: false}}// Router registers routes to be matched and dispatches a handler.//// It implements the http.Handler interface, so it can be registered to serve// requests:////     var router = mux.NewRouter()////     func main() {//         http.Handle("/", router)//     }//// Or, for Google App Engine, register it in a init() function:////     func init() {//         http.Handle("/", router)//     }//// This will send all incoming requests to the router.type Router struct {	// Configurable Handler to be used when no route matches.	NotFoundHandler http.Handler	// Configurable Handler to be used when the request method does not match the route.	MethodNotAllowedHandler http.Handler	// Parent route, if this is a subrouter.	parent parentRoute	// Routes to be matched, in order.	routes []*Route	// Routes by name for URL building.	namedRoutes map[string]*Route	// See Router.StrictSlash(). This defines the flag for new routes.	strictSlash bool	// See Router.SkipClean(). This defines the flag for new routes.	skipClean bool	// If true, do not clear the request context after handling the request.	// This has no effect when go1.7+ is used, since the context is stored	// on the request itself.	KeepContext bool	// see Router.UseEncodedPath(). This defines a flag for all routes.	useEncodedPath bool	// Slice of middlewares to be called after a match is found	middlewares []middleware}// Match attempts to match the given request against the router's registered routes.//// If the request matches a route of this router or one of its subrouters the Route,// Handler, and Vars fields of the the match argument are filled and this function// returns true.//// If the request does not match any of this router's or its subrouters' routes// then this function returns false. If available, a reason for the match failure// will be filled in the match argument's MatchErr field. If the match failure type// (eg: not found) has a registered handler, the handler is assigned to the Handler// field of the match argument.func (r *Router) Match(req *http.Request, match *RouteMatch) bool {	for _, route := range r.routes {		if route.Match(req, match) {			// Build middleware chain if no error was found			if match.MatchErr == nil {				for i := len(r.middlewares) - 1; i >= 0; i-- {					match.Handler = r.middlewares[i].Middleware(match.Handler)				}			}			return true		}	}	if match.MatchErr == ErrMethodMismatch {		if r.MethodNotAllowedHandler != nil {			match.Handler = r.MethodNotAllowedHandler			return true		}		return false	}	// Closest match for a router (includes sub-routers)	if r.NotFoundHandler != nil {		match.Handler = r.NotFoundHandler		match.MatchErr = ErrNotFound		return true	}	match.MatchErr = ErrNotFound	return false}// ServeHTTP dispatches the handler registered in the matched route.//// When there is a match, the route variables can be retrieved calling// mux.Vars(request).func (r *Router) ServeHTTP(w http.ResponseWriter, req *http.Request) {	if !r.skipClean {		path := req.URL.Path		if r.useEncodedPath {			path = req.URL.EscapedPath()		}		// Clean path to canonical form and redirect.		if p := cleanPath(path); p != path {			// Added 3 lines (Philip Schlump) - It was dropping the query string and #whatever from query.			// This matches with fix in go 1.2 r.c. 4 for same problem.  Go Issue:			// http://code.google.com/p/go/issues/detail?id=5252			url := *req.URL			url.Path = p			p = url.String()			w.Header().Set("Location", p)			w.WriteHeader(http.StatusMovedPermanently)			return		}	}	var match RouteMatch	var handler http.Handler	if r.Match(req, &match) {		handler = match.Handler		req = setVars(req, match.Vars)		req = setCurrentRoute(req, match.Route)	}	if handler == nil && match.MatchErr == ErrMethodMismatch {		handler = methodNotAllowedHandler()	}	if handler == nil {		handler = http.NotFoundHandler()	}	if !r.KeepContext {		defer contextClear(req)	}	handler.ServeHTTP(w, req)}// Get returns a route registered with the given name.func (r *Router) Get(name string) *Route {	return r.getNamedRoutes()[name]}// GetRoute returns a route registered with the given name. This method// was renamed to Get() and remains here for backwards compatibility.func (r *Router) GetRoute(name string) *Route {	return r.getNamedRoutes()[name]}// StrictSlash defines the trailing slash behavior for new routes. The initial// value is false.//// When true, if the route path is "/path/", accessing "/path" will perform a redirect// to the former and vice versa. In other words, your application will always// see the path as specified in the route.//// When false, if the route path is "/path", accessing "/path/" will not match// this route and vice versa.//// The re-direct is a HTTP 301 (Moved Permanently). Note that when this is set for// routes with a non-idempotent method (e.g. POST, PUT), the subsequent re-directed// request will be made as a GET by most clients. Use middleware or client settings// to modify this behaviour as needed.//// Special case: when a route sets a path prefix using the PathPrefix() method,// strict slash is ignored for that route because the redirect behavior can't// be determined from a prefix alone. However, any subrouters created from that// route inherit the original StrictSlash setting.func (r *Router) StrictSlash(value bool) *Router {	r.strictSlash = value	return r}// SkipClean defines the path cleaning behaviour for new routes. The initial// value is false. Users should be careful about which routes are not cleaned//// When true, if the route path is "/path//to", it will remain with the double// slash. This is helpful if you have a route like: /fetch/http://xkcd.com/534///// When false, the path will be cleaned, so /fetch/http://xkcd.com/534/ will// become /fetch/http/xkcd.com/534func (r *Router) SkipClean(value bool) *Router {	r.skipClean = value	return r}// UseEncodedPath tells the router to match the encoded original path// to the routes.// For eg. "/path/foo%2Fbar/to" will match the path "/path/{var}/to".//// If not called, the router will match the unencoded path to the routes.// For eg. "/path/foo%2Fbar/to" will match the path "/path/foo/bar/to"func (r *Router) UseEncodedPath() *Router {	r.useEncodedPath = true	return r}// ----------------------------------------------------------------------------// parentRoute// ----------------------------------------------------------------------------func (r *Router) getBuildScheme() string {	if r.parent != nil {		return r.parent.getBuildScheme()	}	return ""}// getNamedRoutes returns the map where named routes are registered.func (r *Router) getNamedRoutes() map[string]*Route {	if r.namedRoutes == nil {		if r.parent != nil {			r.namedRoutes = r.parent.getNamedRoutes()		} else {			r.namedRoutes = make(map[string]*Route)		}	}	return r.namedRoutes}// getRegexpGroup returns regexp definitions from the parent route, if any.func (r *Router) getRegexpGroup() *routeRegexpGroup {	if r.parent != nil {		return r.parent.getRegexpGroup()	}	return nil}func (r *Router) buildVars(m map[string]string) map[string]string {	if r.parent != nil {		m = r.parent.buildVars(m)	}	return m}// ----------------------------------------------------------------------------// Route factories// ----------------------------------------------------------------------------// NewRoute registers an empty route.func (r *Router) NewRoute() *Route {	route := &Route{parent: r, strictSlash: r.strictSlash, skipClean: r.skipClean, useEncodedPath: r.useEncodedPath}	r.routes = append(r.routes, route)	return route}// Handle registers a new route with a matcher for the URL path.// See Route.Path() and Route.Handler().func (r *Router) Handle(path string, handler http.Handler) *Route {	return r.NewRoute().Path(path).Handler(handler)}// HandleFunc registers a new route with a matcher for the URL path.// See Route.Path() and Route.HandlerFunc().func (r *Router) HandleFunc(path string, f func(http.ResponseWriter,	*http.Request)) *Route {	return r.NewRoute().Path(path).HandlerFunc(f)}// Headers registers a new route with a matcher for request header values.// See Route.Headers().func (r *Router) Headers(pairs ...string) *Route {	return r.NewRoute().Headers(pairs...)}// Host registers a new route with a matcher for the URL host.// See Route.Host().func (r *Router) Host(tpl string) *Route {	return r.NewRoute().Host(tpl)}// MatcherFunc registers a new route with a custom matcher function.// See Route.MatcherFunc().func (r *Router) MatcherFunc(f MatcherFunc) *Route {	return r.NewRoute().MatcherFunc(f)}// Methods registers a new route with a matcher for HTTP methods.// See Route.Methods().func (r *Router) Methods(methods ...string) *Route {	return r.NewRoute().Methods(methods...)}// Path registers a new route with a matcher for the URL path.// See Route.Path().func (r *Router) Path(tpl string) *Route {	return r.NewRoute().Path(tpl)}// PathPrefix registers a new route with a matcher for the URL path prefix.// See Route.PathPrefix().func (r *Router) PathPrefix(tpl string) *Route {	return r.NewRoute().PathPrefix(tpl)}// Queries registers a new route with a matcher for URL query values.// See Route.Queries().func (r *Router) Queries(pairs ...string) *Route {	return r.NewRoute().Queries(pairs...)}// Schemes registers a new route with a matcher for URL schemes.// See Route.Schemes().func (r *Router) Schemes(schemes ...string) *Route {	return r.NewRoute().Schemes(schemes...)}// BuildVarsFunc registers a new route with a custom function for modifying// route variables before building a URL.func (r *Router) BuildVarsFunc(f BuildVarsFunc) *Route {	return r.NewRoute().BuildVarsFunc(f)}// Walk walks the router and all its sub-routers, calling walkFn for each route// in the tree. The routes are walked in the order they were added. Sub-routers// are explored depth-first.func (r *Router) Walk(walkFn WalkFunc) error {	return r.walk(walkFn, []*Route{})}// SkipRouter is used as a return value from WalkFuncs to indicate that the// router that walk is about to descend down to should be skipped.var SkipRouter = errors.New("skip this router")// WalkFunc is the type of the function called for each route visited by Walk.// At every invocation, it is given the current route, and the current router,// and a list of ancestor routes that lead to the current route.type WalkFunc func(route *Route, router *Router, ancestors []*Route) errorfunc (r *Router) walk(walkFn WalkFunc, ancestors []*Route) error {	for _, t := range r.routes {		err := walkFn(t, r, ancestors)		if err == SkipRouter {			continue		}		if err != nil {			return err		}		for _, sr := range t.matchers {			if h, ok := sr.(*Router); ok {				ancestors = append(ancestors, t)				err := h.walk(walkFn, ancestors)				if err != nil {					return err				}				ancestors = ancestors[:len(ancestors)-1]			}		}		if h, ok := t.handler.(*Router); ok {			ancestors = append(ancestors, t)			err := h.walk(walkFn, ancestors)			if err != nil {				return err			}			ancestors = ancestors[:len(ancestors)-1]		}	}	return nil}// ----------------------------------------------------------------------------// Context// ----------------------------------------------------------------------------// RouteMatch stores information about a matched route.type RouteMatch struct {	Route   *Route	Handler http.Handler	Vars    map[string]string	// MatchErr is set to appropriate matching error	// It is set to ErrMethodMismatch if there is a mismatch in	// the request method and route method	MatchErr error}type contextKey intconst (	varsKey contextKey = iota	routeKey)// Vars returns the route variables for the current request, if any.func Vars(r *http.Request) map[string]string {	if rv := contextGet(r, varsKey); rv != nil {		return rv.(map[string]string)	}	return nil}// CurrentRoute returns the matched route for the current request, if any.// This only works when called inside the handler of the matched route// because the matched route is stored in the request context which is cleared// after the handler returns, unless the KeepContext option is set on the// Router.func CurrentRoute(r *http.Request) *Route {	if rv := contextGet(r, routeKey); rv != nil {		return rv.(*Route)	}	return nil}func setVars(r *http.Request, val interface{}) *http.Request {	return contextSet(r, varsKey, val)}func setCurrentRoute(r *http.Request, val interface{}) *http.Request {	return contextSet(r, routeKey, val)}// ----------------------------------------------------------------------------// Helpers// ----------------------------------------------------------------------------// cleanPath returns the canonical path for p, eliminating . and .. elements.// Borrowed from the net/http package.func cleanPath(p string) string {	if p == "" {		return "/"	}	if p[0] != '/' {		p = "/" + p	}	np := path.Clean(p)	// path.Clean removes trailing slash except for root;	// put the trailing slash back if necessary.	if p[len(p)-1] == '/' && np != "/" {		np += "/"	}	return np}// uniqueVars returns an error if two slices contain duplicated strings.func uniqueVars(s1, s2 []string) error {	for _, v1 := range s1 {		for _, v2 := range s2 {			if v1 == v2 {				return fmt.Errorf("mux: duplicated route variable %q", v2)			}		}	}	return nil}// checkPairs returns the count of strings passed in, and an error if// the count is not an even number.func checkPairs(pairs ...string) (int, error) {	length := len(pairs)	if length%2 != 0 {		return length, fmt.Errorf(			"mux: number of parameters must be multiple of 2, got %v", pairs)	}	return length, nil}// mapFromPairsToString converts variadic string parameters to a// string to string map.func mapFromPairsToString(pairs ...string) (map[string]string, error) {	length, err := checkPairs(pairs...)	if err != nil {		return nil, err	}	m := make(map[string]string, length/2)	for i := 0; i < length; i += 2 {		m[pairs[i]] = pairs[i+1]	}	return m, nil}// mapFromPairsToRegex converts variadic string parameters to a// string to regex map.func mapFromPairsToRegex(pairs ...string) (map[string]*regexp.Regexp, error) {	length, err := checkPairs(pairs...)	if err != nil {		return nil, err	}	m := make(map[string]*regexp.Regexp, length/2)	for i := 0; i < length; i += 2 {		regex, err := regexp.Compile(pairs[i+1])		if err != nil {			return nil, err		}		m[pairs[i]] = regex	}	return m, nil}// matchInArray returns true if the given string value is in the array.func matchInArray(arr []string, value string) bool {	for _, v := range arr {		if v == value {			return true		}	}	return false}// matchMapWithString returns true if the given key/value pairs exist in a given map.func matchMapWithString(toCheck map[string]string, toMatch map[string][]string, canonicalKey bool) bool {	for k, v := range toCheck {		// Check if key exists.		if canonicalKey {			k = http.CanonicalHeaderKey(k)		}		if values := toMatch[k]; values == nil {			return false		} else if v != "" {			// If value was defined as an empty string we only check that the			// key exists. Otherwise we also check for equality.			valueExists := false			for _, value := range values {				if v == value {					valueExists = true					break				}			}			if !valueExists {				return false			}		}	}	return true}// matchMapWithRegex returns true if the given key/value pairs exist in a given map compiled against// the given regexfunc matchMapWithRegex(toCheck map[string]*regexp.Regexp, toMatch map[string][]string, canonicalKey bool) bool {	for k, v := range toCheck {		// Check if key exists.		if canonicalKey {			k = http.CanonicalHeaderKey(k)		}		if values := toMatch[k]; values == nil {			return false		} else if v != nil {			// If value was defined as an empty string we only check that the			// key exists. Otherwise we also check for equality.			valueExists := false			for _, value := range values {				if v.MatchString(value) {					valueExists = true					break				}			}			if !valueExists {				return false			}		}	}	return true}// methodNotAllowed replies to the request with an HTTP status code 405.func methodNotAllowed(w http.ResponseWriter, r *http.Request) {	w.WriteHeader(http.StatusMethodNotAllowed)}// methodNotAllowedHandler returns a simple request handler// that replies to each request with a status code 405.func methodNotAllowedHandler() http.Handler { return http.HandlerFunc(methodNotAllowed) }
 |