-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandler.go
106 lines (86 loc) · 2.7 KB
/
handler.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
package akumu
import (
"net/http"
"net/http/httptest"
)
// Handler is the akumu's equivalent of the [http.Handler].
//
// Is is the function that can take care of a request and
// handle a correct response for it.
type Handler func(*http.Request) error
// handleResponder is a helper that will handle specific [Responder]
// responses. It also takes care of any parent [Builder].
func handleResponder(writer http.ResponseWriter, request *http.Request, parent *Builder, responder Responder) {
if parent != nil {
parent.
Merge(responder.Respond(request)).
Handle(writer, request)
return
}
responder.
Respond(request).
Handle(writer, request)
}
// handleNoError is called whenever there's a response that does not
// contain any error. For example, returning `nil` in a handler.
func handleNoError(writer http.ResponseWriter, request *http.Request, parent *Builder) {
if parent != nil {
parent.Handle(writer, request)
return
}
Response(http.StatusOK).Handle(writer, request)
}
// handle takes care of responding to a given request.
func handle(writer http.ResponseWriter, request *http.Request, err error, parent *Builder) {
if err == nil {
handleNoError(writer, request, parent)
return
}
if raw, ok := err.(RawResponder); ok {
raw.ServeHTTP(writer, request)
return
}
if responder, ok := err.(Responder); ok {
handleResponder(writer, request, parent, responder)
return
}
if parent != nil {
builder := NewProblem(err, parent.status).
Respond(request)
parent.
Merge(builder).
Handle(writer, request)
return
}
NewProblem(err, http.StatusInternalServerError).
Respond(request).
Handle(writer, request)
}
func RawHandler(handler http.Handler) Handler {
return func(*http.Request) error {
return Raw(handler)
}
}
// ServeHTTP implements the [http.Handler] interface to have
// compatibility with the http package.
func (handler Handler) ServeHTTP(writer http.ResponseWriter, request *http.Request) {
handle(writer, request, handler(request), nil)
}
// HandlerFunc transforms the [Handler] into an [http.HandlerFunc].
func (handler Handler) HandlerFunc() http.HandlerFunc {
return http.HandlerFunc(func(writer http.ResponseWriter, request *http.Request) {
handler.ServeHTTP(writer, request)
})
}
// Record records what a given [Handler] would give as a reponse to a [http.Request].
//
// The response is recorded using a [httptest.ResponseRecorder].
func (handler Handler) Record(request *http.Request) *httptest.ResponseRecorder {
return Record(handler, request)
}
// HandlerFunc transforms a [Handler] into an [http.HandlerFunc].
//
// This function is a simple alias of [Handler.HandlerFunc].
func HandlerFunc(handler Handler) http.HandlerFunc {
return handler.HandlerFunc()
}