Beego/router.go

858 lines
22 KiB
Go
Raw Normal View History

2014-04-12 05:18:18 +00:00
// Beego (http://beego.me/)
// @description beego is an open-source, high-performance web framework for the Go programming language.
// @link http://github.com/astaxie/beego for the canonical source repository
// @license http://github.com/astaxie/beego/blob/master/LICENSE
// @authors astaxie
2012-12-18 07:18:43 +00:00
package beego
import (
2013-12-31 12:47:48 +00:00
"bufio"
"errors"
2013-04-11 06:35:43 +00:00
"fmt"
2013-12-31 12:47:48 +00:00
"net"
2012-12-18 07:18:43 +00:00
"net/http"
2014-06-09 02:11:37 +00:00
"os"
2014-06-08 12:24:01 +00:00
"path"
2014-06-09 02:11:37 +00:00
"path/filepath"
2012-12-18 07:18:43 +00:00
"reflect"
"runtime"
2013-07-27 02:25:14 +00:00
"strconv"
2012-12-18 07:18:43 +00:00
"strings"
"time"
2013-12-03 13:37:39 +00:00
beecontext "github.com/astaxie/beego/context"
"github.com/astaxie/beego/middleware"
"github.com/astaxie/beego/toolbox"
"github.com/astaxie/beego/utils"
2012-12-18 07:18:43 +00:00
)
const (
// default filter execution points
BeforeRouter = iota
BeforeExec
AfterExec
FinishRouter
)
const (
routerTypeBeego = iota
routerTypeRESTFul
routerTypeHandler
)
var (
// supported http methods.
2014-04-05 16:18:21 +00:00
HTTPMETHOD = []string{"get", "post", "put", "delete", "patch", "options", "head", "trace", "connect"}
2014-01-01 09:57:57 +00:00
// these beego.Controller's methods shouldn't reflect to AutoRouter
exceptMethod = []string{"Init", "Prepare", "Finish", "Render", "RenderString",
"RenderBytes", "Redirect", "Abort", "StopRun", "UrlFor", "ServeJson", "ServeJsonp",
"ServeXml", "Input", "ParseForm", "GetString", "GetStrings", "GetInt", "GetBool",
"GetFloat", "GetFile", "SaveToFile", "StartSession", "SetSession", "GetSession",
"DelSession", "SessionRegenerateID", "DestroySession", "IsAjax", "GetSecureCookie",
"SetSecureCookie", "XsrfToken", "CheckXsrfCookie", "XsrfFormHtml",
"GetControllerAndAction"}
)
// To append a slice's value into "exceptMethod", for controller's methods shouldn't reflect to AutoRouter
func ExceptMethodAppend(action string) {
exceptMethod = append(exceptMethod, action)
}
2012-12-18 07:18:43 +00:00
type controllerInfo struct {
controllerType reflect.Type
methods map[string]string
handler http.Handler
runfunction FilterFunc
routerType int
2012-12-18 07:18:43 +00:00
}
// ControllerRegistor containers registered router rules, controller handlers and filters.
2012-12-18 07:18:43 +00:00
type ControllerRegistor struct {
2014-06-09 02:11:37 +00:00
routers map[string]*Tree
2013-12-20 16:34:59 +00:00
enableFilter bool
filters map[int][]*FilterRouter
2012-12-18 07:18:43 +00:00
}
// NewControllerRegistor returns a new ControllerRegistor.
2012-12-18 07:18:43 +00:00
func NewControllerRegistor() *ControllerRegistor {
2013-07-27 02:25:14 +00:00
return &ControllerRegistor{
2014-06-09 02:11:37 +00:00
routers: make(map[string]*Tree),
2014-06-08 12:24:01 +00:00
filters: make(map[int][]*FilterRouter),
2013-07-27 02:25:14 +00:00
}
2012-12-18 07:18:43 +00:00
}
// Add controller handler and pattern rules to ControllerRegistor.
// usage:
// default methods is the same name as method
// Add("/user",&UserController{})
// Add("/api/list",&RestController{},"*:ListFood")
// Add("/api/create",&RestController{},"post:CreateFood")
// Add("/api/update",&RestController{},"put:UpdateFood")
// Add("/api/delete",&RestController{},"delete:DeleteFood")
// Add("/api",&RestController{},"get,post:ApiFunc")
// Add("/simple",&SimpleController{},"get:GetFunc;post:PostFunc")
func (p *ControllerRegistor) Add(pattern string, c ControllerInterface, mappingMethods ...string) {
reflectVal := reflect.ValueOf(c)
t := reflect.Indirect(reflectVal).Type()
methods := make(map[string]string)
if len(mappingMethods) > 0 {
semi := strings.Split(mappingMethods[0], ";")
for _, v := range semi {
colon := strings.Split(v, ":")
if len(colon) != 2 {
panic("method mapping format is invalid")
}
comma := strings.Split(colon[0], ",")
for _, m := range comma {
if m == "*" || utils.InSlice(strings.ToLower(m), HTTPMETHOD) {
if val := reflectVal.MethodByName(colon[1]); val.IsValid() {
methods[strings.ToLower(m)] = colon[1]
} else {
panic(colon[1] + " method doesn't exist in the controller " + t.Name())
}
} else {
panic(v + " is an invalid method mapping. Method doesn't exist " + m)
}
}
}
}
2014-06-08 12:24:01 +00:00
route := &controllerInfo{}
route.methods = methods
route.routerType = routerTypeBeego
route.controllerType = t
2014-06-09 02:11:37 +00:00
if len(methods) == 0 {
for _, m := range HTTPMETHOD {
p.addToRouter(m, pattern, route)
}
} else {
for k, _ := range methods {
if k == "*" {
for _, m := range HTTPMETHOD {
p.addToRouter(m, pattern, route)
}
} else {
p.addToRouter(k, pattern, route)
}
}
}
}
func (p *ControllerRegistor) addToRouter(method, pattern string, r *controllerInfo) {
if t, ok := p.routers[method]; ok {
t.AddRouter(pattern, r)
} else {
t := NewTree()
t.AddRouter(pattern, r)
p.routers[method] = t
}
2014-06-08 12:24:01 +00:00
}
2014-06-08 12:24:01 +00:00
// only when the Runmode is dev will generate router file in the router/auto.go from the controller
// Include(&BankAccount{}, &OrderController{},&RefundController{},&ReceiptController{})
func (p *ControllerRegistor) Include(cList ...ControllerInterface) {
if RunMode == "dev" {
2014-06-09 02:11:37 +00:00
skip := make(map[string]bool, 10)
2014-06-08 12:24:01 +00:00
for _, c := range cList {
reflectVal := reflect.ValueOf(c)
t := reflect.Indirect(reflectVal).Type()
2014-06-09 02:11:37 +00:00
gopath := os.Getenv("GOPATH")
if gopath == "" {
panic("you are in dev mode. So please set gopath")
}
pkgpath := ""
wgopath := filepath.SplitList(gopath)
for _, wg := range wgopath {
wg, _ = filepath.EvalSymlinks(filepath.Join(wg, "src", t.PkgPath()))
if utils.FileExists(wg) {
pkgpath = wg
break
}
}
if pkgpath != "" {
if _, ok := skip[pkgpath]; !ok {
skip[pkgpath] = true
2014-06-09 09:33:04 +00:00
parserPkg(pkgpath, t.PkgPath())
2014-06-09 02:11:37 +00:00
}
}
}
}
for _, c := range cList {
reflectVal := reflect.ValueOf(c)
t := reflect.Indirect(reflectVal).Type()
key := t.PkgPath() + ":" + t.Name()
if comm, ok := GlobalControllerRouter[key]; ok {
2014-06-09 09:33:04 +00:00
for _, a := range comm {
p.Add(a.Router, c, strings.Join(a.AllowHTTPMethods, ",")+":"+a.Method)
2014-06-09 09:33:04 +00:00
}
}
}
}
// add get method
// usage:
// Get("/", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Get(pattern string, f FilterFunc) {
p.AddMethod("get", pattern, f)
}
// add post method
// usage:
// Post("/api", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Post(pattern string, f FilterFunc) {
p.AddMethod("post", pattern, f)
}
// add put method
// usage:
// Put("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Put(pattern string, f FilterFunc) {
p.AddMethod("put", pattern, f)
}
// add delete method
// usage:
// Delete("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Delete(pattern string, f FilterFunc) {
p.AddMethod("delete", pattern, f)
}
// add head method
// usage:
// Head("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Head(pattern string, f FilterFunc) {
p.AddMethod("head", pattern, f)
}
// add patch method
// usage:
// Patch("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Patch(pattern string, f FilterFunc) {
p.AddMethod("patch", pattern, f)
}
// add options method
// usage:
// Options("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Options(pattern string, f FilterFunc) {
p.AddMethod("options", pattern, f)
}
// add all method
// usage:
// Any("/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) Any(pattern string, f FilterFunc) {
p.AddMethod("*", pattern, f)
}
// add http method router
// usage:
// AddMethod("get","/api/:id", func(ctx *context.Context){
// ctx.Output.Body("hello world")
// })
func (p *ControllerRegistor) AddMethod(method, pattern string, f FilterFunc) {
if method != "*" && !utils.InSlice(strings.ToLower(method), HTTPMETHOD) {
panic("not support http method: " + method)
}
route := &controllerInfo{}
route.routerType = routerTypeRESTFul
route.runfunction = f
methods := make(map[string]string)
if method == "*" {
for _, val := range HTTPMETHOD {
methods[val] = val
}
} else {
methods[method] = method
}
route.methods = methods
2014-06-09 02:11:37 +00:00
for k, _ := range methods {
if k == "*" {
for _, m := range HTTPMETHOD {
p.addToRouter(m, pattern, route)
}
} else {
p.addToRouter(k, pattern, route)
}
}
}
2014-06-08 12:24:01 +00:00
// add user defined Handler
func (p *ControllerRegistor) Handler(pattern string, h http.Handler, options ...interface{}) {
route := &controllerInfo{}
route.routerType = routerTypeHandler
route.handler = h
if len(options) > 0 {
2014-06-08 12:24:01 +00:00
if _, ok := options[0].(bool); ok {
pattern = path.Join(pattern, "?:all")
}
2012-12-18 07:18:43 +00:00
}
2014-06-09 02:11:37 +00:00
for _, m := range HTTPMETHOD {
p.addToRouter(m, pattern, route)
}
2012-12-18 07:18:43 +00:00
}
// Add auto router to ControllerRegistor.
// example beego.AddAuto(&MainContorlller{}),
// MainController has method List and Page.
2014-01-01 09:57:57 +00:00
// visit the url /main/list to execute List function
// /main/page to execute Page function.
2013-07-27 02:25:14 +00:00
func (p *ControllerRegistor) AddAuto(c ControllerInterface) {
2014-06-08 12:24:01 +00:00
p.AddAutoPrefix("/", c)
2014-01-01 09:57:57 +00:00
}
// Add auto router to ControllerRegistor with prefix.
// example beego.AddAutoPrefix("/admin",&MainContorlller{}),
// MainController has method List and Page.
// visit the url /admin/main/list to execute List function
// /admin/main/page to execute Page function.
func (p *ControllerRegistor) AddAutoPrefix(prefix string, c ControllerInterface) {
reflectVal := reflect.ValueOf(c)
rt := reflectVal.Type()
ct := reflect.Indirect(reflectVal).Type()
2014-06-08 12:24:01 +00:00
controllerName := strings.ToLower(strings.TrimSuffix(ct.Name(), "Controller"))
2014-01-01 09:57:57 +00:00
for i := 0; i < rt.NumMethod(); i++ {
if !utils.InSlice(rt.Method(i).Name, exceptMethod) {
2014-06-08 12:24:01 +00:00
route := &controllerInfo{}
route.routerType = routerTypeBeego
route.methods = map[string]string{"*": rt.Method(i).Name}
route.controllerType = ct
pattern := path.Join(prefix, controllerName, strings.ToLower(rt.Method(i).Name), "*")
2014-06-09 02:11:37 +00:00
for _, m := range HTTPMETHOD {
p.addToRouter(m, pattern, route)
}
2014-01-01 09:57:57 +00:00
}
2013-07-27 02:25:14 +00:00
}
}
// Add a FilterFunc with pattern rule and action constant.
func (p *ControllerRegistor) InsertFilter(pattern string, pos int, filter FilterFunc) error {
2014-06-08 12:24:01 +00:00
mr := new(FilterRouter)
mr.tree = NewTree()
mr.pattern = pattern
mr.filterFunc = filter
mr.tree.AddRouter(pattern, true)
return p.insertFilterRouter(pos, mr)
}
// add Filter into
func (p *ControllerRegistor) insertFilterRouter(pos int, mr *FilterRouter) error {
2013-11-25 08:04:02 +00:00
p.filters[pos] = append(p.filters[pos], mr)
p.enableFilter = true
return nil
2013-08-11 16:14:42 +00:00
}
// UrlFor does another controller handler in this request function.
// it can access any controller method.
func (p *ControllerRegistor) UrlFor(endpoint string, values ...string) string {
paths := strings.Split(endpoint, ".")
if len(paths) <= 1 {
Warn("urlfor endpoint must like path.controller.method")
return ""
}
if len(values)%2 != 0 {
Warn("urlfor params must key-value pair")
return ""
}
2014-06-08 12:24:01 +00:00
params := make(map[string]string)
if len(values) > 0 {
key := ""
for k, v := range values {
if k%2 == 0 {
key = v
} else {
2014-06-08 12:24:01 +00:00
params[key] = v
}
}
}
controllName := strings.Join(paths[:len(paths)-1], ".")
methodName := paths[len(paths)-1]
2014-06-09 02:11:37 +00:00
for _, t := range p.routers {
ok, url := p.geturl(t, "/", controllName, methodName, params)
if ok {
return url
}
}
2014-06-09 02:11:37 +00:00
return ""
2014-06-08 12:24:01 +00:00
}
func (p *ControllerRegistor) geturl(t *Tree, url, controllName, methodName string, params map[string]string) (bool, string) {
for k, subtree := range t.fixrouters {
u := path.Join(url, k)
ok, u := p.geturl(subtree, u, controllName, methodName, params)
if ok {
return ok, u
}
}
if t.wildcard != nil {
ok, u := p.geturl(t.wildcard, url, controllName, methodName, params)
if ok {
return ok, u
}
}
if t.leaf != nil {
if c, ok := t.leaf.runObject.(*controllerInfo); ok {
if c.routerType == routerTypeBeego && c.controllerType.Name() == controllName {
find := false
if utils.InSlice(strings.ToLower(methodName), HTTPMETHOD) {
if m, ok := c.methods[strings.ToLower(methodName)]; ok && m != methodName {
return false, ""
} else if m, ok = c.methods["*"]; ok && m != methodName {
return false, ""
} else {
2014-06-08 12:24:01 +00:00
find = true
}
} else {
2014-06-08 12:24:01 +00:00
for _, md := range c.methods {
if md == methodName {
find = true
}
}
}
2014-06-08 12:24:01 +00:00
if find {
if t.leaf.regexps == nil {
if len(t.leaf.wildcards) == 0 {
return true, url
}
if len(t.leaf.wildcards) == 1 {
if v, ok := params[t.leaf.wildcards[0]]; ok {
delete(params, t.leaf.wildcards[0])
return true, url + "/" + v + tourl(params)
}
if t.leaf.wildcards[0] == ":splat" {
return true, url + tourl(params)
}
}
2014-06-08 12:24:01 +00:00
if len(t.leaf.wildcards) == 3 && t.leaf.wildcards[0] == "." {
if p, ok := params[":path"]; ok {
if e, isok := params[":ext"]; isok {
delete(params, ":path")
delete(params, ":ext")
return true, url + "/" + p + "." + e + tourl(params)
}
}
2013-05-06 16:17:25 +00:00
}
2014-06-08 12:24:01 +00:00
canskip := false
for _, v := range t.leaf.wildcards {
if v == ":" {
canskip = true
continue
}
if u, ok := params[v]; ok {
url += "/" + u
} else {
if canskip {
canskip = false
continue
} else {
return false, ""
}
}
}
return true, url
2013-12-13 13:25:25 +00:00
} else {
2014-06-08 12:24:01 +00:00
var i int
var startreg bool
url = url + "/"
for _, v := range t.leaf.regexps.String() {
if v == '(' {
startreg = true
continue
} else if v == ')' {
startreg = false
if v, ok := params[t.leaf.wildcards[i]]; ok {
url = url + v
i++
} else {
2013-12-13 13:25:25 +00:00
break
}
2014-06-08 12:24:01 +00:00
} else if !startreg {
url = string(append([]rune(url), v))
2013-12-13 13:25:25 +00:00
}
2013-05-06 16:17:25 +00:00
}
2014-06-08 12:24:01 +00:00
if t.leaf.regexps.MatchString(url) {
return true, url
}
2012-12-18 07:18:43 +00:00
}
2013-04-11 06:35:43 +00:00
}
2012-12-18 07:18:43 +00:00
}
}
2014-06-08 12:24:01 +00:00
}
return false, ""
}
// Implement http.Handler interface.
func (p *ControllerRegistor) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
defer p.recoverPanic(rw, r)
2013-06-20 07:49:43 +00:00
starttime := time.Now()
requestPath := r.URL.Path
2014-06-09 02:11:37 +00:00
method := strings.ToLower(r.Method)
var runrouter reflect.Type
var findrouter bool
var runMethod string
var routerInfo *controllerInfo
2012-12-18 07:18:43 +00:00
w := &responseWriter{writer: rw}
2013-10-28 14:38:50 +00:00
w.Header().Set("Server", BeegoServerName)
2013-12-18 14:33:21 +00:00
// init context
2013-12-20 16:34:59 +00:00
context := &beecontext.Context{
ResponseWriter: w,
Request: r,
Input: beecontext.NewInput(r),
Output: beecontext.NewOutput(),
2013-09-09 16:00:11 +00:00
}
2013-12-20 16:34:59 +00:00
context.Output.Context = context
context.Output.EnableGzip = EnableGzip
2013-12-18 14:33:21 +00:00
if context.Input.IsWebsocket() {
context.ResponseWriter = rw
}
// defined filter function
do_filter := func(pos int) (started bool) {
if p.enableFilter {
if l, ok := p.filters[pos]; ok {
for _, filterR := range l {
if ok, p := filterR.ValidRouter(r.URL.Path); ok {
context.Input.Params = p
filterR.filterFunc(context)
if w.started {
return true
}
}
}
}
}
return false
}
2013-12-16 14:56:35 +00:00
// session init
if SessionOn {
context.Input.CruSession = GlobalSessions.SessionStart(w, r)
defer func() {
context.Input.CruSession.SessionRelease(w)
}()
2013-12-16 14:56:35 +00:00
}
2014-06-09 02:11:37 +00:00
if !utils.InSlice(method, HTTPMETHOD) {
2013-11-07 14:10:46 +00:00
http.Error(w, "Method Not Allowed", 405)
goto Admin
2013-11-07 14:10:46 +00:00
}
2013-09-09 16:00:11 +00:00
if !context.Input.IsGet() && !context.Input.IsHead() {
2014-04-05 16:18:21 +00:00
if CopyRequestBody && !context.Input.IsUpload() {
context.Input.CopyBody()
2012-12-18 07:18:43 +00:00
}
2014-04-05 16:18:21 +00:00
context.Input.ParseFormOrMulitForm(MaxMemory)
2012-12-18 07:18:43 +00:00
}
2014-04-05 16:18:21 +00:00
if do_filter(BeforeRouter) {
goto Admin
2013-09-09 16:00:11 +00:00
}
2012-12-18 07:18:43 +00:00
2014-06-10 03:02:41 +00:00
//static file server
if serverStaticRouter(context) {
goto Admin
}
2014-03-29 06:59:55 +00:00
if context.Input.RunController != nil && context.Input.RunMethod != "" {
findrouter = true
runMethod = context.Input.RunMethod
runrouter = context.Input.RunController
}
if !findrouter {
2014-06-09 02:11:37 +00:00
if t, ok := p.routers[method]; ok {
runObject, p := t.Match(requestPath)
if r, ok := runObject.(*controllerInfo); ok {
routerInfo = r
findrouter = true
if splat, ok := p[":splat"]; ok {
splatlist := strings.Split(splat, "/")
for k, v := range splatlist {
p[strconv.Itoa(k)] = v
}
}
2014-06-09 02:11:37 +00:00
context.Input.Params = p
}
}
2014-06-09 02:11:37 +00:00
}
//if no matches to url, throw a not found exception
if !findrouter {
middleware.Exception("404", rw, r, "")
goto Admin
}
2012-12-18 07:18:43 +00:00
if findrouter {
2012-12-18 07:18:43 +00:00
//execute middleware filters
if do_filter(BeforeExec) {
goto Admin
2012-12-18 07:18:43 +00:00
}
isRunable := false
if routerInfo != nil {
if routerInfo.routerType == routerTypeRESTFul {
if _, ok := routerInfo.methods[strings.ToLower(r.Method)]; ok {
isRunable = true
routerInfo.runfunction(context)
} else {
middleware.Exception("405", rw, r, "Method Not Allowed")
goto Admin
}
} else if routerInfo.routerType == routerTypeHandler {
isRunable = true
routerInfo.handler.ServeHTTP(rw, r)
2014-06-08 12:24:01 +00:00
} else {
runrouter = routerInfo.controllerType
method := strings.ToLower(r.Method)
if method == "post" && strings.ToLower(context.Input.Query("_method")) == "put" {
method = "put"
}
if method == "post" && strings.ToLower(context.Input.Query("_method")) == "delete" {
method = "delete"
}
if m, ok := routerInfo.methods[method]; ok {
runMethod = m
} else if m, ok = routerInfo.methods["*"]; ok {
runMethod = m
} else {
runMethod = strings.Title(method)
}
}
}
2012-12-18 07:18:43 +00:00
2014-06-08 12:24:01 +00:00
// also defined runrouter & runMethod from filter
if !isRunable {
//Invoke the request handler
vc := reflect.New(runrouter)
execController, ok := vc.Interface().(ControllerInterface)
if !ok {
panic("controller is not ControllerInterface")
}
//call the controller init function
execController.Init(context, runrouter.Name(), runMethod, vc.Interface())
2012-12-18 07:18:43 +00:00
//call prepare function
execController.Prepare()
//if XSRF is Enable then check cookie where there has any cookie in the request's cookie _csrf
if EnableXSRF {
execController.XsrfToken()
if r.Method == "POST" || r.Method == "DELETE" || r.Method == "PUT" ||
(r.Method == "POST" && (r.Form.Get("_method") == "delete" || r.Form.Get("_method") == "put")) {
execController.CheckXsrfCookie()
}
2013-08-06 15:21:52 +00:00
}
2014-06-10 03:02:41 +00:00
execController.URLMapping()
if !w.started {
//exec main logic
switch runMethod {
case "Get":
execController.Get()
case "Post":
execController.Post()
case "Delete":
execController.Delete()
case "Put":
execController.Put()
case "Head":
execController.Head()
case "Patch":
execController.Patch()
case "Options":
execController.Options()
default:
2014-06-10 03:02:41 +00:00
if !execController.HandlerFunc(runMethod) {
in := make([]reflect.Value, 0)
method := vc.MethodByName(runMethod)
method.Call(in)
}
}
//render template
if !w.started && !context.Input.IsWebsocket() {
if AutoRender {
if err := execController.Render(); err != nil {
panic(err)
}
}
2012-12-18 07:18:43 +00:00
}
}
2013-09-09 16:00:11 +00:00
// finish all runrouter. release resource
execController.Finish()
}
2013-08-11 16:14:42 +00:00
//execute middleware filters
if do_filter(AfterExec) {
goto Admin
2013-08-11 16:14:42 +00:00
}
2012-12-18 07:18:43 +00:00
}
do_filter(FinishRouter)
2014-06-08 12:24:01 +00:00
2014-04-05 16:18:21 +00:00
Admin:
//admin module record QPS
if EnableAdmin {
2013-11-15 10:08:53 +00:00
timeend := time.Since(starttime)
if FilterMonitorFunc(r.Method, requestPath, timeend) {
if runrouter != nil {
go toolbox.StatisticsMap.AddStatistics(r.Method, requestPath, runrouter.Name(), timeend)
2013-11-15 10:08:53 +00:00
} else {
2013-11-20 13:17:49 +00:00
go toolbox.StatisticsMap.AddStatistics(r.Method, requestPath, "", timeend)
2013-11-15 10:08:53 +00:00
}
2013-11-13 13:37:17 +00:00
}
}
2012-12-18 07:18:43 +00:00
}
2014-06-08 12:24:01 +00:00
func (p *ControllerRegistor) recoverPanic(rw http.ResponseWriter, r *http.Request) {
if err := recover(); err != nil {
if err == USERSTOPRUN {
return
}
if _, ok := err.(middleware.HTTPException); ok {
// catch intented errors, only for HTTP 4XX and 5XX
} else {
if RunMode == "dev" {
if !RecoverPanic {
panic(err)
} else {
if ErrorsShow {
if handler, ok := middleware.ErrorMaps[fmt.Sprint(err)]; ok {
handler(rw, r)
return
}
}
var stack string
Critical("the request url is ", r.URL.Path)
Critical("Handler crashed with error", err)
for i := 1; ; i++ {
_, file, line, ok := runtime.Caller(i)
if !ok {
break
}
Critical(file, line)
stack = stack + fmt.Sprintln(file, line)
}
middleware.ShowErr(err, rw, r, stack)
}
} else {
if !RecoverPanic {
panic(err)
} else {
// in production model show all infomation
if ErrorsShow {
handler := p.getErrorHandler(fmt.Sprint(err))
handler(rw, r)
return
} else {
Critical("the request url is ", r.URL.Path)
Critical("Handler crashed with error", err)
for i := 1; ; i++ {
_, file, line, ok := runtime.Caller(i)
if !ok {
break
}
Critical(file, line)
}
}
}
}
}
}
}
// there always should be error handler that sets error code accordingly for all unhandled errors.
// in order to have custom UI for error page it's necessary to override "500" error.
func (p *ControllerRegistor) getErrorHandler(errorCode string) func(rw http.ResponseWriter, r *http.Request) {
handler := middleware.SimpleServerError
ok := true
if errorCode != "" {
handler, ok = middleware.ErrorMaps[errorCode]
if !ok {
handler, ok = middleware.ErrorMaps["500"]
}
if !ok || handler == nil {
handler = middleware.SimpleServerError
}
}
return handler
}
2012-12-18 07:18:43 +00:00
//responseWriter is a wrapper for the http.ResponseWriter
//started set to true if response was written to then don't execute other handler
type responseWriter struct {
2014-04-05 16:18:21 +00:00
writer http.ResponseWriter
started bool
status int
2012-12-18 07:18:43 +00:00
}
// Header returns the header map that will be sent by WriteHeader.
func (w *responseWriter) Header() http.Header {
return w.writer.Header()
}
// Write writes the data to the connection as part of an HTTP reply,
// and sets `started` to true.
// started means the response has sent out.
2012-12-18 07:18:43 +00:00
func (w *responseWriter) Write(p []byte) (int, error) {
w.started = true
return w.writer.Write(p)
}
// WriteHeader sends an HTTP response header with status code,
// and sets `started` to true.
2012-12-18 07:18:43 +00:00
func (w *responseWriter) WriteHeader(code int) {
w.status = code
w.started = true
w.writer.WriteHeader(code)
}
2013-12-31 12:47:48 +00:00
// hijacker for http
func (w *responseWriter) Hijack() (net.Conn, *bufio.ReadWriter, error) {
hj, ok := w.writer.(http.Hijacker)
if !ok {
return nil, nil, errors.New("webserver doesn't support hijacking")
}
return hj.Hijack()
}
2014-06-08 12:24:01 +00:00
func tourl(params map[string]string) string {
if len(params) == 0 {
return ""
}
u := "?"
for k, v := range params {
u += k + "=" + v + "&"
}
return strings.TrimRight(u, "&")
}