-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathregistry.go
More file actions
113 lines (97 loc) · 2.4 KB
/
registry.go
File metadata and controls
113 lines (97 loc) · 2.4 KB
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
package jsonrpc
import (
"context"
"errors"
"fmt"
"net/http"
"sync"
)
type (
Registry struct {
middlewares []MiddlewareFunc
m *sync.Mutex
r map[string]meta
}
meta struct {
handler HandlerFunc
middlewares []MiddlewareFunc
}
)
func New(middlewares ...MiddlewareFunc) *Registry {
return &Registry{
middlewares: middlewares,
m: &sync.Mutex{},
r: make(map[string]meta),
}
}
func (mr *Registry) ServeHTTP(w http.ResponseWriter, r *http.Request) {
requests, batch, err := ParseRequest(r)
if err != nil {
err := sendResponse(w, []*Response{
{
Version: Version,
Error: err,
},
}, false)
if err != nil {
fmt.Fprint(w, "Failed to encode error objects")
w.WriteHeader(http.StatusInternalServerError)
}
return
}
resp := make([]*Response, len(requests))
for i := range requests {
resp[i] = mr.invokeMethod(r.Context(), requests[i])
}
if err := sendResponse(w, resp, batch); err != nil {
fmt.Fprint(w, "Failed to encode result objects", err)
w.WriteHeader(http.StatusInternalServerError)
}
}
func (mr *Registry) RegisterMethods(methods []Method, middlewares ...MiddlewareFunc) error {
for i := range methods {
if err := mr.RegisterMethod(methods[i], middlewares...); err != nil {
return err
}
}
return nil
}
func (mr *Registry) RegisterMethod(method Method, middlewares ...MiddlewareFunc) error {
if method.method == "" || method.handler == nil {
return errors.New("jsonrpc: method name and function should not be empty")
}
mr.m.Lock()
mr.r[method.method] = meta{
handler: method.handler,
middlewares: middlewares,
}
mr.m.Unlock()
return nil
}
func (mr *Registry) invokeMethod(c context.Context, r *Request) *Response {
var method meta
res := newResponse(r)
method, res.Error = mr.takeMethod(r)
if res.Error != nil {
return res
}
wrappedContext := WithRequestID(c, r.ID)
wrappedContext = WithMethodName(wrappedContext, r.Method)
handler := applyMiddleware(method.handler, mr.middlewares...)
handler = applyMiddleware(handler, method.middlewares...)
res.Result, res.Error = handler(wrappedContext, r.Params)
if res.Error != nil {
res.Result = nil
}
return res
}
func (mr *Registry) takeMethod(r *Request) (meta, *Error) {
if r.Method == "" || r.Version != Version {
return meta{}, ErrInvalidParams()
}
md, ok := mr.r[r.Method]
if !ok {
return meta{}, ErrMethodNotFound()
}
return md, nil
}