2018-10-21 18:20:05 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"strings"
|
|
|
|
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12"
|
|
|
|
"github.com/kataras/iris/v12/core/router"
|
2018-10-21 18:20:05 +02:00
|
|
|
)
|
|
|
|
|
2022-06-17 21:03:18 +02:00
|
|
|
/*
|
|
|
|
A Router should contain all three of the following methods:
|
|
|
|
- HandleRequest should handle the request based on the Context.
|
|
|
|
HandleRequest(ctx iris.Context)
|
|
|
|
- Build should builds the handler, it's being called on router's BuildRouter.
|
|
|
|
Build(provider router.RoutesProvider) error
|
|
|
|
- RouteExists reports whether a particular route exists.
|
|
|
|
RouteExists(ctx iris.Context, method, path string) bool
|
|
|
|
- FireErrorCode(ctx iris.Context) should handle the given ctx.GetStatusCode().
|
2018-10-21 18:20:05 +02:00
|
|
|
|
|
|
|
For a more detailed, complete and useful example
|
|
|
|
you can take a look at the iris' router itself which is located at:
|
2023-08-20 02:12:46 +02:00
|
|
|
https://github.com/kataras/iris/tree/main/core/router/handler.go
|
2018-10-21 18:20:05 +02:00
|
|
|
which completes this exact interface, the `router#RequestHandler`.
|
|
|
|
*/
|
|
|
|
type customRouter struct {
|
|
|
|
// a copy of routes (safer because you will not be able to alter a route on serve-time without a `app.RefreshRouter` call):
|
|
|
|
// []router.Route
|
|
|
|
// or just expect the whole routes provider:
|
|
|
|
provider router.RoutesProvider
|
|
|
|
}
|
|
|
|
|
|
|
|
// HandleRequest a silly example which finds routes based only on the first part of the requested path
|
|
|
|
// which must be a static one as well, the rest goes to fill the parameters.
|
2019-08-11 14:43:47 +02:00
|
|
|
func (r *customRouter) HandleRequest(ctx iris.Context) {
|
2018-10-21 18:20:05 +02:00
|
|
|
path := ctx.Path()
|
|
|
|
ctx.Application().Logger().Infof("Requested resource path: %s", path)
|
|
|
|
|
|
|
|
parts := strings.Split(path, "/")[1:]
|
|
|
|
staticPath := "/" + parts[0]
|
|
|
|
for _, route := range r.provider.GetRoutes() {
|
|
|
|
if strings.HasPrefix(route.Path, staticPath) && route.Method == ctx.Method() {
|
|
|
|
paramParts := parts[1:]
|
|
|
|
for _, paramValue := range paramParts {
|
|
|
|
for _, p := range route.Tmpl().Params {
|
|
|
|
ctx.Params().Set(p.Name, paramValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-10 23:44:54 +02:00
|
|
|
ctx.SetCurrentRoute(route.ReadOnly)
|
2018-10-21 18:20:05 +02:00
|
|
|
ctx.Do(route.Handlers)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if nothing found...
|
|
|
|
ctx.StatusCode(iris.StatusNotFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *customRouter) Build(provider router.RoutesProvider) error {
|
|
|
|
for _, route := range provider.GetRoutes() {
|
|
|
|
// do any necessary validation or conversations based on your custom logic here
|
|
|
|
// but always run the "BuildHandlers" for each registered route.
|
|
|
|
route.BuildHandlers()
|
|
|
|
// [...] r.routes = append(r.routes, *route)
|
|
|
|
}
|
|
|
|
|
|
|
|
r.provider = provider
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-08-11 14:43:47 +02:00
|
|
|
func (r *customRouter) RouteExists(ctx iris.Context, method, path string) bool {
|
2018-10-21 18:20:05 +02:00
|
|
|
// [...]
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-06-08 04:19:12 +02:00
|
|
|
func (r *customRouter) FireErrorCode(ctx iris.Context) {
|
2020-05-10 23:44:54 +02:00
|
|
|
// responseStatusCode := ctx.GetStatusCode() // set by prior ctx.StatusCode calls
|
|
|
|
// [...]
|
|
|
|
}
|
|
|
|
|
2018-10-21 18:20:05 +02:00
|
|
|
func main() {
|
|
|
|
app := iris.New()
|
|
|
|
|
|
|
|
// In case you are wondering, the parameter types and macros like "{param:string $func()}" still work inside
|
|
|
|
// your custom router if you fetch by the Route's Handler
|
|
|
|
// because they are middlewares under the hood, so you don't have to implement the logic of handling them manually,
|
|
|
|
// though you have to match what requested path is what route and fill the ctx.Params(), this is the work of your custom router.
|
2019-08-11 14:43:47 +02:00
|
|
|
app.Get("/hello/{name}", func(ctx iris.Context) {
|
2018-10-21 18:20:05 +02:00
|
|
|
name := ctx.Params().Get("name")
|
|
|
|
ctx.Writef("Hello %s\n", name)
|
|
|
|
})
|
|
|
|
|
2019-08-11 14:43:47 +02:00
|
|
|
app.Get("/cs/{num:uint64 min(10) else 400}", func(ctx iris.Context) {
|
2018-10-21 18:20:05 +02:00
|
|
|
num := ctx.Params().GetUint64Default("num", 0)
|
|
|
|
ctx.Writef("num is: %d\n", num)
|
|
|
|
})
|
|
|
|
|
|
|
|
// To replace the existing router with a customized one by using the iris/context.Context
|
|
|
|
// you have to use the `app.BuildRouter` method before `app.Run` and after the routes registered.
|
|
|
|
// You should pass your custom router's instance as the second input arg, which must completes the `router#RequestHandler`
|
|
|
|
// interface as shown above.
|
|
|
|
//
|
|
|
|
// To see how you can build something even more low-level without direct iris' context support (you can do that manually as well)
|
|
|
|
// navigate to the "custom-wrapper" example instead.
|
|
|
|
myCustomRouter := new(customRouter)
|
|
|
|
app.BuildRouter(app.ContextPool, myCustomRouter, app.APIBuilder, true)
|
|
|
|
|
2020-04-28 04:22:58 +02:00
|
|
|
app.Listen(":8080")
|
2018-10-21 18:20:05 +02:00
|
|
|
}
|