-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathmux.go
132 lines (113 loc) · 3.31 KB
/
mux.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
package route
import (
"errors"
"fmt"
"net/http"
"strings"
)
// Mux implements router compatible with http.Handler
type Mux struct {
// NotFound sets handler for routes that are not found
notFound http.Handler
router Router
aliases []alias
}
type alias struct {
match string
replace string
}
// NewMux returns new Mux router
func NewMux() *Mux {
return &Mux{
router: New(),
notFound: ¬Found{},
}
}
// AddAlias adds an alias for matchers in an expression. If the string
// in `match` matches any part of an expression added via `Mux.Handle()`
// then the match is replaced with the value of `alias`.
func (m *Mux) AddAlias(match, replace string) {
m.aliases = append(m.aliases, alias{match: match, replace: replace})
}
func (m *Mux) applyAliases(expr string) (string, bool) {
alias := expr
for _, a := range m.aliases {
alias = strings.Replace(alias, a.match, a.replace, -1)
}
return alias, alias != expr
}
// InitHandlers This adds a map of handlers and expressions in a single call. This allows
// init to load many rules on first startup, thus reducing the time it takes to
// create the initial mux.
func (m *Mux) InitHandlers(handlers map[string]interface{}) error {
if len(m.aliases) == 0 {
return m.router.InitRoutes(handlers)
}
// Apply aliases to routes
modified := make(map[string]interface{}, len(handlers))
for k, v := range handlers {
// If an alias matched, add the modified route to the handlers passed
if alias, ok := m.applyAliases(k); ok {
modified[alias] = v
}
modified[k] = v
}
return m.router.InitRoutes(modified)
}
// Handle adds http handler for route expression
func (m *Mux) Handle(expr string, handler http.Handler) error {
if err := m.router.UpsertRoute(expr, handler); err != nil {
return err
}
if alias, ok := m.applyAliases(expr); ok {
if err := m.router.UpsertRoute(alias, handler); err != nil {
return fmt.Errorf("while adding alias handler: %s", err)
}
}
return nil
}
// HandleFunc adds http handler function for route expression
func (m *Mux) HandleFunc(expr string, handler func(http.ResponseWriter, *http.Request)) error {
return m.Handle(expr, http.HandlerFunc(handler))
}
func (m *Mux) Remove(expr string) error {
if err := m.router.RemoveRoute(expr); err != nil {
return err
}
if alias, ok := m.applyAliases(expr); ok {
if err := m.router.RemoveRoute(alias); err != nil {
return fmt.Errorf("while removing alias handler: %s", err)
}
}
return nil
}
// ServeHTTP routes the request and passes it to handler
func (m *Mux) ServeHTTP(w http.ResponseWriter, r *http.Request) {
h, err := m.router.Route(r)
if err != nil || h == nil {
m.notFound.ServeHTTP(w, r)
return
}
h.(http.Handler).ServeHTTP(w, r)
}
func (m *Mux) SetNotFound(n http.Handler) error {
if n == nil {
return errors.New("not found handler cannot be nil: operation rejected")
}
m.notFound = n
return nil
}
func (m *Mux) GetNotFound() http.Handler {
return m.notFound
}
func (m *Mux) IsValid(expr string) bool {
return IsValid(expr)
}
// NotFound is a generic http.Handler for request
type notFound struct{}
// ServeHTTP returns a simple 404 Not found response
func (notFound) ServeHTTP(w http.ResponseWriter, _ *http.Request) {
w.Header().Set("Content-Type", "text/plain")
w.WriteHeader(http.StatusNotFound)
_, _ = fmt.Fprint(w, http.StatusText(http.StatusNotFound))
}