// Copyright 2023 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // This file implements a decision tree for fast matching of requests to // patterns. // // The root of the tree branches on the host of the request. // The next level branches on the method. // The remaining levels branch on consecutive segments of the path. // // The "more specific wins" precedence rule can result in backtracking. // For example, given the patterns // /a/b/z // /a/{x}/c // we will first try to match the path "/a/b/c" with /a/b/z, and // when that fails we will try against /a/{x}/c. package http import ( "strings" ) // A routingNode is a node in the decision tree. // The same struct is used for leaf and interior nodes. type routingNode struct { // A leaf node holds a single pattern and the Handler it was registered // with. pattern *pattern handler Handler // An interior node maps parts of the incoming request to child nodes. // special children keys: // "/" trailing slash (resulting from {$}) // "" single wildcard // "*" multi wildcard children mapping[string, *routingNode] emptyChild *routingNode // optimization: child with key "" } // addPattern adds a pattern and its associated Handler to the tree // at root. func (root *routingNode) addPattern(p *pattern, h Handler) { // First level of tree is host. n := root.addChild(p.host) // Second level of tree is method. n = n.addChild(p.method) // Remaining levels are path. n.addSegments(p.segments, p, h) } // addSegments adds the given segments to the tree rooted at n. // If there are no segments, then n is a leaf node that holds // the given pattern and handler. func (n *routingNode) addSegments(segs []segment, p *pattern, h Handler) { if len(segs) == 0 { n.set(p, h) return } seg := segs[0] if seg.multi { if len(segs) != 1 { panic("multi wildcard not last") } n.addChild("*").set(p, h) } else if seg.wild { n.addChild("").addSegments(segs[1:], p, h) } else { n.addChild(seg.s).addSegments(segs[1:], p, h) } } // set sets the pattern and handler for n, which // must be a leaf node. func (n *routingNode) set(p *pattern, h Handler) { if n.pattern != nil || n.handler != nil { panic("non-nil leaf fields") } n.pattern = p n.handler = h } // addChild adds a child node with the given key to n // if one does not exist, and returns the child. func (n *routingNode) addChild(key string) *routingNode { if key == "" { if n.emptyChild == nil { n.emptyChild = &routingNode{} } return n.emptyChild } if c := n.findChild(key); c != nil { return c } c := &routingNode{} n.children.add(key, c) return c } // findChild returns the child of n with the given key, or nil // if there is no child with that key. func (n *routingNode) findChild(key string) *routingNode { if key == "" { return n.emptyChild } r, _ := n.children.find(key) return r } // match returns the leaf node under root that matches the arguments, and a list // of values for pattern wildcards in the order that the wildcards appear. // For example, if the request path is "/a/b/c" and the pattern is "/{x}/b/{y}", // then the second return value will be []string{"a", "c"}. func (root *routingNode) match(host, method, path string) (*routingNode, []string) { if host != "" { // There is a host. If there is a pattern that specifies that host and it // matches, we are done. If the pattern doesn't match, fall through to // try patterns with no host. if l, m := root.findChild(host).matchMethodAndPath(method, path); l != nil { return l, m } } return root.emptyChild.matchMethodAndPath(method, path) } // matchMethodAndPath matches the method and path. // Its return values are the same as [routingNode.match]. // The receiver should be a child of the root. func (n *routingNode) matchMethodAndPath(method, path string) (*routingNode, []string) { if n == nil { return nil, nil } if l, m := n.findChild(method).matchPath(path, nil); l != nil { // Exact match of method name. return l, m } if method == "HEAD" { // GET matches HEAD too. if l, m := n.findChild("GET").matchPath(path, nil); l != nil { return l, m } } // No exact match; try patterns with no method. return n.emptyChild.matchPath(path, nil) } // matchPath matches a path. // Its return values are the same as [routingNode.match]. // matchPath calls itself recursively. The matches argument holds the wildcard matches // found so far. func (n *routingNode) matchPath(path string, matches []string) (*routingNode, []string) { if n == nil { return nil, nil } // If path is empty, then we are done. // If n is a leaf node, we found a match; return it. // If n is an interior node (which means it has a nil pattern), // then we failed to match. if path == "" { if n.pattern == nil { return nil, nil } return n, matches } // Get the first segment of path. seg, rest := firstSegment(path) // First try matching against patterns that have a literal for this position. // We know by construction that such patterns are more specific than those // with a wildcard at this position (they are either more specific, equivalent, // or overlap, and we ruled out the first two when the patterns were registered). if n, m := n.findChild(seg).matchPath(rest, matches); n != nil { return n, m } // If matching a literal fails, try again with patterns that have a single // wildcard (represented by an empty string in the child mapping). // Again, by construction, patterns with a single wildcard must be more specific than // those with a multi wildcard. // We skip this step if the segment is a trailing slash, because single wildcards // don't match trailing slashes. if seg != "/" { if n, m := n.emptyChild.matchPath(rest, append(matches, seg)); n != nil { return n, m } } // Lastly, match the pattern (there can be at most one) that has a multi // wildcard in this position to the rest of the path. if c := n.findChild("*"); c != nil { // Don't record a match for a nameless wildcard (which arises from a // trailing slash in the pattern). if c.pattern.lastSegment().s != "" { matches = append(matches, pathUnescape(path[1:])) // remove initial slash } return c, matches } return nil, nil } // firstSegment splits path into its first segment, and the rest. // The path must begin with "/". // If path consists of only a slash, firstSegment returns ("/", ""). // The segment is returned unescaped, if possible. func firstSegment(path string) (seg, rest string) { if path == "/" { return "/", "" } path = path[1:] // drop initial slash i := strings.IndexByte(path, '/') if i < 0 { i = len(path) } return pathUnescape(path[:i]), path[i:] } // matchingMethods adds to methodSet all the methods that would result in a // match if passed to routingNode.match with the given host and path. func (root *routingNode) matchingMethods(host, path string, methodSet map[string]bool) { if host != "" { root.findChild(host).matchingMethodsPath(path, methodSet) } root.emptyChild.matchingMethodsPath(path, methodSet) if methodSet["GET"] { methodSet["HEAD"] = true } } func (n *routingNode) matchingMethodsPath(path string, set map[string]bool) { if n == nil { return } n.children.eachPair(func(method string, c *routingNode) bool { if p, _ := c.matchPath(path, nil); p != nil { set[method] = true } return true }) // Don't look at the empty child. If there were an empty // child, it would match on any method, but we only // call this when we fail to match on a method. }