-
Notifications
You must be signed in to change notification settings - Fork 56
/
handler.go
107 lines (85 loc) · 2.4 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
107
package routing
import (
"encoding/json"
"errors"
"fmt"
"net/http"
apierrors "code.cloudfoundry.org/korifi/api/errors"
"code.cloudfoundry.org/korifi/api/presenter"
"github.com/go-http-utils/headers"
"github.com/go-logr/logr"
)
type Response struct {
httpStatus int
body interface{}
headers map[string][]string
}
func NewResponse(httpStatus int) *Response {
return &Response{
httpStatus: httpStatus,
headers: map[string][]string{},
}
}
func (r *Response) WithHeader(key, value string) *Response {
r.headers[key] = append(r.headers[key], value)
return r
}
func (r *Response) WithBody(body interface{}) *Response {
r.body = body
return r
}
//counterfeiter:generate -o fake -fake-name Handler . Handler
type Handler func(r *http.Request) (*Response, error)
func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
logger := logr.FromContextOrDiscard(r.Context())
handlerResponse, err := h(r)
if err != nil {
logger.Info("handler returned error", "reason", err)
PresentError(logger, w, err)
return
}
if err := handlerResponse.writeTo(w); err != nil {
_ = apierrors.LogAndReturn(logger, err, "failed to write result to the HTTP response", "handlerResponse", handlerResponse, "method", r.Method, "URL", r.URL)
}
}
func PresentError(logger logr.Logger, w http.ResponseWriter, err error) {
var apiError apierrors.ApiError
if errors.As(err, &apiError) {
writeErr := NewResponse(apiError.HttpStatus()).
WithBody(presenter.ErrorsResponse{
Errors: []presenter.PresentedError{
{
Detail: apiError.Detail(),
Title: apiError.Title(),
Code: apiError.Code(),
},
},
}).
writeTo(w)
if writeErr != nil {
_ = apierrors.LogAndReturn(logger, writeErr, "failed to write error to the HTTP response")
}
return
}
PresentError(logger, w, apierrors.NewUnknownError(err))
}
func (response *Response) writeTo(w http.ResponseWriter) error {
for header, headerValues := range response.headers {
for _, value := range headerValues {
w.Header().Add(header, value)
}
}
if response.body == nil {
w.WriteHeader(response.httpStatus)
return nil
}
w.Header().Set(headers.ContentType, "application/json")
w.WriteHeader(response.httpStatus)
encoder := json.NewEncoder(w)
encoder.SetEscapeHTML(false)
err := encoder.Encode(response.body)
if err != nil {
return fmt.Errorf("failed to encode and write response: %w", err)
}
return nil
}