2016-11-07 14:15:22 +01:00
/ *
Package iris the fastest go web framework in ( this ) Earth .
Basic usage
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
package main
import "github.com/kataras/iris"
func main ( ) {
iris . Get ( "/hi_json" , func ( c * iris . Context ) {
c . JSON ( iris . StatusOK , iris . Map {
"Name" : "Iris" ,
"Released" : "13 March 2016" ,
2017-01-02 20:20:17 +01:00
"Stars" : "5883" ,
2016-11-07 14:15:22 +01:00
} )
} )
iris . ListenLETSENCRYPT ( "mydomain.com" )
}
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
package main
import "github.com/kataras/iris"
func main ( ) {
s1 := iris . New ( )
s1 . Get ( "/hi_json" , func ( c * iris . Context ) {
2017-01-02 20:20:17 +01:00
c . JSON ( iris . StatusOK , iris . Map {
2016-11-07 14:15:22 +01:00
"Name" : "Iris" ,
"Released" : "13 March 2016" ,
2017-01-02 20:20:17 +01:00
"Stars" : "5883" ,
2016-11-07 14:15:22 +01:00
} )
} )
s2 := iris . New ( )
s2 . Get ( "/hi_raw_html" , func ( c * iris . Context ) {
c . HTML ( iris . StatusOK , "<b> Iris </b> welcomes <h1>you!</h1>" )
} )
go s1 . Listen ( ":8080" )
s2 . Listen ( ":1993" )
}
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
For middleware , template engines , response engines , sessions , websockets , mails , subdomains ,
dynamic subdomains , routes , party of subdomains & routes , ssh and much more
2017-01-02 20:20:17 +01:00
visit https : //docs.iris-go.com
2016-11-07 14:15:22 +01:00
* /
2016-06-30 16:33:08 +02:00
package iris // import "github.com/kataras/iris"
2016-05-30 16:08:09 +02:00
import (
2016-11-15 19:20:29 +01:00
"bytes"
2016-06-14 07:45:40 +02:00
"fmt"
2016-09-07 06:36:23 +02:00
"log"
2016-09-27 15:28:38 +02:00
"net"
2017-01-02 20:20:17 +01:00
"net/http"
2016-09-07 06:36:23 +02:00
"net/url"
"os"
2016-09-27 15:28:38 +02:00
"os/signal"
2016-09-07 06:36:23 +02:00
"path"
"reflect"
"strconv"
"strings"
"sync"
"time"
2016-10-02 02:30:37 +02:00
"github.com/kataras/go-errors"
"github.com/kataras/go-fs"
"github.com/kataras/go-serializer"
"github.com/kataras/go-sessions"
"github.com/kataras/go-template"
"github.com/kataras/go-template/html"
2016-05-30 16:08:09 +02:00
)
const (
2016-10-21 02:06:50 +02:00
// IsLongTermSupport flag is true when the below version number is a long-term-support version
2016-10-31 20:21:11 +01:00
IsLongTermSupport = false
2016-10-21 02:06:50 +02:00
// Version is the current version number of the Iris web framework
2017-01-04 14:16:53 +01:00
Version = "6.0.1"
2016-07-06 20:24:34 +02:00
banner = ` _____ _
2016-06-06 20:04:38 +02:00
| _ _ | ( _ )
| | ____ _ ___
| | | __ || | / __ |
_ | | _ | | | | \ __ \
2016-07-06 20:24:34 +02:00
| _____ | _ | | _ || ___ / ` + Version + ` `
)
2016-09-27 15:28:38 +02:00
// Default iris instance entry and its public fields, use it with iris.$anyPublicFuncOrField
2016-07-06 20:24:34 +02:00
var (
Default * Framework
2016-09-09 07:09:03 +02:00
Config * Configuration
2016-09-07 06:36:23 +02:00
Logger * log . Logger // if you want colors in your console then you should use this https://github.com/iris-contrib/logger instead.
2016-07-06 20:24:34 +02:00
Plugins PluginContainer
2017-01-02 20:20:17 +01:00
Router http . Handler
2016-09-01 05:01:53 +02:00
Websocket * WebsocketServer
2016-07-06 20:24:34 +02:00
// Available is a channel type of bool, fired to true when the server is opened and all plugins ran
// never fires false, if the .Close called then the channel is re-allocating.
2016-09-27 15:28:38 +02:00
// the channel remains open until you close it.
2016-07-06 20:24:34 +02:00
//
2016-09-27 15:28:38 +02:00
// look at the http_test.go file for a usage example
2016-07-06 20:24:34 +02:00
Available chan bool
2016-05-30 16:08:09 +02:00
)
2016-10-13 16:25:01 +02:00
// ResetDefault resets the iris.Default which is the instance which is used on the default iris station for
// iris.Get(all api functions)
// iris.Config
// iris.Logger
// iris.Plugins
// iris.Router
// iris.Websocket
2016-10-31 07:19:00 +01:00
// iris.Available channel
2016-10-13 16:25:01 +02:00
// useful mostly when you are not using the form of app := iris.New() inside your tests, to make sure that you're using a new iris instance
func ResetDefault ( ) {
2016-07-06 20:24:34 +02:00
Default = New ( )
Config = Default . Config
Logger = Default . Logger
Plugins = Default . Plugins
2016-09-27 15:28:38 +02:00
Router = Default . Router
2016-07-06 20:24:34 +02:00
Websocket = Default . Websocket
Available = Default . Available
}
func init ( ) {
2016-10-13 16:25:01 +02:00
ResetDefault ( )
2016-07-06 20:24:34 +02:00
}
// -------------------------------------------------------------------------------------
// -------------------------------------------------------------------------------------
// --------------------------------Framework implementation-----------------------------
// -------------------------------------------------------------------------------------
// -------------------------------------------------------------------------------------
2016-06-14 07:45:40 +02:00
type (
// FrameworkAPI contains the main Iris Public API
FrameworkAPI interface {
MuxAPI
2016-09-09 07:09:03 +02:00
Set ( ... OptionSetter )
2016-09-16 20:16:48 +02:00
Must ( error )
2016-09-27 15:28:38 +02:00
Build ( )
Serve ( net . Listener ) error
2016-06-14 07:45:40 +02:00
Listen ( string )
ListenTLS ( string , string , string )
2016-10-28 20:21:57 +02:00
ListenLETSENCRYPT ( string , ... string )
2016-06-14 07:45:40 +02:00
ListenUNIX ( string , os . FileMode )
2016-09-27 15:28:38 +02:00
Close ( ) error
Reserve ( ) error
2017-01-02 20:20:17 +01:00
AcquireCtx ( http . ResponseWriter , * http . Request ) * Context
2016-09-18 06:21:35 +02:00
ReleaseCtx ( * Context )
2016-09-27 15:28:38 +02:00
CheckForUpdates ( bool )
2016-09-04 21:02:31 +02:00
UseSessionDB ( sessions . Database )
2016-09-10 06:23:02 +02:00
UseSerializer ( string , serializer . Serializer )
2016-09-02 05:05:44 +02:00
UseTemplate ( template . Engine ) * template . Loader
2016-09-29 16:05:22 +02:00
UsePreRender ( PreRender )
2016-07-07 23:59:00 +02:00
UseGlobal ( ... Handler )
UseGlobalFunc ( ... HandlerFunc )
2016-06-14 07:45:40 +02:00
Lookup ( string ) Route
Lookups ( ) [ ] Route
Path ( string , ... interface { } ) string
URL ( string , ... interface { } ) string
2016-07-13 05:28:09 +02:00
TemplateString ( string , interface { } , ... map [ string ] interface { } ) string
2016-09-09 07:09:03 +02:00
TemplateSourceString ( string , interface { } ) string
2016-09-10 06:23:02 +02:00
SerializeToString ( string , interface { } , ... map [ string ] interface { } ) string
2016-10-31 07:19:00 +01:00
Cache ( HandlerFunc , time . Duration ) HandlerFunc
2016-06-14 07:45:40 +02:00
}
2016-05-30 16:08:09 +02:00
2017-01-02 20:20:17 +01:00
// MuxAPI the visible api for the serveMux
MuxAPI interface {
Party ( string , ... HandlerFunc ) MuxAPI
// middleware serial, appending
Use ( ... Handler ) MuxAPI
UseFunc ( ... HandlerFunc ) MuxAPI
Done ( ... Handler ) MuxAPI
DoneFunc ( ... HandlerFunc ) MuxAPI
// main handlers
Handle ( string , string , ... Handler ) RouteNameFunc
HandleFunc ( string , string , ... HandlerFunc ) RouteNameFunc
API ( string , HandlerAPI , ... HandlerFunc )
// http methods
Get ( string , ... HandlerFunc ) RouteNameFunc
Post ( string , ... HandlerFunc ) RouteNameFunc
Put ( string , ... HandlerFunc ) RouteNameFunc
Delete ( string , ... HandlerFunc ) RouteNameFunc
Connect ( string , ... HandlerFunc ) RouteNameFunc
Head ( string , ... HandlerFunc ) RouteNameFunc
Options ( string , ... HandlerFunc ) RouteNameFunc
Patch ( string , ... HandlerFunc ) RouteNameFunc
Trace ( string , ... HandlerFunc ) RouteNameFunc
Any ( string , ... HandlerFunc )
// static content
StaticServe ( string , ... string ) RouteNameFunc
StaticContent ( string , string , [ ] byte ) RouteNameFunc
StaticEmbedded ( string , string , func ( string ) ( [ ] byte , error ) , func ( ) [ ] string ) RouteNameFunc
Favicon ( string , ... string ) RouteNameFunc
// static file system
StaticHandler ( string , string , bool , bool ) HandlerFunc
StaticWeb ( string , string ) RouteNameFunc
// party layout for template engines
Layout ( string ) MuxAPI
// errors
OnError ( int , HandlerFunc )
EmitError ( int , * Context )
2016-07-06 20:24:34 +02:00
}
2017-01-02 20:20:17 +01:00
// RouteNameFunc the func returns from the MuxAPi's methods, optionally sets the name of the Route (*route)
RouteNameFunc func ( string )
2016-07-06 20:24:34 +02:00
)
2016-06-17 06:18:09 +02:00
2017-01-02 20:20:17 +01:00
// Framework is our God |\| Google.Search('Greek mythology Iris')
//
// Implements the FrameworkAPI
type Framework struct {
* muxAPI
// HTTP Server runtime fields is the iris' defined main server, developer can use unlimited number of servers
// note: they're available after .Build, and .Serve/Listen/ListenTLS/ListenLETSENCRYPT/ListenUNIX
ln net . Listener
srv * http . Server
Available chan bool
//
// Router field which can change the default iris' mux behavior
// if you want to get benefit with iris' context make use of:
// ctx:= iris.AcquireCtx(http.ResponseWriter, *http.Request) to get the context at the beginning of your handler
// iris.ReleaseCtx(ctx) to release/put the context to the pool, at the very end of your custom handler.
Router http . Handler
contextPool sync . Pool
once sync . Once
Config * Configuration
sessions sessions . Sessions
serializers serializer . Serializers
templates * templateEngines
Logger * log . Logger
Plugins PluginContainer
Websocket * WebsocketServer
}
2016-07-06 20:24:34 +02:00
var _ FrameworkAPI = & Framework { }
2016-05-30 16:08:09 +02:00
2016-09-09 07:09:03 +02:00
// New creates and returns a new Iris instance.
//
// Receives (optional) multi options, use iris.Option and your editor should show you the available options to set
// all options are inside ./configuration.go
// example 1: iris.New(iris.OptionIsDevelopment(true), iris.OptionCharset("UTF-8"), irisOptionSessionsCookie("mycookieid"),iris.OptionWebsocketEndpoint("my_endpoint"))
// example 2: iris.New(iris.Configuration{IsDevelopment:true, Charset: "UTF-8", Sessions: iris.SessionsConfiguration{Cookie:"mycookieid"}, Websocket: iris.WebsocketConfiguration{Endpoint:"/my_endpoint"}})
// both ways are totally valid and equal
func New ( setters ... OptionSetter ) * Framework {
s := & Framework { }
s . Set ( setters ... )
2016-09-07 06:36:23 +02:00
2016-10-31 07:19:00 +01:00
// logger & plugins
2016-09-09 07:09:03 +02:00
{
// set the Logger, which it's configuration should be declared before .Listen because the servemux and plugins needs that
s . Logger = log . New ( s . Config . LoggerOut , s . Config . LoggerPreffix , log . LstdFlags )
2016-09-05 02:56:28 +02:00
s . Plugins = newPluginContainer ( s . Logger )
2016-09-09 07:09:03 +02:00
}
// rendering
{
2016-09-10 06:23:02 +02:00
s . serializers = serializer . Serializers { }
2016-07-13 05:28:09 +02:00
// set the templates
2016-09-02 05:05:44 +02:00
s . templates = newTemplateEngines ( map [ string ] interface { } {
"url" : s . URL ,
"urlpath" : s . Path ,
} )
2016-09-09 07:09:03 +02:00
}
2016-09-13 06:54:44 +02:00
// websocket & sessions
2016-09-09 07:09:03 +02:00
{
2017-01-02 20:20:17 +01:00
// in order to be able to call $instance.Websocket.OnConnection
// the whole ws configuration and websocket server is really initialized only on first OnConnection
s . Websocket = NewWebsocketServer ( s )
2016-09-13 06:54:44 +02:00
// set the sessions, look .initialize for its GC
s . sessions = sessions . New ( sessions . DisableAutoGC ( true ) )
2016-09-09 07:09:03 +02:00
}
2016-09-27 15:28:38 +02:00
// routing
2016-09-09 07:09:03 +02:00
{
2016-07-06 20:24:34 +02:00
// set the servemux, which will provide us the public API also, with its context pool
2016-08-14 04:44:36 +02:00
mux := newServeMux ( s . Logger )
2016-10-02 06:28:36 +02:00
mux . setCorrectPath ( ! s . Config . DisablePathCorrection ) // correctPath is re-setted on .Set and after build*
2016-10-02 02:30:37 +02:00
2016-07-07 23:59:00 +02:00
mux . onLookup = s . Plugins . DoPreLookup
2016-09-09 07:09:03 +02:00
s . contextPool . New = func ( ) interface { } {
2016-09-19 00:48:37 +02:00
return & Context { framework : s }
2016-09-09 07:09:03 +02:00
}
2016-09-18 20:57:15 +02:00
// set the public router API (and party)
s . muxAPI = & muxAPI { mux : mux , relativePath : "/" }
2016-09-09 07:09:03 +02:00
s . Available = make ( chan bool )
2016-07-06 20:24:34 +02:00
}
2016-05-30 16:08:09 +02:00
2016-07-06 20:24:34 +02:00
return s
}
2016-06-14 19:29:01 +02:00
2016-09-27 15:28:38 +02:00
// Set sets an option aka configuration field to the default iris instance
func Set ( setters ... OptionSetter ) {
Default . Set ( setters ... )
}
2016-08-17 11:57:54 +02:00
2016-09-27 15:28:38 +02:00
// Set sets an option aka configuration field to this iris instance
func ( s * Framework ) Set ( setters ... OptionSetter ) {
if s . Config == nil {
defaultConfiguration := DefaultConfiguration ( )
s . Config = & defaultConfiguration
2016-08-17 11:57:54 +02:00
}
2016-09-16 09:25:56 +02:00
2016-09-27 15:28:38 +02:00
for _ , setter := range setters {
setter . Set ( s . Config )
2016-09-16 20:16:48 +02:00
}
2016-10-02 02:30:37 +02:00
2016-10-02 06:28:36 +02:00
if s . muxAPI != nil && s . mux != nil { // if called after .New, which it does, correctPath is the only field we need to be updated before .Listen, so:
s . mux . setCorrectPath ( ! s . Config . DisablePathCorrection )
2016-10-02 02:30:37 +02:00
}
2016-07-06 20:24:34 +02:00
}
2016-09-27 15:28:38 +02:00
// Must panics on error, it panics on registed iris' logger
func Must ( err error ) {
Default . Must ( err )
2016-09-18 06:21:35 +02:00
}
2016-09-27 15:28:38 +02:00
// Must panics on error, it panics on registed iris' logger
func ( s * Framework ) Must ( err error ) {
if err != nil {
2017-01-02 20:20:17 +01:00
// s.Logger.Panicf("%s. Trace:\n%s", err, debug.Stack())
s . Logger . Panic ( err )
2016-09-27 15:28:38 +02:00
}
2016-07-06 20:24:34 +02:00
}
2016-09-27 15:28:38 +02:00
// Build builds the whole framework's parts together
// DO NOT CALL IT MANUALLY IF YOU ARE NOT:
2017-01-02 20:20:17 +01:00
// SERVE IRIS BEHIND AN EXTERNAL CUSTOM net/http.Server, CAN BE CALLED ONCE PER IRIS INSTANCE FOR YOUR SAFETY
2016-09-27 15:28:38 +02:00
func Build ( ) {
Default . Build ( )
2016-09-18 05:55:44 +02:00
}
2016-09-27 15:28:38 +02:00
// Build builds the whole framework's parts together
// DO NOT CALL IT MANUALLY IF YOU ARE NOT:
2017-01-04 14:16:53 +01:00
// SERVE IRIS BEHIND AN EXTERNAL CUSTOM nethttp.Server, CAN BE CALLED ONCE PER IRIS INSTANCE FOR YOUR SAFETY
2016-09-27 15:28:38 +02:00
func ( s * Framework ) Build ( ) {
s . once . Do ( func ( ) {
2016-10-11 13:10:19 +02:00
// .Build, normally*, auto-called after station's listener setted but before the real Serve, so here set the host, scheme
// and the mux hostname(*this is here because user may not call .Serve/.Listen functions if listen by a custom server)
if s . Config . VHost == "" { // if not setted by Listen functions
if s . ln != nil { // but user called .Serve
// then take the listener's addr
s . Config . VHost = s . ln . Addr ( ) . String ( )
} else {
// if no .Serve or .Listen called, then the user should set the VHost manually,
// however set it to a default value here for any case
s . Config . VHost = DefaultServerAddr
}
}
// if user didn't specified a scheme then get it from the VHost, which is already setted at before statements
if s . Config . VScheme == "" {
s . Config . VScheme = ParseScheme ( s . Config . VHost )
}
s . Plugins . DoPreBuild ( s ) // once after configuration has been setted. *nothing stops you to change the VHost and VScheme at this point*
2016-09-30 13:02:03 +02:00
// re-nwe logger's attrs
s . Logger . SetPrefix ( s . Config . LoggerPreffix )
s . Logger . SetOutput ( s . Config . LoggerOut )
2016-09-27 15:28:38 +02:00
// prepare the serializers, if not any other serializers setted for the default serializer types(json,jsonp,xml,markdown,text,data) then the defaults are setted:
serializer . RegisterDefaults ( s . serializers )
2016-09-18 05:55:44 +02:00
2016-09-27 15:28:38 +02:00
// prepare the templates if enabled
if ! s . Config . DisableTemplateEngines {
2016-09-18 06:21:35 +02:00
2016-09-27 15:28:38 +02:00
s . templates . Reload = s . Config . IsDevelopment
// check and prepare the templates
if len ( s . templates . Entries ) == 0 { // no template engines were registered, let's use the default
s . UseTemplate ( html . New ( ) )
}
2016-09-18 05:55:44 +02:00
2016-09-27 15:28:38 +02:00
if err := s . templates . Load ( ) ; err != nil {
s . Logger . Panic ( err ) // panic on templates loading before listening if we have an error.
}
2016-09-18 05:55:44 +02:00
}
2016-09-27 15:28:38 +02:00
// init, starts the session manager if the Cookie configuration field is not empty
if s . Config . Sessions . Cookie != "" {
// re-set the configuration field for any case
s . sessions . Set ( s . Config . Sessions , sessions . DisableAutoGC ( false ) )
2016-07-06 20:24:34 +02:00
}
2016-10-02 06:28:36 +02:00
// prepare the mux runtime fields again, for any case
s . mux . setCorrectPath ( ! s . Config . DisablePathCorrection )
2016-10-11 21:35:12 +02:00
s . mux . setFireMethodNotAllowed ( s . Config . FireMethodNotAllowed )
2016-10-02 06:28:36 +02:00
2016-09-27 15:28:38 +02:00
// prepare the server's handler, we do that check because iris supports
// custom routers (you can take the routes registed by iris using iris.Lookups function)
if s . Router == nil {
// build and get the default mux' handler(*Context)
serve := s . mux . BuildHandler ( )
2017-01-02 20:20:17 +01:00
// build the net/http.Handler to bind it to the servers
defaultHandler := http . HandlerFunc ( func ( w http . ResponseWriter , r * http . Request ) {
ctx := s . AcquireCtx ( w , r )
2016-09-27 15:28:38 +02:00
serve ( ctx )
s . ReleaseCtx ( ctx )
2017-01-02 20:20:17 +01:00
} )
2016-07-06 20:24:34 +02:00
2016-09-27 15:28:38 +02:00
s . Router = defaultHandler
}
2016-07-06 20:24:34 +02:00
2016-09-27 15:28:38 +02:00
// set the mux' hostname (for multi subdomain routing)
s . mux . hostname = ParseHostname ( s . Config . VHost )
if s . ln != nil { // user called Listen functions or Serve,
// create the main server
2017-01-02 20:20:17 +01:00
s . srv = & http . Server {
ReadTimeout : s . Config . ReadTimeout ,
WriteTimeout : s . Config . WriteTimeout ,
MaxHeaderBytes : s . Config . MaxHeaderBytes ,
TLSNextProto : s . Config . TLSNextProto ,
ConnState : s . Config . ConnState ,
Handler : s . Router ,
Addr : s . Config . VHost ,
2016-09-29 19:17:42 +02:00
}
2017-01-02 20:20:17 +01:00
if s . Config . TLSNextProto != nil {
s . srv . TLSNextProto = s . Config . TLSNextProto
}
if s . Config . ConnState != nil {
s . srv . ConnState = s . Config . ConnState
2016-09-27 15:28:38 +02:00
}
}
2016-06-14 07:45:40 +02:00
2016-09-27 15:28:38 +02:00
// updates, to cover the default station's irs.Config.checkForUpdates
// note: we could use the IsDevelopment configuration field to do that BUT
// the developer may want to check for updates without, for example, re-build template files (comes from IsDevelopment) on each request
if s . Config . CheckForUpdatesSync {
s . CheckForUpdates ( false )
} else if s . Config . CheckForUpdates {
go s . CheckForUpdates ( false )
}
} )
2016-09-16 20:16:48 +02:00
}
2016-09-27 15:28:38 +02:00
var (
errServerAlreadyStarted = errors . New ( "Server is already started and listening" )
2016-09-16 20:16:48 +02:00
)
2016-09-27 15:28:38 +02:00
// Serve serves incoming connections from the given listener.
//
// Serve blocks until the given listener returns permanent error.
func Serve ( ln net . Listener ) error {
return Default . Serve ( ln )
2016-09-16 20:16:48 +02:00
}
2016-09-27 15:28:38 +02:00
// Serve serves incoming connections from the given listener.
//
// Serve blocks until the given listener returns permanent error.
func ( s * Framework ) Serve ( ln net . Listener ) error {
if s . IsRunning ( ) {
return errServerAlreadyStarted
}
// maybe a 'race' here but user should not call .Serve more than one time especially in more than one go routines...
s . ln = ln
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
s . Build ( )
s . Plugins . DoPreListen ( s )
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
// This didn't helped me ,here, but maybe can help you:
// https://www.oreilly.com/learning/run-strikingly-fast-parallel-file-searches-in-go-with-sync-errgroup?utm_source=golangweekly&utm_medium=email
// new experimental package: errgroup
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
defer func ( ) {
if err := recover ( ) ; err != nil {
s . Logger . Panic ( err )
2016-09-16 20:16:48 +02:00
}
2016-09-27 15:28:38 +02:00
} ( )
// start the server in goroutine, .Available will block instead
2017-01-02 20:20:17 +01:00
go func ( ) { s . Must ( s . srv . Serve ( ln ) ) } ( )
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
if ! s . Config . DisableBanner {
bannerMessage := fmt . Sprintf ( "%s: Running at %s" , time . Now ( ) . Format ( s . Config . TimeFormat ) , s . Config . VHost )
// we don't print it via Logger because:
// 1. The banner is only 'useful' when the developer logs to terminal and no file
// 2. Prefix & LstdFlags options of the default s.Logger
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
fmt . Printf ( "%s\n\n%s\n" , banner , bannerMessage )
2016-09-16 20:16:48 +02:00
}
2016-09-27 15:28:38 +02:00
s . Plugins . DoPostListen ( s )
2016-09-16 20:16:48 +02:00
2016-09-27 15:28:38 +02:00
go func ( ) { s . Available <- true } ( )
2016-10-04 00:18:17 +02:00
ch := make ( chan os . Signal , 1 )
signal . Notify ( ch , os . Interrupt )
2016-09-27 15:28:38 +02:00
<- ch
if err := s . Close ( ) ; err != nil {
2016-10-04 00:18:17 +02:00
if s . Config . IsDevelopment {
s . Logger . Printf ( "Error while closing the server: %s\n" , err )
}
2016-09-27 15:28:38 +02:00
return err
2016-06-14 07:45:40 +02:00
}
2016-09-27 15:28:38 +02:00
os . Exit ( 1 )
return nil
2016-07-05 15:26:47 +02:00
}
2016-06-14 07:45:40 +02:00
// Listen starts the standalone http server
// which listens to the addr parameter which as the form of
// host:port
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
2016-06-14 07:45:40 +02:00
func Listen ( addr string ) {
Default . Listen ( addr )
}
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
// Listen starts the standalone http server
// which listens to the addr parameter which as the form of
// host:port
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
2016-06-14 07:45:40 +02:00
func ( s * Framework ) Listen ( addr string ) {
2016-09-27 15:28:38 +02:00
addr = ParseHost ( addr )
if s . Config . VHost == "" {
s . Config . VHost = addr
// this will be set as the front-end listening addr
}
2017-01-04 14:16:53 +01:00
ln , err := TCPKeepAlive ( addr )
2016-09-27 15:28:38 +02:00
if err != nil {
s . Logger . Panic ( err )
}
s . Must ( s . Serve ( ln ) )
2016-06-14 07:45:40 +02:00
}
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
// ListenTLS Starts a https server with certificates,
// if you use this method the requests of the form of 'http://' will fail
// only https:// connections are allowed
// which listens to the addr parameter which as the form of
// host:port
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
2016-09-09 07:09:03 +02:00
// ex: iris.ListenTLS(":8080","yourfile.cert","yourfile.key")
2016-06-14 07:45:40 +02:00
func ListenTLS ( addr string , certFile string , keyFile string ) {
Default . ListenTLS ( addr , certFile , keyFile )
}
2016-06-06 12:25:09 +02:00
2016-06-14 07:45:40 +02:00
// ListenTLS Starts a https server with certificates,
// if you use this method the requests of the form of 'http://' will fail
// only https:// connections are allowed
// which listens to the addr parameter which as the form of
// host:port
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
2016-09-09 07:09:03 +02:00
// ex: iris.ListenTLS(":8080","yourfile.cert","yourfile.key")
2016-06-14 07:45:40 +02:00
func ( s * Framework ) ListenTLS ( addr string , certFile , keyFile string ) {
2016-09-27 15:28:38 +02:00
addr = ParseHost ( addr )
if s . Config . VHost == "" {
s . Config . VHost = addr
// this will be set as the front-end listening addr
}
ln , err := TLS ( addr , certFile , keyFile )
if err != nil {
s . Logger . Panic ( err )
2016-07-07 02:36:48 +02:00
}
2016-09-27 15:28:38 +02:00
s . Must ( s . Serve ( ln ) )
2016-06-14 07:45:40 +02:00
}
2016-06-06 12:25:09 +02:00
2016-09-27 15:28:38 +02:00
// ListenLETSENCRYPT starts a server listening at the specific nat address
2016-07-29 00:33:20 +02:00
// using key & certification taken from the letsencrypt.org 's servers
2016-09-27 15:28:38 +02:00
// it's also starts a second 'http' server to redirect all 'http://$ADDR_HOSTNAME:80' to the' https://$ADDR'
2016-10-28 20:21:57 +02:00
// it creates a cache file to store the certifications, for performance reasons, this file by-default is "./letsencrypt.cache"
// if you skip the second parameter then the cache file is "./letsencrypt.cache"
2016-10-28 20:55:00 +02:00
// if you want to disable cache then simple pass as second argument an empty empty string ""
2016-10-28 20:21:57 +02:00
//
2016-11-13 10:12:59 +01:00
// example: https://github.com/iris-contrib/examples/blob/master/letsencrypt/main.go
2016-10-28 20:21:57 +02:00
//
// supports localhost domains for testing,
// NOTE: if you are ready for production then use `$app.Serve(iris.LETSENCRYPTPROD("mydomain.com"))` instead
func ListenLETSENCRYPT ( addr string , cacheFileOptional ... string ) {
Default . ListenLETSENCRYPT ( addr , cacheFileOptional ... )
2016-09-27 15:28:38 +02:00
}
// ListenLETSENCRYPT starts a server listening at the specific nat address
// using key & certification taken from the letsencrypt.org 's servers
// it's also starts a second 'http' server to redirect all 'http://$ADDR_HOSTNAME:80' to the' https://$ADDR'
2016-10-28 20:21:57 +02:00
// it creates a cache file to store the certifications, for performance reasons, this file by-default is "./letsencrypt.cache"
// if you skip the second parameter then the cache file is "./letsencrypt.cache"
2016-10-28 20:55:00 +02:00
// if you want to disable cache then simple pass as second argument an empty empty string ""
2016-10-28 20:21:57 +02:00
//
2016-11-13 10:12:59 +01:00
// example: https://github.com/iris-contrib/examples/blob/master/letsencrypt/main.go
2016-10-28 20:21:57 +02:00
//
// supports localhost domains for testing,
// NOTE: if you are ready for production then use `$app.Serve(iris.LETSENCRYPTPROD("mydomain.com"))` instead
func ( s * Framework ) ListenLETSENCRYPT ( addr string , cacheFileOptional ... string ) {
2016-09-27 15:28:38 +02:00
addr = ParseHost ( addr )
if s . Config . VHost == "" {
s . Config . VHost = addr
// this will be set as the front-end listening addr
}
2016-10-28 20:21:57 +02:00
ln , err := LETSENCRYPT ( addr , cacheFileOptional ... )
2016-09-27 15:28:38 +02:00
if err != nil {
s . Logger . Panic ( err )
2016-07-29 00:33:20 +02:00
}
2017-01-04 14:16:53 +01:00
// starts a second server which listening on HOST:80 to redirect all requests to the HTTPS://HOST:PORT
Proxy ( ParseHostname ( addr ) + ":80" , "https://" + addr )
2016-09-27 15:28:38 +02:00
s . Must ( s . Serve ( ln ) )
2016-07-29 00:33:20 +02:00
}
2016-06-14 07:45:40 +02:00
// ListenUNIX starts the process of listening to the new requests using a 'socket file', this works only on unix
2016-07-07 02:36:48 +02:00
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
2016-09-09 07:09:03 +02:00
// ex: iris.ListenUNIX(":8080", Mode: os.FileMode)
2016-06-14 07:45:40 +02:00
func ListenUNIX ( addr string , mode os . FileMode ) {
Default . ListenUNIX ( addr , mode )
}
// ListenUNIX starts the process of listening to the new requests using a 'socket file', this works only on unix
2016-06-30 04:58:04 +02:00
//
2016-09-27 15:28:38 +02:00
// It panics on error if you need a func to return an error, use the Serve
// ex: iris.ListenUNIX(":8080", Mode: os.FileMode)
2016-07-07 02:36:48 +02:00
func ( s * Framework ) ListenUNIX ( addr string , mode os . FileMode ) {
2016-09-27 15:28:38 +02:00
// *on unix listen we don't parse the host, because sometimes it causes problems to the user
if s . Config . VHost == "" {
s . Config . VHost = addr
// this will be set as the front-end listening addr
}
ln , err := UNIX ( addr , mode )
if err != nil {
s . Logger . Panic ( err )
}
2016-07-05 14:29:32 +02:00
2016-09-27 15:28:38 +02:00
s . Must ( s . Serve ( ln ) )
2016-07-05 14:29:32 +02:00
}
2016-09-27 15:28:38 +02:00
// IsRunning returns true if server is running
func IsRunning ( ) bool {
return Default . IsRunning ( )
}
2016-07-05 14:29:32 +02:00
2016-09-27 15:28:38 +02:00
// IsRunning returns true if server is running
func ( s * Framework ) IsRunning ( ) bool {
return s != nil && s . ln != nil && s . ln . Addr ( ) != nil && s . ln . Addr ( ) . String ( ) != ""
}
2016-07-06 20:24:34 +02:00
2016-07-07 02:36:48 +02:00
// Close terminates all the registered servers and returns an error if any
// if you want to panic on this error use the iris.Must(iris.Close())
func Close ( ) error {
return Default . Close ( )
2016-06-14 07:45:40 +02:00
}
2016-07-07 02:36:48 +02:00
// Close terminates all the registered servers and returns an error if any
// if you want to panic on this error use the iris.Must(iris.Close())
func ( s * Framework ) Close ( ) error {
2016-09-27 15:28:38 +02:00
if s . IsRunning ( ) {
s . Plugins . DoPreClose ( s )
s . Available = make ( chan bool )
return s . ln . Close ( )
}
return nil
}
// Reserve re-starts the server using the last .Serve's listener
func Reserve ( ) error {
return Default . Reserve ( )
}
// Reserve re-starts the server using the last .Serve's listener
func ( s * Framework ) Reserve ( ) error {
return s . Serve ( s . ln )
}
// AcquireCtx gets an Iris' Context from pool
// see .ReleaseCtx & .Serve
2017-01-02 20:20:17 +01:00
func AcquireCtx ( w http . ResponseWriter , r * http . Request ) * Context {
return Default . AcquireCtx ( w , r )
2016-09-27 15:28:38 +02:00
}
// AcquireCtx gets an Iris' Context from pool
// see .ReleaseCtx & .Serve
2017-01-02 20:20:17 +01:00
func ( s * Framework ) AcquireCtx ( w http . ResponseWriter , r * http . Request ) * Context {
2016-09-27 15:28:38 +02:00
ctx := s . contextPool . Get ( ) . ( * Context ) // Changed to use the pool's New 09/07/2016, ~ -4k nanoseconds(9 bench tests) per requests (better performance)
2017-01-02 20:20:17 +01:00
ctx . ResponseWriter = acquireResponseWriter ( w )
ctx . Request = r
2016-09-27 15:28:38 +02:00
return ctx
}
// ReleaseCtx puts the Iris' Context back to the pool in order to be re-used
// see .AcquireCtx & .Serve
func ReleaseCtx ( ctx * Context ) {
Default . ReleaseCtx ( ctx )
}
// ReleaseCtx puts the Iris' Context back to the pool in order to be re-used
// see .AcquireCtx & .Serve
func ( s * Framework ) ReleaseCtx ( ctx * Context ) {
2017-01-04 14:16:53 +01:00
// flush the body (on recorder) or just the status code (on basic response writer)
// when all finished
2017-01-02 20:20:17 +01:00
ctx . ResponseWriter . flushResponse ( )
2016-10-13 16:25:01 +02:00
ctx . Middleware = nil
2016-09-27 15:28:38 +02:00
ctx . session = nil
2017-01-02 20:20:17 +01:00
ctx . Request = nil
2017-01-04 14:16:53 +01:00
///TODO:
ctx . ResponseWriter . releaseMe ( )
2017-01-02 20:20:17 +01:00
ctx . values . Reset ( )
2016-09-27 15:28:38 +02:00
s . contextPool . Put ( ctx )
}
// global once because is not necessary to check for updates on more than one iris station*
var updateOnce sync . Once
const (
githubOwner = "kataras"
githubRepo = "iris"
)
// CheckForUpdates will try to search for newer version of Iris based on the https://github.com/kataras/iris/releases
// If a newer version found then the app will ask the he dev/user if want to update the 'x' version
// if 'y' is pressed then the updater will try to install the latest version
// the updater, will notify the dev/user that the update is finished and should restart the App manually.
func CheckForUpdates ( force bool ) {
Default . CheckForUpdates ( force )
}
// CheckForUpdates will try to search for newer version of Iris based on the https://github.com/kataras/iris/releases
// If a newer version found then the app will ask the he dev/user if want to update the 'x' version
// if 'y' is pressed then the updater will try to install the latest version
// the updater, will notify the dev/user that the update is finished and should restart the App manually.
// Note: exported func CheckForUpdates exists because of the reason that an update can be executed while Iris is running
func ( s * Framework ) CheckForUpdates ( force bool ) {
updated := false
checker := func ( ) {
writer := s . Config . LoggerOut
if writer == nil {
writer = os . Stdout // we need a writer because the update process will not be silent.
}
fs . DefaultUpdaterAlreadyInstalledMessage = "INFO: Running with the latest version(%s)\n"
updater , err := fs . GetUpdater ( githubOwner , githubRepo , Version )
if err != nil {
writer . Write ( [ ] byte ( "Update failed: " + err . Error ( ) ) )
2016-12-31 13:40:19 +01:00
return
2016-09-27 15:28:38 +02:00
}
updated = updater . Run ( fs . Stdout ( writer ) , fs . Stderr ( writer ) , fs . Silent ( false ) )
}
if force {
checker ( )
} else {
updateOnce . Do ( checker )
}
if updated { // if updated, then do not run the web server
if s . Logger != nil {
s . Logger . Println ( "exiting now..." )
}
os . Exit ( 1 )
}
2016-06-16 19:08:59 +02:00
}
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
// UseSessionDB registers a session database, you can register more than one
// accepts a session database which implements a Load(sid string) map[string]interface{} and an Update(sid string, newValues map[string]interface{})
// the only reason that a session database will be useful for you is when you want to keep the session's values/data after the app restart
// a session database doesn't have write access to the session, it doesn't accept the context, so forget 'cookie database' for sessions, I will never allow that, for your protection.
//
// Note: Don't worry if no session database is registered, your context.Session will continue to work.
2016-09-04 21:02:31 +02:00
func UseSessionDB ( db sessions . Database ) {
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
Default . UseSessionDB ( db )
}
2016-07-13 05:28:09 +02:00
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
// UseSessionDB registers a session database, you can register more than one
// accepts a session database which implements a Load(sid string) map[string]interface{} and an Update(sid string, newValues map[string]interface{})
// the only reason that a session database will be useful for you is when you want to keep the session's values/data after the app restart
// a session database doesn't have write access to the session, it doesn't accept the context, so forget 'cookie database' for sessions, I will never allow that, for your protection.
//
// Note: Don't worry if no session database is registered, your context.Session will continue to work.
2016-09-04 21:02:31 +02:00
func ( s * Framework ) UseSessionDB ( db sessions . Database ) {
s . sessions . UseDatabase ( db )
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
}
2016-07-13 05:28:09 +02:00
2016-09-10 06:23:02 +02:00
// UseSerializer accepts a Serializer and the key or content type on which the developer wants to register this serializer
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// the gzip and charset are automatically supported by Iris, by passing the iris.RenderOptions{} map on the context.Render
// context.Render renders this response or a template engine if no response engine with the 'key' found
// with these engines you can inject the context.JSON,Text,Data,JSONP,XML also
2016-09-10 06:23:02 +02:00
// to do that just register with UseSerializer(mySerializer,"application/json") and so on
// look at the https://github.com/kataras/go-serializer for examples
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
//
2016-09-10 06:23:02 +02:00
// if more than one serializer with the same key/content type exists then the results will be appended to the final request's body
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// this allows the developer to be able to create 'middleware' responses engines
//
// Note: if you pass an engine which contains a dot('.') as key, then the engine will not be registered.
// you don't have to import and use github.com/iris-contrib/json, jsonp, xml, data, text, markdown
// because iris uses these by default if no other response engine is registered for these content types
2016-09-10 06:23:02 +02:00
func UseSerializer ( forContentType string , e serializer . Serializer ) {
Default . UseSerializer ( forContentType , e )
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
}
2016-09-10 06:23:02 +02:00
// UseSerializer accepts a Serializer and the key or content type on which the developer wants to register this serializer
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// the gzip and charset are automatically supported by Iris, by passing the iris.RenderOptions{} map on the context.Render
// context.Render renders this response or a template engine if no response engine with the 'key' found
// with these engines you can inject the context.JSON,Text,Data,JSONP,XML also
2016-09-10 06:23:02 +02:00
// to do that just register with UseSerializer(mySerializer,"application/json") and so on
// look at the https://github.com/kataras/go-serializer for examples
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
//
2016-09-10 06:23:02 +02:00
// if more than one serializer with the same key/content type exists then the results will be appended to the final request's body
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// this allows the developer to be able to create 'middleware' responses engines
//
// Note: if you pass an engine which contains a dot('.') as key, then the engine will not be registered.
// you don't have to import and use github.com/iris-contrib/json, jsonp, xml, data, text, markdown
// because iris uses these by default if no other response engine is registered for these content types
2016-09-10 06:23:02 +02:00
func ( s * Framework ) UseSerializer ( forContentType string , e serializer . Serializer ) {
s . serializers . For ( forContentType , e )
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
}
2016-09-29 16:05:22 +02:00
// UsePreRender adds a Template's PreRender
// PreRender is typeof func(*iris.Context, filenameOrSource string, binding interface{}, options ...map[string]interface{}) bool
// PreRenders helps developers to pass middleware between the route Handler and a context.Render call
// all parameter receivers can be changed before passing it to the actual context's Render
// so, you can change the filenameOrSource, the page binding, the options, and even add cookies, session value or a flash message through ctx
// the return value of a PreRender is a boolean, if returns false then the next PreRender will not be executed, keep note
// that the actual context's Render will be called at any case.
func UsePreRender ( pre PreRender ) {
Default . UsePreRender ( pre )
}
// UsePreRender adds a Template's PreRender
// PreRender is typeof func(*iris.Context, filenameOrSource string, binding interface{}, options ...map[string]interface{}) bool
// PreRenders helps developers to pass middleware between the route Handler and a context.Render call
// all parameter receivers can be changed before passing it to the actual context's Render
// so, you can change the filenameOrSource, the page binding, the options, and even add cookies, session value or a flash message through ctx
// the return value of a PreRender is a boolean, if returns false then the next PreRender will not be executed, keep note
// that the actual context's Render will be called at any case.
func ( s * Framework ) UsePreRender ( pre PreRender ) {
s . templates . usePreRender ( pre )
}
2016-07-13 13:05:32 +02:00
// UseTemplate adds a template engine to the iris view system
2016-07-13 05:28:09 +02:00
// it does not build/load them yet
2016-09-02 05:05:44 +02:00
func UseTemplate ( e template . Engine ) * template . Loader {
2016-07-13 13:05:32 +02:00
return Default . UseTemplate ( e )
2016-07-13 05:28:09 +02:00
}
2016-07-13 13:05:32 +02:00
// UseTemplate adds a template engine to the iris view system
2016-07-13 05:28:09 +02:00
// it does not build/load them yet
2016-09-02 05:05:44 +02:00
func ( s * Framework ) UseTemplate ( e template . Engine ) * template . Loader {
return s . templates . AddEngine ( e )
2016-07-13 05:28:09 +02:00
}
2016-07-07 23:59:00 +02:00
// UseGlobal registers Handler middleware to the beginning, prepends them instead of append
2016-06-17 06:18:09 +02:00
//
// Use it when you want to add a global middleware to all parties, to all routes in all subdomains
// It can be called after other, (but before .Listen of course)
2016-07-07 23:59:00 +02:00
func UseGlobal ( handlers ... Handler ) {
2016-07-13 13:05:32 +02:00
Default . UseGlobal ( handlers ... )
2016-06-17 06:18:09 +02:00
}
2016-07-07 23:59:00 +02:00
// UseGlobalFunc registers HandlerFunc middleware to the beginning, prepends them instead of append
2016-06-17 06:18:09 +02:00
//
// Use it when you want to add a global middleware to all parties, to all routes in all subdomains
// It can be called after other, (but before .Listen of course)
2016-07-07 23:59:00 +02:00
func UseGlobalFunc ( handlersFn ... HandlerFunc ) {
2016-07-13 13:05:32 +02:00
Default . UseGlobalFunc ( handlersFn ... )
2016-06-17 06:18:09 +02:00
}
2016-07-07 23:59:00 +02:00
// UseGlobal registers Handler middleware to the beginning, prepends them instead of append
2016-06-17 06:18:09 +02:00
//
// Use it when you want to add a global middleware to all parties, to all routes in all subdomains
// It can be called after other, (but before .Listen of course)
2016-07-07 23:59:00 +02:00
func ( s * Framework ) UseGlobal ( handlers ... Handler ) {
2016-06-17 06:18:09 +02:00
for _ , r := range s . mux . lookups {
r . middleware = append ( handlers , r . middleware ... )
}
}
2016-07-07 23:59:00 +02:00
// UseGlobalFunc registers HandlerFunc middleware to the beginning, prepends them instead of append
2016-06-17 06:18:09 +02:00
//
// Use it when you want to add a global middleware to all parties, to all routes in all subdomains
// It can be called after other, (but before .Listen of course)
2016-07-07 23:59:00 +02:00
func ( s * Framework ) UseGlobalFunc ( handlersFn ... HandlerFunc ) {
2016-07-13 13:05:32 +02:00
s . UseGlobal ( convertToHandlers ( handlersFn ) ... )
2016-06-17 06:18:09 +02:00
}
2016-06-14 07:45:40 +02:00
// Lookup returns a registed route by its name
func Lookup ( routeName string ) Route {
return Default . Lookup ( routeName )
}
// Lookups returns all registed routes
func Lookups ( ) [ ] Route {
return Default . Lookups ( )
}
// Lookup returns a registed route by its name
func ( s * Framework ) Lookup ( routeName string ) Route {
2016-08-16 12:17:26 +02:00
r := s . mux . lookup ( routeName )
if nil == r {
return nil
}
return r
2016-06-14 07:45:40 +02:00
}
// Lookups returns all registed routes
func ( s * Framework ) Lookups ( ) ( routes [ ] Route ) {
// silly but...
for i := range s . mux . lookups {
routes = append ( routes , s . mux . lookups [ i ] )
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
return
}
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
// Path used to check arguments with the route's named parameters and return the correct url
// if parse failed returns empty string
func Path ( routeName string , args ... interface { } ) string {
return Default . Path ( routeName , args ... )
2016-05-30 16:08:09 +02:00
}
2016-12-12 11:18:59 +01:00
func joinPathArguments ( args ... interface { } ) [ ] interface { } {
arguments := args [ 0 : ]
for i , v := range arguments {
if arr , ok := v . ( [ ] string ) ; ok {
if len ( arr ) > 0 {
interfaceArr := make ( [ ] interface { } , len ( arr ) )
for j , sv := range arr {
interfaceArr [ j ] = sv
}
// replace the current slice
// with the first string element (always as interface{})
arguments [ i ] = interfaceArr [ 0 ]
// append the rest of them to the slice itself
// the range is not affected by these things in go,
// so we are safe to do it.
arguments = append ( args , interfaceArr [ 1 : ] ... )
}
}
}
return arguments
}
2016-06-14 07:45:40 +02:00
// Path used to check arguments with the route's named parameters and return the correct url
// if parse failed returns empty string
func ( s * Framework ) Path ( routeName string , args ... interface { } ) string {
r := s . mux . lookup ( routeName )
if r == nil {
return ""
}
argsLen := len ( args )
// we have named parameters but arguments not given
if argsLen == 0 && r . formattedParts > 0 {
return ""
} else if argsLen == 0 && r . formattedParts == 0 {
// it's static then just return the path
return r . path
}
// we have arguments but they are much more than the named parameters
// 1 check if we have /*, if yes then join all arguments to one as path and pass that as parameter
if argsLen > r . formattedParts {
if r . path [ len ( r . path ) - 1 ] == matchEverythingByte {
// we have to convert each argument to a string in this case
argsString := make ( [ ] string , argsLen , argsLen )
for i , v := range args {
if s , ok := v . ( string ) ; ok {
argsString [ i ] = s
} else if num , ok := v . ( int ) ; ok {
argsString [ i ] = strconv . Itoa ( num )
} else if b , ok := v . ( bool ) ; ok {
argsString [ i ] = strconv . FormatBool ( b )
} else if arr , ok := v . ( [ ] string ) ; ok {
if len ( arr ) > 0 {
argsString [ i ] = arr [ 0 ]
argsString = append ( argsString , arr [ 1 : ] ... )
}
}
}
parameter := strings . Join ( argsString , slash )
result := fmt . Sprintf ( r . formattedPath , parameter )
return result
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
// 2 if !1 return false
return ""
2016-05-30 16:08:09 +02:00
}
2016-12-12 11:18:59 +01:00
arguments := joinPathArguments ( args ... )
2016-06-14 07:45:40 +02:00
return fmt . Sprintf ( r . formattedPath , arguments ... )
}
2017-01-02 20:20:17 +01:00
// DecodeQuery returns the uri parameter as url (string)
2016-07-21 19:33:00 +02:00
// useful when you want to pass something to a database and be valid to retrieve it via context.Param
// use it only for special cases, when the default behavior doesn't suits you.
//
// http://www.blooberry.com/indexdot/html/topics/urlencoding.htm
// it uses just the url.QueryUnescape
2017-01-02 20:20:17 +01:00
func DecodeQuery ( path string ) string {
if path == "" {
2016-07-21 19:33:00 +02:00
return ""
}
2017-01-02 20:20:17 +01:00
encodedPath , err := url . QueryUnescape ( path )
if err != nil {
return path
}
2016-07-21 19:33:00 +02:00
return encodedPath
}
2017-01-02 20:20:17 +01:00
// DecodeURL returns the decoded uri
2016-07-21 19:33:00 +02:00
// useful when you want to pass something to a database and be valid to retrieve it via context.Param
// use it only for special cases, when the default behavior doesn't suits you.
//
// http://www.blooberry.com/indexdot/html/topics/urlencoding.htm
2017-01-02 20:20:17 +01:00
// it uses just the url.Parse
func DecodeURL ( uri string ) string {
u , err := url . Parse ( uri )
if err != nil {
2016-07-21 19:33:00 +02:00
return ""
}
2017-01-02 20:20:17 +01:00
return u . String ( )
2016-07-21 19:33:00 +02:00
}
2016-06-14 07:45:40 +02:00
// URL returns the subdomain+ host + Path(...optional named parameters if route is dynamic)
// returns an empty string if parse is failed
func URL ( routeName string , args ... interface { } ) ( url string ) {
return Default . URL ( routeName , args ... )
2016-06-01 14:29:38 +02:00
}
2016-06-14 07:45:40 +02:00
// URL returns the subdomain+ host + Path(...optional named parameters if route is dynamic)
// returns an empty string if parse is failed
func ( s * Framework ) URL ( routeName string , args ... interface { } ) ( url string ) {
r := s . mux . lookup ( routeName )
if r == nil {
return
}
2016-09-27 15:28:38 +02:00
scheme := s . Config . VScheme // if s.Config.VScheme was setted, that will be used instead of the real, in order to make easy to run behind nginx
host := s . Config . VHost // if s.Config.VHost was setted, that will be used instead of the real, in order to make easy to run behind nginx
2016-12-12 11:18:59 +01:00
arguments := joinPathArguments ( args ... )
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
// if it's dynamic subdomain then the first argument is the subdomain part
if r . subdomain == dynamicSubdomainIndicator {
if len ( arguments ) == 0 { // it's a wildcard subdomain but not arguments
return
}
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
if subdomain , ok := arguments [ 0 ] . ( string ) ; ok {
host = subdomain + "." + host
} else {
// it is not array because we join them before. if not pass a string then this is not a subdomain part, return empty uri
return
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
arguments = arguments [ 1 : ]
2016-05-30 16:08:09 +02:00
}
2016-07-03 10:38:03 +02:00
if parsedPath := s . Path ( routeName , arguments ... ) ; parsedPath != "" {
2016-06-14 07:45:40 +02:00
url = scheme + host + parsedPath
2016-06-06 20:04:38 +02:00
}
2016-06-14 07:45:40 +02:00
return
}
2016-05-30 16:08:09 +02:00
2016-07-13 05:28:09 +02:00
// TemplateString executes a template from the default template engine and returns its result as string, useful when you want it for sending rich e-mails
2016-06-14 07:45:40 +02:00
// returns empty string on error
2016-07-13 05:28:09 +02:00
func TemplateString ( templateFile string , pageContext interface { } , options ... map [ string ] interface { } ) string {
return Default . TemplateString ( templateFile , pageContext , options ... )
2016-06-14 07:45:40 +02:00
}
2016-05-30 16:08:09 +02:00
2016-07-13 05:28:09 +02:00
// TemplateString executes a template from the default template engine and returns its result as string, useful when you want it for sending rich e-mails
2016-06-14 07:45:40 +02:00
// returns empty string on error
2016-07-13 05:28:09 +02:00
func ( s * Framework ) TemplateString ( templateFile string , pageContext interface { } , options ... map [ string ] interface { } ) string {
if s . Config . DisableTemplateEngines {
return ""
}
2016-09-02 05:05:44 +02:00
res , err := s . templates . ExecuteString ( templateFile , pageContext , options ... )
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
if err != nil {
return ""
}
return res
}
2016-09-09 07:09:03 +02:00
// TemplateSourceString executes a template source(raw string contents) from the first template engines which supports raw parsing returns its result as string,
// useful when you want it for sending rich e-mails
// returns empty string on error
func TemplateSourceString ( src string , pageContext interface { } ) string {
return Default . TemplateSourceString ( src , pageContext )
}
// TemplateSourceString executes a template source(raw string contents) from the first template engines which supports raw parsing returns its result as string,
// useful when you want it for sending rich e-mails
// returns empty string on error
func ( s * Framework ) TemplateSourceString ( src string , pageContext interface { } ) string {
if s . Config . DisableTemplateEngines {
return ""
}
res , err := s . templates . ExecuteRawString ( src , pageContext )
if err != nil {
res = ""
}
return res
}
2016-09-10 06:23:02 +02:00
// SerializeToString returns the string of a serializer,
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// does not render it to the client
// returns empty string on error
2016-09-10 06:23:02 +02:00
func SerializeToString ( keyOrContentType string , obj interface { } , options ... map [ string ] interface { } ) string {
return Default . SerializeToString ( keyOrContentType , obj , options ... )
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
}
2016-09-10 06:23:02 +02:00
// SerializeToString returns the string of a serializer,
Update to 4.0.0-alpha.3 - Response Engines, 'inject' the context.JSON/JSONP/Text/Data/Markdown/Render, Read HISTORY.md
## 4.0.0-alpha.2 -> 4.0.0-alpha.3
**New**
A **Response Engine** gives you the freedom to create/change the
render/response writer for
- `context.JSON`
- `context.JSONP`
- `context.XML`
- `context.Text`
- `context.Markdown`
- `context.Data`
- `context.Render("my_custom_type",mystructOrData{},
iris.RenderOptions{"gzip":false,"charset":"UTF-8"})`
- `context.MarkdownString`
- `iris.ResponseString(...)`
**Fix**
- https://github.com/kataras/iris/issues/294
**Small changes**
- `iris.Config.Charset`, before alpha.3 was `iris.Config.Rest.Charset` &
`iris.Config.Render.Template.Charset`, but you can override it at
runtime by passinth a map `iris.RenderOptions` on the `context.Render`
call .
- `iris.Config.IsDevelopment` , before alpha.1 was
`iris.Config.Render.Template.IsDevelopment`
**Websockets changes**
No need to import the `github.com/kataras/iris/websocket` to use the
`Connection` iteral, the websocket moved inside `kataras/iris` , now all
exported variables' names have the prefix of `Websocket`, so the old
`websocket.Connection` is now `iris.WebsocketConnection`.
Generally, no other changes on the 'frontend API', for response engines
examples and how you can register your own to add more features on
existing response engines or replace them, look
[here](https://github.com/iris-contrib/response).
**BAD SIDE**: E-Book is still pointing on the v3 release, but will be
updated soon.
2016-07-18 16:40:42 +02:00
// does not render it to the client
// returns empty string on error
2016-09-10 06:23:02 +02:00
func ( s * Framework ) SerializeToString ( keyOrContentType string , obj interface { } , options ... map [ string ] interface { } ) string {
res , err := s . serializers . SerializeToString ( keyOrContentType , obj , options ... )
2016-06-14 07:45:40 +02:00
if err != nil {
2016-10-27 02:17:09 +02:00
if s . Config . IsDevelopment {
s . Logger . Printf ( "Error on SerializeToString, Key(content-type): %s. Trace: %s\n" , keyOrContentType , err )
}
2016-06-14 07:45:40 +02:00
return ""
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
return res
}
2016-05-30 16:08:09 +02:00
2016-10-31 07:19:00 +01:00
// Cache is just a wrapper for a route's handler which you want to enable body caching
// Usage: iris.Get("/", iris.Cache(func(ctx *iris.Context){
// ctx.WriteString("Hello, world!") // or a template or anything else
// }, time.Duration(10*time.Second))) // duration of expiration
2016-11-15 19:20:29 +01:00
// if <=2 seconds then it tries to find it though request header's "cache-control" maxage value
2016-10-31 07:19:00 +01:00
//
// Note that it depends on a station instance's cache service.
// Do not try to call it from default' station if you use the form of app := iris.New(),
// use the app.Cache instead of iris.Cache
func Cache ( bodyHandler HandlerFunc , expiration time . Duration ) HandlerFunc {
return Default . Cache ( bodyHandler , expiration )
}
// Cache is just a wrapper for a route's handler which you want to enable body caching
// Usage: iris.Get("/", iris.Cache(func(ctx *iris.Context){
// ctx.WriteString("Hello, world!") // or a template or anything else
// }, time.Duration(10*time.Second))) // duration of expiration
// if <=time.Second then it tries to find it though request header's "cache-control" maxage value
//
// Note that it depends on a station instance's cache service.
// Do not try to call it from default' station if you use the form of app := iris.New(),
// use the app.Cache instead of iris.Cache
func ( s * Framework ) Cache ( bodyHandler HandlerFunc , expiration time . Duration ) HandlerFunc {
2016-11-15 19:20:29 +01:00
ce := newCachedMuxEntry ( s , bodyHandler , expiration )
return ce . Serve
2016-10-31 07:19:00 +01:00
}
2016-06-14 07:45:40 +02:00
// -------------------------------------------------------------------------------------
// -------------------------------------------------------------------------------------
// ----------------------------------MuxAPI implementation------------------------------
// -------------------------------------------------------------------------------------
// -------------------------------------------------------------------------------------
2016-07-06 20:24:34 +02:00
2017-01-02 20:20:17 +01:00
type muxAPI struct {
mux * serveMux
doneMiddleware Middleware
apiRoutes [ ] * route // used to register the .Done middleware
relativePath string
middleware Middleware
}
2016-07-06 20:24:34 +02:00
var _ MuxAPI = & muxAPI { }
2016-05-30 16:08:09 +02:00
2016-06-14 07:45:40 +02:00
var (
// errAPIContextNotFound returns an error with message: 'From .API: "Context *iris.Context could not be found..'
errAPIContextNotFound = errors . New ( "From .API: Context *iris.Context could not be found." )
// errDirectoryFileNotFound returns an error with message: 'Directory or file %s couldn't found. Trace: +error trace'
errDirectoryFileNotFound = errors . New ( "Directory or file %s couldn't found. Trace: %s" )
)
// Party is just a group joiner of routes which have the same prefix and share same middleware(s) also.
// Party can also be named as 'Join' or 'Node' or 'Group' , Party chosen because it has more fun
func Party ( relativePath string , handlersFn ... HandlerFunc ) MuxAPI {
return Default . Party ( relativePath , handlersFn ... )
}
// Party is just a group joiner of routes which have the same prefix and share same middleware(s) also.
// Party can also be named as 'Join' or 'Node' or 'Group' , Party chosen because it has more fun
func ( api * muxAPI ) Party ( relativePath string , handlersFn ... HandlerFunc ) MuxAPI {
parentPath := api . relativePath
dot := string ( subdomainIndicator [ 0 ] )
if len ( parentPath ) > 0 && parentPath [ 0 ] == slashByte && strings . HasSuffix ( relativePath , dot ) { // if ends with . , example: admin., it's subdomain->
parentPath = parentPath [ 1 : ] // remove first slash
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
fullpath := parentPath + relativePath
middleware := convertToHandlers ( handlersFn )
// append the parent's +child's handlers
middleware = joinMiddleware ( api . middleware , middleware )
2016-07-25 15:02:51 +02:00
return & muxAPI { relativePath : fullpath , mux : api . mux , apiRoutes : make ( [ ] * route , 0 ) , middleware : middleware , doneMiddleware : api . doneMiddleware }
2016-05-30 16:08:09 +02:00
}
2016-06-17 06:18:09 +02:00
// Use registers Handler middleware
2016-12-16 09:20:05 +01:00
func Use ( handlers ... Handler ) MuxAPI {
return Default . Use ( handlers ... )
2016-06-14 07:45:40 +02:00
}
2016-06-17 06:18:09 +02:00
// UseFunc registers HandlerFunc middleware
2016-12-16 09:20:05 +01:00
func UseFunc ( handlersFn ... HandlerFunc ) MuxAPI {
return Default . UseFunc ( handlersFn ... )
2016-06-14 07:45:40 +02:00
}
2016-07-25 14:53:12 +02:00
// Done registers Handler 'middleware' the only difference from .Use is that it
// should be used BEFORE any party route registered or AFTER ALL party's routes have been registered.
//
// returns itself
func Done ( handlers ... Handler ) MuxAPI {
return Default . Done ( handlers ... )
}
// DoneFunc registers HandlerFunc 'middleware' the only difference from .Use is that it
// should be used BEFORE any party route registered or AFTER ALL party's routes have been registered.
//
// returns itself
func DoneFunc ( handlersFn ... HandlerFunc ) MuxAPI {
return Default . DoneFunc ( handlersFn ... )
}
2016-06-17 06:18:09 +02:00
// Use registers Handler middleware
2016-12-16 09:20:05 +01:00
// returns itself
func ( api * muxAPI ) Use ( handlers ... Handler ) MuxAPI {
2016-06-14 07:45:40 +02:00
api . middleware = append ( api . middleware , handlers ... )
2016-12-16 09:20:05 +01:00
return api
2016-06-14 07:45:40 +02:00
}
2016-06-17 06:18:09 +02:00
// UseFunc registers HandlerFunc middleware
2016-12-16 09:20:05 +01:00
// returns itself
func ( api * muxAPI ) UseFunc ( handlersFn ... HandlerFunc ) MuxAPI {
return api . Use ( convertToHandlers ( handlersFn ) ... )
2016-06-14 07:45:40 +02:00
}
2016-07-25 14:53:12 +02:00
// Done registers Handler 'middleware' the only difference from .Use is that it
// should be used BEFORE any party route registered or AFTER ALL party's routes have been registered.
//
// returns itself
func ( api * muxAPI ) Done ( handlers ... Handler ) MuxAPI {
if len ( api . apiRoutes ) > 0 { // register these middleware on previous-party-defined routes, it called after the party's route methods (Handle/HandleFunc/Get/Post/Put/Delete/...)
for i , n := 0 , len ( api . apiRoutes ) ; i < n ; i ++ {
api . apiRoutes [ i ] . middleware = append ( api . apiRoutes [ i ] . middleware , handlers ... )
}
} else {
// register them on the doneMiddleware, which will be used on Handle to append these middlweare as the last handler(s)
api . doneMiddleware = append ( api . doneMiddleware , handlers ... )
}
return api
}
// Done registers HandlerFunc 'middleware' the only difference from .Use is that it
// should be used BEFORE any party route registered or AFTER ALL party's routes have been registered.
//
// returns itself
func ( api * muxAPI ) DoneFunc ( handlersFn ... HandlerFunc ) MuxAPI {
return api . Done ( convertToHandlers ( handlersFn ) ... )
}
2016-12-16 09:20:05 +01:00
// UseTransaction adds transaction(s) middleware
// the difference from manually adding them to the ctx.BeginTransaction
// is that if a transaction is requested scope and is failed then the (next) handler is not executed.
//
// Returns itself.
//
// See https://github.com/iris-contrib/examples/tree/master/transactions to manually add transactions
// and https://github.com/kataras/iris/blob/master/context_test.go for more
2017-01-02 20:20:17 +01:00
// func UseTransaction(pipes ...TransactionFunc) MuxAPI {
// return Default.UseTransaction(pipes...)
// }
2016-12-16 09:20:05 +01:00
// UseTransaction adds transaction(s) middleware
// the difference from manually adding them to the ctx.BeginTransaction
// is that if a transaction is requested scope and is failed then the (next) handler is not executed.
//
// Returns itself.
//
// See https://github.com/iris-contrib/examples/tree/master/transactions to manually add transactions
// and https://github.com/kataras/iris/blob/master/context_test.go for more
2017-01-02 20:20:17 +01:00
// func (api *muxAPI) UseTransaction(pipes ...TransactionFunc) MuxAPI {
// return api.UseFunc(func(ctx *Context) {
// for i := range pipes {
// ctx.BeginTransaction(pipes[i])
// if ctx.TransactionsSkipped() {
// ctx.StopExecution()
// }
// }
// ctx.Next()
// })
// }
2016-12-16 09:20:05 +01:00
// DoneTransaction registers Transaction 'middleware' the only difference from .UseTransaction is that
// is executed always last, after all of each route's handlers, returns itself.
//
// See https://github.com/iris-contrib/examples/tree/master/transactions to manually add transactions
// and https://github.com/kataras/iris/blob/master/context_test.go for more
2017-01-02 20:20:17 +01:00
// func DoneTransaction(pipes ...TransactionFunc) MuxAPI {
// return Default.DoneTransaction(pipes...)
// }
2016-12-16 09:20:05 +01:00
// DoneTransaction registers Transaction 'middleware' the only difference from .UseTransaction is that
// is executed always last, after all of each route's handlers, returns itself.
//
// See https://github.com/iris-contrib/examples/tree/master/transactions to manually add transactions
// and https://github.com/kataras/iris/blob/master/context_test.go for more
2017-01-02 20:20:17 +01:00
// func (api *muxAPI) DoneTransaction(pipes ...TransactionFunc) MuxAPI {
// return api.DoneFunc(func(ctx *Context) {
// for i := range pipes {
// ctx.BeginTransaction(pipes[i])
// }
// })
// }
2016-12-16 09:20:05 +01:00
2016-06-14 07:45:40 +02:00
// Handle registers a route to the server's router
// if empty method is passed then registers handler(s) for all methods, same as .Any, but returns nil as result
func Handle ( method string , registedPath string , handlers ... Handler ) RouteNameFunc {
return Default . Handle ( method , registedPath , handlers ... )
}
// HandleFunc registers and returns a route with a method string, path string and a handler
// registedPath is the relative url path
func HandleFunc ( method string , registedPath string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . HandleFunc ( method , registedPath , handlersFn ... )
}
// Handle registers a route to the server's router
// if empty method is passed then registers handler(s) for all methods, same as .Any, but returns nil as result
func ( api * muxAPI ) Handle ( method string , registedPath string , handlers ... Handler ) RouteNameFunc {
if method == "" { // then use like it was .Any
for _ , k := range AllMethods {
api . Handle ( k , registedPath , handlers ... )
}
return nil
}
2016-10-02 02:30:37 +02:00
fullpath := api . relativePath + registedPath // for now, keep the last "/" if any, "/xyz/"
2016-06-14 07:45:40 +02:00
middleware := joinMiddleware ( api . middleware , handlers )
// here we separate the subdomain and relative path
subdomain := ""
path := fullpath
if dotWSlashIdx := strings . Index ( path , subdomainIndicator ) ; dotWSlashIdx > 0 {
subdomain = fullpath [ 0 : dotWSlashIdx + 1 ] // admin.
path = fullpath [ dotWSlashIdx + 1 : ] // /
}
2016-10-05 12:21:40 +02:00
2016-10-03 16:23:53 +02:00
// we splitted the path and subdomain parts so we're ready to check only the path,
// otherwise we will had problems with subdomains
2016-10-05 12:21:40 +02:00
// if the user wants beta:= iris.Party("/beta"); beta.Get("/") to be registered as
//: /beta/ then should disable the path correction OR register it like: beta.Get("//")
// this is only for the party's roots in order to have expected paths,
// as we do with iris.Get("/") which is localhost:8080 as RFC points, not localhost:8080/
2016-10-04 00:18:17 +02:00
if api . mux . correctPath && registedPath == slash { // check the given relative path
2016-10-02 02:30:37 +02:00
// remove last "/" if any, "/xyz/"
if len ( path ) > 1 { // if it's the root, then keep it*
if path [ len ( path ) - 1 ] == slashByte {
// ok we are inside /xyz/
}
}
2016-10-04 00:18:17 +02:00
}
2016-06-14 07:45:40 +02:00
path = strings . Replace ( path , "//" , "/" , - 1 ) // fix the path if double //
2016-07-25 14:53:12 +02:00
if len ( api . doneMiddleware ) > 0 {
middleware = append ( middleware , api . doneMiddleware ... ) // register the done middleware, if any
}
2017-01-02 20:20:17 +01:00
r := api . mux . register ( method , subdomain , path , middleware )
2016-07-25 14:53:12 +02:00
api . apiRoutes = append ( api . apiRoutes , r )
2016-07-25 16:18:50 +02:00
// should we remove the api.apiRoutes on the .Party (new children party) ?, No, because the user maybe use this party later
// should we add to the 'inheritance tree' the api.apiRoutes, No, these are for this specific party only, because the user propably, will have unexpected behavior when using Use/UseFunc, Done/DoneFunc
2016-07-25 14:53:12 +02:00
return r . setName
2016-06-14 07:45:40 +02:00
}
// HandleFunc registers and returns a route with a method string, path string and a handler
// registedPath is the relative url path
func ( api * muxAPI ) HandleFunc ( method string , registedPath string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . Handle ( method , registedPath , convertToHandlers ( handlersFn ) ... )
}
// API converts & registers a custom struct to the router
// receives two parameters
// first is the request path (string)
// second is the custom struct (interface{}) which can be anything that has a *iris.Context as field.
// third is the common middlewares, it's optional
2016-05-30 16:08:09 +02:00
//
2016-06-14 07:45:40 +02:00
// Note that API's routes have their default-name to the full registed path,
// no need to give a special name for it, because it's not supposed to be used inside your templates.
//
// Recommend to use when you retrieve data from an external database,
// and the router-performance is not the (only) thing which slows the server's overall performance.
//
// This is a slow method, if you care about router-performance use the Handle/HandleFunc/Get/Post/Put/Delete/Trace/Options... instead
func API ( path string , restAPI HandlerAPI , middleware ... HandlerFunc ) {
Default . API ( path , restAPI , middleware ... )
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
// API converts & registers a custom struct to the router
// receives two parameters
// first is the request path (string)
// second is the custom struct (interface{}) which can be anything that has a *iris.Context as field.
// third is the common middleware, it's optional
2016-05-30 16:08:09 +02:00
//
2016-06-14 07:45:40 +02:00
// Note that API's routes have their default-name to the full registed path,
// no need to give a special name for it, because it's not supposed to be used inside your templates.
//
// Recommend to use when you retrieve data from an external database,
// and the router-performance is not the (only) thing which slows the server's overall performance.
//
// This is a slow method, if you care about router-performance use the Handle/HandleFunc/Get/Post/Put/Delete/Trace/Options... instead
func ( api * muxAPI ) API ( path string , restAPI HandlerAPI , middleware ... HandlerFunc ) {
// here we need to find the registed methods and convert them to handler funcs
// methods are collected by method naming: Get(),GetBy(...), Post(),PostBy(...), Put() and so on
2016-07-25 16:18:50 +02:00
if len ( path ) == 0 {
path = "/"
}
if path [ 0 ] != slashByte {
// the route's paths always starts with "/", when the client navigates, the router works without "/" also ,
// but the developer should always prepend the slash ("/") to register the routes
path = "/" + path
}
2016-06-14 07:45:40 +02:00
typ := reflect . ValueOf ( restAPI ) . Type ( )
contextField , found := typ . FieldByName ( "Context" )
if ! found {
2016-09-01 05:01:53 +02:00
panic ( errAPIContextNotFound )
2016-06-14 07:45:40 +02:00
}
// check & register the Get(),Post(),Put(),Delete() and so on
for _ , methodName := range AllMethods {
methodCapitalName := strings . Title ( strings . ToLower ( methodName ) )
if method , found := typ . MethodByName ( methodCapitalName ) ; found {
methodFunc := method . Func
if ! methodFunc . IsValid ( ) || methodFunc . Type ( ) . NumIn ( ) > 1 { // for any case
continue
}
func ( path string , typ reflect . Type , contextField reflect . StructField , methodFunc reflect . Value , method string ) {
var handlersFn [ ] HandlerFunc
handlersFn = append ( handlersFn , middleware ... )
handlersFn = append ( handlersFn , func ( ctx * Context ) {
newController := reflect . New ( typ ) . Elem ( )
newController . FieldByName ( "Context" ) . Set ( reflect . ValueOf ( ctx ) )
methodFunc . Call ( [ ] reflect . Value { newController } )
} )
// register route
api . HandleFunc ( method , path , handlersFn ... )
} ( path , typ , contextField , methodFunc , methodName )
}
}
// check for GetBy/PostBy(id string, something_else string) , these must be requested by the same order.
// (we could do this in the same top loop but I don't want)
// GET, DELETE -> with url named parameters (/users/:id/:secondArgumentIfExists)
// POST, PUT -> with post values (form)
// all other with URL Parameters (?something=this&else=other
//
// or no, I changed my mind, let all be named parameters and let users to decide what info they need,
// using the Context to take more values (post form,url params and so on).-
2016-10-25 14:58:18 +02:00
paramPrefix := [ ] byte ( "param" )
2016-06-14 07:45:40 +02:00
for _ , methodName := range AllMethods {
methodWithBy := strings . Title ( strings . ToLower ( methodName ) ) + "By"
if method , found := typ . MethodByName ( methodWithBy ) ; found {
methodFunc := method . Func
if ! methodFunc . IsValid ( ) || methodFunc . Type ( ) . NumIn ( ) < 2 { //it's By but it has not receive any arguments so its not api's
continue
}
methodFuncType := methodFunc . Type ( )
numInLen := methodFuncType . NumIn ( ) // how much data we should receive from the request
registedPath := path
for i := 1 ; i < numInLen ; i ++ { // from 1 because the first is the 'object'
if registedPath [ len ( registedPath ) - 1 ] == slashByte {
2016-10-25 14:58:18 +02:00
registedPath += ":" + string ( paramPrefix ) + strconv . Itoa ( i )
2016-06-14 07:45:40 +02:00
} else {
2016-10-25 14:58:18 +02:00
registedPath += "/:" + string ( paramPrefix ) + strconv . Itoa ( i )
2016-06-14 07:45:40 +02:00
}
}
func ( registedPath string , typ reflect . Type , contextField reflect . StructField , methodFunc reflect . Value , paramsLen int , method string ) {
var handlersFn [ ] HandlerFunc
handlersFn = append ( handlersFn , middleware ... )
handlersFn = append ( handlersFn , func ( ctx * Context ) {
newController := reflect . New ( typ ) . Elem ( )
newController . FieldByName ( "Context" ) . Set ( reflect . ValueOf ( ctx ) )
args := make ( [ ] reflect . Value , paramsLen + 1 , paramsLen + 1 )
args [ 0 ] = newController
2016-07-21 22:08:42 +02:00
j := 1
2016-10-25 14:58:18 +02:00
2017-01-02 20:20:17 +01:00
ctx . VisitValues ( func ( k [ ] byte , v interface { } ) {
2016-10-25 14:58:18 +02:00
if bytes . HasPrefix ( k , paramPrefix ) {
args [ j ] = reflect . ValueOf ( v . ( string ) )
2016-07-25 16:18:50 +02:00
2016-07-21 22:08:42 +02:00
j ++ // the first parameter is the context, other are the path parameters, j++ to be align with (API's registered)paramsLen
}
2016-10-25 14:58:18 +02:00
} )
2016-07-21 22:08:42 +02:00
2016-06-14 07:45:40 +02:00
methodFunc . Call ( args )
} )
// register route
api . HandleFunc ( method , registedPath , handlersFn ... )
} ( registedPath , typ , contextField , methodFunc , numInLen - 1 , methodName )
}
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
// Get registers a route for the Get http method
func Get ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Get ( path , handlersFn ... )
}
// Post registers a route for the Post http method
func Post ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Post ( path , handlersFn ... )
}
// Put registers a route for the Put http method
func Put ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Put ( path , handlersFn ... )
}
// Delete registers a route for the Delete http method
func Delete ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Delete ( path , handlersFn ... )
}
// Connect registers a route for the Connect http method
func Connect ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Connect ( path , handlersFn ... )
}
// Head registers a route for the Head http method
func Head ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Head ( path , handlersFn ... )
}
// Options registers a route for the Options http method
func Options ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Options ( path , handlersFn ... )
}
// Patch registers a route for the Patch http method
func Patch ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Patch ( path , handlersFn ... )
}
// Trace registers a route for the Trace http method
func Trace ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return Default . Trace ( path , handlersFn ... )
}
// Any registers a route for ALL of the http methods (Get,Post,Put,Head,Patch,Options,Connect,Delete)
func Any ( registedPath string , handlersFn ... HandlerFunc ) {
Default . Any ( registedPath , handlersFn ... )
}
// Get registers a route for the Get http method
func ( api * muxAPI ) Get ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodGet , path , handlersFn ... )
}
// Post registers a route for the Post http method
func ( api * muxAPI ) Post ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodPost , path , handlersFn ... )
}
// Put registers a route for the Put http method
func ( api * muxAPI ) Put ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodPut , path , handlersFn ... )
}
// Delete registers a route for the Delete http method
func ( api * muxAPI ) Delete ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodDelete , path , handlersFn ... )
}
// Connect registers a route for the Connect http method
func ( api * muxAPI ) Connect ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodConnect , path , handlersFn ... )
}
// Head registers a route for the Head http method
func ( api * muxAPI ) Head ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodHead , path , handlersFn ... )
}
// Options registers a route for the Options http method
func ( api * muxAPI ) Options ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodOptions , path , handlersFn ... )
}
// Patch registers a route for the Patch http method
func ( api * muxAPI ) Patch ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodPatch , path , handlersFn ... )
}
// Trace registers a route for the Trace http method
func ( api * muxAPI ) Trace ( path string , handlersFn ... HandlerFunc ) RouteNameFunc {
return api . HandleFunc ( MethodTrace , path , handlersFn ... )
}
// Any registers a route for ALL of the http methods (Get,Post,Put,Head,Patch,Options,Connect,Delete)
func ( api * muxAPI ) Any ( registedPath string , handlersFn ... HandlerFunc ) {
for _ , k := range AllMethods {
api . HandleFunc ( k , registedPath , handlersFn ... )
}
}
2016-12-12 11:18:59 +01:00
// if / then returns /*wildcard or /something then /something/*wildcard
// if empty then returns /*wildcard too
func validateWildcard ( reqPath string , paramName string ) string {
if reqPath [ len ( reqPath ) - 1 ] != slashByte {
reqPath += slash
}
reqPath += "*" + paramName
return reqPath
}
func ( api * muxAPI ) registerResourceRoute ( reqPath string , h HandlerFunc ) RouteNameFunc {
api . Head ( reqPath , h )
return api . Get ( reqPath , h )
}
2016-06-14 07:45:40 +02:00
// StaticServe serves a directory as web resource
// it's the simpliest form of the Static* functions
// Almost same usage as StaticWeb
// accepts only one required parameter which is the systemPath ( the same path will be used to register the GET&HEAD routes)
// if second parameter is empty, otherwise the requestPath is the second parameter
// it uses gzip compression (compression on each request, no file cache)
func StaticServe ( systemPath string , requestPath ... string ) RouteNameFunc {
return Default . StaticServe ( systemPath , requestPath ... )
}
// StaticServe serves a directory as web resource
// it's the simpliest form of the Static* functions
// Almost same usage as StaticWeb
// accepts only one required parameter which is the systemPath ( the same path will be used to register the GET&HEAD routes)
// if second parameter is empty, otherwise the requestPath is the second parameter
// it uses gzip compression (compression on each request, no file cache)
func ( api * muxAPI ) StaticServe ( systemPath string , requestPath ... string ) RouteNameFunc {
var reqPath string
if len ( requestPath ) == 0 {
2016-09-01 05:34:55 +02:00
reqPath = strings . Replace ( systemPath , fs . PathSeparator , slash , - 1 ) // replaces any \ to /
reqPath = strings . Replace ( reqPath , "//" , slash , - 1 ) // for any case, replaces // to /
reqPath = strings . Replace ( reqPath , "." , "" , - 1 ) // replace any dots (./mypath -> /mypath)
2016-06-14 07:45:40 +02:00
} else {
reqPath = requestPath [ 0 ]
}
return api . Get ( reqPath + "/*file" , func ( ctx * Context ) {
filepath := ctx . Param ( "file" )
2016-09-01 05:34:55 +02:00
spath := strings . Replace ( filepath , "/" , fs . PathSeparator , - 1 )
2016-06-14 07:45:40 +02:00
spath = path . Join ( systemPath , spath )
2016-09-01 05:34:55 +02:00
if ! fs . DirectoryExists ( spath ) {
2016-06-14 07:45:40 +02:00
ctx . NotFound ( )
return
}
ctx . ServeFile ( spath , true )
} )
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
// StaticContent serves bytes, memory cached, on the reqPath
// a good example of this is how the websocket server uses that to auto-register the /iris-ws.js
func StaticContent ( reqPath string , contentType string , content [ ] byte ) RouteNameFunc {
return Default . StaticContent ( reqPath , contentType , content )
2016-05-30 16:08:09 +02:00
}
2016-06-14 07:45:40 +02:00
// StaticContent serves bytes, memory cached, on the reqPath
// a good example of this is how the websocket server uses that to auto-register the /iris-ws.js
2016-08-16 12:17:26 +02:00
func ( api * muxAPI ) StaticContent ( reqPath string , cType string , content [ ] byte ) RouteNameFunc { // func(string) because we use that on websockets
2016-06-14 07:45:40 +02:00
modtime := time . Now ( )
h := func ( ctx * Context ) {
2016-12-12 11:18:59 +01:00
ctx . SetClientCachedBody ( StatusOK , content , cType , modtime )
2016-06-14 07:45:40 +02:00
}
2016-12-12 11:18:59 +01:00
return api . registerResourceRoute ( reqPath , h )
2016-05-30 16:08:09 +02:00
}
2016-10-17 02:37:57 +02:00
// StaticEmbedded used when files are distrubuted inside the app executable, using go-bindata mostly
// First parameter is the request path, the path which the files in the vdir(second parameter) will be served to, for example "/static"
// Second parameter is the (virtual) directory path, for example "./assets"
// Third parameter is the Asset function
// Forth parameter is the AssetNames function
//
// For more take a look at the
// example: https://github.com/iris-contrib/examples/tree/master/static_files_embedded
func StaticEmbedded ( requestPath string , vdir string , assetFn func ( name string ) ( [ ] byte , error ) , namesFn func ( ) [ ] string ) RouteNameFunc {
return Default . StaticEmbedded ( requestPath , vdir , assetFn , namesFn )
}
// StaticEmbedded used when files are distrubuted inside the app executable, using go-bindata mostly
// First parameter is the request path, the path which the files in the vdir will be served to, for example "/static"
// Second parameter is the (virtual) directory path, for example "./assets"
// Third parameter is the Asset function
// Forth parameter is the AssetNames function
//
// For more take a look at the
// example: https://github.com/iris-contrib/examples/tree/master/static_files_embedded
func ( api * muxAPI ) StaticEmbedded ( requestPath string , vdir string , assetFn func ( name string ) ( [ ] byte , error ) , namesFn func ( ) [ ] string ) RouteNameFunc {
// check if requestPath already contains an asterix-match to anything symbol: /path/*
requestPath = strings . Replace ( requestPath , "//" , "/" , - 1 )
matchEverythingIdx := strings . IndexByte ( requestPath , matchEverythingByte )
paramName := "path"
if matchEverythingIdx != - 1 {
// found so it should has a param name, take it
paramName = requestPath [ matchEverythingIdx + 1 : ]
} else {
// make the requestPath
if requestPath [ len ( requestPath ) - 1 ] == slashByte {
// ends with / remove it
requestPath = requestPath [ 0 : len ( requestPath ) - 2 ]
}
requestPath += slash + "*" + paramName // $requestPath/*path
}
if len ( vdir ) > 0 {
if vdir [ 0 ] == '.' { // first check for .wrong
vdir = vdir [ 1 : ]
}
if vdir [ 0 ] == '/' || vdir [ 0 ] == os . PathSeparator { // second check for /something, (or ./something if we had dot on 0 it will be removed
vdir = vdir [ 1 : ]
}
}
// collect the names we are care for, because not all Asset used here, we need the vdir's assets.
allNames := namesFn ( )
var names [ ] string
for _ , path := range allNames {
// check if path is the path name we care for
if ! strings . HasPrefix ( path , vdir ) {
continue
}
path = strings . Replace ( path , "\\" , "/" , - 1 ) // replace system paths with double slashes
path = strings . Replace ( path , "./" , "/" , - 1 ) // replace ./assets/favicon.ico to /assets/favicon.ico in order to be ready for compare with the reqPath later
path = path [ len ( vdir ) : ] // set it as the its 'relative' ( we should re-setted it when assetFn will be used)
names = append ( names , path )
}
if len ( names ) == 0 {
// we don't start the server yet, so:
2016-10-25 15:49:57 +02:00
panic ( "iris.StaticEmbedded: Unable to locate any embedded files located to the (virtual) directory: " + vdir )
2016-10-17 02:37:57 +02:00
}
modtime := time . Now ( )
h := func ( ctx * Context ) {
reqPath := ctx . Param ( paramName )
for _ , path := range names {
if path != reqPath {
continue
}
2016-10-17 03:43:31 +02:00
cType := fs . TypeByExtension ( path )
fullpath := vdir + path
buf , err := assetFn ( fullpath )
if err != nil {
continue
}
2016-12-12 11:18:59 +01:00
ctx . SetClientCachedBody ( StatusOK , buf , cType , modtime )
2016-10-17 02:37:57 +02:00
return
}
2016-12-12 11:18:59 +01:00
// not found or error
2016-10-17 02:37:57 +02:00
ctx . EmitError ( StatusNotFound )
}
2016-12-12 11:18:59 +01:00
return api . registerResourceRoute ( requestPath , h )
2016-10-17 02:37:57 +02:00
}
2016-06-14 07:45:40 +02:00
// Favicon serves static favicon
// accepts 2 parameters, second is optional
// favPath (string), declare the system directory path of the __.ico
// requestPath (string), it's the route's path, by default this is the "/favicon.ico" because some browsers tries to get this by default first,
// you can declare your own path if you have more than one favicon (desktop, mobile and so on)
//
// this func will add a route for you which will static serve the /yuorpath/yourfile.ico to the /yourfile.ico (nothing special that you can't handle by yourself)
// Note that you have to call it on every favicon you have to serve automatically (dekstop, mobile and so on)
//
// panics on error
func Favicon ( favPath string , requestPath ... string ) RouteNameFunc {
return Default . Favicon ( favPath , requestPath ... )
2016-05-30 16:08:09 +02:00
}
2016-06-01 14:29:38 +02:00
2016-06-14 07:45:40 +02:00
// Favicon serves static favicon
// accepts 2 parameters, second is optional
// favPath (string), declare the system directory path of the __.ico
// requestPath (string), it's the route's path, by default this is the "/favicon.ico" because some browsers tries to get this by default first,
// you can declare your own path if you have more than one favicon (desktop, mobile and so on)
//
// this func will add a route for you which will static serve the /yuorpath/yourfile.ico to the /yourfile.ico (nothing special that you can't handle by yourself)
// Note that you have to call it on every favicon you have to serve automatically (dekstop, mobile and so on)
//
// panics on error
func ( api * muxAPI ) Favicon ( favPath string , requestPath ... string ) RouteNameFunc {
f , err := os . Open ( favPath )
if err != nil {
panic ( errDirectoryFileNotFound . Format ( favPath , err . Error ( ) ) )
}
defer f . Close ( )
fi , _ := f . Stat ( )
if fi . IsDir ( ) { // if it's dir the try to get the favicon.ico
fav := path . Join ( favPath , "favicon.ico" )
f , err = os . Open ( fav )
if err != nil {
//we try again with .png
return api . Favicon ( path . Join ( favPath , "favicon.png" ) )
}
favPath = fav
fi , _ = f . Stat ( )
}
2016-09-09 07:09:03 +02:00
2016-09-01 05:34:55 +02:00
cType := fs . TypeByExtension ( favPath )
2016-06-14 07:45:40 +02:00
// copy the bytes here in order to cache and not read the ico on each request.
cacheFav := make ( [ ] byte , fi . Size ( ) )
if _ , err = f . Read ( cacheFav ) ; err != nil {
panic ( errDirectoryFileNotFound . Format ( favPath , "Couldn't read the data bytes for Favicon: " + err . Error ( ) ) )
}
2016-09-09 07:09:03 +02:00
modtime := ""
2016-06-14 07:45:40 +02:00
h := func ( ctx * Context ) {
2016-09-09 07:09:03 +02:00
if modtime == "" {
modtime = fi . ModTime ( ) . UTC ( ) . Format ( ctx . framework . Config . TimeFormat )
}
if t , err := time . Parse ( ctx . framework . Config . TimeFormat , ctx . RequestHeader ( ifModifiedSince ) ) ; err == nil && fi . ModTime ( ) . Before ( t . Add ( StaticCacheDuration ) ) {
2017-01-02 20:20:17 +01:00
ctx . ResponseWriter . Header ( ) . Del ( contentType )
ctx . ResponseWriter . Header ( ) . Del ( contentLength )
2016-06-14 07:45:40 +02:00
ctx . SetStatusCode ( StatusNotModified )
return
}
2017-01-02 20:20:17 +01:00
ctx . ResponseWriter . Header ( ) . Set ( contentType , cType )
ctx . ResponseWriter . Header ( ) . Set ( lastModified , modtime )
2016-06-14 07:45:40 +02:00
ctx . SetStatusCode ( StatusOK )
2017-01-04 14:16:53 +01:00
ctx . Write ( cacheFav )
2016-06-14 07:45:40 +02:00
}
reqPath := "/favicon" + path . Ext ( fi . Name ( ) ) //we could use the filename, but because standards is /favicon.ico/.png.
if len ( requestPath ) > 0 {
reqPath = requestPath [ 0 ]
}
2016-12-12 11:18:59 +01:00
return api . registerResourceRoute ( reqPath , h )
2016-06-01 14:29:38 +02:00
}
2016-06-14 19:29:01 +02:00
2017-01-02 20:20:17 +01:00
// StripPrefix returns a handler that serves HTTP requests
// by removing the given prefix from the request URL's Path
// and invoking the handler h. StripPrefix handles a
// request for a path that doesn't begin with prefix by
// replying with an HTTP 404 not found error.
func StripPrefix ( prefix string , h HandlerFunc ) HandlerFunc {
if prefix == "" {
return h
}
return func ( ctx * Context ) {
if p := strings . TrimPrefix ( ctx . Request . URL . Path , prefix ) ; len ( p ) < len ( ctx . Request . URL . Path ) {
ctx . Request . URL . Path = p
h ( ctx )
} else {
ctx . NotFound ( )
}
}
}
// StaticHandler returns a new Handler which serves static files
func StaticHandler ( reqPath string , systemPath string , showList bool , enableGzip bool ) HandlerFunc {
return Default . StaticHandler ( reqPath , systemPath , showList , enableGzip )
}
// StaticHandler returns a new Handler which serves static files
func ( api * muxAPI ) StaticHandler ( reqPath string , systemPath string , showList bool , enableGzip bool ) HandlerFunc {
h := NewStaticHandlerBuilder ( systemPath ) .
Path ( api . relativePath + reqPath ) .
Listing ( showList ) .
Gzip ( enableGzip ) .
Build ( )
managedStaticHandler := func ( ctx * Context ) {
h ( ctx )
prevStatusCode := ctx . ResponseWriter . StatusCode ( )
if prevStatusCode >= 400 { // we have an error
// fire the custom error handler
api . mux . fireError ( prevStatusCode , ctx )
}
// go to the next middleware
if ctx . Pos < len ( ctx . Middleware ) - 1 {
ctx . Next ( )
}
}
return managedStaticHandler
}
// StaticWeb returns a handler that serves HTTP requests
// with the contents of the file system rooted at directory.
//
// first parameter: the route path
// second parameter: the system directory
// for more options look iris.StaticHandler.
//
// iris.StaticWeb("/static", "./static")
//
// As a special case, the returned file server redirects any request
// ending in "/index.html" to the same path, without the final
// "index.html".
//
// StaticWeb calls the StaticHandler(reqPath, systemPath, listingDirectories: false, gzip: false ).
func StaticWeb ( reqPath string , systemPath string ) RouteNameFunc {
return Default . StaticWeb ( reqPath , systemPath )
}
// StaticWeb returns a handler that serves HTTP requests
// with the contents of the file system rooted at directory.
//
// first parameter: the route path
// second parameter: the system directory
// for more options look iris.StaticHandler.
//
// iris.StaticWeb("/static", "./static")
//
// As a special case, the returned file server redirects any request
// ending in "/index.html" to the same path, without the final
// "index.html".
//
// StaticWeb calls the StaticHandler(reqPath, systemPath, listingDirectories: false, gzip: false ).
func ( api * muxAPI ) StaticWeb ( reqPath string , systemPath string ) RouteNameFunc {
h := api . StaticHandler ( reqPath , systemPath , false , false )
routePath := validateWildcard ( reqPath , "file" )
return api . registerResourceRoute ( routePath , h )
}
2016-09-29 16:05:22 +02:00
// Layout oerrides the parent template layout with a more specific layout for this Party
// returns this Party, to continue as normal
// example:
// my := iris.Party("/my").Layout("layouts/mylayout.html")
// {
// my.Get("/", func(ctx *iris.Context) {
// ctx.MustRender("page1.html", nil)
// })
// }
//
func Layout ( tmplLayoutFile string ) MuxAPI {
return Default . Layout ( tmplLayoutFile )
}
2016-06-14 19:29:01 +02:00
// Layout oerrides the parent template layout with a more specific layout for this Party
// returns this Party, to continue as normal
// example:
// my := iris.Party("/my").Layout("layouts/mylayout.html")
// {
// my.Get("/", func(ctx *iris.Context) {
// ctx.MustRender("page1.html", nil)
// })
// }
//
func ( api * muxAPI ) Layout ( tmplLayoutFile string ) MuxAPI {
api . UseFunc ( func ( ctx * Context ) {
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
ctx . Set ( TemplateLayoutContextKey , tmplLayoutFile )
2016-06-14 19:29:01 +02:00
ctx . Next ( )
} )
2016-09-29 16:05:22 +02:00
2016-06-14 19:29:01 +02:00
return api
}
2016-08-14 04:44:36 +02:00
// OnError registers a custom http error handler
func OnError ( statusCode int , handlerFn HandlerFunc ) {
Default . OnError ( statusCode , handlerFn )
}
// EmitError fires a custom http error handler to the client
//
// if no custom error defined with this statuscode, then iris creates one, and once at runtime
func EmitError ( statusCode int , ctx * Context ) {
Default . EmitError ( statusCode , ctx )
}
// OnError registers a custom http error handler
func ( api * muxAPI ) OnError ( statusCode int , handlerFn HandlerFunc ) {
path := strings . Replace ( api . relativePath , "//" , "/" , - 1 ) // fix the path if double //
staticPath := path
// find the static path (on Party the path should be ALWAYS a static path, as we all know,
// but do this check for any case)
dynamicPathIdx := strings . IndexByte ( path , parameterStartByte ) // check for /mypath/:param
if dynamicPathIdx == - 1 {
dynamicPathIdx = strings . IndexByte ( path , matchEverythingByte ) // check for /mypath/*param
}
if dynamicPathIdx > 1 { //yes after / and one character more ( /*param or /:param will break the root path, and this is not allowed even on error handlers).
staticPath = api . relativePath [ 0 : dynamicPathIdx ]
}
if staticPath == "/" {
api . mux . registerError ( statusCode , handlerFn ) // register the user-specific error message, as the global error handler, for now.
return
}
//after this, we have more than one error handler for one status code, and that's dangerous some times, but use it for non-globals error catching by your own risk
// NOTES:
// subdomains error will not work if same path of a non-subdomain (maybe a TODO for later)
// errors for parties should be registered from the biggest path length to the smaller.
// get the previous
prevErrHandler := api . mux . errorHandlers [ statusCode ]
if prevErrHandler == nil {
/ *
make a new one with the standard error message ,
this will be used as the last handler if no other error handler catches the error ( by prefix ( ? ) )
* /
prevErrHandler = HandlerFunc ( func ( ctx * Context ) {
2017-01-04 14:16:53 +01:00
if w , ok := ctx . IsRecording ( ) ; ok {
w . Reset ( )
}
2016-08-14 04:44:36 +02:00
ctx . SetStatusCode ( statusCode )
2017-01-04 14:16:53 +01:00
ctx . WriteString ( statusText [ statusCode ] )
2016-08-14 04:44:36 +02:00
} )
}
func ( statusCode int , staticPath string , prevErrHandler Handler , newHandler Handler ) { // to separate the logic
errHandler := HandlerFunc ( func ( ctx * Context ) {
2017-01-02 20:20:17 +01:00
if strings . HasPrefix ( ctx . Path ( ) , staticPath ) { // yes the user should use OnError from longest to lower static path's length in order this to work, so we can find another way, like a builder on the end.
2016-08-14 04:44:36 +02:00
newHandler . Serve ( ctx )
return
}
// serve with the user-specific global ("/") pure iris.OnError receiver Handler or the standar handler if OnError called only from inside a no-relative Party.
prevErrHandler . Serve ( ctx )
} )
api . mux . registerError ( statusCode , errHandler )
} ( statusCode , staticPath , prevErrHandler , handlerFn )
}
// EmitError fires a custom http error handler to the client
//
// if no custom error defined with this statuscode, then iris creates one, and once at runtime
func ( api * muxAPI ) EmitError ( statusCode int , ctx * Context ) {
api . mux . fireError ( statusCode , ctx )
}