package iris import ( "bytes" stdContext "context" "errors" "fmt" "io" "log" "net" "net/http" "os" "regexp" "strings" "sync" "time" "github.com/kataras/iris/v12/context" "github.com/kataras/iris/v12/core/host" "github.com/kataras/iris/v12/core/netutil" "github.com/kataras/iris/v12/core/router" "github.com/kataras/iris/v12/i18n" "github.com/kataras/iris/v12/middleware/accesslog" "github.com/kataras/iris/v12/middleware/cors" "github.com/kataras/iris/v12/middleware/recover" "github.com/kataras/iris/v12/middleware/requestid" "github.com/kataras/iris/v12/view" "github.com/kataras/golog" "github.com/kataras/tunnel" "github.com/tdewolff/minify/v2" "github.com/tdewolff/minify/v2/css" "github.com/tdewolff/minify/v2/html" "github.com/tdewolff/minify/v2/js" "github.com/tdewolff/minify/v2/json" "github.com/tdewolff/minify/v2/svg" "github.com/tdewolff/minify/v2/xml" ) // Version is the current version of the Iris Web Framework. const Version = "12.2.0-beta1" // Byte unit helpers. const ( B = 1 << (10 * iota) KB MB GB TB PB EB ) // Application is responsible to manage the state of the application. // It contains and handles all the necessary parts to create a fast web server. type Application struct { // routing embedded | exposing APIBuilder's and Router's public API. *router.APIBuilder *router.Router router.HTTPErrorHandler // if Router is Downgraded this is nil. ContextPool *context.Pool // See SetContextErrorHandler, defaults to nil. contextErrorHandler context.ErrorHandler // config contains the configuration fields // all fields defaults to something that is working, developers don't have to set it. config *Configuration // the golog logger instance, defaults to "Info" level messages (all except "Debug") logger *golog.Logger // I18n contains localization and internationalization support. // Use the `Load` or `LoadAssets` to locale language files. // // See `Context#Tr` method for request-based translations. I18n *i18n.I18n // Validator is the request body validator, defaults to nil. Validator context.Validator // Minifier to minify responses. minifier *minify.M // view engine view view.View // used for build builded bool defaultMode bool // OnBuild is a single function which // is fired on the first `Build` method call. // If reports an error then the execution // is stopped and the error is logged. // It's nil by default except when `Switch` instead of `New` or `Default` // is used to initialize the Application. // Users can wrap it to accept more events. OnBuild func() error mu sync.Mutex // name is the application name and the log prefix for // that Application instance's Logger. See `SetName` and `String`. // Defaults to IRIS_APP_NAME envrinoment variable otherwise empty. name string // Hosts contains a list of all servers (Host Supervisors) that this app is running on. // // Hosts may be empty only if application ran(`app.Run`) with `iris.Raw` option runner, // otherwise it contains a single host (`app.Hosts[0]`). // // Additional Host Supervisors can be added to that list by calling the `app.NewHost` manually. // // Hosts field is available after `Run` or `NewHost`. Hosts []*host.Supervisor hostConfigurators []host.Configurator } // New creates and returns a fresh empty iris *Application instance. func New() *Application { config := DefaultConfiguration() app := &Application{ config: &config, Router: router.NewRouter(), I18n: i18n.New(), minifier: newMinifier(), } logger := newLogger(app) app.logger = logger app.APIBuilder = router.NewAPIBuilder(logger) app.ContextPool = context.New(func() interface{} { return context.NewContext(app) }) context.RegisterApplication(app) return app } // Default returns a new Application. // Default with "debug" Logger Level. // Localization enabled on "./locales" directory // and HTML templates on "./views" or "./templates" directory. // It runs with the AccessLog on "./access.log", // CORS (allow all), Recovery and Request ID middleware already attached. func Default() *Application { app := New() // Set default log level. app.logger.SetLevel("debug") app.logger.Debugf(`Log level set to "debug"`) // Register the accesslog middleware. logFile, err := os.OpenFile("./access.log", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0600) if err == nil { // Close the file on shutdown. app.ConfigureHost(func(su *Supervisor) { su.RegisterOnShutdown(func() { logFile.Close() }) }) ac := accesslog.New(logFile) ac.AddOutput(app.logger.Printer) app.UseRouter(ac.Handler) app.logger.Debugf("Using <%s> to log requests", logFile.Name()) } // Register the requestid middleware // before recover so current Context.GetID() contains the info on panic logs. app.UseRouter(requestid.New()) app.logger.Debugf("Using to identify requests") // Register the recovery, after accesslog and recover, // before end-developer's middleware. app.UseRouter(recover.New()) // Register CORS (allow any origin to pass through) middleware. app.UseRouter(cors.New(). ExtractOriginFunc(cors.DefaultOriginExtractor). ReferrerPolicy(cors.NoReferrerWhenDowngrade). AllowOriginFunc(cors.AllowAnyOrigin). Handler()) app.defaultMode = true return app } func newLogger(app *Application) *golog.Logger { logger := golog.Default.Child(app) if name := os.Getenv("IRIS_APP_NAME"); name != "" { app.name = name logger.SetChildPrefix(name) } return logger } // SetName sets a unique name to this Iris Application. // It sets a child prefix for the current Application's Logger. // Look `String` method too. // // It returns this Application. func (app *Application) SetName(appName string) *Application { app.mu.Lock() defer app.mu.Unlock() if app.name == "" { app.logger.SetChildPrefix(appName) } app.name = appName return app } // String completes the fmt.Stringer interface and it returns // the application's name. // If name was not set by `SetName` or `IRIS_APP_NAME` environment variable // then this will return an empty string. func (app *Application) String() string { return app.name } // WWW creates and returns a "www." subdomain. // The difference from `app.Subdomain("www")` or `app.Party("www.")` is that the `app.WWW()` method // wraps the router so all http(s)://mydomain.com will be redirect to http(s)://www.mydomain.com. // Other subdomains can be registered using the app: `sub := app.Subdomain("mysubdomain")`, // child subdomains can be registered using the www := app.WWW(); www.Subdomain("wwwchildSubdomain"). func (app *Application) WWW() router.Party { return app.SubdomainRedirect(app, app.Subdomain("www")) } // SubdomainRedirect registers a router wrapper which // redirects(StatusMovedPermanently) a (sub)domain to another subdomain or to the root domain as fast as possible, // before the router's try to execute route's handler(s). // // It receives two arguments, they are the from and to/target locations, // 'from' can be a wildcard subdomain as well (app.WildcardSubdomain()) // 'to' is not allowed to be a wildcard for obvious reasons, // 'from' can be the root domain(app) when the 'to' is not the root domain and visa-versa. // // Usage: // www := app.Subdomain("www") <- same as app.Party("www.") // app.SubdomainRedirect(app, www) // This will redirect all http(s)://mydomain.com/%anypath% to http(s)://www.mydomain.com/%anypath%. // // One or more subdomain redirects can be used to the same app instance. // // If you need more information about this implementation then you have to navigate through // the `core/router#NewSubdomainRedirectWrapper` function instead. // // Example: https://github.com/kataras/iris/tree/master/_examples/routing/subdomains/redirect func (app *Application) SubdomainRedirect(from, to router.Party) router.Party { sd := router.NewSubdomainRedirectWrapper(app.ConfigurationReadOnly().GetVHost, from.GetRelPath(), to.GetRelPath()) app.Router.AddRouterWrapper(sd) return to } // Configure can called when modifications to the framework instance needed. // It accepts the framework instance // and returns an error which if it's not nil it's printed to the logger. // See configuration.go for more. // // Returns itself in order to be used like `app:= New().Configure(...)` func (app *Application) Configure(configurators ...Configurator) *Application { for _, cfg := range configurators { if cfg != nil { cfg(app) } } return app } // ConfigurationReadOnly returns an object which doesn't allow field writing. func (app *Application) ConfigurationReadOnly() context.ConfigurationReadOnly { return app.config } // Logger returns the golog logger instance(pointer) that is being used inside the "app". // // Available levels: // - "disable" // - "fatal" // - "error" // - "warn" // - "info" // - "debug" // Usage: app.Logger().SetLevel("error") // Or set the level through Configurartion's LogLevel or WithLogLevel functional option. // Defaults to "info" level. // // Callers can use the application's logger which is // the same `golog.Default.LastChild()` logger, // to print custom logs too. // Usage: // app.Logger().Error/Errorf("...") // app.Logger().Warn/Warnf("...") // app.Logger().Info/Infof("...") // app.Logger().Debug/Debugf("...") // // Setting one or more outputs: app.Logger().SetOutput(io.Writer...) // Adding one or more outputs : app.Logger().AddOutput(io.Writer...) // // Adding custom levels requires import of the `github.com/kataras/golog` package: // First we create our level to a golog.Level // in order to be used in the Log functions. // var SuccessLevel golog.Level = 6 // Register our level, just three fields. // golog.Levels[SuccessLevel] = &golog.LevelMetadata{ // Name: "success", // RawText: "[SUCC]", // // ColorfulText (Green Color[SUCC]) // ColorfulText: "\x1b[32m[SUCC]\x1b[0m", // } // Usage: // app.Logger().SetLevel("success") // app.Logger().Logf(SuccessLevel, "a custom leveled log message") func (app *Application) Logger() *golog.Logger { return app.logger } // IsDebug reports whether the application is running // under debug/development mode. // It's just a shortcut of Logger().Level >= golog.DebugLevel. // The same method existss as Context.IsDebug() too. func (app *Application) IsDebug() bool { return app.logger.Level >= golog.DebugLevel } // I18nReadOnly returns the i18n's read-only features. // See `I18n` method for more. func (app *Application) I18nReadOnly() context.I18nReadOnly { return app.I18n } // Validate validates a value and returns nil if passed or // the failure reason if does not. func (app *Application) Validate(v interface{}) error { if app.Validator == nil { return nil } // val := reflect.ValueOf(v) // if val.Kind() == reflect.Ptr && !val.IsNil() { // val = val.Elem() // } // if val.Kind() == reflect.Struct && val.Type() != timeType { // return app.Validator.Struct(v) // } // no need to check the kind, underline lib does it but in the future this may change (look above). err := app.Validator.Struct(v) if err != nil { if !strings.HasPrefix(err.Error(), "validator: ") { return err } } return nil } func newMinifier() *minify.M { m := minify.New() m.AddFunc("text/css", css.Minify) m.AddFunc("text/html", html.Minify) m.AddFunc("image/svg+xml", svg.Minify) m.AddFuncRegexp(regexp.MustCompile("^(application|text)/(x-)?(java|ecma)script$"), js.Minify) m.AddFuncRegexp(regexp.MustCompile("[/+]json$"), json.Minify) m.AddFuncRegexp(regexp.MustCompile("[/+]xml$"), xml.Minify) return m } // Minify is a middleware which minifies the responses // based on the response content type. // Note that minification might be slower, caching is advised. // Customize the minifier through `Application.Minifier()`. // Usage: // app.Use(iris.Minify) func Minify(ctx Context) { w := ctx.Application().Minifier().ResponseWriter(ctx.ResponseWriter().Naive(), ctx.Request()) // Note(@kataras): // We don't use defer w.Close() // because this response writer holds a sync.WaitGroup under the hoods // and we MUST be sure that its wg.Wait is called on request cancelation // and not in the end of handlers chain execution // (which if running a time-consuming task it will delay its resource release). ctx.OnCloseErr(w.Close) ctx.ResponseWriter().SetWriter(w) ctx.Next() } // Minifier returns the minifier instance. // By default it can minifies: // - text/html // - text/css // - image/svg+xml // - application/text(javascript, ecmascript, json, xml). // Use that instance to add custom Minifiers before server ran. func (app *Application) Minifier() *minify.M { return app.minifier } // RegisterView registers a view engine for the application. // Children can register their own too. If no Party view Engine is registered // then this one will be used to render the templates instead. func (app *Application) RegisterView(viewEngine view.Engine) { app.view.Register(viewEngine) } // View executes and writes the result of a template file to the writer. // // First parameter is the writer to write the parsed template. // Second parameter is the relative, to templates directory, template filename, including extension. // Third parameter is the layout, can be empty string. // Forth parameter is the bindable data to the template, can be nil. // // Use context.View to render templates to the client instead. // Returns an error on failure, otherwise nil. func (app *Application) View(writer io.Writer, filename string, layout string, bindingData interface{}) error { if !app.view.Registered() { err := errors.New("view engine is missing, use `RegisterView`") app.logger.Error(err) return err } return app.view.ExecuteWriter(writer, filename, layout, bindingData) } // GetContextPool returns the Iris sync.Pool which holds the contexts values. // Iris automatically releases the request context, so you don't have to use it. // It's only useful to manually release the context on cases that connection // is hijacked by a third-party middleware and the http handler return too fast. func (app *Application) GetContextPool() *context.Pool { return app.ContextPool } // SetContextErrorHandler can optionally register a handler to handle // and fire a customized error body to the client on JSON write failures. // // ExampleCode: // // type contextErrorHandler struct{} // func (e *contextErrorHandler) HandleContextError(ctx iris.Context, err error) { // errors.InvalidArgument.Err(ctx, err) // } // ... // app.SetContextErrorHandler(new(contextErrorHandler)) func (app *Application) SetContextErrorHandler(errHandler context.ErrorHandler) *Application { app.contextErrorHandler = errHandler return app } // GetContextErrorHandler returns the handler which handles errors // on JSON write failures. func (app *Application) GetContextErrorHandler() context.ErrorHandler { return app.contextErrorHandler } // ConfigureHost accepts one or more `host#Configuration`, these configurators functions // can access the host created by `app.Run` or `app.Listen`, // they're being executed when application is ready to being served to the public. // // It's an alternative way to interact with a host that is automatically created by // `app.Run`. // // These "configurators" can work side-by-side with the `iris#Addr, iris#Server, iris#TLS, iris#AutoTLS, iris#Listener` // final arguments("hostConfigs") too. // // Note that these application's host "configurators" will be shared with the rest of // the hosts that this app will may create (using `app.NewHost`), meaning that // `app.NewHost` will execute these "configurators" everytime that is being called as well. // // These "configurators" should be registered before the `app.Run` or `host.Serve/Listen` functions. func (app *Application) ConfigureHost(configurators ...host.Configurator) *Application { app.mu.Lock() app.hostConfigurators = append(app.hostConfigurators, configurators...) app.mu.Unlock() return app } const serverLoggerPrefix = "[HTTP Server] " type customHostServerLogger struct { // see #1875 parent io.Writer ignoreLogs [][]byte } var newLineBytes = []byte("\n") func newCustomHostServerLogger(w io.Writer, ignoreLogs []string) *customHostServerLogger { prefixAsByteSlice := []byte(serverLoggerPrefix) // build the ignore lines. ignoreLogsAsByteSlice := make([][]byte, 0, len(ignoreLogs)) for _, s := range ignoreLogs { ignoreLogsAsByteSlice = append(ignoreLogsAsByteSlice, append(prefixAsByteSlice, []byte(s)...)) // append([]byte(s), newLineBytes...) } return &customHostServerLogger{ parent: w, ignoreLogs: ignoreLogsAsByteSlice, } } func (l *customHostServerLogger) Write(p []byte) (int, error) { for _, ignoredLogBytes := range l.ignoreLogs { if bytes.Equal(bytes.TrimSuffix(p, newLineBytes), ignoredLogBytes) { return 0, nil } } return l.parent.Write(p) } // NewHost accepts a standard *http.Server object, // completes the necessary missing parts of that "srv" // and returns a new, ready-to-use, host (supervisor). func (app *Application) NewHost(srv *http.Server) *host.Supervisor { app.mu.Lock() defer app.mu.Unlock() // set the server's handler to the framework's router if srv.Handler == nil { srv.Handler = app.Router } // check if different ErrorLog provided, if not bind it with the framework's logger. if srv.ErrorLog == nil { serverLogger := newCustomHostServerLogger(app.logger.Printer.Output, app.config.IgnoreServerErrors) srv.ErrorLog = log.New(serverLogger, serverLoggerPrefix, 0) } if addr := srv.Addr; addr == "" { addr = ":8080" if len(app.Hosts) > 0 { if v := app.Hosts[0].Server.Addr; v != "" { addr = v } } srv.Addr = addr } // app.logger.Debugf("Host: addr is %s", srv.Addr) // create the new host supervisor // bind the constructed server and return it su := host.New(srv) if app.config.vhost == "" { // vhost now is useful for router subdomain on wildcard subdomains, // in order to correct decide what to do on: // mydomain.com -> invalid // localhost -> invalid // sub.mydomain.com -> valid // sub.localhost -> valid // we need the host (without port if 80 or 443) in order to validate these, so: app.config.vhost = netutil.ResolveVHost(srv.Addr) } // app.logger.Debugf("Host: virtual host is %s", app.config.vhost) // the below schedules some tasks that will run among the server if !app.config.DisableStartupLog { printer := app.logger.Printer.Output hostPrinter := host.WriteStartupLogOnServe(printer) if len(app.Hosts) == 0 { // print the version info on the first running host. su.RegisterOnServe(func(h host.TaskHost) { hasBuildInfo := BuildTime != "" && BuildRevision != "" tab := " " if hasBuildInfo { tab = " " } fmt.Fprintf(printer, "Iris Version:%s%s\n", tab, Version) if hasBuildInfo { fmt.Fprintf(printer, "Build Time: %s\nBuild Revision: %s\n", BuildTime, BuildRevision) } fmt.Fprintln(printer) hostPrinter(h) }) } else { su.RegisterOnServe(hostPrinter) } // app.logger.Debugf("Host: register startup notifier") } if !app.config.DisableInterruptHandler { // when CTRL/CMD+C pressed. shutdownTimeout := 10 * time.Second host.RegisterOnInterrupt(host.ShutdownOnInterrupt(su, shutdownTimeout)) // app.logger.Debugf("Host: register server shutdown on interrupt(CTRL+C/CMD+C)") } su.IgnoredErrors = append(su.IgnoredErrors, app.config.IgnoreServerErrors...) if len(su.IgnoredErrors) > 0 { app.logger.Debugf("Host: server will ignore the following errors: %s", su.IgnoredErrors) } su.Configure(app.hostConfigurators...) app.Hosts = append(app.Hosts, su) return su } // Shutdown gracefully terminates all the application's server hosts and any tunnels. // Returns an error on the first failure, otherwise nil. func (app *Application) Shutdown(ctx stdContext.Context) error { app.mu.Lock() defer app.mu.Unlock() for i, su := range app.Hosts { app.logger.Debugf("Host[%d]: Shutdown now", i) if err := su.Shutdown(ctx); err != nil { app.logger.Debugf("Host[%d]: Error while trying to shutdown", i) return err } } for _, t := range app.config.Tunneling.Tunnels { if t.Name == "" { continue } if err := app.config.Tunneling.StopTunnel(t); err != nil { return err } } return nil } // Build sets up, once, the framework. // It builds the default router with its default macros // and the template functions that are very-closed to iris. // // If error occurred while building the Application, the returns type of error will be an *errgroup.Group // which let the callers to inspect the errors and cause, usage: // // import "github.com/kataras/iris/v12/core/errgroup" // // errgroup.Walk(app.Build(), func(typ interface{}, err error) { // app.Logger().Errorf("%s: %s", typ, err) // }) func (app *Application) Build() error { if app.builded { return nil } if cb := app.OnBuild; cb != nil { if err := cb(); err != nil { return err } } // start := time.Now() app.builded = true // even if fails. // check if a prior app.Logger().SetLevel called and if not // then set the defined configuration's log level. if app.logger.Level == golog.InfoLevel /* the default level */ { app.logger.SetLevel(app.config.LogLevel) } if app.defaultMode { // the app.I18n and app.View will be not available until Build. if !app.I18n.Loaded() { for _, s := range []string{"./locales/*/*", "./locales/*", "./translations"} { if _, err := os.Stat(s); err != nil { continue } if err := app.I18n.Load(s); err != nil { continue } app.I18n.SetDefault("en-US") break } } if !app.view.Registered() { for _, s := range []string{"./views", "./templates", "./web/views"} { if _, err := os.Stat(s); err != nil { continue } app.RegisterView(HTML(s, ".html")) break } } } if app.I18n.Loaded() { // {{ tr "lang" "key" arg1 arg2 }} app.view.AddFunc("tr", app.I18n.Tr) app.Router.PrependRouterWrapper(app.I18n.Wrapper()) } if app.view.Registered() { app.logger.Debugf("Application: view engine %q is registered", app.view.Name()) // view engine // here is where we declare the closed-relative framework functions. // Each engine has their defaults, i.e yield,render,render_r,partial, params... rv := router.NewRoutePathReverser(app.APIBuilder) app.view.AddFunc("urlpath", rv.Path) // app.view.AddFunc("url", rv.URL) if err := app.view.Load(); err != nil { app.logger.Errorf("View Builder: %v", err) return err } } if !app.Router.Downgraded() { // router if _, err := injectLiveReload(app); err != nil { app.logger.Errorf("LiveReload: init: failed: %v", err) return err } if app.config.ForceLowercaseRouting { // This should always be executed first. app.Router.PrependRouterWrapper(func(w http.ResponseWriter, r *http.Request, next http.HandlerFunc) { r.Host = strings.ToLower(r.Host) r.URL.Host = strings.ToLower(r.URL.Host) r.URL.Path = strings.ToLower(r.URL.Path) next(w, r) }) } // create the request handler, the default routing handler routerHandler := router.NewDefaultHandler(app.config, app.logger) err := app.Router.BuildRouter(app.ContextPool, routerHandler, app.APIBuilder, false) if err != nil { app.logger.Error(err) return err } app.HTTPErrorHandler = routerHandler if app.config.Timeout > 0 { app.Router.SetTimeoutHandler(app.config.Timeout, app.config.TimeoutMessage) app.ConfigureHost(func(su *Supervisor) { if su.Server.ReadHeaderTimeout == 0 { su.Server.ReadHeaderTimeout = app.config.Timeout + 5*time.Second } if su.Server.ReadTimeout == 0 { su.Server.ReadTimeout = app.config.Timeout + 10*time.Second } if su.Server.WriteTimeout == 0 { su.Server.WriteTimeout = app.config.Timeout + 15*time.Second } if su.Server.IdleTimeout == 0 { su.Server.IdleTimeout = app.config.Timeout + 25*time.Second } }) } // re-build of the router from outside can be done with // app.RefreshRouter() } // if end := time.Since(start); end.Seconds() > 5 { // app.logger.Debugf("Application: build took %s", time.Since(start)) return nil } // Runner is just an interface which accepts the framework instance // and returns an error. // // It can be used to register a custom runner with `Run` in order // to set the framework's server listen action. // // Currently `Runner` is being used to declare the builtin server listeners. // // See `Run` for more. type Runner func(*Application) error // Listener can be used as an argument for the `Run` method. // It can start a server with a custom net.Listener via server's `Serve`. // // Second argument is optional, it accepts one or more // `func(*host.Configurator)` that are being executed // on that specific host that this function will create to start the server. // Via host configurators you can configure the back-end host supervisor, // i.e to add events for shutdown, serve or error. // An example of this use case can be found at: // https://github.com/kataras/iris/blob/master/_examples/http-server/notify-on-shutdown/main.go // Look at the `ConfigureHost` too. // // See `Run` for more. func Listener(l net.Listener, hostConfigs ...host.Configurator) Runner { return func(app *Application) error { app.config.vhost = netutil.ResolveVHost(l.Addr().String()) return app.NewHost(&http.Server{Addr: l.Addr().String()}). Configure(hostConfigs...). Serve(l) } } // Server can be used as an argument for the `Run` method. // It can start a server with a *http.Server. // // Second argument is optional, it accepts one or more // `func(*host.Configurator)` that are being executed // on that specific host that this function will create to start the server. // Via host configurators you can configure the back-end host supervisor, // i.e to add events for shutdown, serve or error. // An example of this use case can be found at: // https://github.com/kataras/iris/blob/master/_examples/http-server/notify-on-shutdown/main.go // Look at the `ConfigureHost` too. // // See `Run` for more. func Server(srv *http.Server, hostConfigs ...host.Configurator) Runner { return func(app *Application) error { return app.NewHost(srv). Configure(hostConfigs...). ListenAndServe() } } // Addr can be used as an argument for the `Run` method. // It accepts a host address which is used to build a server // and a listener which listens on that host and port. // // Addr should have the form of [host]:port, i.e localhost:8080 or :8080. // // Second argument is optional, it accepts one or more // `func(*host.Configurator)` that are being executed // on that specific host that this function will create to start the server. // Via host configurators you can configure the back-end host supervisor, // i.e to add events for shutdown, serve or error. // An example of this use case can be found at: // https://github.com/kataras/iris/blob/master/_examples/http-server/notify-on-shutdown/main.go // Look at the `ConfigureHost` too. // // See `Run` for more. func Addr(addr string, hostConfigs ...host.Configurator) Runner { return func(app *Application) error { return app.NewHost(&http.Server{Addr: addr}). Configure(hostConfigs...). ListenAndServe() } } var ( // TLSNoRedirect is a `host.Configurator` which can be passed as last argument // to the `TLS` runner function. It disables the automatic // registration of redirection from "http://" to "https://" requests. // Applies only to the `TLS` runner. // See `AutoTLSNoRedirect` to register a custom fallback server for `AutoTLS` runner. TLSNoRedirect = func(su *host.Supervisor) { su.NoRedirect() } // AutoTLSNoRedirect is a `host.Configurator`. // It registers a fallback HTTP/1.1 server for the `AutoTLS` one. // The function accepts the letsencrypt wrapper and it // should return a valid instance of http.Server which its handler should be the result // of the "acmeHandler" wrapper. // Usage: // getServer := func(acme func(http.Handler) http.Handler) *http.Server { // srv := &http.Server{Handler: acme(yourCustomHandler), ...otherOptions} // go srv.ListenAndServe() // return srv // } // app.Run(iris.AutoTLS(":443", "example.com example2.com", "mail@example.com", getServer)) // // Note that if Server.Handler is nil then the server is automatically ran // by the framework and the handler set to automatic redirection, it's still // a valid option when the caller wants just to customize the server's fields (except Addr). // With this host configurator the caller can customize the server // that letsencrypt relies to perform the challenge. // LetsEncrypt Certification Manager relies on http://example.com/.well-known/acme-challenge/. AutoTLSNoRedirect = func(getFallbackServer func(acmeHandler func(fallback http.Handler) http.Handler) *http.Server) host.Configurator { return func(su *host.Supervisor) { su.NoRedirect() su.Fallback = getFallbackServer } } ) // TLS can be used as an argument for the `Run` method. // It will start the Application's secure server. // // Use it like you used to use the http.ListenAndServeTLS function. // // Addr should have the form of [host]:port, i.e localhost:443 or :443. // "certFileOrContents" & "keyFileOrContents" should be filenames with their extensions // or raw contents of the certificate and the private key. // // Last argument is optional, it accepts one or more // `func(*host.Configurator)` that are being executed // on that specific host that this function will create to start the server. // Via host configurators you can configure the back-end host supervisor, // i.e to add events for shutdown, serve or error. // An example of this use case can be found at: // https://github.com/kataras/iris/blob/master/_examples/http-server/notify-on-shutdown/main.go // Look at the `ConfigureHost` too. // // See `Run` for more. func TLS(addr string, certFileOrContents, keyFileOrContents string, hostConfigs ...host.Configurator) Runner { return func(app *Application) error { return app.NewHost(&http.Server{Addr: addr}). Configure(hostConfigs...). ListenAndServeTLS(certFileOrContents, keyFileOrContents) } } // AutoTLS can be used as an argument for the `Run` method. // It will start the Application's secure server using // certifications created on the fly by the "autocert" golang/x package, // so localhost may not be working, use it at "production" machine. // // Addr should have the form of [host]:port, i.e mydomain.com:443. // // The whitelisted domains are separated by whitespace in "domain" argument, // i.e "iris-go.com", can be different than "addr". // If empty, all hosts are currently allowed. This is not recommended, // as it opens a potential attack where clients connect to a server // by IP address and pretend to be asking for an incorrect host name. // Manager will attempt to obtain a certificate for that host, incorrectly, // eventually reaching the CA's rate limit for certificate requests // and making it impossible to obtain actual certificates. // // For an "e-mail" use a non-public one, letsencrypt needs that for your own security. // // Note: `AutoTLS` will start a new server for you // which will redirect all http versions to their https, including subdomains as well. // // Last argument is optional, it accepts one or more // `func(*host.Configurator)` that are being executed // on that specific host that this function will create to start the server. // Via host configurators you can configure the back-end host supervisor, // i.e to add events for shutdown, serve or error. // An example of this use case can be found at: // https://github.com/kataras/iris/blob/master/_examples/http-server/notify-on-shutdown/main.go // Look at the `ConfigureHost` too. // // Usage: // app.Run(iris.AutoTLS("iris-go.com:443", "iris-go.com www.iris-go.com", "mail@example.com")) // // See `Run` and `core/host/Supervisor#ListenAndServeAutoTLS` for more. func AutoTLS( addr string, domain string, email string, hostConfigs ...host.Configurator) Runner { return func(app *Application) error { return app.NewHost(&http.Server{Addr: addr}). Configure(hostConfigs...). ListenAndServeAutoTLS(domain, email, "letscache") } } // Raw can be used as an argument for the `Run` method. // It accepts any (listen) function that returns an error, // this function should be block and return an error // only when the server exited or a fatal error caused. // // With this option you're not limited to the servers // that iris can run by-default. // // See `Run` for more. func Raw(f func() error) Runner { return func(app *Application) error { app.logger.Debugf("HTTP Server will start from unknown, external function") return f() } } var ( // ErrServerClosed is logged by the standard net/http server when the server is terminated. // Ignore it by passing this error to the `iris.WithoutServerError` configurator // on `Application.Run/Listen` method. // // An alias of the `http#ErrServerClosed`. ErrServerClosed = http.ErrServerClosed // ErrURLQuerySemicolon is logged by the standard net/http server when // the request contains a semicolon (;) wihch, after go1.17 it's not used as a key-value separator character. // // Ignore it by passing this error to the `iris.WithoutServerError` configurator // on `Application.Run/Listen` method. // // An alias of the `http#ErrServerClosed`. ErrURLQuerySemicolon = errors.New("http: URL query contains semicolon, which is no longer a supported separator; parts of the query may be stripped when parsed; see golang.org/issue/25192") ) // Listen builds the application and starts the server // on the TCP network address "host:port" which // handles requests on incoming connections. // // Listen always returns a non-nil error. // Ignore specific errors by using an `iris.WithoutServerError(iris.ErrServerClosed)` // as a second input argument. // // Listen is a shortcut of `app.Run(iris.Addr(hostPort, withOrWithout...))`. // See `Run` for details. func (app *Application) Listen(hostPort string, withOrWithout ...Configurator) error { return app.Run(Addr(hostPort), withOrWithout...) } // Run builds the framework and starts the desired `Runner` with or without configuration edits. // // Run should be called only once per Application instance, it blocks like http.Server. // // If more than one server needed to run on the same iris instance // then create a new host and run it manually by `go NewHost(*http.Server).Serve/ListenAndServe` etc... // or use an already created host: // h := NewHost(*http.Server) // Run(Raw(h.ListenAndServe), WithCharset("utf-8"), WithRemoteAddrHeader("CF-Connecting-IP")) // // The Application can go online with any type of server or iris's host with the help of // the following runners: // `Listener`, `Server`, `Addr`, `TLS`, `AutoTLS` and `Raw`. func (app *Application) Run(serve Runner, withOrWithout ...Configurator) error { app.Configure(withOrWithout...) if err := app.Build(); err != nil { app.logger.Error(err) return err } app.ConfigureHost(func(host *Supervisor) { host.SocketSharding = app.config.SocketSharding host.KeepAlive = app.config.KeepAlive }) app.tryStartTunneling() if len(app.Hosts) > 0 { app.logger.Debugf("Application: running using %d host(s)", len(app.Hosts)+1 /* +1 the current */) } // this will block until an error(unless supervisor's DeferFlow called from a Task). err := serve(app) if err != nil { app.logger.Error(err) } return err } // https://ngrok.com/docs func (app *Application) tryStartTunneling() { if len(app.config.Tunneling.Tunnels) == 0 { return } app.ConfigureHost(func(su *host.Supervisor) { su.RegisterOnServe(func(h host.TaskHost) { publicAddrs, err := tunnel.Start(app.config.Tunneling) if err != nil { app.logger.Errorf("Host: tunneling error: %v", err) return } publicAddr := publicAddrs[0] // to make subdomains resolution still based on this new remote, public addresses. app.config.vhost = publicAddr[strings.Index(publicAddr, "://")+3:] directLog := []byte(fmt.Sprintf("• Public Address: %s\n", publicAddr)) app.logger.Printer.Write(directLog) // nolint:errcheck }) }) }