Publish the new version :airplane: | Look description please!
# FAQ
### Looking for free support?
http://support.iris-go.com
https://kataras.rocket.chat/channel/iris
### Looking for previous versions?
https://github.com/kataras/iris#version
### Should I upgrade my Iris?
Developers are not forced to upgrade if they don't really need it. Upgrade whenever you feel ready.
> Iris uses the [vendor directory](https://docs.google.com/document/d/1Bz5-UB7g2uPBdOx-rw5t9MxJwkfpx90cqG9AFL0JAYo) feature, so you get truly reproducible builds, as this method guards against upstream renames and deletes.
**How to upgrade**: Open your command-line and execute this command: `go get -u github.com/kataras/iris`.
For further installation support, please click [here](http://support.iris-go.com/d/16-how-to-install-iris-web-framework).
### About our new home page
http://iris-go.com
Thanks to [Santosh Anand](https://github.com/santoshanand) the http://iris-go.com has been upgraded and it's really awesome!
[Santosh](https://github.com/santoshanand) is a freelancer, he has a great knowledge of nodejs and express js, Android, iOS, React Native, Vue.js etc, if you need a developer to find or create a solution for your problem or task, please contact with him.
The amount of the next two or three donations you'll send they will be immediately transferred to his own account balance, so be generous please!
Read more at https://github.com/kataras/iris/blob/master/HISTORY.md
Former-commit-id: eec2d71bbe011d6b48d2526eb25919e36e5ad94e
2017-06-03 22:22:52 +02:00
// Copyright 2017 Gerasimos Maropoulos, ΓΜ. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package router
import (
"net/http"
"os"
"path"
"strings"
"time"
"github.com/kataras/iris/context"
"github.com/kataras/iris/core/errors"
"github.com/kataras/iris/core/router/macro"
)
const (
// MethodNone is a Virtual method
// to store the "offline" routes
MethodNone = "NONE"
)
var (
// AllMethods contains the valid http methods:
// "GET", "POST", "PUT", "DELETE", "CONNECT", "HEAD",
// "PATCH", "OPTIONS", "TRACE".
AllMethods = [ ... ] string {
"GET" ,
"POST" ,
"PUT" ,
"DELETE" ,
"CONNECT" ,
"HEAD" ,
"PATCH" ,
"OPTIONS" ,
"TRACE" ,
}
)
// repository passed to all parties(subrouters), it's the object witch keeps
// all the routes.
type repository struct {
routes [ ] * Route
}
func ( r * repository ) register ( route * Route ) {
r . routes = append ( r . routes , route )
}
func ( r * repository ) get ( routeName string ) * Route {
for _ , r := range r . routes {
if r . Name == routeName {
return r
}
}
return nil
}
func ( r * repository ) getAll ( ) [ ] * Route {
return r . routes
}
2017-06-10 02:56:42 +02:00
// RoutesProvider should be implemented by
// iteral which contains the registered routes.
Publish the new version :airplane: | Look description please!
# FAQ
### Looking for free support?
http://support.iris-go.com
https://kataras.rocket.chat/channel/iris
### Looking for previous versions?
https://github.com/kataras/iris#version
### Should I upgrade my Iris?
Developers are not forced to upgrade if they don't really need it. Upgrade whenever you feel ready.
> Iris uses the [vendor directory](https://docs.google.com/document/d/1Bz5-UB7g2uPBdOx-rw5t9MxJwkfpx90cqG9AFL0JAYo) feature, so you get truly reproducible builds, as this method guards against upstream renames and deletes.
**How to upgrade**: Open your command-line and execute this command: `go get -u github.com/kataras/iris`.
For further installation support, please click [here](http://support.iris-go.com/d/16-how-to-install-iris-web-framework).
### About our new home page
http://iris-go.com
Thanks to [Santosh Anand](https://github.com/santoshanand) the http://iris-go.com has been upgraded and it's really awesome!
[Santosh](https://github.com/santoshanand) is a freelancer, he has a great knowledge of nodejs and express js, Android, iOS, React Native, Vue.js etc, if you need a developer to find or create a solution for your problem or task, please contact with him.
The amount of the next two or three donations you'll send they will be immediately transferred to his own account balance, so be generous please!
Read more at https://github.com/kataras/iris/blob/master/HISTORY.md
Former-commit-id: eec2d71bbe011d6b48d2526eb25919e36e5ad94e
2017-06-03 22:22:52 +02:00
type RoutesProvider interface { // api builder
GetRoutes ( ) [ ] * Route
GetRoute ( routeName string ) * Route
}
// APIBuilder the visible API for constructing the router
// and child routers.
type APIBuilder struct {
// the api builder global macros registry
2017-06-12 03:47:16 +02:00
macros * macro . Map
Publish the new version :airplane: | Look description please!
# FAQ
### Looking for free support?
http://support.iris-go.com
https://kataras.rocket.chat/channel/iris
### Looking for previous versions?
https://github.com/kataras/iris#version
### Should I upgrade my Iris?
Developers are not forced to upgrade if they don't really need it. Upgrade whenever you feel ready.
> Iris uses the [vendor directory](https://docs.google.com/document/d/1Bz5-UB7g2uPBdOx-rw5t9MxJwkfpx90cqG9AFL0JAYo) feature, so you get truly reproducible builds, as this method guards against upstream renames and deletes.
**How to upgrade**: Open your command-line and execute this command: `go get -u github.com/kataras/iris`.
For further installation support, please click [here](http://support.iris-go.com/d/16-how-to-install-iris-web-framework).
### About our new home page
http://iris-go.com
Thanks to [Santosh Anand](https://github.com/santoshanand) the http://iris-go.com has been upgraded and it's really awesome!
[Santosh](https://github.com/santoshanand) is a freelancer, he has a great knowledge of nodejs and express js, Android, iOS, React Native, Vue.js etc, if you need a developer to find or create a solution for your problem or task, please contact with him.
The amount of the next two or three donations you'll send they will be immediately transferred to his own account balance, so be generous please!
Read more at https://github.com/kataras/iris/blob/master/HISTORY.md
Former-commit-id: eec2d71bbe011d6b48d2526eb25919e36e5ad94e
2017-06-03 22:22:52 +02:00
// the api builder global handlers per status code registry (used for custom http errors)
errorCodeHandlers * ErrorCodeHandlers
// the api builder global routes repository
routes * repository
// the api builder global route path reverser object
// used by the view engine but it can be used anywhere.
reverser * RoutePathReverser
// the per-party middleware
middleware context . Handlers
// the per-party routes (useful only for done middleware)
apiRoutes [ ] * Route
// the per-party done middleware
doneHandlers context . Handlers
// the per-party
relativePath string
}
var _ Party = & APIBuilder { }
var _ RoutesProvider = & APIBuilder { } // passed to the default request handler (routerHandler)
// NewAPIBuilder creates & returns a new builder
// which is responsible to build the API and the router handler.
func NewAPIBuilder ( ) * APIBuilder {
rb := & APIBuilder {
macros : defaultMacros ( ) ,
errorCodeHandlers : defaultErrorCodeHandlers ( ) ,
relativePath : "/" ,
routes : new ( repository ) ,
}
return rb
}
// Handle registers a route to the server's rb.
// if empty method is passed then handler(s) are being registered to all methods, same as .Any.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Handle ( method string , registeredPath string , handlers ... context . Handler ) ( * Route , error ) {
// if registeredPath[0] != '/' {
// return nil, errors.New("path should start with slash and should not be empty")
// }
if method == "" || method == "ALL" || method == "ANY" { // then use like it was .Any
return nil , rb . Any ( registeredPath , handlers ... )
}
// no clean path yet because of subdomain indicator/separator which contains a dot.
// but remove the first slash if the relative has already ending with a slash
// it's not needed because later on we do normalize/clean the path, but better do it here too
// for any future updates.
if rb . relativePath [ len ( rb . relativePath ) - 1 ] == '/' {
if registeredPath [ 0 ] == '/' {
registeredPath = registeredPath [ 1 : ]
}
}
fullpath := rb . relativePath + registeredPath // for now, keep the last "/" if any, "/xyz/"
routeHandlers := joinHandlers ( rb . middleware , handlers )
// here we separate the subdomain and relative path
subdomain , path := exctractSubdomain ( fullpath )
if len ( rb . doneHandlers ) > 0 {
routeHandlers = append ( routeHandlers , rb . doneHandlers ... ) // register the done middleware, if any
}
r , err := NewRoute ( method , subdomain , path , routeHandlers , rb . macros )
if err != nil {
return nil , err
}
// global
rb . routes . register ( r )
// per -party
rb . apiRoutes = append ( rb . apiRoutes , r )
// should we remove the rb.apiRoutes on the .Party (new children party) ?, No, because the user maybe use this party later
// should we add to the 'inheritance tree' the rb.apiRoutes, No, these are for this specific party only, because the user propably, will have unexpected behavior when using Use/Use, Done/DoneFunc
return r , nil
}
// Party is just a group joiner of routes which have the same prefix and share same middleware(s) also.
// Party could also be named as 'Join' or 'Node' or 'Group' , Party chosen because it is fun.
func ( rb * APIBuilder ) Party ( relativePath string , handlers ... context . Handler ) Party {
parentPath := rb . relativePath
dot := string ( SubdomainIndicator [ 0 ] )
if len ( parentPath ) > 0 && parentPath [ 0 ] == '/' && strings . HasSuffix ( relativePath , dot ) { // if ends with . , example: admin., it's subdomain->
parentPath = parentPath [ 1 : ] // remove first slash
}
fullpath := parentPath + relativePath
// append the parent's +child's handlers
middleware := joinHandlers ( rb . middleware , handlers )
return & APIBuilder {
// global/api builder
macros : rb . macros ,
routes : rb . routes ,
errorCodeHandlers : rb . errorCodeHandlers ,
doneHandlers : rb . doneHandlers ,
// per-party/children
middleware : middleware ,
relativePath : fullpath ,
}
}
2017-06-10 02:56:42 +02:00
// Macros returns the macro map which is responsible
// to register custom macro functions for all routes.
2017-06-12 03:47:16 +02:00
//
// Learn more at: https://github.com/kataras/iris/tree/master/_examples/beginner/routing/dynamic-path
func ( rb * APIBuilder ) Macros ( ) * macro . Map {
Publish the new version :airplane: | Look description please!
# FAQ
### Looking for free support?
http://support.iris-go.com
https://kataras.rocket.chat/channel/iris
### Looking for previous versions?
https://github.com/kataras/iris#version
### Should I upgrade my Iris?
Developers are not forced to upgrade if they don't really need it. Upgrade whenever you feel ready.
> Iris uses the [vendor directory](https://docs.google.com/document/d/1Bz5-UB7g2uPBdOx-rw5t9MxJwkfpx90cqG9AFL0JAYo) feature, so you get truly reproducible builds, as this method guards against upstream renames and deletes.
**How to upgrade**: Open your command-line and execute this command: `go get -u github.com/kataras/iris`.
For further installation support, please click [here](http://support.iris-go.com/d/16-how-to-install-iris-web-framework).
### About our new home page
http://iris-go.com
Thanks to [Santosh Anand](https://github.com/santoshanand) the http://iris-go.com has been upgraded and it's really awesome!
[Santosh](https://github.com/santoshanand) is a freelancer, he has a great knowledge of nodejs and express js, Android, iOS, React Native, Vue.js etc, if you need a developer to find or create a solution for your problem or task, please contact with him.
The amount of the next two or three donations you'll send they will be immediately transferred to his own account balance, so be generous please!
Read more at https://github.com/kataras/iris/blob/master/HISTORY.md
Former-commit-id: eec2d71bbe011d6b48d2526eb25919e36e5ad94e
2017-06-03 22:22:52 +02:00
return rb . macros
}
// GetRoutes returns the routes information,
// some of them can be changed at runtime some others not.
//
// Needs refresh of the router to Method or Path or Handlers changes to take place.
func ( rb * APIBuilder ) GetRoutes ( ) [ ] * Route {
return rb . routes . getAll ( )
}
// GetRoute returns the registered route based on its name, otherwise nil.
// One note: "routeName" should be case-sensitive.
func ( rb * APIBuilder ) GetRoute ( routeName string ) * Route {
return rb . routes . get ( routeName )
}
// Use appends Handler(s) to the current Party's routes and child routes.
// If the current Party is the root, then it registers the middleware to all child Parties' routes too.
func ( rb * APIBuilder ) Use ( handlers ... context . Handler ) {
rb . middleware = append ( rb . middleware , handlers ... )
}
// Done appends to the very end, Handler(s) to the current Party's routes and child routes
// The difference from .Use is that this/or these Handler(s) are being always running last.
func ( rb * APIBuilder ) Done ( handlers ... context . Handler ) {
if len ( rb . 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 ( rb . apiRoutes ) ; i < n ; i ++ {
routeInfo := rb . apiRoutes [ i ]
routeInfo . Handlers = append ( routeInfo . Handlers , handlers ... )
}
} else {
// register them on the doneHandlers, which will be used on Handle to append these middlweare as the last handler(s)
rb . doneHandlers = append ( rb . doneHandlers , handlers ... )
}
}
// UseGlobal registers Handler middleware to the beginning, prepends them instead of append
//
// Use it when you want to add a global middleware to all parties, to all routes in all subdomains
// It should be called right before Listen functions
func ( rb * APIBuilder ) UseGlobal ( handlers ... context . Handler ) {
for _ , r := range rb . routes . routes {
r . Handlers = append ( handlers , r . Handlers ... ) // prepend the handlers
}
rb . middleware = append ( handlers , rb . middleware ... ) // set as middleware on the next routes too
// rb.Use(handlers...)
}
// None registers an "offline" route
// see context.ExecRoute(routeName) and
// party.Routes().Online(handleResultRouteInfo, "GET") and
// Offline(handleResultRouteInfo)
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) None ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( MethodNone , path , handlers ... )
}
// Get registers a route for the Get http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Get ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodGet , path , handlers ... )
}
// Post registers a route for the Post http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Post ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodPost , path , handlers ... )
}
// Put registers a route for the Put http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Put ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodPut , path , handlers ... )
}
// Delete registers a route for the Delete http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Delete ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodDelete , path , handlers ... )
}
// Connect registers a route for the Connect http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Connect ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodConnect , path , handlers ... )
}
// Head registers a route for the Head http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Head ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodHead , path , handlers ... )
}
// Options registers a route for the Options http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Options ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodOptions , path , handlers ... )
}
// Patch registers a route for the Patch http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Patch ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodPatch , path , handlers ... )
}
// Trace registers a route for the Trace http method.
//
// Returns a *Route and an error which will be filled if route wasn't registered successfully.
func ( rb * APIBuilder ) Trace ( path string , handlers ... context . Handler ) ( * Route , error ) {
return rb . Handle ( http . MethodTrace , path , handlers ... )
}
// Any registers a route for ALL of the http methods
// (Get,Post,Put,Head,Patch,Options,Connect,Delete).
func ( rb * APIBuilder ) Any ( registeredPath string , handlers ... context . Handler ) error {
for _ , k := range AllMethods {
if _ , err := rb . Handle ( k , registeredPath , handlers ... ) ; err != nil {
return err
}
}
return nil
}
// StaticCacheDuration expiration duration for INACTIVE file handlers, it's the only one global configuration
// which can be changed.
var StaticCacheDuration = 20 * time . Second
const (
lastModifiedHeaderKey = "Last-Modified"
ifModifiedSinceHeaderKey = "If-Modified-Since"
contentDispositionHeaderKey = "Content-Disposition"
cacheControlHeaderKey = "Cache-Control"
contentEncodingHeaderKey = "Content-Encoding"
acceptEncodingHeaderKey = "Accept-Encoding"
// contentLengthHeaderKey represents the header["Content-Length"]
contentLengthHeaderKey = "Content-Length"
contentTypeHeaderKey = "Content-Type"
varyHeaderKey = "Vary"
)
func ( rb * APIBuilder ) registerResourceRoute ( reqPath string , h context . Handler ) ( * Route , error ) {
if _ , err := rb . Head ( reqPath , h ) ; err != nil {
return nil , err
}
return rb . Get ( reqPath , h )
}
// StaticHandler returns a new Handler which is ready
// to serve all kind of static files.
//
// Note:
// The only difference from package-level `StaticHandler`
// is that this `StaticHandler`` receives a request path which
// is appended to the party's relative path and stripped here,
// so `iris.StripPath` is useless and should not being used here.
//
// Usage:
// app := iris.New()
// ...
// mySubdomainFsServer := app.Party("mysubdomain.")
// h := mySubdomainFsServer.StaticHandler("/static", "./static_files", false, false)
// /* http://mysubdomain.mydomain.com/static/css/style.css */
// mySubdomainFsServer.Get("/static", h)
// ...
//
func ( rb * APIBuilder ) StaticHandler ( reqPath string , systemPath string , showList bool , enableGzip bool , exceptRoutes ... * Route ) context . Handler {
return StripPrefix ( rb . relativePath + reqPath ,
StaticHandler ( systemPath , showList , enableGzip ) )
}
// 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 and HEAD method routes.
// If second parameter is empty, otherwise the requestPath is the second parameter
// it uses gzip compression (compression on each request, no file cache).
//
// Returns the GET *Route.
func ( rb * APIBuilder ) StaticServe ( systemPath string , requestPath ... string ) ( * Route , error ) {
var reqPath string
if len ( requestPath ) == 0 {
reqPath = strings . Replace ( systemPath , string ( os . PathSeparator ) , "/" , - 1 ) // replaces any \ to /
reqPath = strings . Replace ( reqPath , "//" , "/" , - 1 ) // for any case, replaces // to /
reqPath = strings . Replace ( reqPath , "." , "" , - 1 ) // replace any dots (./mypath -> /mypath)
} else {
reqPath = requestPath [ 0 ]
}
return rb . Get ( joinPath ( reqPath , WildcardParam ( "file" ) ) , func ( ctx context . Context ) {
filepath := ctx . Params ( ) . Get ( "file" )
spath := strings . Replace ( filepath , "/" , string ( os . PathSeparator ) , - 1 )
spath = path . Join ( systemPath , spath )
if ! DirectoryExists ( spath ) {
ctx . NotFound ( )
return
}
if err := ctx . ServeFile ( spath , true ) ; err != nil {
ctx . StatusCode ( http . StatusInternalServerError )
}
} )
}
// StaticContent registers a GET and HEAD method routes to the requestPath
// that are ready to serve raw static bytes, memory cached.
//
// Returns the GET *Route.
func ( rb * APIBuilder ) StaticContent ( reqPath string , cType string , content [ ] byte ) ( * Route , error ) {
modtime := time . Now ( )
h := func ( ctx context . Context ) {
if err := ctx . WriteWithExpiration ( http . StatusOK , content , cType , modtime ) ; err != nil {
ctx . Application ( ) . Log ( "error while serving []byte via StaticContent: %s" , err . Error ( ) )
}
}
return rb . registerResourceRoute ( reqPath , h )
}
// StaticEmbedded used when files are distributed 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.
//
// Returns the GET *Route.
//
// Example: https://github.com/kataras/iris/tree/master/_examples/intermediate/serve-embedded-files
func ( rb * APIBuilder ) StaticEmbedded ( requestPath string , vdir string , assetFn func ( name string ) ( [ ] byte , error ) , namesFn func ( ) [ ] string ) ( * Route , error ) {
paramName := "path"
requestPath = joinPath ( requestPath , WildcardParam ( paramName ) )
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
}
names = append ( names , cleanPath ( path ) )
// 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 {
return nil , errors . New ( "unable to locate any embedded files located to the (virtual) directory: " + vdir )
}
modtime := time . Now ( )
h := func ( ctx context . Context ) {
reqPath := ctx . Params ( ) . Get ( paramName )
for _ , path := range names {
// in order to map "/" as "/index.html"
// as requested here: https://github.com/kataras/iris/issues/633#issuecomment-281691851
if path == "/index.html" {
if reqPath [ len ( reqPath ) - 1 ] == '/' {
reqPath = "/index.html"
}
}
if path != reqPath {
continue
}
cType := TypeByExtension ( path )
fullpath := vdir + path
buf , err := assetFn ( fullpath )
if err != nil {
continue
}
if err := ctx . WriteWithExpiration ( http . StatusOK , buf , cType , modtime ) ; err != nil {
ctx . StatusCode ( http . StatusInternalServerError )
ctx . StopExecution ( )
}
return
}
// not found or error
ctx . NotFound ( )
}
return rb . registerResourceRoute ( requestPath , h )
}
// errDirectoryFileNotFound returns an error with message: 'Directory or file %s couldn't found. Trace: +error trace'
var errDirectoryFileNotFound = errors . New ( "Directory or file %s couldn't found. Trace: %s" )
// 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 (desktop, mobile and so on).
//
// Returns the GET *Route.
func ( rb * APIBuilder ) Favicon ( favPath string , requestPath ... string ) ( * Route , error ) {
favPath = Abs ( favPath )
f , err := os . Open ( favPath )
if err != nil {
return nil , errDirectoryFileNotFound . Format ( favPath , err . Error ( ) )
}
// ignore error f.Close()
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 rb . Favicon ( path . Join ( favPath , "favicon.png" ) )
}
favPath = fav
fi , _ = f . Stat ( )
}
cType := TypeByFilename ( favPath )
// 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 {
// Here we are before actually run the server.
// So we could panic but we don't,
// we just interrupt with a message
// to the (user-defined) logger.
return nil , errDirectoryFileNotFound .
Format ( favPath , "favicon: couldn't read the data bytes for file: " + err . Error ( ) )
}
modtime := ""
h := func ( ctx context . Context ) {
if modtime == "" {
modtime = fi . ModTime ( ) . UTC ( ) . Format ( ctx . Application ( ) . ConfigurationReadOnly ( ) . GetTimeFormat ( ) )
}
if t , err := time . Parse ( ctx . Application ( ) . ConfigurationReadOnly ( ) . GetTimeFormat ( ) , ctx . GetHeader ( ifModifiedSinceHeaderKey ) ) ; err == nil && fi . ModTime ( ) . Before ( t . Add ( StaticCacheDuration ) ) {
ctx . ResponseWriter ( ) . Header ( ) . Del ( contentTypeHeaderKey )
ctx . ResponseWriter ( ) . Header ( ) . Del ( contentLengthHeaderKey )
ctx . StatusCode ( http . StatusNotModified )
return
}
ctx . ResponseWriter ( ) . Header ( ) . Set ( contentTypeHeaderKey , cType )
ctx . ResponseWriter ( ) . Header ( ) . Set ( lastModifiedHeaderKey , modtime )
ctx . StatusCode ( http . StatusOK )
if _ , err := ctx . Write ( cacheFav ) ; err != nil {
ctx . Application ( ) . Log ( "error while trying to serve the favicon: %s" , err . Error ( ) )
}
}
reqPath := "/favicon" + path . Ext ( fi . Name ( ) ) //we could use the filename, but because standards is /favicon.ico/.png.
if len ( requestPath ) > 0 && requestPath [ 0 ] != "" {
reqPath = requestPath [ 0 ]
}
return rb . registerResourceRoute ( reqPath , h )
}
// 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
// third OPTIONAL parameter: the exception routes
// (= give priority to these routes instead of the static handler)
// for more options look rb.StaticHandler.
//
// rb.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(requestPath, systemPath, listingDirectories: false, gzip: false ).
//
// Returns the GET *Route.
func ( rb * APIBuilder ) StaticWeb ( reqPath string , systemPath string , exceptRoutes ... * Route ) ( * Route , error ) {
h := rb . StaticHandler ( reqPath , systemPath , false , false , exceptRoutes ... )
paramName := "file"
routePath := joinPath ( reqPath , WildcardParam ( paramName ) )
handler := func ( ctx context . Context ) {
h ( ctx )
// re-check the content type here for any case,
// although the new code does it automatically but it's good to have it here.
if ctx . GetStatusCode ( ) >= 200 && ctx . GetStatusCode ( ) < 400 {
if fname := ctx . Params ( ) . Get ( paramName ) ; fname != "" {
cType := TypeByFilename ( fname )
ctx . ContentType ( cType )
}
}
}
return rb . registerResourceRoute ( routePath , handler )
}
// OnErrorCode registers an error http status code
// based on the "statusCode" >= 400.
// The handler is being wrapepd by a generic
// handler which will try to reset
// the body if recorder was enabled
// and/or disable the gzip if gzip response recorder
// was active.
func ( rb * APIBuilder ) OnErrorCode ( statusCode int , handler context . Handler ) {
rb . errorCodeHandlers . Register ( statusCode , handler )
}
// FireErrorCode executes an error http status code handler
// based on the context's status code.
//
// If a handler is not already registered,
// then it creates & registers a new trivial handler on the-fly.
func ( rb * APIBuilder ) FireErrorCode ( ctx context . Context ) {
rb . errorCodeHandlers . Fire ( ctx )
}
// Layout oerrides the parent template layout with a more specific layout for this Party
// returns this Party, to continue as normal
// Usage:
// app := iris.New()
// my := app.Party("/my").Layout("layouts/mylayout.html")
// {
// my.Get("/", func(ctx context.Context) {
// ctx.MustRender("page1.html", nil)
// })
// }
func ( rb * APIBuilder ) Layout ( tmplLayoutFile string ) Party {
rb . Use ( func ( ctx context . Context ) {
ctx . ViewLayout ( tmplLayoutFile )
ctx . Next ( )
} )
return rb
}
// joinHandlers uses to create a copy of all Handlers and return them in order to use inside the node
func joinHandlers ( Handlers1 context . Handlers , Handlers2 context . Handlers ) context . Handlers {
nowLen := len ( Handlers1 )
totalLen := nowLen + len ( Handlers2 )
// create a new slice of Handlers in order to store all handlers, the already handlers(Handlers) and the new
newHandlers := make ( context . Handlers , totalLen )
//copy the already Handlers to the just created
copy ( newHandlers , Handlers1 )
//start from there we finish, and store the new Handlers too
copy ( newHandlers [ nowLen : ] , Handlers2 )
return newHandlers
}