mirror of
https://github.com/astaxie/beego.git
synced 2024-11-22 13:00:54 +00:00
beego: support namespace
ns := beego.NewNamespace("/v1/api/") ns.Cond(func(ctx *context.Context)bool{ if ctx.Input.Domain() == "www.beego.me" { return true } return false }) .Filter("before", Authenticate) .Router("/order", &admin.OrderController{}) .Get("/version",func (ctx *context.Context) { ctx.Output.Body([]byte("1.0.0")) }) .Post("/login",func (ctx *context.Context) { if ctx.Query("username") == "admin" && ctx.Query("username") == "password" { } }) .Namespace( NewNamespace("/shop"). Get("/order/:id", func(ctx *context.Context) { ctx.Output.Body([]byte(ctx.Input.Param(":id"))) }), )
This commit is contained in:
parent
2ed9b2bffd
commit
e657dcfd5f
4
app.go
4
app.go
@ -185,8 +185,8 @@ func (app *App) Any(rootpath string, f FilterFunc) *App {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add router for http.Handler
|
// add router for http.Handler
|
||||||
func (app *App) Handler(rootpath string, h http.Handler) *App {
|
func (app *App) Handler(rootpath string, h http.Handler, options ...interface{}) *App {
|
||||||
app.Handlers.Handler(rootpath, h)
|
app.Handlers.Handler(rootpath, h, options...)
|
||||||
return app
|
return app
|
||||||
}
|
}
|
||||||
|
|
||||||
|
4
beego.go
4
beego.go
@ -170,8 +170,8 @@ func Any(rootpath string, f FilterFunc) *App {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// register router for own Handler
|
// register router for own Handler
|
||||||
func Handler(rootpath string, h http.Handler) *App {
|
func Handler(rootpath string, h http.Handler, options ...interface{}) *App {
|
||||||
BeeApp.Handler(rootpath, h)
|
BeeApp.Handler(rootpath, h, options...)
|
||||||
return BeeApp
|
return BeeApp
|
||||||
}
|
}
|
||||||
|
|
||||||
|
137
namespace.go
Normal file
137
namespace.go
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
// 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
|
||||||
|
package beego
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
beecontext "github.com/astaxie/beego/context"
|
||||||
|
)
|
||||||
|
|
||||||
|
type namespaceCond func(*beecontext.Context) bool
|
||||||
|
|
||||||
|
type Namespace struct {
|
||||||
|
prefix string
|
||||||
|
condition namespaceCond
|
||||||
|
handlers *ControllerRegistor
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNamespace(prefix string) *Namespace {
|
||||||
|
cr := NewControllerRegistor()
|
||||||
|
return &Namespace{
|
||||||
|
prefix: prefix,
|
||||||
|
handlers: cr,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Cond(cond namespaceCond) *Namespace {
|
||||||
|
n.condition = cond
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Filter(action string, filter FilterFunc) *Namespace {
|
||||||
|
if action == "before" {
|
||||||
|
action = "BeforeRouter"
|
||||||
|
} else if action == "after" {
|
||||||
|
action = "FinishRouter"
|
||||||
|
}
|
||||||
|
n.handlers.AddFilter("*", action, filter)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Router(rootpath string, c ControllerInterface, mappingMethods ...string) *Namespace {
|
||||||
|
n.handlers.Add(rootpath, c, mappingMethods...)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) AutoRouter(c ControllerInterface) *Namespace {
|
||||||
|
n.handlers.AddAuto(c)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) AutoPrefix(prefix string, c ControllerInterface) *Namespace {
|
||||||
|
n.handlers.AddAutoPrefix(prefix, c)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Get(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Get(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Post(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Post(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Delete(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Delete(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Put(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Put(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Head(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Head(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Options(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Options(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Patch(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Patch(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Any(rootpath string, f FilterFunc) *Namespace {
|
||||||
|
n.handlers.Any(rootpath, f)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Handler(rootpath string, h http.Handler) *Namespace {
|
||||||
|
n.handlers.Handler(rootpath, h)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) Namespace(ns *Namespace) *Namespace {
|
||||||
|
n.handlers.Handler(ns.prefix, ns, true)
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Namespace) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
|
||||||
|
//trim the preifix from URL.Path
|
||||||
|
r.URL.Path = strings.TrimPrefix(r.URL.Path, n.prefix)
|
||||||
|
// init context
|
||||||
|
context := &beecontext.Context{
|
||||||
|
ResponseWriter: rw,
|
||||||
|
Request: r,
|
||||||
|
Input: beecontext.NewInput(r),
|
||||||
|
Output: beecontext.NewOutput(),
|
||||||
|
}
|
||||||
|
context.Output.Context = context
|
||||||
|
context.Output.EnableGzip = EnableGzip
|
||||||
|
|
||||||
|
if context.Input.IsWebsocket() {
|
||||||
|
context.ResponseWriter = rw
|
||||||
|
}
|
||||||
|
if n.condition != nil && !n.condition(context) {
|
||||||
|
http.Error(rw, "Method Not Allowed", 405)
|
||||||
|
}
|
||||||
|
n.handlers.ServeHTTP(rw, r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddNamespace(nl ...*Namespace) {
|
||||||
|
for _, n := range nl {
|
||||||
|
Handler(n.prefix, n, true)
|
||||||
|
}
|
||||||
|
}
|
137
namespace_test.go
Normal file
137
namespace_test.go
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
// 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
|
||||||
|
|
||||||
|
package beego
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"strconv"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/astaxie/beego/context"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNamespaceGet(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/user", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Get("/user", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte("v1_user"))
|
||||||
|
})
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "v1_user" {
|
||||||
|
t.Errorf("TestNamespaceGet can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespacePost(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("POST", "/v1/user/123", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Post("/user/:id", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte(ctx.Input.Param(":id")))
|
||||||
|
})
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "123" {
|
||||||
|
t.Errorf("TestNamespacePost can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceNest(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/admin/order", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Namespace(
|
||||||
|
NewNamespace("/admin").
|
||||||
|
Get("/order", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte("order"))
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "order" {
|
||||||
|
t.Errorf("TestNamespaceNest can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceNestParam(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/admin/order/123", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Namespace(
|
||||||
|
NewNamespace("/admin").
|
||||||
|
Get("/order/:id", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte(ctx.Input.Param(":id")))
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "123" {
|
||||||
|
t.Errorf("TestNamespaceNestParam can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceFilter(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/user/123", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Filter("before", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte("this is Filter"))
|
||||||
|
}).
|
||||||
|
Get("/user/:id", func(ctx *context.Context) {
|
||||||
|
ctx.Output.Body([]byte(ctx.Input.Param(":id")))
|
||||||
|
})
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "this is Filter" {
|
||||||
|
t.Errorf("TestNamespaceFilter can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceRouter(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/api/list", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Router("/api/list", &TestController{}, "*:List")
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "i am list" {
|
||||||
|
t.Errorf("TestNamespaceRouter can't run, get the response is " + w.Body.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceAutoFunc(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/test/list", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.AutoRouter(&TestController{})
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Body.String() != "i am list" {
|
||||||
|
t.Errorf("user define func can't run")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNamespaceCond(t *testing.T) {
|
||||||
|
r, _ := http.NewRequest("GET", "/v1/test/list", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
ns := NewNamespace("/v1")
|
||||||
|
ns.Cond(func(ctx *context.Context) bool {
|
||||||
|
if ctx.Input.Domain() == "beego.me" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}).
|
||||||
|
AutoRouter(&TestController{})
|
||||||
|
ns.ServeHTTP(w, r)
|
||||||
|
if w.Code != 405 {
|
||||||
|
t.Errorf("TestNamespaceCond can't run get the result " + strconv.Itoa(w.Code))
|
||||||
|
}
|
||||||
|
}
|
17
router.go
17
router.go
@ -69,6 +69,7 @@ type controllerInfo struct {
|
|||||||
handler http.Handler
|
handler http.Handler
|
||||||
runfunction FilterFunc
|
runfunction FilterFunc
|
||||||
routerType int
|
routerType int
|
||||||
|
isPrefix bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// ControllerRegistor containers registered router rules, controller handlers and filters.
|
// ControllerRegistor containers registered router rules, controller handlers and filters.
|
||||||
@ -277,11 +278,16 @@ func (p *ControllerRegistor) AddMethod(method, pattern string, f FilterFunc) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ControllerRegistor) Handler(pattern string, h http.Handler) {
|
func (p *ControllerRegistor) Handler(pattern string, h http.Handler, options ...interface{}) {
|
||||||
paramnums, params, parts := p.splitRoute(pattern)
|
paramnums, params, parts := p.splitRoute(pattern)
|
||||||
route := &controllerInfo{}
|
route := &controllerInfo{}
|
||||||
route.routerType = routerTypeHandler
|
route.routerType = routerTypeHandler
|
||||||
route.handler = h
|
route.handler = h
|
||||||
|
if len(options) > 0 {
|
||||||
|
if v, ok := options[0].(bool); ok {
|
||||||
|
route.isPrefix = v
|
||||||
|
}
|
||||||
|
}
|
||||||
if paramnums == 0 {
|
if paramnums == 0 {
|
||||||
route.pattern = pattern
|
route.pattern = pattern
|
||||||
p.fixrouters = append(p.fixrouters, route)
|
p.fixrouters = append(p.fixrouters, route)
|
||||||
@ -446,6 +452,7 @@ func (p *ControllerRegistor) AddFilter(pattern, action string, filter FilterFunc
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
switch action {
|
switch action {
|
||||||
case "BeforeRouter":
|
case "BeforeRouter":
|
||||||
p.filters[BeforeRouter] = append(p.filters[BeforeRouter], mr)
|
p.filters[BeforeRouter] = append(p.filters[BeforeRouter], mr)
|
||||||
@ -760,6 +767,14 @@ func (p *ControllerRegistor) ServeHTTP(rw http.ResponseWriter, r *http.Request)
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if route.routerType == routerTypeHandler && route.isPrefix &&
|
||||||
|
strings.HasPrefix(requestPath, route.pattern) {
|
||||||
|
|
||||||
|
routerInfo = route
|
||||||
|
runrouter = route.controllerType
|
||||||
|
findrouter = true
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user