2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2015-02-26 23:34:43 +08:00
package beego
2013-09-10 00:00:11 +08:00
import (
"fmt"
"html/template"
"net/http"
2015-02-26 23:34:43 +08:00
"reflect"
2013-09-10 00:00:11 +08:00
"runtime"
2013-09-11 17:00:39 +08:00
"strconv"
2015-02-26 23:34:43 +08:00
"strings"
"github.com/astaxie/beego/context"
"github.com/astaxie/beego/utils"
2013-09-10 00:00:11 +08:00
)
2015-02-26 23:34:43 +08:00
const (
errorTypeHandler = iota
errorTypeController
2013-09-11 16:31:18 +08:00
)
2015-02-26 23:34:43 +08:00
2013-09-10 00:00:11 +08:00
var tpl = `
2013-12-06 00:44:54 -05:00
< ! DOCTYPE html >
< html >
2013-09-10 00:00:11 +08:00
< head >
< meta http - equiv = "Content-Type" content = "text/html; charset=UTF-8" / >
< title > beego application error < / title >
< style >
html , body , body * { padding : 0 ; margin : 0 ; }
# header { background : # ffd ; border - bottom : solid 2 px # A31515 ; padding : 20 px 10 px ; }
# header h2 { }
# footer { border - top : solid 1 px # aaa ; padding : 5 px 10 px ; font - size : 12 px ; color : green ; }
# content { padding : 5 px ; }
# content . stack b { font - size : 13 px ; color : red ; }
# content . stack pre { padding - left : 10 px ; }
table { }
td . t { text - align : right ; padding - right : 5 px ; color : # 888 ; }
2013-12-06 00:44:54 -05:00
< / style >
2013-09-10 00:00:11 +08:00
< script type = "text/javascript" >
< / script >
2013-12-06 00:44:54 -05:00
< / head >
2013-09-10 00:00:11 +08:00
< body >
< div id = "header" >
< h2 > { { . AppError } } < / h2 >
< / div >
< div id = "content" >
< table >
< tr >
< td class = "t" > Request Method : < / td > < td > { { . RequestMethod } } < / td >
< / tr >
< tr >
< td class = "t" > Request URL : < / td > < td > { { . RequestURL } } < / td >
< / tr >
< tr >
< td class = "t" > RemoteAddr : < / td > < td > { { . RemoteAddr } } < / td >
< / tr >
< / table >
< div class = "stack" >
< b > Stack < / b >
< pre > { { . Stack } } < / pre >
< / div >
< / div >
< div id = "footer" >
< p > beego { { . BeegoVersion } } ( beego framework ) < / p >
< p > golang version : { { . GoVersion } } < / p >
< / div >
< / body >
2013-12-06 00:44:54 -05:00
< / html >
2013-09-10 00:00:11 +08:00
`
2014-01-17 16:03:01 +08:00
// render default application error page with error and stack string.
2015-12-21 17:16:58 +08:00
func showErr ( err interface { } , ctx * context . Context , stack string ) {
2013-09-10 00:00:11 +08:00
t , _ := template . New ( "beegoerrortemp" ) . Parse ( tpl )
2015-12-21 17:16:58 +08:00
data := map [ string ] string {
"AppError" : fmt . Sprintf ( "%s:%v" , BConfig . AppName , err ) ,
"RequestMethod" : ctx . Input . Method ( ) ,
"RequestURL" : ctx . Input . URI ( ) ,
"RemoteAddr" : ctx . Input . IP ( ) ,
"Stack" : stack ,
"BeegoVersion" : VERSION ,
"GoVersion" : runtime . Version ( ) ,
}
2016-08-30 21:00:27 +08:00
if ctx . Output . Status != 0 {
ctx . ResponseWriter . WriteHeader ( ctx . Output . Status )
} else {
ctx . ResponseWriter . WriteHeader ( 500 )
}
2015-02-26 23:34:43 +08:00
t . Execute ( ctx . ResponseWriter , data )
2013-09-10 00:00:11 +08:00
}
var errtpl = `
< ! DOCTYPE html >
< html lang = "en" >
< head >
< meta http - equiv = "Content-Type" content = "text/html; charset=UTF-8" >
< title > { { . Title } } < / title >
< style type = "text/css" >
* {
margin : 0 ;
padding : 0 ;
}
body {
background - color : # EFEFEF ;
font : .9 em "Lucida Sans Unicode" , "Lucida Grande" , sans - serif ;
}
# wrapper {
width : 600 px ;
margin : 40 px auto 0 ;
text - align : center ;
- moz - box - shadow : 5 px 5 px 10 px rgba ( 0 , 0 , 0 , 0.3 ) ;
- webkit - box - shadow : 5 px 5 px 10 px rgba ( 0 , 0 , 0 , 0.3 ) ;
box - shadow : 5 px 5 px 10 px rgba ( 0 , 0 , 0 , 0.3 ) ;
}
# wrapper h1 {
color : # FFF ;
text - align : center ;
margin - bottom : 20 px ;
}
# wrapper a {
display : block ;
font - size : .9 em ;
padding - top : 20 px ;
color : # FFF ;
text - decoration : none ;
text - align : center ;
}
# container {
width : 600 px ;
padding - bottom : 15 px ;
background - color : # FFFFFF ;
}
. navtop {
height : 40 px ;
background - color : # 24 B2EB ;
padding : 13 px ;
}
. content {
padding : 10 px 10 px 25 px ;
background : # FFFFFF ;
margin : ;
color : # 333 ;
}
a . button {
color : white ;
padding : 15 px 20 px ;
text - shadow : 1 px 1 px 0 # 00 A5FF ;
font - weight : bold ;
text - align : center ;
border : 1 px solid # 24 B2EB ;
margin : 0 px 200 px ;
clear : both ;
background - color : # 24 B2EB ;
border - radius : 100 px ;
- moz - border - radius : 100 px ;
- webkit - border - radius : 100 px ;
}
a . button : hover {
text - decoration : none ;
background - color : # 24 B2EB ;
}
< / style >
< / head >
< body >
< div id = "wrapper" >
< div id = "container" >
< div class = "navtop" >
< h1 > { { . Title } } < / h1 >
< / div >
< div id = "content" >
{ { . Content } }
< a href = "/" title = "Home" class = "button" > Go Home < / a > < br / >
2014-02-10 12:55:53 +08:00
< br > Powered by beego { { . BeegoVersion } }
2013-09-10 00:00:11 +08:00
< / div >
< / div >
< / div >
< / body >
< / html >
`
2015-02-26 23:34:43 +08:00
type errorInfo struct {
controllerType reflect . Type
handler http . HandlerFunc
method string
errorType int
}
2016-01-18 00:18:21 +08:00
// ErrorMaps holds map of http handlers for each error string.
2015-09-18 10:32:21 +08:00
// there is 10 kinds default error(40x and 50x)
2015-09-17 23:05:45 +08:00
var ErrorMaps = make ( map [ string ] * errorInfo , 10 )
2013-09-10 00:00:11 +08:00
2015-02-26 23:49:24 +08:00
// show 401 unauthorized error.
func unauthorized ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
401 ,
"<br>The page you have requested can't be authorized." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br>The credentials you supplied are incorrect" +
"<br>There are errors in the website address" +
"</ul>" ,
)
2013-09-10 00:00:11 +08:00
}
2015-02-26 23:49:24 +08:00
// show 402 Payment Required
func paymentRequired ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
402 ,
"<br>The page you have requested Payment Required." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br>The credentials you supplied are incorrect" +
"<br>There are errors in the website address" +
"</ul>" ,
)
2013-09-10 00:00:11 +08:00
}
2014-01-17 16:03:01 +08:00
// show 403 forbidden error.
2015-02-26 23:49:24 +08:00
func forbidden ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
403 ,
"<br>The page you have requested is forbidden." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br>Your address may be blocked" +
"<br>The site may be disabled" +
"<br>You need to log in" +
"</ul>" ,
)
2013-09-10 00:00:11 +08:00
}
2016-03-17 09:35:14 +08:00
// show 404 not found error.
2015-02-26 23:49:24 +08:00
func notFound ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
404 ,
"<br>The page you have requested has flown the coop." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br>The page has moved" +
"<br>The page no longer exists" +
"<br>You were looking for your puppy and got lost" +
"<br>You like 404 pages" +
"</ul>" ,
)
2015-02-26 23:49:24 +08:00
}
// show 405 Method Not Allowed
func methodNotAllowed ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
405 ,
"<br>The method you have requested Not Allowed." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br>The method specified in the Request-Line is not allowed for the resource identified by the Request-URI" +
"<br>The response MUST include an Allow header containing a list of valid methods for the requested resource." +
"</ul>" ,
)
2013-09-10 00:00:11 +08:00
}
2014-01-17 16:03:01 +08:00
// show 500 internal server error.
2015-02-26 23:49:24 +08:00
func internalServerError ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
500 ,
"<br>The page you have requested is down right now." +
"<br><br><ul>" +
"<br>Please try again later and report the error to the website administrator" +
"<br></ul>" ,
)
2013-09-10 00:00:11 +08:00
}
2015-02-26 23:49:24 +08:00
// show 501 Not Implemented.
func notImplemented ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
501 ,
"<br>The page you have requested is Not Implemented." +
"<br><br><ul>" +
"<br>Please try again later and report the error to the website administrator" +
"<br></ul>" ,
)
2015-02-26 23:49:24 +08:00
}
// show 502 Bad Gateway.
func badGateway ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
502 ,
"<br>The page you have requested is down right now." +
"<br><br><ul>" +
"<br>The server, while acting as a gateway or proxy, received an invalid response from the upstream server it accessed in attempting to fulfill the request." +
"<br>Please try again later and report the error to the website administrator" +
"<br></ul>" ,
)
2015-02-26 23:49:24 +08:00
}
// show 503 service unavailable error.
func serviceUnavailable ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
503 ,
"<br>The page you have requested is unavailable." +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br><br>The page is overloaded" +
"<br>Please try again later." +
"</ul>" ,
)
2015-02-26 23:49:24 +08:00
}
// show 504 Gateway Timeout.
func gatewayTimeout ( rw http . ResponseWriter , r * http . Request ) {
2016-03-17 09:35:14 +08:00
responseError ( rw , r ,
504 ,
"<br>The page you have requested is unavailable" +
"<br>Perhaps you are here because:" +
"<br><br><ul>" +
"<br><br>The server, while acting as a gateway or proxy, did not receive a timely response from the upstream server specified by the URI." +
"<br>Please try again later." +
"</ul>" ,
)
}
func responseError ( rw http . ResponseWriter , r * http . Request , errCode int , errContent string ) {
2015-02-26 23:49:24 +08:00
t , _ := template . New ( "beegoerrortemp" ) . Parse ( errtpl )
2015-12-21 17:16:58 +08:00
data := map [ string ] interface { } {
2016-03-17 09:35:14 +08:00
"Title" : http . StatusText ( errCode ) ,
2015-12-21 17:16:58 +08:00
"BeegoVersion" : VERSION ,
2016-04-04 21:32:43 +08:00
"Content" : template . HTML ( errContent ) ,
2015-12-21 17:16:58 +08:00
}
2015-02-26 23:49:24 +08:00
t . Execute ( rw , data )
2013-12-06 00:44:54 -05:00
}
2015-02-26 23:34:43 +08:00
// ErrorHandler registers http.HandlerFunc to each http err code string.
// usage:
// beego.ErrorHandler("404",NotFound)
// beego.ErrorHandler("500",InternalServerError)
2015-09-08 21:41:38 +08:00
func ErrorHandler ( code string , h http . HandlerFunc ) * App {
2015-12-21 17:16:58 +08:00
ErrorMaps [ code ] = & errorInfo {
errorType : errorTypeHandler ,
handler : h ,
method : code ,
}
2015-02-26 23:34:43 +08:00
return BeeApp
}
// ErrorController registers ControllerInterface to each http err code string.
// usage:
2015-09-08 21:41:38 +08:00
// beego.ErrorController(&controllers.ErrorController{})
2015-02-26 23:34:43 +08:00
func ErrorController ( c ControllerInterface ) * App {
reflectVal := reflect . ValueOf ( c )
rt := reflectVal . Type ( )
ct := reflect . Indirect ( reflectVal ) . Type ( )
for i := 0 ; i < rt . NumMethod ( ) ; i ++ {
2015-09-18 17:59:28 +08:00
methodName := rt . Method ( i ) . Name
if ! utils . InSlice ( methodName , exceptMethod ) && strings . HasPrefix ( methodName , "Error" ) {
errName := strings . TrimPrefix ( methodName , "Error" )
2015-12-21 17:16:58 +08:00
ErrorMaps [ errName ] = & errorInfo {
errorType : errorTypeController ,
controllerType : ct ,
method : methodName ,
}
2015-02-26 23:34:43 +08:00
}
}
return BeeApp
}
2016-06-01 21:30:04 +08:00
// Exception Write HttpStatus with errCode and Exec error handler if exist.
func Exception ( errCode uint64 , ctx * context . Context ) {
exception ( strconv . FormatUint ( errCode , 10 ) , ctx )
}
2014-01-17 16:03:01 +08:00
// show error string as simple text message.
2015-09-17 10:33:12 +08:00
// if error string is empty, show 503 or 500 error as default.
func exception ( errCode string , ctx * context . Context ) {
2015-09-18 17:03:00 +08:00
atoi := func ( code string ) int {
v , err := strconv . Atoi ( code )
if err == nil {
return v
2015-09-18 10:31:10 +08:00
}
2016-03-17 09:46:34 +08:00
if ctx . Output . Status == 0 {
return 503
}
return ctx . Output . Status
2015-09-18 17:03:00 +08:00
}
for _ , ec := range [ ] string { errCode , "503" , "500" } {
2015-09-17 10:33:12 +08:00
if h , ok := ErrorMaps [ ec ] ; ok {
2015-09-18 17:03:00 +08:00
executeError ( h , ctx , atoi ( ec ) )
2015-09-17 10:33:12 +08:00
return
}
2013-09-11 17:00:39 +08:00
}
2015-09-17 10:33:12 +08:00
//if 50x error has been removed from errorMap
2015-09-18 17:03:00 +08:00
ctx . ResponseWriter . WriteHeader ( atoi ( errCode ) )
2015-09-17 10:33:12 +08:00
ctx . WriteString ( errCode )
2015-02-26 23:34:43 +08:00
}
2015-03-30 20:35:57 +08:00
func executeError ( err * errorInfo , ctx * context . Context , code int ) {
2015-02-26 23:34:43 +08:00
if err . errorType == errorTypeHandler {
2016-02-01 22:39:41 +08:00
ctx . ResponseWriter . WriteHeader ( code )
2015-02-26 23:34:43 +08:00
err . handler ( ctx . ResponseWriter , ctx . Request )
return
2015-02-23 11:50:13 +08:00
}
2015-02-26 23:34:43 +08:00
if err . errorType == errorTypeController {
2015-03-30 20:35:57 +08:00
ctx . Output . SetStatus ( code )
2015-02-26 23:34:43 +08:00
//Invoke the request handler
vc := reflect . New ( err . controllerType )
execController , ok := vc . Interface ( ) . ( ControllerInterface )
if ! ok {
panic ( "controller is not ControllerInterface" )
}
//call the controller init function
execController . Init ( ctx , err . controllerType . Name ( ) , err . method , vc . Interface ( ) )
//call prepare function
execController . Prepare ( )
execController . URLMapping ( )
method := vc . MethodByName ( err . method )
2015-12-21 17:16:58 +08:00
method . Call ( [ ] reflect . Value { } )
2015-02-26 23:34:43 +08:00
//render template
2015-12-09 23:35:04 +08:00
if BConfig . WebConfig . AutoRender {
2015-03-30 20:35:57 +08:00
if err := execController . Render ( ) ; err != nil {
panic ( err )
2015-02-26 23:34:43 +08:00
}
}
// finish all runrouter. release resource
execController . Finish ( )
2015-02-23 11:50:13 +08:00
}
2013-09-11 17:00:39 +08:00
}