This repository has been archived by the owner on Mar 11, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathtransport_http.go
106 lines (89 loc) · 2.23 KB
/
transport_http.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 command
import (
"encoding/json"
"errors"
"io"
"net/http"
httptransport "github.com/go-kit/kit/transport/http"
"golang.org/x/net/context"
)
type HTTPHandlers struct {
NewCommandHandler http.Handler
}
func MakeHTTPHandlers(ctx context.Context, endpoints Endpoints, opts ...httptransport.ServerOption) HTTPHandlers {
h := HTTPHandlers{
NewCommandHandler: httptransport.NewServer(
ctx,
endpoints.NewCommandEndpoint,
decodeRequest,
encodeResponse,
opts...,
),
}
return h
}
type errorer interface {
error() error
}
type statuser interface {
status() int
}
// EncodeError is used by the HTTP transport to encode service errors in HTTP.
// The EncodeError should be passed to the Go-Kit httptransport as the
// ServerErrorEncoder to encode error responses with JSON.
func EncodeError(ctx context.Context, err error, w http.ResponseWriter) {
// unwrap Go-Kit Error
var domain string
if e, ok := err.(httptransport.Error); ok {
err = e.Err
domain = e.Domain
}
enc := json.NewEncoder(w)
enc.SetIndent("", " ")
switch domain {
case httptransport.DomainDecode:
w.WriteHeader(http.StatusBadRequest)
case httptransport.DomainDo:
w.WriteHeader(http.StatusServiceUnavailable)
default:
w.WriteHeader(codeFromErr(err))
}
enc.Encode(map[string]interface{}{
"error": err.Error(),
})
}
func codeFromErr(err error) int {
switch err {
case errEmptyRequest:
return http.StatusBadRequest
default:
return http.StatusInternalServerError
}
}
func errorDecoder(r *http.Response) error {
var w errorWrapper
if err := json.NewDecoder(r.Body).Decode(&w); err != nil {
return err
}
return errors.New(w.Error)
}
type errorWrapper struct {
Error string `json:"error"`
}
func decodeRequest(ctx context.Context, r *http.Request) (interface{}, error) {
var req newCommandRequest
err := json.NewDecoder(io.LimitReader(r.Body, 10000)).Decode(&req)
return req, err
}
func encodeResponse(ctx context.Context, w http.ResponseWriter, response interface{}) error {
if e, ok := response.(errorer); ok && e.error() != nil {
EncodeError(ctx, e.error(), w)
return nil
}
if s, ok := response.(statuser); ok {
w.WriteHeader(s.status())
}
enc := json.NewEncoder(w)
enc.SetIndent("", " ")
return enc.Encode(response)
}