-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathroute.go
162 lines (146 loc) · 5.64 KB
/
route.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
package rmhttp
import (
"fmt"
"net/http"
"slices"
"strings"
"time"
)
// ------------------------------------------------------------------------------------------------
// ROUTE
// ------------------------------------------------------------------------------------------------
// A Route encapsulates all of the information that the router will need to satisfy an HTTP
// request. Alongside supplying standard information such as what HTTP method and URL
// pattern a handler should be bound to, the Route also allows the enclosed handler
// to be configured with their own timeout, headers, and middleware.
type Route struct {
Method string
Pattern string
Handler Handler
Middleware []MiddlewareFunc
Timeout Timeout
Headers map[string]string
Parent *Group
}
// NewRoute validates the input, then creates, initialises and returns a pointer to a Route. The
// validation step ensures that a valid HTTP method has been passed (http.MethodGet will be
// used, if not). The method will also be transformed to uppercase, and the pattern to
// lowercase.
func NewRoute(method string, pattern string, handler Handler) *Route {
m := strings.ToUpper(method)
if !slices.Contains(ValidHTTPMethods(), m) {
method = http.MethodGet
}
return &Route{
Method: method,
Pattern: strings.ToLower(pattern),
Handler: handler,
Headers: make(map[string]string),
}
}
// ComputedPattern dynamically calculates the pattern for the Route. It returns the URL pattern as a
// string.
func (route *Route) ComputedPattern() string {
return route.buildPattern(route.Pattern, route.Parent)
}
// buildPattern builds a URL pattern by conatenating any parent Group patterns together with the
// Route pattern.
func (route *Route) buildPattern(pattern string, parent *Group) string {
if parent == nil {
return pattern
}
pattern = fmt.Sprintf("%s%s", parent.Pattern, pattern)
return route.buildPattern(pattern, parent.Parent)
}
// ComputedHeaders dynamically calculates the HTTP headers that have been added to the Route and
// any parent Groups.
func (route *Route) ComputedHeaders() map[string]string {
return route.findHeaders(route.Headers, route.Parent)
}
// findHeaders collects all of the headers set on the Route, plus any parent groups.
func (route *Route) findHeaders(headers map[string]string, parent *Group) map[string]string {
if parent == nil {
return headers
}
// Only add a parent header if it hasn't already been set in the child.
for key, value := range parent.Headers {
if _, ok := headers[key]; !ok {
headers[key] = value
}
}
return route.findHeaders(headers, parent.Parent)
}
// Timeout returns the Timeout object that has been added to the Route.
func (route *Route) ComputedTimeout() Timeout {
if !route.Timeout.Enabled {
return route.findEnabledTimeout(route.Parent)
}
return route.Timeout
}
// findEnabledTimeout searches for an enabled Timeout in any parent Group.
func (route *Route) findEnabledTimeout(parent *Group) Timeout {
if parent == nil {
return Timeout{}
}
if parent.Timeout.Enabled {
return parent.Timeout
}
return route.findEnabledTimeout(parent.Parent)
}
// Middleware returns the slice of MiddlewareFuncs that have been added to the Route.
func (route *Route) ComputedMiddleware() []MiddlewareFunc {
m := route.Middleware
return m
}
// WithMiddleware adds Middleware handlers to the receiver Route.
//
// Each middleware handler will be wrapped to create a call stack with the order in which the
// middleware is added being maintained. So, for example, if the user added A and B
// middleware via this method, the resulting callstack would be as follows -
//
// Middleware A -> Middleware B -> Route Handler -> Middleware B -> Middleware A
//
// (This actually a slight simplification, as internal middleware such as HTTP Logging, CORS, HTTP
// Error Handling and Route Panic Recovery may also be inserted into the call stack, depending
// on how the App is configured).
//
// The middlewares argument is variadic, allowing the user to add multiple middleware functions
// in a single call.
//
// This method will return a pointer to the receiver Route, allowing the user to chain any of the
// other builder methods that Route implements.
func (route *Route) WithMiddleware(middlewares ...func(Handler) Handler) *Route {
for _, mw := range middlewares {
route.Middleware = append(route.Middleware, MiddlewareFunc(mw))
}
return route
}
// Use is a convenience method for adding middleware handlers to a Route. It uses WithMiddleware
// behind the scenes.
//
// This method will return a pointer to the receiver Route, allowing the user to chain any of the
// other builder methods that Route implements.
func (route *Route) Use(middlewares ...func(Handler) Handler) *Route {
return route.WithMiddleware(middlewares...)
}
// WithTimeout sets a request timeout amount and message for this route.
//
// This method will return a pointer to the receiver Route, allowing the user to chain any of the
// other builder methods that Route implements.
func (route *Route) WithTimeout(timeout time.Duration, message string) *Route {
route.Timeout = NewTimeout(timeout, message)
return route
}
// WithHeader sets an HTTP header for this route. Calling this method more than once will either
// overwrite an existing header, or add a new one.
//
// This method will return a pointer to the receiver Route, allowing the user to chain any of the
// other builder methods that Route implements.
func (route *Route) WithHeader(key, value string) *Route {
route.Headers[key] = value
return route
}
// String is used internally to calculate a string signature for use as map keys, etc.
func (route *Route) String() string {
return fmt.Sprint(route.Method, " ", route.Pattern)
}