package router import ( "fmt" "strings" "github.com/kataras/iris/context" "github.com/kataras/iris/macro" "github.com/kataras/iris/macro/handler" ) // Route contains the information about a registered Route. // If any of the following fields are changed then the // caller should Refresh the router. type Route struct { Name string `json:"name"` // "userRoute" Method string `json:"method"` // "GET" methodBckp string // if Method changed to something else (which is possible at runtime as well, via RefreshRouter) then this field will be filled with the old one. Subdomain string `json:"subdomain"` // "admin." tmpl macro.Template // Tmpl().Src: "/api/user/{id:uint64}" // temp storage, they're appended to the Handlers on build. // Execution happens before Handlers, can be empty. beginHandlers context.Handlers // Handlers are the main route's handlers, executed by order. // Cannot be empty. Handlers context.Handlers `json:"-"` MainHandlerName string `json:"mainHandlerName"` // temp storage, they're appended to the Handlers on build. // Execution happens after Begin and main Handler(s), can be empty. doneHandlers context.Handlers Path string `json:"path"` // the underline router's representation, i.e "/api/user/:id" // FormattedPath all dynamic named parameters (if any) replaced with %v, // used by Application to validate param values of a Route based on its name. FormattedPath string `json:"formattedPath"` // StaticSites if not empty, refers to the system (or virtual if embedded) directory // and sub directories that this "GET" route was registered to serve files and folders // that contain index.html (a site). The index handler may registered by other // route, manually or automatic by the framework, // get the route by `Application#GetRouteByPath(staticSite.RequestPath)`. StaticSites []context.StaticSite `json:"staticSites"` topLink *Route } // NewRoute returns a new route based on its method, // subdomain, the path (unparsed or original), // handlers and the macro container which all routes should share. // It parses the path based on the "macros", // handlers are being changed to validate the macros at serve time, if needed. func NewRoute(method, subdomain, unparsedPath, mainHandlerName string, handlers context.Handlers, macros macro.Macros) (*Route, error) { tmpl, err := macro.Parse(unparsedPath, macros) if err != nil { return nil, err } path := convertMacroTmplToNodePath(tmpl) // prepend the macro handler to the route, now, // right before the register to the tree, so APIBuilder#UseGlobal will work as expected. if handler.CanMakeHandler(tmpl) { macroEvaluatorHandler := handler.MakeHandler(tmpl) handlers = append(context.Handlers{macroEvaluatorHandler}, handlers...) } path = cleanPath(path) // maybe unnecessary here. defaultName := method + subdomain + tmpl.Src formattedPath := formatPath(path) route := &Route{ Name: defaultName, Method: method, methodBckp: method, Subdomain: subdomain, tmpl: tmpl, Path: path, Handlers: handlers, MainHandlerName: mainHandlerName, FormattedPath: formattedPath, } return route, nil } // Use adds explicit begin handlers to this route. // Alternatively the end-dev can prepend to the `Handlers` field. // Should be used before the `BuildHandlers` which is // called by the framework itself on `Application#Run` (build state). // // Used internally at `APIBuilder#UseGlobal` -> `beginGlobalHandlers` -> `APIBuilder#Handle`. func (r *Route) Use(handlers ...context.Handler) { if len(handlers) == 0 { return } r.beginHandlers = append(r.beginHandlers, handlers...) } // Done adds explicit finish handlers to this route. // Alternatively the end-dev can append to the `Handlers` field. // Should be used before the `BuildHandlers` which is // called by the framework itself on `Application#Run` (build state). // // Used internally at `APIBuilder#DoneGlobal` -> `doneGlobalHandlers` -> `APIBuilder#Handle`. func (r *Route) Done(handlers ...context.Handler) { if len(handlers) == 0 { return } r.doneHandlers = append(r.doneHandlers, handlers...) } // ChangeMethod will try to change the HTTP Method of this route instance. // A call of `RefreshRouter` is required after this type of change in order to change to be really applied. func (r *Route) ChangeMethod(newMethod string) bool { if newMethod != r.Method { r.methodBckp = r.Method r.Method = newMethod return true } return false } // SetStatusOffline will try make this route unavailable. // A call of `RefreshRouter` is required after this type of change in order to change to be really applied. func (r *Route) SetStatusOffline() bool { return r.ChangeMethod(MethodNone) } // RestoreStatus will try to restore the status of this route instance, i.e if `SetStatusOffline` called on a "GET" route, // then this function will make this route available with "GET" HTTP Method. // Note if that you want to set status online for an offline registered route then you should call the `ChangeMethod` instead. // It will return true if the status restored, otherwise false. // A call of `RefreshRouter` is required after this type of change in order to change to be really applied. func (r *Route) RestoreStatus() bool { return r.ChangeMethod(r.methodBckp) } // BuildHandlers is executed automatically by the router handler // at the `Application#Build` state. Do not call it manually, unless // you were defined your own request mux handler. func (r *Route) BuildHandlers() { if len(r.beginHandlers) > 0 { r.Handlers = append(r.beginHandlers, r.Handlers...) r.beginHandlers = r.beginHandlers[0:0] } if len(r.doneHandlers) > 0 { r.Handlers = append(r.Handlers, r.doneHandlers...) r.doneHandlers = r.doneHandlers[0:0] } // note: no mutex needed, this should be called in-sync when server is not running of course. } // String returns the form of METHOD, SUBDOMAIN, TMPL PATH. func (r Route) String() string { return fmt.Sprintf("%s %s%s", r.Method, r.Subdomain, r.Tmpl().Src) } // Equal compares the method, subdomain and the // underline representation of the route's path, // instead of the `String` function which returns the front representation. func (r *Route) Equal(other *Route) bool { return r.Method == other.Method && r.Subdomain == other.Subdomain && r.Path == other.Path } // DeepEqual compares the method, subdomain, the // underline representation of the route's path, // and the template source. func (r *Route) DeepEqual(other *Route) bool { return r.Equal(other) && r.tmpl.Src == other.tmpl.Src } // Tmpl returns the path template, // it contains the parsed template // for the route's path. // May contain zero named parameters. // // Developer can get his registered path // via Tmpl().Src, Route.Path is the path // converted to match the underline router's specs. func (r Route) Tmpl() macro.Template { return r.tmpl } // RegisteredHandlersLen returns the end-developer's registered handlers, all except the macro evaluator handler // if was required by the build process. func (r Route) RegisteredHandlersLen() int { n := len(r.Handlers) if handler.CanMakeHandler(r.tmpl) { n-- } return n } // IsOnline returns true if the route is marked as "online" (state). func (r Route) IsOnline() bool { return r.Method != MethodNone } // formats the parsed to the underline path syntax. // path = "/api/users/:id" // return "/api/users/%v" // // path = "/files/*file" // return /files/%v // // path = "/:username/messages/:messageid" // return "/%v/messages/%v" // we don't care about performance here, it's prelisten. func formatPath(path string) string { if strings.Contains(path, ParamStart) || strings.Contains(path, WildcardParamStart) { var ( startRune = ParamStart[0] wildcardStartRune = WildcardParamStart[0] ) var formattedParts []string parts := strings.Split(path, "/") for _, part := range parts { if len(part) == 0 { continue } if part[0] == startRune || part[0] == wildcardStartRune { // is param or wildcard param part = "%v" } formattedParts = append(formattedParts, part) } return "/" + strings.Join(formattedParts, "/") } // the whole path is static just return it return path } // StaticPath returns the static part of the original, registered route path. // if /user/{id} it will return /user // if /user/{id}/friend/{friendid:uint64} it will return /user too // if /assets/{filepath:path} it will return /assets. func (r Route) StaticPath() string { src := r.tmpl.Src bidx := strings.IndexByte(src, '{') if bidx == -1 || len(src) <= bidx { return src // no dynamic part found } if bidx <= 1 { // found at first{...} or second index (/{...}), // although first index should never happen because of the prepended slash. return "/" } return src[:bidx-1] // (/static/{...} -> /static) } // ResolvePath returns the formatted path's %v replaced with the args. func (r Route) ResolvePath(args ...string) string { rpath, formattedPath := r.Path, r.FormattedPath if rpath == formattedPath { // static, no need to pass args return rpath } // check if we have /*, if yes then join all arguments to one as path and pass that as parameter if rpath[len(rpath)-1] == WildcardParamStart[0] { parameter := strings.Join(args, "/") return fmt.Sprintf(formattedPath, parameter) } // else return the formattedPath with its args, // the order matters. for _, s := range args { formattedPath = strings.Replace(formattedPath, "%v", s, 1) } return formattedPath } // Trace returns some debug infos as a string sentence. // Should be called after Build. func (r Route) Trace() string { printfmt := fmt.Sprintf("%s:", r.Method) if r.Subdomain != "" { printfmt += fmt.Sprintf(" %s", r.Subdomain) } printfmt += fmt.Sprintf(" %s ", r.Tmpl().Src) mainHandlerName := r.MainHandlerName if !strings.HasSuffix(mainHandlerName, ")") { mainHandlerName += "()" } if l := r.RegisteredHandlersLen(); l > 1 { printfmt += fmt.Sprintf("-> %s and %d more", mainHandlerName, l-1) } else { printfmt += fmt.Sprintf("-> %s", mainHandlerName) } // printfmt := fmt.Sprintf("%s: %s >> %s", r.Method, r.Subdomain+r.Tmpl().Src, r.MainHandlerName) // if l := len(r.Handlers); l > 0 { // printfmt += fmt.Sprintf(" and %d more", l) // } return printfmt // without new line. } type routeReadOnlyWrapper struct { *Route } func (rd routeReadOnlyWrapper) Method() string { return rd.Route.Method } func (rd routeReadOnlyWrapper) Name() string { return rd.Route.Name } func (rd routeReadOnlyWrapper) Subdomain() string { return rd.Route.Subdomain } func (rd routeReadOnlyWrapper) Path() string { return rd.Route.tmpl.Src } func (rd routeReadOnlyWrapper) Trace() string { return rd.Route.Trace() } func (rd routeReadOnlyWrapper) Tmpl() macro.Template { return rd.Route.Tmpl() } func (rd routeReadOnlyWrapper) MainHandlerName() string { return rd.Route.MainHandlerName } func (rd routeReadOnlyWrapper) StaticSites() []context.StaticSite { return rd.Route.StaticSites }