-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrouter.go
123 lines (116 loc) · 3.12 KB
/
router.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package http
import (
"strings"
)
// Router Router is a tree indexing by path,
// holding the handler chain for request processing.
// The root is Router with `/` path and children are Routers with subpath.
type Router struct {
title string
comment string
path string
realPath string
method string
handlerChain []HandlerFunc
children []*Router
}
// Group Group is a Router node, which children are Routers.
// Every Router can create Groups as children.
func (this *Router) Group(path string) *Router {
router := &Router{
path: path,
realPath: this.realPath + path,
handlerChain: append([]HandlerFunc{}, this.handlerChain...),
children: []*Router{},
}
this.children = append(this.children, router)
return router
}
// Use Use register a middleware on the Router, which will work on all children.
func (this *Router) Use(handler HandlerFunc) *Router {
this.handlerChain = append(this.handlerChain, handler)
for _, router := range this.children {
router.handlerChain = append(router.handlerChain, handler)
}
return this
}
// Handle Handle register a handler on the Router.
func (this *Router) Handle(title, path string, handlers ...HandlerFunc) *Router {
if len(path) < 1 || path[0] != '/' || strings.Contains(path, "//") {
panic("add router faild, invalid path " + path)
}
if sepIndex := strings.Index(path[1:], "/") + 1; sepIndex > 1 {
root := path[:sepIndex]
subpath := path[sepIndex:]
var group *Router = nil
for _, router := range this.children {
if router.path == root {
group = router
}
}
if group == nil {
group = this.Group(root)
}
return group.Handle(title, subpath, handlers...)
}
handlerChain := append([]HandlerFunc{}, this.handlerChain...)
handlerChain = append(handlerChain, handlers...)
router := &Router{
title: title,
path: path,
realPath: this.realPath + path,
handlerChain: handlerChain,
children: []*Router{},
}
this.children = append(this.children, router)
log.Log("DEBUG", "add router", router.realPath)
return router
}
// Comment Comment add comment on Router, using in doc.
func (this *Router) Comment(comment string) *Router {
this.comment = comment
return this
}
func (this *Router) find(path string) *Router {
// path should not like:
// 1. ""
// 2. "xxx"
// 3. "//"
// 4. "//xxx"
// path is ok like:
// 1. "/"
// 2. "/xxx"
// 3. "/xxx/"
// 4. "/xxx/xxx"
// 5. "/xxx/xxx/"
// 6. ...
if len(path) < 1 || path[0] != '/' || strings.HasPrefix(path, "//") {
log.Log("DEBUG", "invalid path", path)
return nil
}
// path should not contain chars
if strings.ContainsAny(path, "\"\"'%&();+[]{}:*<>=") {
log.Log("DEBUG", "illegal path charactor", path)
return nil
}
sepIndex := strings.Index(path[1:], "/") + 1
if sepIndex < 1 {
// find in this level
for _, router := range this.children {
if router.path == path {
return router
}
}
} else {
root := path[:sepIndex]
subpath := path[sepIndex:]
// find in next level
for _, router := range this.children {
if router.path == root {
subrouter := router.find(subpath)
return subrouter
}
}
}
return nil
}