-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathgroup.go
283 lines (262 loc) · 8.53 KB
/
group.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
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
// Package group provides functionality for managing grouped routes within the Quick framework.
//
// This package allows for the creation and organization of route groups, enabling
// the reuse of middleware and the application of common prefixes to related routes.
// It simplifies route management by structuring them into logical collections.
//
// Features:
// - Grouping of related routes under a shared prefix.
// - Support for middleware application at the group level.
// - Simplified registration of HTTP methods (GET, POST, PUT, DELETE, etc.).
// - Automatic handling of parameter extraction in routes.
package quick
import (
"net/http"
"strings"
"github.com/jeffotoni/quick/internal/concat"
)
// Constants for route processing
const (
methodSeparator = "#"
errInvalidExtractor = "Invalid function signature for paramExtractor"
)
// Group represents a collection of routes that share a common prefix.
//
// Fields:
// - prefix: The URL prefix shared by all routes in the group.
// - routes: A list of registered routes within the group.
// - middlewares: A list of middleware functions applied to the group.
// - quick: A reference to the Quick router.
type Group struct {
prefix string
routes []Route
middlewares []func(http.Handler) http.Handler
quick *Quick
}
// Use adds middleware to the group.
//
// This function allows adding middleware to a specific group of routes.
// Middleware functions are executed **before** the route handlers,
// allowing for request modifications, logging, authentication, etc.
//
// Parameters:
// - mw: A middleware function that modifies the HTTP handler.
//
// Example Usage:
//
// q := quick.New()
//
// group := q.Group("/v1")
//
// group.Get("/user", func(c *quick.Ctx) error {
// return c.Status(200).SendString("[GET] [GROUP] /v1/user ok!!!")
// })
func (g *Group) Use(mw func(http.Handler) http.Handler) {
g.middlewares = append(g.middlewares, mw)
}
// Group creates a new route group with a shared prefix.
//
// This function allows organizing routes under a common prefix, making it easier
// to manage related endpoints (e.g., `/api`, `/v1`, `/admin`).
// All routes registered within this group will automatically inherit the specified prefix.
//
// Grouping routes is useful for:
// - API versioning (`/v1`, `/v2`).
// - Organizing authentication-protected routes (`/auth`, `/admin`).
// - Applying shared middlewares to a set of routes.
//
// Parameters:
// - prefix: The common prefix for all routes in this group.
//
// Returns:
// - *Group: A new Group instance that can be used to define related routes.
//
// Example Usage:
//
// group := q.Group("/api")
//
// group.Get("/user", func(c *quick.Ctx) error {
// return c.Status(200).SendString("[GET] [GROUP] /v1/user ok!!!")
// })
func (q *Quick) Group(prefix string) *Group {
g := &Group{
prefix: prefix,
routes: []Route{},
quick: q,
}
q.groups = append(q.groups, *g)
return g
}
// normalizePattern constructs the full path with the group prefix.
//
// Parameters:
// - prefix: The group's base URL path.
// - pattern: The specific route pattern.
//
// Returns:
// - string: The normalized URL path.
//
// Example:
//
// fullPath := normalizePattern("/api", "/users") // "/api/users"
func normalizePattern(prefix, pattern string) string {
return concat.String(strings.TrimRight(prefix, "/"), "/", strings.TrimLeft(pattern, "/"))
}
// resolveParamExtractor ensures the correct function signature for paramExtractor.
//
// Parameters:
// - q: The Quick router instance.
// - handlerFunc: The function handling the route.
// - paramExtractor: A function for extracting parameters.
// - path: The normalized route path.
// - params: URL parameters.
//
// Returns:
// - http.HandlerFunc: A wrapped handler with parameter extraction.
//
// Example:
//
// handler := resolveParamExtractor(q, handlerFunc, extractParams, "/users/:id", "id")
func resolveParamExtractor(q *Quick, handlerFunc HandleFunc, paramExtractor interface{}, path, params string) http.HandlerFunc {
switch fn := paramExtractor.(type) {
case func(*Quick, HandleFunc) http.HandlerFunc:
return fn(q, handlerFunc)
case func(*Quick, string, string, HandleFunc) http.HandlerFunc:
return fn(q, path, params, handlerFunc)
default:
panic(errInvalidExtractor)
}
}
// applyMiddlewares applies all middlewares to a handler.
//
// Parameters:
// - handler: The original HTTP handler function.
// - middlewares: A list of middleware functions.
//
// Returns:
// - http.HandlerFunc: The wrapped handler with applied middlewares.
func applyMiddlewares(handler http.HandlerFunc, middlewares []func(http.Handler) http.Handler) http.HandlerFunc {
for _, mw := range middlewares {
handler = mw(handler).(http.HandlerFunc) // CORREÇÃO: Garante conversão correta
}
return handler
}
// createAndRegisterRoute creates a new route and registers it in the Quick router.
//
// Parameters:
// - g: The group to which the route belongs.
// - method: The HTTP method (GET, POST, etc.).
// - pattern: The route pattern.
// - compiledPattern: The compiled route pattern with parameters.
// - params: URL parameters for the route.
// - handler: The HTTP handler function.
func createAndRegisterRoute(g *Group, method, pattern, compiledPattern, params string, handler http.HandlerFunc) {
route := Route{
Pattern: compiledPattern,
Path: pattern,
Params: params,
handler: handler,
Method: method,
Group: g.prefix,
}
g.quick.appendRoute(&route)
// FIX: Adjust path in mux to maintain compatibility with tests
if method == http.MethodGet {
g.quick.mux.HandleFunc(pattern, handler)
} else {
g.quick.mux.HandleFunc(concat.String(strings.ToLower(method), methodSeparator, pattern), handler)
}
}
// Handle registers a new route dynamically.
//
// Parameters:
// - method: The HTTP method (GET, POST, etc.).
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
// - paramExtractor: The function to extract parameters.
//
// Example:
//
// g.Handle("GET", "/users/:id", userHandler, extractParamsGet)
func (g *Group) Handle(method, pattern string, handlerFunc HandleFunc, paramExtractor any) {
// Normalize pattern and extract parameters
pattern = normalizePattern(g.prefix, pattern)
path, params, compiledPattern := extractParamsPattern(pattern)
// Resolve parameter extractor and apply middlewares
handler := resolveParamExtractor(g.quick, handlerFunc, paramExtractor, path, params)
handler = applyMiddlewares(handler, g.middlewares)
// Register route
createAndRegisterRoute(g, method, pattern, compiledPattern, params, handler)
}
// Get registers a new GET route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Get("/users", listUsersHandler)
func (g *Group) Get(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodGet, pattern, handlerFunc, extractParamsGet)
}
// Post registers a new POST route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Post("/users", createUserHandler)
func (g *Group) Post(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodPost, pattern, handlerFunc, extractParamsPost)
}
// Put registers a new PUT route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Put("/users/:id", updateUserHandler)
func (g *Group) Put(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodPut, pattern, handlerFunc, extractParamsPut)
}
// Delete registers a new DELETE route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Delete("/users/:id", deleteUserHandler)
func (g *Group) Delete(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodDelete, pattern, handlerFunc, extractParamsDelete)
}
// Patch registers a new PATCH route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Patch("/users/:id", partialUpdateHandler)
func (g *Group) Patch(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodPatch, pattern, handlerFunc, extractParamsPatch)
}
// Options registers a new OPTIONS route.
//
// Parameters:
// - pattern: The route pattern.
// - handlerFunc: The function handling the request.
//
// Example:
//
// g.Options("/users", optionsHandler)
func (g *Group) Options(pattern string, handlerFunc HandleFunc) {
g.Handle(http.MethodOptions, pattern, handlerFunc, extractParamsOptions)
}