package context import ( "path/filepath" "reflect" "regexp" "runtime" "strings" "sync" ) var ( handlerNames = make(map[*regexp.Regexp]string) handlerNamesMu sync.RWMutex ) // PackageName is the Iris Go module package name. var PackageName = strings.TrimSuffix(reflect.TypeOf(Handlers{}).PkgPath(), "/context") // SetHandlerName sets a handler name that could be // fetched through `HandlerName`. The "original" should be // the Go's original regexp-featured (can be retrieved through a `HandlerName` call) function name. // The "replacement" should be the custom, human-text of that function name. // // If the name starts with "iris" then it replaces that string with the // full Iris module package name, // e.g. iris/middleware/logger.(*requestLoggerMiddleware).ServeHTTP-fm to // github.com/kataras/iris/v12/middleware/logger.(*requestLoggerMiddleware).ServeHTTP-fm // for convenient between Iris versions. func SetHandlerName(original string, replacement string) { if strings.HasPrefix(original, "iris") { original = PackageName + strings.TrimPrefix(original, "iris") } handlerNamesMu.Lock() handlerNames[regexp.MustCompile(original)] = replacement handlerNamesMu.Unlock() } // A Handler responds to an HTTP request. // It writes reply headers and data to the Context.ResponseWriter() and then return. // Returning signals that the request is finished; // it is not valid to use the Context after or concurrently with the completion of the Handler call. // // Depending on the HTTP client software, HTTP protocol version, // and any intermediaries between the client and the iris server, // it may not be possible to read from the Context.Request().Body after writing to the Context.ResponseWriter(). // Cautious handlers should read the Context.Request().Body first, and then reply. // // Except for reading the body, handlers should not modify the provided Context. // // If Handler panics, the server (the caller of Handler) assumes that the effect of the panic was isolated to the active request. // It recovers the panic, logs a stack trace to the server error log, and hangs up the connection. type Handler func(Context) // Handlers is just a type of slice of []Handler. // // See `Handler` for more. type Handlers []Handler func valueOf(v interface{}) reflect.Value { if val, ok := v.(reflect.Value); ok { return val } return reflect.ValueOf(v) } // HandlerName returns the handler's function name. // See `Context.HandlerName` method to get function name of the current running handler in the chain. // See `SetHandlerName` too. func HandlerName(h interface{}) string { pc := valueOf(h).Pointer() name := runtime.FuncForPC(pc).Name() handlerNamesMu.RLock() for regex, newName := range handlerNames { if regex.MatchString(name) { name = newName break } } handlerNamesMu.RUnlock() return name } // HandlerFileLine returns the handler's file and line information. // See `context.HandlerFileLine` to get the file, line of the current running handler in the chain. func HandlerFileLine(h interface{}) (file string, line int) { pc := valueOf(h).Pointer() return runtime.FuncForPC(pc).FileLine(pc) } // HandlerFileLineRel same as `HandlerFileLine` but it returns the path as relative to the "workingDir". func HandlerFileLineRel(h interface{}, workingDir string) (file string, line int) { file, line = HandlerFileLine(h) if relFile, err := filepath.Rel(workingDir, file); err == nil { if !strings.HasPrefix(relFile, "..") { // Only if it's relative to this path, not parent. file = "./" + relFile } } return } // MainHandlerName tries to find the main handler that end-developer // registered on the provided chain of handlers and returns its function name. func MainHandlerName(handlers Handlers) (name string, index int) { for i := 0; i < len(handlers); i++ { name = HandlerName(handlers[i]) index = i if !strings.HasPrefix(name, "github.com/kataras/iris/v12") || strings.HasPrefix(name, "github.com/kataras/iris/v12/core/router.StripPrefix") || strings.HasPrefix(name, "github.com/kataras/iris/v12/core/router.FileServer") { break } } return } // Filter is just a type of func(Handler) bool which reports whether an action must be performed // based on the incoming request. // // See `NewConditionalHandler` for more. type Filter func(Context) bool // NewConditionalHandler returns a single Handler which can be registered // as a middleware. // Filter is just a type of Handler which returns a boolean. // Handlers here should act like middleware, they should contain `ctx.Next` to proceed // to the next handler of the chain. Those "handlers" are registered to the per-request context. // // // It checks the "filter" and if passed then // it, correctly, executes the "handlers". // // If passed, this function makes sure that the Context's information // about its per-request handler chain based on the new "handlers" is always updated. // // If not passed, then simply the Next handler(if any) is executed and "handlers" are ignored. // // Example can be found at: _examples/routing/conditional-chain. func NewConditionalHandler(filter Filter, handlers ...Handler) Handler { return func(ctx Context) { if filter(ctx) { // Note that we don't want just to fire the incoming handlers, we must make sure // that it won't break any further handler chain // information that may be required for the next handlers. // // The below code makes sure that this conditional handler does not break // the ability that iris provides to its end-devs // to check and modify the per-request handlers chain at runtime. currIdx := ctx.HandlerIndex(-1) currHandlers := ctx.Handlers() if currIdx == len(currHandlers)-1 { // if this is the last handler of the chain // just add to the last the new handlers and call Next to fire those. ctx.AddHandler(handlers...) ctx.Next() return } // otherwise insert the new handlers in the middle of the current executed chain and the next chain. newHandlers := append(currHandlers[:currIdx+1], append(handlers, currHandlers[currIdx+1:]...)...) ctx.SetHandlers(newHandlers) ctx.Next() return } // if not pass, then just execute the next. ctx.Next() } }