2013-09-09 16:00:11 +00:00
|
|
|
|
package beego
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"regexp"
|
2013-11-26 08:47:50 +00:00
|
|
|
|
"strings"
|
2013-09-09 16:00:11 +00:00
|
|
|
|
)
|
|
|
|
|
|
2013-12-20 13:16:26 +00:00
|
|
|
|
// FilterRouter defines filter operation before controller handler execution.
|
|
|
|
|
// it can match patterned url and do filter function when action arrives.
|
2013-09-09 16:00:11 +00:00
|
|
|
|
type FilterRouter struct {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
pattern string
|
|
|
|
|
regex *regexp.Regexp
|
|
|
|
|
filterFunc FilterFunc
|
|
|
|
|
hasregex bool
|
|
|
|
|
params map[int]string
|
|
|
|
|
parseParams map[string]string
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-12-20 13:16:26 +00:00
|
|
|
|
// ValidRouter check current request is valid for this filter.
|
|
|
|
|
// if matched, returns parsed params in this request by defined filter router pattern.
|
2013-11-26 08:47:50 +00:00
|
|
|
|
func (mr *FilterRouter) ValidRouter(router string) (bool, map[string]string) {
|
2013-09-09 16:00:11 +00:00
|
|
|
|
if mr.pattern == "" {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
return true, nil
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|
2013-09-19 15:14:47 +00:00
|
|
|
|
if mr.pattern == "*" {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
return true, nil
|
2013-09-19 15:14:47 +00:00
|
|
|
|
}
|
2013-09-09 16:00:11 +00:00
|
|
|
|
if router == mr.pattern {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
return true, nil
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|
2013-12-30 07:06:51 +00:00
|
|
|
|
//pattern /admin router /admin/ match
|
|
|
|
|
//pattern /admin/ router /admin don't match, because url will 301 in router
|
|
|
|
|
if n := len(router); n > 1 && router[n-1] == '/' && router[:n-2] == mr.pattern {
|
|
|
|
|
return true, nil
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-09 16:00:11 +00:00
|
|
|
|
if mr.hasregex {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
if !mr.regex.MatchString(router) {
|
|
|
|
|
return false, nil
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|
|
|
|
|
matches := mr.regex.FindStringSubmatch(router)
|
2013-10-12 08:56:30 +00:00
|
|
|
|
if len(matches) > 0 {
|
|
|
|
|
if len(matches[0]) == len(router) {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
params := make(map[string]string)
|
|
|
|
|
for i, match := range matches[1:] {
|
|
|
|
|
params[mr.params[i]] = match
|
|
|
|
|
}
|
|
|
|
|
return true, params
|
2013-10-12 08:56:30 +00:00
|
|
|
|
}
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2013-11-26 08:47:50 +00:00
|
|
|
|
return false, nil
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-30 07:06:51 +00:00
|
|
|
|
func buildFilter(pattern string, filter FilterFunc) (*FilterRouter, error) {
|
2013-11-26 08:47:50 +00:00
|
|
|
|
mr := new(FilterRouter)
|
|
|
|
|
mr.params = make(map[int]string)
|
|
|
|
|
mr.filterFunc = filter
|
|
|
|
|
parts := strings.Split(pattern, "/")
|
|
|
|
|
j := 0
|
|
|
|
|
for i, part := range parts {
|
|
|
|
|
if strings.HasPrefix(part, ":") {
|
2013-12-30 07:06:51 +00:00
|
|
|
|
expr := "(.*)"
|
2013-11-26 08:47:50 +00:00
|
|
|
|
//a user may choose to override the default expression
|
|
|
|
|
// similar to expressjs: ‘/user/:id([0-9]+)’
|
|
|
|
|
if index := strings.Index(part, "("); index != -1 {
|
|
|
|
|
expr = part[index:]
|
|
|
|
|
part = part[:index]
|
|
|
|
|
//match /user/:id:int ([0-9]+)
|
|
|
|
|
//match /post/:username:string ([\w]+)
|
|
|
|
|
} else if lindex := strings.LastIndex(part, ":"); lindex != 0 {
|
|
|
|
|
switch part[lindex:] {
|
|
|
|
|
case ":int":
|
|
|
|
|
expr = "([0-9]+)"
|
|
|
|
|
part = part[:lindex]
|
|
|
|
|
case ":string":
|
|
|
|
|
expr = `([\w]+)`
|
|
|
|
|
part = part[:lindex]
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
mr.params[j] = part
|
|
|
|
|
parts[i] = expr
|
|
|
|
|
j++
|
|
|
|
|
}
|
|
|
|
|
if strings.HasPrefix(part, "*") {
|
2013-12-30 07:06:51 +00:00
|
|
|
|
expr := "(.*)"
|
2013-11-26 08:47:50 +00:00
|
|
|
|
if part == "*.*" {
|
|
|
|
|
mr.params[j] = ":path"
|
|
|
|
|
parts[i] = "([^.]+).([^.]+)"
|
|
|
|
|
j++
|
|
|
|
|
mr.params[j] = ":ext"
|
|
|
|
|
j++
|
|
|
|
|
} else {
|
|
|
|
|
mr.params[j] = ":splat"
|
|
|
|
|
parts[i] = expr
|
|
|
|
|
j++
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
//url like someprefix:id(xxx).html
|
|
|
|
|
if strings.Contains(part, ":") && strings.Contains(part, "(") && strings.Contains(part, ")") {
|
|
|
|
|
var out []rune
|
|
|
|
|
var start bool
|
|
|
|
|
var startexp bool
|
|
|
|
|
var param []rune
|
|
|
|
|
var expt []rune
|
|
|
|
|
for _, v := range part {
|
|
|
|
|
if start {
|
|
|
|
|
if v != '(' {
|
|
|
|
|
param = append(param, v)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if startexp {
|
|
|
|
|
if v != ')' {
|
|
|
|
|
expt = append(expt, v)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if v == ':' {
|
|
|
|
|
param = make([]rune, 0)
|
|
|
|
|
param = append(param, ':')
|
|
|
|
|
start = true
|
|
|
|
|
} else if v == '(' {
|
|
|
|
|
startexp = true
|
|
|
|
|
start = false
|
|
|
|
|
mr.params[j] = string(param)
|
|
|
|
|
j++
|
|
|
|
|
expt = make([]rune, 0)
|
|
|
|
|
expt = append(expt, '(')
|
|
|
|
|
} else if v == ')' {
|
|
|
|
|
startexp = false
|
|
|
|
|
expt = append(expt, ')')
|
|
|
|
|
out = append(out, expt...)
|
|
|
|
|
} else {
|
|
|
|
|
out = append(out, v)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
parts[i] = string(out)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if j != 0 {
|
|
|
|
|
pattern = strings.Join(parts, "/")
|
|
|
|
|
regex, regexErr := regexp.Compile(pattern)
|
|
|
|
|
if regexErr != nil {
|
2013-12-30 07:06:51 +00:00
|
|
|
|
return nil, regexErr
|
2013-11-26 08:47:50 +00:00
|
|
|
|
}
|
|
|
|
|
mr.regex = regex
|
|
|
|
|
mr.hasregex = true
|
|
|
|
|
}
|
|
|
|
|
mr.pattern = pattern
|
2013-12-30 07:06:51 +00:00
|
|
|
|
return mr, nil
|
2013-09-09 16:00:11 +00:00
|
|
|
|
}
|