-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrouter.go
180 lines (158 loc) · 4.65 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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
package pine
import (
"context"
"net/http"
"regexp"
"strings"
)
var allMethods = []string{"GET", "POST", "DELETE", "PUT", "HEAD", "PATCH", "CONNECT", "OPTIONS", "TRACE"}
var variableRegex = regexp.MustCompile(`(?m)\{([a-zA-Z]*)\}`)
func notFoundHandler(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(404)
w.Write([]byte("not found."))
}
func New() *router {
return &router{
RootNode: &node{
Path: "", // this should match anything
children: []*node{},
Name: "root",
},
NotFoundHandler: http.HandlerFunc(notFoundHandler),
Middlewares: []MiddlewareFunc{},
}
}
func Var(r *http.Request, k string) string {
return r.Context().Value("pine:" + k).(string)
}
type MiddlewareFunc func(http.Handler) http.Handler
type router struct {
RootNode *node
NotFoundHandler http.Handler
Middlewares []MiddlewareFunc
}
// HandleFunc Inserts a handler function for all HTTP methods relating to the specified path
func (r *router) HandleFunc(tpl string, handler http.HandlerFunc) {
insertHandlerForAllMethods(r.RootNode, tpl, http.HandlerFunc(handler))
}
// Handle Inserts a http.Handler for all HTTP methods relating to the specified path
func (r *router) Handle(tpl string, handler http.Handler) {
insertHandlerForAllMethods(r.RootNode, tpl, handler)
}
func (r *router) Use(middleware MiddlewareFunc) {
r.Middlewares = append(r.Middlewares, middleware)
}
func (r *router) ServeHTTP(w http.ResponseWriter, req *http.Request) {
handler := findMatchingHandler(r.RootNode, req)
if handler == nil {
handler = r.NotFoundHandler
}
varsMap := makeVarsFromRequest(r.RootNode, req)
ctx := req.Context()
for k, v := range varsMap {
ctx = context.WithValue(ctx, "pine:"+k, v)
}
for i := len(r.Middlewares) - 1; i >= 0; i-- {
handler = r.Middlewares[i](handler)
}
handler.ServeHTTP(w, req.WithContext(ctx))
}
type node struct {
children []*node
handler http.Handler
Path string
Method string
Name string
variable string
}
func findMatchingHandler(root *node, r *http.Request) http.Handler {
request_path := r.URL.Path
// get the folders and skip the first element as that is an empty string
request_path_parts := strings.Split(request_path, "/")[1:]
branch, _ := findMatchingPathNode(root, request_path_parts, 0)
leaf := findMatchingMethodNode(branch, r.Method)
return leaf.handler
}
func findMatchingPathNode(root *node, request_path []string, count int) (*node, int) {
if len(request_path) == 0 {
return root, count + 1
}
for _, child := range root.children {
if child.Path == request_path[0] {
if child.Method != "" {
return root, count + 1
}
return findMatchingPathNode(child, request_path[1:], count+1)
}
if child.Path == "" {
if child.Method != "" {
return root, count + 1
}
return findMatchingPathNode(child, request_path[1:], count+1)
}
}
// not sure about the root case maybe just return the root node?
return root, count
}
func findMatchingMethodNode(root *node, method string) *node {
for _, child := range root.children {
if child.Method == method {
return child
}
}
return root
}
func makeVarsFromRequest(start *node, r *http.Request) map[string]string {
request_path := r.URL.Path
vars := make(map[string]string)
request_path_parts := strings.Split(request_path, "/")[1:]
// loop over children and fill up the vars as we encounter them
var root *node = start
for {
if len(request_path_parts) == 0 {
break
}
for _, child := range root.children {
if child.Path == request_path_parts[0] {
root = child
break
}
if child.Path == "" && child.variable != "" {
vars[child.variable] = request_path_parts[0]
}
}
request_path_parts = request_path_parts[1:]
}
return vars
}
func insertHandlerForAllMethods(root *node, tmpl string, handler http.Handler) {
path_parts := strings.Split(tmpl, "/")[1:]
// 1. find the farthest node
var farthest *node = root
farthest, count := findMatchingPathNode(root, path_parts, 0)
// 2. insert as much as needed
path_parts_to_insert := path_parts[count:]
for len(path_parts_to_insert) > 0 {
path_part := path_parts_to_insert[0]
new_node := &node{
children: []*node{},
Path: path_part,
}
if variableRegex.MatchString(path_part) {
new_node.Path = ""
new_node.variable = variableRegex.FindStringSubmatch(path_part)[1]
}
farthest.children = append(farthest.children, new_node)
path_parts_to_insert = path_parts_to_insert[1:] // pop off the left
farthest = new_node
}
// 3. insert all methods for leaves
for _, method := range allMethods {
new_node := &node{
Method: method,
children: []*node{},
handler: handler,
}
farthest.children = append(farthest.children, new_node)
}
}