Documentation
¶
Index ¶
- Constants
- Variables
- func CleanPath(p string) string
- func DefaultHandleRecovery(c Context, err any)
- func NewTestContext(w http.ResponseWriter, r *http.Request) (*Router, Context)
- func Reverse(t *Tree, method, path string) string
- func Walk(tree *Tree, fn WalkFunc) error
- type Context
- type ContextCloser
- type HandlerFunc
- type Iterator
- func (it *Iterator) Handler() HandlerFunc
- func (it *Iterator) Method() string
- func (it *Iterator) Next()
- func (it *Iterator) Path() string
- func (it *Iterator) Rewind()
- func (it *Iterator) SeekMethod(method string)
- func (it *Iterator) SeekMethodPrefix(method, key string)
- func (it *Iterator) SeekPrefix(key string)
- func (it *Iterator) Valid() bool
- type MiddlewareFunc
- type MiddlewareScope
- type Option
- func DefaultOptions() Option
- func WithAutoOptions(enable bool) Option
- func WithMiddleware(m ...MiddlewareFunc) Option
- func WithMiddlewareFor(scope MiddlewareScope, m ...MiddlewareFunc) Option
- func WithNoMethod(enable bool) Option
- func WithNoMethodHandler(handler HandlerFunc) Option
- func WithNoRouteHandler(handler HandlerFunc) Option
- func WithOptionsHandler(handler HandlerFunc) Option
- func WithRedirectTrailingSlash(enable bool) Option
- type Param
- type Params
- type RecoveryFunc
- type ResponseWriter
- type RouteConflictError
- type Router
- func (fox *Router) Handle(method, path string, handler HandlerFunc) error
- func (fox *Router) MustHandle(method, path string, handler HandlerFunc)
- func (fox *Router) NewTree() *Tree
- func (fox *Router) Remove(method, path string) error
- func (fox *Router) ServeHTTP(w http.ResponseWriter, r *http.Request)
- func (fox *Router) Swap(new *Tree) (old *Tree)
- func (fox *Router) Tree() *Tree
- func (fox *Router) Update(method, path string, handler HandlerFunc) error
- type Tree
- func (t *Tree) Handle(method, path string, handler HandlerFunc) error
- func (t *Tree) Has(method, path string) bool
- func (t *Tree) Lookup(method, path string, lazy bool) (handler HandlerFunc, cc ContextCloser, tsr bool)
- func (t *Tree) LookupMethods(path string) (methods []string)
- func (t *Tree) Methods() []string
- func (t *Tree) Remove(method, path string) error
- func (t *Tree) Update(method, path string, handler HandlerFunc) error
- type WalkFunc
Examples ¶
Constants ¶
const ( MIMEApplicationJSON = "application/json" MIMEApplicationJSONCharsetUTF8 = MIMEApplicationJSON + "; " + charsetUTF8 MIMEApplicationJavaScript = "application/javascript" MIMEApplicationJavaScriptCharsetUTF8 = MIMEApplicationJavaScript + "; " + charsetUTF8 MIMEApplicationXML = "application/xml" MIMEApplicationXMLCharsetUTF8 = MIMEApplicationXML + "; " + charsetUTF8 MIMETextXML = "text/xml" MIMETextXMLCharsetUTF8 = MIMETextXML + "; " + charsetUTF8 MIMEApplicationForm = "application/x-www-form-urlencoded" MIMEApplicationProtobuf = "application/protobuf" MIMEApplicationMsgpack = "application/msgpack" MIMETextHTML = "text/html" MIMETextHTMLCharsetUTF8 = MIMETextHTML + "; " + charsetUTF8 MIMETextPlain = "text/plain" MIMETextPlainCharsetUTF8 = MIMETextPlain + "; " + charsetUTF8 MIMEMultipartForm = "multipart/form-data" MIMEOctetStream = "application/octet-stream" )
MIME types
const ( HeaderAccept = "Accept" HeaderAcceptEncoding = "Accept-Encoding" // HeaderAllow is the name of the "Allow" header field used to list the set of methods // advertised as supported by the target resource. Returning an Allow header is mandatory // for status 405 (method not found) and useful for the OPTIONS method in responses. // See RFC 7231: https://datatracker.ietf.org/doc/html/rfc7231#section-7.4.1 HeaderAllow = "Allow" HeaderAuthorization = "Authorization" HeaderContentDisposition = "Content-Disposition" HeaderContentEncoding = "Content-Encoding" HeaderContentLength = "Content-Length" HeaderContentType = "Content-Type" HeaderCookie = "Cookie" HeaderSetCookie = "Set-Cookie" HeaderIfModifiedSince = "If-Modified-Since" HeaderLastModified = "Last-Modified" HeaderLocation = "Location" HeaderRetryAfter = "Retry-After" HeaderUpgrade = "Upgrade" HeaderVary = "Vary" HeaderWWWAuthenticate = "WWW-Authenticate" HeaderXForwardedFor = "X-Forwarded-For" HeaderXForwardedProto = "X-Forwarded-Proto" HeaderXForwardedProtocol = "X-Forwarded-Protocol" HeaderXForwardedSsl = "X-Forwarded-Ssl" HeaderXUrlScheme = "X-Url-Scheme" HeaderXHTTPMethodOverride = "X-HTTP-Method-Override" HeaderXRealIP = "X-Real-Ip" HeaderXRequestID = "X-Request-Id" HeaderXCorrelationID = "X-Correlation-Id" HeaderXRequestedWith = "X-Requested-With" HeaderServer = "Server" HeaderOrigin = "Origin" HeaderCacheControl = "Cache-Control" HeaderConnection = "Connection" HeaderETag = "ETag" // Access control HeaderAccessControlRequestMethod = "Access-Control-Request-Method" HeaderAccessControlRequestHeaders = "Access-Control-Request-Headers" HeaderAccessControlAllowOrigin = "Access-Control-Allow-Origin" HeaderAccessControlAllowMethods = "Access-Control-Allow-Methods" HeaderAccessControlAllowHeaders = "Access-Control-Allow-Headers" HeaderAccessControlAllowCredentials = "Access-Control-Allow-Credentials" HeaderAccessControlExposeHeaders = "Access-Control-Expose-Headers" HeaderAccessControlMaxAge = "Access-Control-Max-Age" // Security HeaderStrictTransportSecurity = "Strict-Transport-Security" HeaderXContentTypeOptions = "X-Content-Type-Options" HeaderXXSSProtection = "X-XSS-Protection" HeaderXFrameOptions = "X-Frame-Options" HeaderContentSecurityPolicy = "Content-Security-Policy" HeaderContentSecurityPolicyReportOnly = "Content-Security-Policy-Report-Only" // nolint:gosec HeaderXCSRFToken = "X-CSRF-Token" HeaderReferrerPolicy = "Referrer-Policy" )
Headers
Variables ¶
var ( ErrRouteNotFound = errors.New("route not found") ErrRouteExist = errors.New("route already registered") ErrRouteConflict = errors.New("route conflict") ErrInvalidRoute = errors.New("invalid route") ErrDiscardedResponseWriter = errors.New("discarded response writer") ErrInvalidRedirectCode = errors.New("invalid redirect code") )
var SkipMethod = errors.New("skip method")
SkipMethod is used as a return value from WalkFunc to indicate that the method named in the call is to be skipped.
Functions ¶
func CleanPath ¶
CleanPath is the URL version of path.Clean, it returns a canonical URL path for p, eliminating . and .. elements.
The following rules are applied iteratively until no further processing can be done:
- Replace multiple slashes with a single slash.
- Eliminate each . path name element (the current directory).
- Eliminate each inner .. path name element (the parent directory) along with the non-.. element that precedes it.
- Eliminate .. elements that begin a rooted path: that is, replace "/.." by "/" at the beginning of a path.
If the result of this process is an empty string, "/" is returned
func DefaultHandleRecovery ¶ added in v0.7.2
DefaultHandleRecovery is a default implementation of the RecoveryFunc. It logs the recovered panic error to stderr, including the stack trace. If the response has not been written yet and the error is not caused by a broken connection, it sets the status code to http.StatusInternalServerError and writes a generic error message.
func NewTestContext ¶ added in v0.7.0
NewTestContext returns a new Router and its associated Context, designed only for testing purpose.
func Reverse ¶ added in v0.5.0
Reverse perform a lookup on the tree for the given method and path and return the matching registered route if any. This function is safe for concurrent use by multiple goroutine and while mutation on Tree are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
func Walk ¶ added in v0.5.0
Walk allow to walk over all registered route in lexicographical order. If the function return the special value SkipMethod, Walk skips the current method. This function is safe for concurrent use by multiple goroutine and while mutation are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
Types ¶
type Context ¶ added in v0.7.0
type Context interface { // Ctx returns the context associated with the current request. Ctx() ctx.Context // Request returns the current *http.Request. Request() *http.Request // SetRequest sets the *http.Request. SetRequest(r *http.Request) // Writer method returns a custom ResponseWriter implementation. The returned ResponseWriter object implements additional // http.Flusher, http.Hijacker, io.ReaderFrom interfaces for HTTP/1.x requests and http.Flusher, http.Pusher interfaces // for HTTP/2 requests. These additional interfaces provide extra functionality and are used by underlying HTTP protocols // for specific tasks. // // In actual workload scenarios, the custom ResponseWriter satisfies interfaces for HTTP/1.x and HTTP/2 protocols, // however, if testing with e.g. httptest.Recorder, only the http.Flusher is available to the underlying ResponseWriter. // Therefore, while asserting interfaces like http.Hijacker will not fail, invoking Hijack method will panic if the // underlying ResponseWriter does not implement this interface. // // To facilitate testing with e.g. httptest.Recorder, use the WrapTestContext helper function which only exposes the // http.Flusher interface for the ResponseWriter. Writer() ResponseWriter // SetWriter sets the ResponseWriter. SetWriter(w ResponseWriter) // TeeWriter append an additional writer (sink) to which the response body will be written. // This API is EXPERIMENTAL and is likely to change in future release. TeeWriter(w io.Writer) // Path returns the registered path for the handler. Path() string // Params returns a Params slice containing the matched // wildcard parameters. Params() Params // Param retrieve a matching wildcard parameter by name. Param(name string) string // QueryParams parses the Request RawQuery and returns the corresponding values. QueryParams() url.Values // QueryParam returns the first query value associated with the given key. QueryParam(name string) string // SetHeader sets the response header for the given key to the specified value. SetHeader(key, value string) // Header retrieves the value of the request header for the given key. Header(key string) string // String sends a formatted string with the specified status code. String(code int, format string, values ...any) error // Blob sends a byte slice with the specified status code and content type. Blob(code int, contentType string, buf []byte) error // Stream sends data from an io.Reader with the specified status code and content type. Stream(code int, contentType string, r io.Reader) error // Redirect sends an HTTP redirect response with the given status code and URL. Redirect(code int, url string) error // Clone returns a copy of the Context that is safe to use after the HandlerFunc returns. Clone() Context // Tree is a local copy of the Tree in use to serve the request. Tree() *Tree // Fox returns the Router in use to serve the request. Fox() *Router // Reset resets the Context to its initial state, attaching the provided Router, // http.ResponseWriter, and *http.Request. Reset(fox *Router, w http.ResponseWriter, r *http.Request) }
Context represents the context of the current HTTP request. It provides methods to access request data and to write a response. Be aware that the Context API is not thread-safe and its lifetime should be limited to the duration of the HandlerFunc execution, as the underlying implementation may be reused a soon as the handler return. (see Clone method).
func NewTestContextOnly ¶ added in v0.7.0
NewTestContextOnly returns a new Context associated with the provided Router, designed only for testing purpose.
type ContextCloser ¶ added in v0.7.0
type ContextCloser interface { Context Close() }
ContextCloser extends Context for manually created instances, adding a Close method to release resources after use.
type HandlerFunc ¶
type HandlerFunc func(c Context)
HandlerFunc is a function type that responds to an HTTP request. It enforces the same contract as http.Handler but provides additional feature like matched wildcard route segments via the Context type. The Context is freed once the HandlerFunc returns and may be reused later to save resources. If you need to hold the context longer, you have to copy it (see Clone method).
Similar to http.Handler, to abort a HandlerFunc so the client sees an interrupted response, panic with the value http.ErrAbortHandler.
HandlerFunc functions should be thread-safe, as they will be called concurrently.
func DefaultMethodNotAllowedHandler ¶ added in v0.7.6
func DefaultMethodNotAllowedHandler() HandlerFunc
DefaultMethodNotAllowedHandler returns a simple HandlerFunc that replies to each request with a “405 Method Not Allowed” reply.
func DefaultNotFoundHandler ¶ added in v0.7.6
func DefaultNotFoundHandler() HandlerFunc
DefaultNotFoundHandler returns a simple HandlerFunc that replies to each request with a “404 page not found” reply.
func WrapF ¶
func WrapF(f http.HandlerFunc) HandlerFunc
WrapF is an adapter for wrapping http.HandlerFunc and returns a HandlerFunc function.
func WrapH ¶
func WrapH(h http.Handler) HandlerFunc
WrapH is an adapter for wrapping http.Handler and returns a HandlerFunc function.
func WrapTestContext ¶ added in v0.9.0
func WrapTestContext(next HandlerFunc) HandlerFunc
WrapTestContext method is a helper function provided for testing purposes. It wraps the provided HandlerFunc, returning a new HandlerFunc that only exposes the http.Flusher interface of the ResponseWriter. This is useful when testing implementations that rely on interface assertions with e.g. httptest.Recorder, since its only supports the http.Flusher interface.
type Iterator ¶ added in v0.3.0
type Iterator struct {
// contains filtered or unexported fields
}
func NewIterator ¶ added in v0.5.0
NewIterator returns an Iterator that traverses all registered routes in lexicographic order. An Iterator is safe to use when the router is serving request, when routing updates are ongoing or in parallel with other Iterators. Note that changes that happen while iterating over routes may not be reflected by the Iterator. This api is EXPERIMENTAL and is likely to change in future release.
Example ¶
r := New() it := NewIterator(r.Tree()) // Iterate over all routes for it.Rewind(); it.Valid(); it.Next() { fmt.Println(it.Method(), it.Path()) } // Iterate over all routes for the GET method for it.SeekMethod(http.MethodGet); it.Valid(); it.Next() { fmt.Println(it.Method(), it.Path()) } // Iterate over all routes starting with /users for it.SeekPrefix("/users"); it.Valid(); it.Next() { fmt.Println(it.Method(), it.Path()) } // Iterate over all route starting with /users for the GET method for it.SeekMethodPrefix(http.MethodGet, "/user"); it.Valid(); it.Next() { fmt.Println(it.Method(), it.Path()) }
Output:
func (*Iterator) Handler ¶ added in v0.3.0
func (it *Iterator) Handler() HandlerFunc
Handler return the registered handler for the current route.
func (*Iterator) Next ¶ added in v0.3.0
func (it *Iterator) Next()
Next advance the iterator to the next route. Always check it.Valid() after a it.Next().
func (*Iterator) Rewind ¶ added in v0.3.0
func (it *Iterator) Rewind()
Rewind reset the iterator cursor all the way to zero-th position which is the first method and route. It does not keep track of whether the cursor started with SeekPrefix, SeekMethod or SeekMethodPrefix.
func (*Iterator) SeekMethod ¶ added in v0.3.0
SeekMethod reset the iterator cursor to the first route for the given method. It does not keep tracking of previous seek.
func (*Iterator) SeekMethodPrefix ¶ added in v0.3.0
SeekMethodPrefix reset the iterator cursor to the first route starting with key for the given method. It does not keep tracking of previous seek.
func (*Iterator) SeekPrefix ¶ added in v0.3.0
SeekPrefix reset the iterator cursor to the first route starting with key. It does not keep tracking of previous seek.
type MiddlewareFunc ¶ added in v0.7.0
type MiddlewareFunc func(next HandlerFunc) HandlerFunc
MiddlewareFunc is a function type for implementing HandlerFunc middleware. The returned HandlerFunc usually wraps the input HandlerFunc, allowing you to perform operations before and/or after the wrapped HandlerFunc is executed. MiddlewareFunc functions should be thread-safe, as they will be called concurrently.
func Recovery ¶ added in v0.7.0
func Recovery(handle RecoveryFunc) MiddlewareFunc
Recovery is a middleware that captures panics and recovers from them. It takes a custom handle function that will be called with the Context and the value recovered from the panic. Note that the middleware check if the panic is caused by http.ErrAbortHandler and re-panic if true allowing the http server to handle it as an abort.
func WrapM ¶ added in v0.7.0
WrapM is an adapter for converting http.Handler middleware into a MiddlewareFunc. The boolean parameter, useOriginalWriter, determines how the middleware interacts with the ResponseWriter:
- If useOriginalWriter is false, the middleware is provided with the ResponseWriter from the Fox router. This is suitable for middlewares that may write a response and stop further execution (like an authorization middleware). The Fox's ResponseWriter allows to keep track of the response status and size.
- If useOriginalWriter is true, the middleware is provided with the original http.ResponseWriter from Go's net/http package. This is required for middlewares that need to wrap the ResponseWriter with their own implementation (like a gzip middleware). The Wrap function is used to ensure that the Fox's ResponseWriter wraps the middleware's ResponseWriter implementation.
This API is EXPERIMENTAL and is likely to change in future release.
Example ¶
This example demonstrates the usage of the WrapM function which is used to wrap an http.Handler middleware and returns a MiddlewareFunc function compatible with Fox.
// Case 1: Middleware that may write a response and stop execution. // This is commonly seen in middleware that implements authorization checks. If the check does not pass, // the middleware can stop the execution of the subsequent handlers and write an error response. // The "authorizationMiddleware" in this example checks for a specific token in the request's Authorization header. // If the token is not the expected value, it sends an HTTP 401 Unauthorized error and stops the execution. // In this case, the WrapM function is used with the "useOriginalWriter" set to false, as we want to use // the custom ResponseWriter provided by the Fox framework to capture the status code and response size. authorizationMiddleware := func(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { token := r.Header.Get("Authorization") if token != "valid-token" { http.Error(w, "Invalid token", http.StatusUnauthorized) return } next.ServeHTTP(w, r) }) } _ = New(WithMiddleware(WrapM(authorizationMiddleware, false))) // Case 2: Middleware that wraps the ResponseWriter with its own implementation. // This is typically used in middleware that transforms the response in some way, for instance by applying gzip compression. // The "gzipMiddleware" in this example wraps the original ResponseWriter with a gzip writer, which compresses the response data. // In this case, the WrapM function is used with the "useOriginalWriter" set to true, as the middleware needs to wrap the original // http.ResponseWriter with its own implementation. After wrapping, the Fox framework's ResponseWriter is updated to the new ResponseWriter. gzipMiddleware := func(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { gz := gzip.NewWriter(w) defer gz.Close() // Create a new ResponseWriter that writes to the gzip writer gzw := gzipResponseWriter{Writer: gz, ResponseWriter: w} r.Header.Set("Content-Encoding", "gzip") next.ServeHTTP(gzw, r) }) } _ = New(WithMiddleware(WrapM(gzipMiddleware, true)))
Output:
type MiddlewareScope ¶ added in v0.7.6
type MiddlewareScope uint8
MiddlewareScope is a type that represents different scopes for applying middleware.
const ( // RouteHandlers scope applies middleware only to regular routes registered in the router. RouteHandlers MiddlewareScope = 1 << (8 - 1 - iota) // NoRouteHandler scope applies middleware to the NoRoute handler. NoRouteHandler // NoMethodHandler scope applies middleware to the NoMethod handler. NoMethodHandler // RedirectHandler scope applies middleware to the internal redirect trailing slash handler. RedirectHandler // OptionsHandler scope applies middleware to the internal automatic OPTIONS handler. OptionsHandler // AllHandlers is a combination of all the above scopes, which means the middleware will be applied to all types of handlers. AllHandlers = RouteHandlers | NoRouteHandler | NoMethodHandler | RedirectHandler | OptionsHandler )
type Option ¶ added in v0.5.0
type Option interface {
// contains filtered or unexported methods
}
func DefaultOptions ¶ added in v0.7.0
func DefaultOptions() Option
DefaultOptions configure the router to use the Recovery middleware for the RouteHandlers scope and enable automatic OPTIONS response. Note that DefaultOptions push the Recovery middleware to the first position of the middleware chains.
func WithAutoOptions ¶ added in v0.9.0
WithAutoOptions enables automatic response to OPTIONS requests with, by default, a 200 OK status code. Use the WithOptionsHandler option to customize the response. When this option is enabled, the router automatically determines the "Allow" header value based on the methods registered for the given route. Note that custom OPTIONS handler take priority over automatic replies. This option is automatically enabled when providing a custom handler with the option WithOptionsHandler. This api is EXPERIMENTAL and is likely to change in future release.
func WithMiddleware ¶ added in v0.7.0
func WithMiddleware(m ...MiddlewareFunc) Option
WithMiddleware attaches a global middleware to the router. Middlewares provided will be chained in the order they were added. Note that this option apply middleware to all handler, including NotFound, MethodNotAllowed and the internal redirect handler.
Example ¶
// Define a custom middleware to measure the time taken for request processing and // log the URL, route, time elapsed, and status code metrics := func(next HandlerFunc) HandlerFunc { return func(c Context) { start := time.Now() next(c) log.Printf( "url=%s; route=%s; time=%d; status=%d", c.Request().URL, c.Path(), time.Since(start), c.Writer().Status(), ) } } r := New(WithMiddleware(metrics)) r.MustHandle(http.MethodGet, "/hello/{name}", func(c Context) { _ = c.String(200, "Hello %s\n", c.Param("name")) })
Output:
func WithMiddlewareFor ¶ added in v0.7.6
func WithMiddlewareFor(scope MiddlewareScope, m ...MiddlewareFunc) Option
WithMiddlewareFor attaches middleware to the router for a specified scope. Middlewares provided will be chained in the order they were added. The scope parameter determines which types of handlers the middleware will be applied to. Possible scopes include RouteHandlers (regular routes), NoRouteHandler, NoMethodHandler, RedirectHandler, and any combination of these. Use this option when you need fine-grained control over where the middleware is applied. This api is EXPERIMENTAL and is likely to change in future release.
func WithNoMethod ¶ added in v0.9.0
WithNoMethod enable to returns 405 Method Not Allowed instead of 404 Not Found when the route exist for another http verb. The "Allow" header it automatically set before calling the handler. Note that this option is automatically enabled when providing a custom handler with the option WithNoMethodHandler.
func WithNoMethodHandler ¶ added in v0.9.0
func WithNoMethodHandler(handler HandlerFunc) Option
WithNoMethodHandler register an HandlerFunc which is called when the request cannot be routed, but the same route exist for other methods. The "Allow" header it automatically set before calling the handler. By default, the DefaultMethodNotAllowedHandler is used. Note that this option automatically enable WithNoMethod.
func WithNoRouteHandler ¶ added in v0.9.0
func WithNoRouteHandler(handler HandlerFunc) Option
WithNoRouteHandler register an HandlerFunc which is called when no matching route is found. By default, the DefaultNotFoundHandler is used.
func WithOptionsHandler ¶ added in v0.9.0
func WithOptionsHandler(handler HandlerFunc) Option
WithOptionsHandler register an HandlerFunc which is called on automatic OPTIONS requests. By default, the router respond with a 200 OK status code. The "Allow" header it automatically set before calling the handler. Note that custom OPTIONS handler take priority over automatic replies. This option automatically enable WithAutoOptions. This api is EXPERIMENTAL and is likely to change in future release.
func WithRedirectTrailingSlash ¶ added in v0.5.0
WithRedirectTrailingSlash enable automatic redirection fallback when the current request does not match but another handler is found with/without an additional trailing slash. E.g. /foo/bar/ request does not match but /foo/bar would match. The client is redirected with a http status code 301 for GET requests and 308 for all other methods.
type RecoveryFunc ¶ added in v0.7.0
RecoveryFunc is a function type that defines how to handle panics that occur during the handling of an HTTP request.
type ResponseWriter ¶ added in v0.7.0
type ResponseWriter interface { http.ResponseWriter // Status recorded after Write and WriteHeader. Status() int // Written returns true if the response has been written. Written() bool // Size returns the size of the written response. Size() int // Unwrap returns the underlying http.ResponseWriter. Unwrap() http.ResponseWriter // Wrap replaces the underlying http.ResponseWriter for the current ResponseWriter. It does not reset the status, // written state, or response size of the current ResponseWriter. Caution: You should pass the original // http.ResponseWriter to this method, not the ResponseWriter itself, to avoid wrapping the ResponseWriter within itself. Wrap(w http.ResponseWriter) // WriteString writes the provided string to the underlying connection // as part of an HTTP reply. The method returns the number of bytes written // and an error, if any. WriteString(s string) (int, error) }
ResponseWriter extends http.ResponseWriter and provides methods to retrieve the recorded status code, written state, and response size. ResponseWriter object implements additional http.Flusher, http.Hijacker, io.ReaderFrom interfaces for HTTP/1.x requests and http.Flusher, http.Pusher interfaces for HTTP/2 requests.
type RouteConflictError ¶
type RouteConflictError struct { Method string Path string Matched []string // contains filtered or unexported fields }
RouteConflictError is a custom error type used to represent conflicts when registering or updating routes in the router. It holds information about the conflicting method, path, and the matched routes that caused the conflict.
func (*RouteConflictError) Error ¶
func (e *RouteConflictError) Error() string
Error returns a formatted error message for the RouteConflictError.
func (*RouteConflictError) Unwrap ¶
func (e *RouteConflictError) Unwrap() error
Unwrap returns the sentinel value ErrRouteConflict.
type Router ¶
type Router struct {
// contains filtered or unexported fields
}
Router is a lightweight high performance HTTP request router that support mutation on its routing tree while handling request concurrently.
func New ¶
New returns a ready to use instance of Fox router.
Example ¶
This example demonstrates how to create a simple router using the default options, which include the Recovery middleware. A basic route is defined, along with a custom middleware to log the request metrics.
// Create a new router with default options, which include the Recovery middleware r := New(DefaultOptions()) // Define a custom middleware to measure the time taken for request processing and // log the URL, route, time elapsed, and status code metrics := func(next HandlerFunc) HandlerFunc { return func(c Context) { start := time.Now() next(c) log.Printf("url=%s; route=%s; time=%d; status=%d", c.Request().URL, c.Path(), time.Since(start), c.Writer().Status()) } } // Define a route with the path "/hello/{name}", apply the custom "metrics" middleware, // and set a simple handler that greets the user by their name r.MustHandle(http.MethodGet, "/hello/{name}", metrics(func(c Context) { _ = c.String(200, "Hello %s\n", c.Param("name")) })) // Start the HTTP server using the router as the handler and listen on port 8080 log.Fatalln(http.ListenAndServe(":8080", r))
Output:
func (*Router) Handle ¶ added in v0.7.0
func (fox *Router) Handle(method, path string, handler HandlerFunc) error
Handle registers a new handler for the given method and path. This function return an error if the route is already registered or conflict with another. It's perfectly safe to add a new handler while the tree is in use for serving requests. This function is safe for concurrent use by multiple goroutine. To override an existing route, use Update.
func (*Router) MustHandle ¶ added in v0.7.0
func (fox *Router) MustHandle(method, path string, handler HandlerFunc)
MustHandle registers a new handler for the given method and path. This function is a convenience wrapper for the Handle function. It will panic if the route is already registered or conflicts with another route. It's perfectly safe to add a new handler while the tree is in use for serving requests. This function is safe for concurrent use by multiple goroutines. To override an existing route, use Update.
func (*Router) NewTree ¶ added in v0.5.0
NewTree returns a fresh routing Tree. It's safe to create multiple Tree concurrently. However, a Tree itself is not thread-safe and all its APIs that perform write operations should be run serially. Note that a Tree give direct access to the underlying sync.Mutex. This API is EXPERIMENTAL and is likely to change in future release.
func (*Router) Remove ¶
Remove delete an existing handler for the given method and path. If the route does not exist, the function return an ErrRouteNotFound. It's perfectly safe to remove a handler while the tree is in use for serving requests. This function is safe for concurrent use by multiple goroutine.
func (*Router) Swap ¶ added in v0.5.0
Swap atomically replaces the currently in-use routing tree with the provided new tree, and returns the previous tree. This API is EXPERIMENTAL and is likely to change in future release.
func (*Router) Tree ¶ added in v0.5.0
Tree atomically loads and return the currently in-use routing tree. This API is EXPERIMENTAL and is likely to change in future release.
Example ¶
This example demonstrates some important considerations when using the Tree API.
r := New() // Each tree as its own sync.Mutex that is used to lock write on the tree. Since the router tree may be swapped at // any given time, you MUST always copy the pointer locally, This ensures that you do not inadvertently cause a // deadlock by locking/unlocking the wrong tree. tree := r.Tree() upsert := func(method, path string, handler HandlerFunc) error { tree.Lock() defer tree.Unlock() if tree.Has(method, path) { return tree.Update(method, path, handler) } return tree.Handle(method, path, handler) } _ = upsert(http.MethodGet, "/foo/bar", func(c Context) { // Note the tree accessible from fox.Context is already a local copy so the golden rule above does not apply. c.Tree().Lock() defer c.Tree().Unlock() _ = c.String(200, "foo bar") }) // Bad, instead make a local copy of the tree! _ = func(method, path string, handler HandlerFunc) error { r.Tree().Lock() defer r.Tree().Unlock() if r.Tree().Has(method, path) { return r.Tree().Update(method, path, handler) } return r.Tree().Handle(method, path, handler) }
Output:
func (*Router) Update ¶
func (fox *Router) Update(method, path string, handler HandlerFunc) error
Update override an existing handler for the given method and path. If the route does not exist, the function return an ErrRouteNotFound. It's perfectly safe to update a handler while the tree is in use for serving requests. This function is safe for concurrent use by multiple goroutine. To add new handler, use Handle method.
type Tree ¶ added in v0.5.0
Tree implements a Concurrent Radix Tree that supports lock-free reads while allowing concurrent writes. The caller is responsible for ensuring that all writes are run serially.
IMPORTANT: Each tree as its own sync.Mutex that may be used to serialize write. Since the router tree may be swapped at any given time, you MUST always copy the pointer locally to avoid inadvertently causing a deadlock by locking/unlocking the wrong Tree.
Good: t := r.Tree() t.Lock() defer t.Unlock()
Dramatically bad, may cause deadlock r.Tree().Lock() defer r.Tree().Unlock()
func (*Tree) Handle ¶ added in v0.7.0
func (t *Tree) Handle(method, path string, handler HandlerFunc) error
Handle registers a new handler for the given method and path. This function return an error if the route is already registered or conflict with another. It's perfectly safe to add a new handler while the tree is in use for serving requests. However, this function is NOT thread-safe and should be run serially, along with all other Tree APIs that perform write operations. To override an existing route, use Update.
func (*Tree) Has ¶ added in v0.8.0
Has allows to check if the given method and path exactly match a registered route. This function is safe for concurrent use by multiple goroutine and while mutation on Tree are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
func (*Tree) Lookup ¶ added in v0.8.0
func (t *Tree) Lookup(method, path string, lazy bool) (handler HandlerFunc, cc ContextCloser, tsr bool)
Lookup allow to do manual lookup of a route for the given method and path and return the matched HandlerFunc along with a ContextCloser and trailing slash redirect recommendation. If lazy is set to true, wildcard parameter are not parsed. You should always close the ContextCloser if NOT nil by calling cc.Close(). Note that the returned ContextCloser does not have a router, request and response writer attached (use the Reset method). This function is safe for concurrent use by multiple goroutine and while mutation on Tree are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
Example ¶
This example demonstrates how to create a custom middleware that cleans the request path and performs a manual lookup on the tree. If the cleaned path matches a registered route, the client is redirected with a 301 status code (Moved Permanently).
redirectFixedPath := MiddlewareFunc(func(next HandlerFunc) HandlerFunc { return func(c Context) { req := c.Request() cleanedPath := CleanPath(req.URL.Path) handler, cc, _ := c.Tree().Lookup(req.Method, cleanedPath, true) // You should always close a non-nil Context. if cc != nil { defer cc.Close() } // 301 redirect and returns. if handler != nil { req.URL.Path = cleanedPath http.Redirect(c.Writer(), req, req.URL.String(), http.StatusMovedPermanently) return } next(c) } }) f := New( // Register the middleware for the NoRouteHandler scope. WithMiddlewareFor(NoRouteHandler, redirectFixedPath), ) f.MustHandle(http.MethodGet, "/foo/bar", func(c Context) { _ = c.String(http.StatusOK, "foo bar") })
Output:
func (*Tree) LookupMethods ¶ added in v0.9.0
LookupMethods lookup and returns all HTTP methods associated with a route that match the given path. This function is safe for concurrent use by multiple goroutine and while mutation on Tree are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
func (*Tree) Methods ¶ added in v0.7.4
Methods returns a sorted slice of HTTP methods that are currently in use to route requests. This function is safe for concurrent use by multiple goroutine and while mutation on Tree are ongoing. This API is EXPERIMENTAL and is likely to change in future release.
func (*Tree) Remove ¶ added in v0.5.0
Remove delete an existing handler for the given method and path. If the route does not exist, the function return an ErrRouteNotFound. It's perfectly safe to remove a handler while the tree is in use for serving requests. However, this function is NOT thread-safe and should be run serially, along with all other Tree APIs that perform write operations.
func (*Tree) Update ¶ added in v0.5.0
func (t *Tree) Update(method, path string, handler HandlerFunc) error
Update override an existing handler for the given method and path. If the route does not exist, the function return an ErrRouteNotFound. It's perfectly safe to update a handler while the tree is in use for serving requests. However, this function is NOT thread-safe and should be run serially, along with all other Tree APIs that perform write operations. To add new handler, use Handle method.
type WalkFunc ¶
type WalkFunc func(method, path string, handler HandlerFunc) error
WalkFunc is the type of the function called by Walk to visit each registered routes.