mirror of
https://github.com/astaxie/beego.git
synced 2024-11-21 22:00:54 +00:00
397 lines
9.9 KiB
Go
397 lines
9.9 KiB
Go
// Copyright 2014 beego Author. All Rights Reserved.
|
|
//
|
|
// 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
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// 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.
|
|
|
|
package beego
|
|
|
|
import (
|
|
"net/http"
|
|
"strings"
|
|
|
|
beecontext "github.com/astaxie/beego/context"
|
|
)
|
|
|
|
type namespaceCond func(*beecontext.Context) bool
|
|
|
|
// LinkNamespace used as link action
|
|
type LinkNamespace func(*Namespace)
|
|
|
|
// Namespace is store all the info
|
|
type Namespace struct {
|
|
prefix string
|
|
handlers *ControllerRegister
|
|
}
|
|
|
|
// NewNamespace get new Namespace
|
|
func NewNamespace(prefix string, params ...LinkNamespace) *Namespace {
|
|
ns := &Namespace{
|
|
prefix: prefix,
|
|
handlers: NewControllerRegister(),
|
|
}
|
|
for _, p := range params {
|
|
p(ns)
|
|
}
|
|
return ns
|
|
}
|
|
|
|
// Cond set condition function
|
|
// if cond return true can run this namespace, else can't
|
|
// usage:
|
|
// ns.Cond(func (ctx *context.Context) bool{
|
|
// if ctx.Input.Domain() == "api.beego.me" {
|
|
// return true
|
|
// }
|
|
// return false
|
|
// })
|
|
// Cond as the first filter
|
|
func (n *Namespace) Cond(cond namespaceCond) *Namespace {
|
|
fn := func(ctx *beecontext.Context) {
|
|
if !cond(ctx) {
|
|
exception("405", ctx)
|
|
}
|
|
}
|
|
if v := n.handlers.filters[BeforeRouter]; len(v) > 0 {
|
|
mr := new(FilterRouter)
|
|
mr.tree = NewTree()
|
|
mr.pattern = "*"
|
|
mr.filterFunc = fn
|
|
mr.tree.AddRouter("*", true)
|
|
n.handlers.filters[BeforeRouter] = append([]*FilterRouter{mr}, v...)
|
|
} else {
|
|
n.handlers.InsertFilter("*", BeforeRouter, fn)
|
|
}
|
|
return n
|
|
}
|
|
|
|
// Filter add filter in the Namespace
|
|
// action has before & after
|
|
// FilterFunc
|
|
// usage:
|
|
// Filter("before", func (ctx *context.Context){
|
|
// _, ok := ctx.Input.Session("uid").(int)
|
|
// if !ok && ctx.Request.RequestURI != "/login" {
|
|
// ctx.Redirect(302, "/login")
|
|
// }
|
|
// })
|
|
func (n *Namespace) Filter(action string, filter ...FilterFunc) *Namespace {
|
|
var a int
|
|
if action == "before" {
|
|
a = BeforeRouter
|
|
} else if action == "after" {
|
|
a = FinishRouter
|
|
}
|
|
for _, f := range filter {
|
|
n.handlers.InsertFilter("*", a, f)
|
|
}
|
|
return n
|
|
}
|
|
|
|
// Router same as beego.Rourer
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Router
|
|
func (n *Namespace) Router(rootpath string, c ControllerInterface, mappingMethods ...string) *Namespace {
|
|
n.handlers.Add(rootpath, c, mappingMethods...)
|
|
return n
|
|
}
|
|
|
|
// AutoRouter same as beego.AutoRouter
|
|
// refer: https://godoc.org/github.com/astaxie/beego#AutoRouter
|
|
func (n *Namespace) AutoRouter(c ControllerInterface) *Namespace {
|
|
n.handlers.AddAuto(c)
|
|
return n
|
|
}
|
|
|
|
// AutoPrefix same as beego.AutoPrefix
|
|
// refer: https://godoc.org/github.com/astaxie/beego#AutoPrefix
|
|
func (n *Namespace) AutoPrefix(prefix string, c ControllerInterface) *Namespace {
|
|
n.handlers.AddAutoPrefix(prefix, c)
|
|
return n
|
|
}
|
|
|
|
// Get same as beego.Get
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Get
|
|
func (n *Namespace) Get(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Get(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Post same as beego.Post
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Post
|
|
func (n *Namespace) Post(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Post(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Delete same as beego.Delete
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Delete
|
|
func (n *Namespace) Delete(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Delete(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Put same as beego.Put
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Put
|
|
func (n *Namespace) Put(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Put(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Head same as beego.Head
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Head
|
|
func (n *Namespace) Head(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Head(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Options same as beego.Options
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Options
|
|
func (n *Namespace) Options(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Options(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Patch same as beego.Patch
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Patch
|
|
func (n *Namespace) Patch(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Patch(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Any same as beego.Any
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Any
|
|
func (n *Namespace) Any(rootpath string, f FilterFunc) *Namespace {
|
|
n.handlers.Any(rootpath, f)
|
|
return n
|
|
}
|
|
|
|
// Handler same as beego.Handler
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Handler
|
|
func (n *Namespace) Handler(rootpath string, h http.Handler) *Namespace {
|
|
n.handlers.Handler(rootpath, h)
|
|
return n
|
|
}
|
|
|
|
// Include add include class
|
|
// refer: https://godoc.org/github.com/astaxie/beego#Include
|
|
func (n *Namespace) Include(cList ...ControllerInterface) *Namespace {
|
|
n.handlers.Include(cList...)
|
|
return n
|
|
}
|
|
|
|
// Namespace add nest Namespace
|
|
// usage:
|
|
//ns := beego.NewNamespace(“/v1”).
|
|
//Namespace(
|
|
// beego.NewNamespace("/shop").
|
|
// Get("/:id", func(ctx *context.Context) {
|
|
// ctx.Output.Body([]byte("shopinfo"))
|
|
// }),
|
|
// beego.NewNamespace("/order").
|
|
// Get("/:id", func(ctx *context.Context) {
|
|
// ctx.Output.Body([]byte("orderinfo"))
|
|
// }),
|
|
// beego.NewNamespace("/crm").
|
|
// Get("/:id", func(ctx *context.Context) {
|
|
// ctx.Output.Body([]byte("crminfo"))
|
|
// }),
|
|
//)
|
|
func (n *Namespace) Namespace(ns ...*Namespace) *Namespace {
|
|
for _, ni := range ns {
|
|
for k, v := range ni.handlers.routers {
|
|
if _, ok := n.handlers.routers[k]; ok {
|
|
addPrefix(v, ni.prefix)
|
|
n.handlers.routers[k].AddTree(ni.prefix, v)
|
|
} else {
|
|
t := NewTree()
|
|
t.AddTree(ni.prefix, v)
|
|
addPrefix(t, ni.prefix)
|
|
n.handlers.routers[k] = t
|
|
}
|
|
}
|
|
if ni.handlers.enableFilter {
|
|
for pos, filterList := range ni.handlers.filters {
|
|
for _, mr := range filterList {
|
|
t := NewTree()
|
|
t.AddTree(ni.prefix, mr.tree)
|
|
mr.tree = t
|
|
n.handlers.insertFilterRouter(pos, mr)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return n
|
|
}
|
|
|
|
// AddNamespace register Namespace into beego.Handler
|
|
// support multi Namespace
|
|
func AddNamespace(nl ...*Namespace) {
|
|
for _, n := range nl {
|
|
for k, v := range n.handlers.routers {
|
|
if _, ok := BeeApp.Handlers.routers[k]; ok {
|
|
addPrefix(v, n.prefix)
|
|
BeeApp.Handlers.routers[k].AddTree(n.prefix, v)
|
|
} else {
|
|
t := NewTree()
|
|
t.AddTree(n.prefix, v)
|
|
addPrefix(t, n.prefix)
|
|
BeeApp.Handlers.routers[k] = t
|
|
}
|
|
}
|
|
if n.handlers.enableFilter {
|
|
for pos, filterList := range n.handlers.filters {
|
|
for _, mr := range filterList {
|
|
t := NewTree()
|
|
t.AddTree(n.prefix, mr.tree)
|
|
mr.tree = t
|
|
BeeApp.Handlers.insertFilterRouter(pos, mr)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func addPrefix(t *Tree, prefix string) {
|
|
for _, v := range t.fixrouters {
|
|
addPrefix(v, prefix)
|
|
}
|
|
if t.wildcard != nil {
|
|
addPrefix(t.wildcard, prefix)
|
|
}
|
|
for _, l := range t.leaves {
|
|
if c, ok := l.runObject.(*ControllerInfo); ok {
|
|
if !strings.HasPrefix(c.pattern, prefix) {
|
|
c.pattern = prefix + c.pattern
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// NSCond is Namespace Condition
|
|
func NSCond(cond namespaceCond) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Cond(cond)
|
|
}
|
|
}
|
|
|
|
// NSBefore Namespace BeforeRouter filter
|
|
func NSBefore(filterList ...FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Filter("before", filterList...)
|
|
}
|
|
}
|
|
|
|
// NSAfter add Namespace FinishRouter filter
|
|
func NSAfter(filterList ...FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Filter("after", filterList...)
|
|
}
|
|
}
|
|
|
|
// NSInclude Namespace Include ControllerInterface
|
|
func NSInclude(cList ...ControllerInterface) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Include(cList...)
|
|
}
|
|
}
|
|
|
|
// NSRouter call Namespace Router
|
|
func NSRouter(rootpath string, c ControllerInterface, mappingMethods ...string) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Router(rootpath, c, mappingMethods...)
|
|
}
|
|
}
|
|
|
|
// NSGet call Namespace Get
|
|
func NSGet(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Get(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSPost call Namespace Post
|
|
func NSPost(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Post(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSHead call Namespace Head
|
|
func NSHead(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Head(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSPut call Namespace Put
|
|
func NSPut(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Put(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSDelete call Namespace Delete
|
|
func NSDelete(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Delete(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSAny call Namespace Any
|
|
func NSAny(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Any(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSOptions call Namespace Options
|
|
func NSOptions(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Options(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSPatch call Namespace Patch
|
|
func NSPatch(rootpath string, f FilterFunc) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Patch(rootpath, f)
|
|
}
|
|
}
|
|
|
|
// NSAutoRouter call Namespace AutoRouter
|
|
func NSAutoRouter(c ControllerInterface) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.AutoRouter(c)
|
|
}
|
|
}
|
|
|
|
// NSAutoPrefix call Namespace AutoPrefix
|
|
func NSAutoPrefix(prefix string, c ControllerInterface) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.AutoPrefix(prefix, c)
|
|
}
|
|
}
|
|
|
|
// NSNamespace add sub Namespace
|
|
func NSNamespace(prefix string, params ...LinkNamespace) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
n := NewNamespace(prefix, params...)
|
|
ns.Namespace(n)
|
|
}
|
|
}
|
|
|
|
// NSHandler add handler
|
|
func NSHandler(rootpath string, h http.Handler) LinkNamespace {
|
|
return func(ns *Namespace) {
|
|
ns.Handler(rootpath, h)
|
|
}
|
|
}
|