-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathunary.go
102 lines (95 loc) · 2.3 KB
/
unary.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
package htadaptor
import (
"context"
"errors"
"fmt"
"net/http"
)
// NewUnaryFuncAdaptor creates a new adaptor for a
// function that takes a validatable struct and returns a struct.
func NewUnaryFuncAdaptor[T any, V *T, O any](
domainCall func(context.Context, V) (O, error),
withOptions ...Option,
) (*UnaryFuncAdaptor[T, V, O], error) {
o := &options{}
err := WithOptions(append(
withOptions,
WithDefaultStatusCode(),
func(o *options) (err error) {
if err = o.Validate(); err != nil {
return err
}
if o.Encoder == nil {
if err = WithDefaultEncoder()(o); err != nil {
return err
}
}
if o.Decoder == nil {
if err = WithDefaultDecoder()(o); err != nil {
return err
}
}
if domainCall == nil {
return errors.New("cannot use a <nil> domain call")
}
return nil
},
)...)(o)
if err != nil {
return nil, fmt.Errorf("unable to initialize unary adaptor: %w", err)
}
return &UnaryFuncAdaptor[T, V, O]{
domainCall: domainCall,
statusCode: o.StatusCode,
decoder: o.Decoder,
encoder: o.Encoder,
errorHandler: o.ErrorHandler,
logger: o.Logger,
}, nil
}
// UnaryFuncAdaptor extracts a struct from request
// and calls a domain function with it expecting
// a struct response.
type UnaryFuncAdaptor[T any, V *T, O any] struct {
domainCall func(context.Context, V) (O, error)
statusCode int
decoder Decoder
encoder Encoder
errorHandler ErrorHandler
logger Logger
}
func (a *UnaryFuncAdaptor[T, V, O]) executeDomainCall(
w http.ResponseWriter,
r *http.Request,
) (err error) {
var request V = new(T)
// request := new(V)
if err = a.decoder.Decode(request, r); err != nil {
return NewDecodingError(err)
}
ctx := r.Context()
if validatable, ok := any(request).(Validatable); ok {
if err = validatable.Validate(ctx); err != nil {
return err
}
}
response, err := a.domainCall(ctx, request)
if err != nil {
return err
}
if err = a.encoder.Encode(w, r, a.statusCode, response); err != nil {
return NewEncodingError(err)
}
return nil
}
// ServeHTTP satisfies [http.Handler] interface.
func (a *UnaryFuncAdaptor[T, V, O]) ServeHTTP(
w http.ResponseWriter,
r *http.Request,
) {
err := a.executeDomainCall(w, r)
if err != nil {
err = a.errorHandler.HandleError(w, r, err)
}
a.logger.LogRequest(r, err)
}