|
8 | 8 | "errors"
|
9 | 9 | "fmt"
|
10 | 10 | "html"
|
| 11 | + "slices" |
11 | 12 | "sort"
|
12 | 13 | "strings"
|
13 | 14 | "sync/atomic"
|
@@ -302,6 +303,13 @@ func (app *App) register(methods []string, pathRaw string, group *Group, handler
|
302 | 303 | if method != methodUse && app.methodInt(method) == -1 {
|
303 | 304 | panic(fmt.Sprintf("add: invalid http method %s\n", method))
|
304 | 305 | }
|
| 306 | + |
| 307 | + // Duplicate Route Handling |
| 308 | + if app.routeExists(method, pathRaw) { |
| 309 | + matchPathFunc := func(r *Route) bool { return r.Path == pathRaw } |
| 310 | + app.deleteRoute([]string{method}, matchPathFunc) |
| 311 | + } |
| 312 | + |
305 | 313 | // is mounted app
|
306 | 314 | isMount := group != nil && group.app != app
|
307 | 315 | // A route requires atleast one ctx handler
|
@@ -375,6 +383,72 @@ func (app *App) register(methods []string, pathRaw string, group *Group, handler
|
375 | 383 | }
|
376 | 384 | }
|
377 | 385 |
|
| 386 | +func (app *App) routeExists(method, pathRaw string) bool { |
| 387 | + pathToCheck := pathRaw |
| 388 | + if !app.config.CaseSensitive { |
| 389 | + pathToCheck = utils.ToLower(pathToCheck) |
| 390 | + } |
| 391 | + |
| 392 | + return slices.ContainsFunc(app.stack[app.methodInt(method)], func(r *Route) bool { |
| 393 | + routePath := r.path |
| 394 | + if !app.config.CaseSensitive { |
| 395 | + routePath = utils.ToLower(routePath) |
| 396 | + } |
| 397 | + |
| 398 | + return routePath == pathToCheck |
| 399 | + }) |
| 400 | +} |
| 401 | + |
| 402 | +// RemoveRoute is used to remove a route from the stack by path. |
| 403 | +// This only needs to be called to remove a route, route registration prevents duplicate routes. |
| 404 | +// You should call RebuildTree after using this to ensure consistency of the tree. |
| 405 | +func (app *App) RemoveRoute(path string, methods ...string) { |
| 406 | + pathMatchFunc := func(r *Route) bool { return r.Path == path } |
| 407 | + app.deleteRoute(methods, pathMatchFunc) |
| 408 | +} |
| 409 | + |
| 410 | +// RemoveRouteByName is used to remove a route from the stack by name. |
| 411 | +// This only needs to be called to remove a route, route registration prevents duplicate routes. |
| 412 | +// You should call RebuildTree after using this to ensure consistency of the tree. |
| 413 | +func (app *App) RemoveRouteByName(name string, methods ...string) { |
| 414 | + matchFunc := func(r *Route) bool { return r.Name == name } |
| 415 | + app.deleteRoute(methods, matchFunc) |
| 416 | +} |
| 417 | + |
| 418 | +func (app *App) deleteRoute(methods []string, matchFunc func(r *Route) bool) { |
| 419 | + app.mutex.Lock() |
| 420 | + defer app.mutex.Unlock() |
| 421 | + |
| 422 | + for _, method := range methods { |
| 423 | + // Uppercase HTTP methods |
| 424 | + method = utils.ToUpper(method) |
| 425 | + |
| 426 | + // Get unique HTTP method identifier |
| 427 | + m := app.methodInt(method) |
| 428 | + if m == -1 { |
| 429 | + continue // Skip invalid HTTP methods |
| 430 | + } |
| 431 | + |
| 432 | + // Find the index of the route to remove |
| 433 | + index := slices.IndexFunc(app.stack[m], matchFunc) |
| 434 | + if index == -1 { |
| 435 | + continue // Route not found |
| 436 | + } |
| 437 | + |
| 438 | + route := app.stack[m][index] |
| 439 | + |
| 440 | + // Decrement global handler count |
| 441 | + atomic.AddUint32(&app.handlersCount, ^uint32(len(route.Handlers)-1)) //nolint:gosec // Not a concern |
| 442 | + // Decrement global route position |
| 443 | + atomic.AddUint32(&app.routesCount, ^uint32(0)) |
| 444 | + |
| 445 | + // Remove route from tree stack |
| 446 | + app.stack[m] = slices.Delete(app.stack[m], index, index+1) |
| 447 | + } |
| 448 | + |
| 449 | + app.routesRefreshed = true |
| 450 | +} |
| 451 | + |
378 | 452 | func (app *App) addRoute(method string, route *Route, isMounted ...bool) {
|
379 | 453 | app.mutex.Lock()
|
380 | 454 | defer app.mutex.Unlock()
|
@@ -415,7 +489,7 @@ func (app *App) addRoute(method string, route *Route, isMounted ...bool) {
|
415 | 489 | // This method is useful when you want to register routes dynamically after the app has started.
|
416 | 490 | // It is not recommended to use this method on production environments because rebuilding
|
417 | 491 | // the tree is performance-intensive and not thread-safe in runtime. Since building the tree
|
418 |
| -// is only done in the startupProcess of the app, this method does not makes sure that the |
| 492 | +// is only done in the startupProcess of the app, this method does not make sure that the |
419 | 493 | // routeTree is being safely changed, as it would add a great deal of overhead in the request.
|
420 | 494 | // Latest benchmark results showed a degradation from 82.79 ns/op to 94.48 ns/op and can be found in:
|
421 | 495 | // https://github.com/gofiber/fiber/issues/2769#issuecomment-2227385283
|
|
0 commit comments