forked from hyperledger/fabric
-
Notifications
You must be signed in to change notification settings - Fork 6
/
handler_registry.go
170 lines (139 loc) · 4.52 KB
/
handler_registry.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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
/*
Copyright IBM Corp. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0
*/
package chaincode
import (
"sync"
"github.com/pkg/errors"
)
// HandlerRegistry maintains chaincode Handler instances.
type HandlerRegistry struct {
allowUnsolicitedRegistration bool // from cs.userRunsCC
mutex sync.Mutex // lock covering handlers and launching
handlers map[string]*Handler // chaincode cname to associated handler
launching map[string]*LaunchState // launching chaincodes to LaunchState
}
type LaunchState struct {
mutex sync.Mutex
notified bool
done chan struct{}
err error
}
func NewLaunchState() *LaunchState {
return &LaunchState{
done: make(chan struct{}),
}
}
func (l *LaunchState) Done() <-chan struct{} {
return l.done
}
func (l *LaunchState) Err() error {
l.mutex.Lock()
err := l.err
l.mutex.Unlock()
return err
}
func (l *LaunchState) Notify(err error) {
l.mutex.Lock()
if !l.notified {
l.notified = true
l.err = err
close(l.done)
}
l.mutex.Unlock()
}
// NewHandlerRegistry constructs a HandlerRegistry.
func NewHandlerRegistry(allowUnsolicitedRegistration bool) *HandlerRegistry {
return &HandlerRegistry{
handlers: map[string]*Handler{},
launching: map[string]*LaunchState{},
allowUnsolicitedRegistration: allowUnsolicitedRegistration,
}
}
// Launching indicates that chaincode is being launched. The LaunchState that
// is returned provides mechanisms to determine when the operation has
// completed and whether or not it failed. The bool indicates whether or not
// the chaincode has already been started.
func (r *HandlerRegistry) Launching(cname string) (*LaunchState, bool) {
r.mutex.Lock()
defer r.mutex.Unlock()
// launch happened or already happening
if launchState, ok := r.launching[cname]; ok {
return launchState, true
}
// handler registered without going through launch
if _, ok := r.handlers[cname]; ok {
launchState := NewLaunchState()
launchState.Notify(nil)
return launchState, true
}
// first attempt to launch so the runtime needs to start
launchState := NewLaunchState()
r.launching[cname] = launchState
return launchState, false
}
// Ready indicates that the chaincode registration has completed and the
// READY response has been sent to the chaincode.
func (r *HandlerRegistry) Ready(cname string) {
r.mutex.Lock()
defer r.mutex.Unlock()
launchStatus := r.launching[cname]
if launchStatus != nil {
launchStatus.Notify(nil)
}
}
// Failed indicates that registration of a launched chaincode has failed.
func (r *HandlerRegistry) Failed(cname string, err error) {
r.mutex.Lock()
defer r.mutex.Unlock()
launchStatus := r.launching[cname]
if launchStatus != nil {
launchStatus.Notify(err)
}
}
// Handler retrieves the handler for a chaincode instance.
func (r *HandlerRegistry) Handler(cname string) *Handler {
r.mutex.Lock()
h := r.handlers[cname]
r.mutex.Unlock()
return h
}
// Register adds a chaincode handler to the registry.
// An error will be returned if a handler is already registered for the
// chaincode. An error will also be returned if the chaincode has not already
// been "launched", and unsolicited registration is not allowed.
func (r *HandlerRegistry) Register(h *Handler) error {
r.mutex.Lock()
defer r.mutex.Unlock()
key := h.chaincodeID.Name
if r.handlers[key] != nil {
chaincodeLogger.Debugf("duplicate registered handler(key:%s) return error", key)
return errors.Errorf("duplicate chaincodeID: %s", h.chaincodeID.Name)
}
// This chaincode was not launched by the peer but is attempting
// to register. Only allowed in development mode.
if r.launching[key] == nil && !r.allowUnsolicitedRegistration {
return errors.Errorf("peer will not accept external chaincode connection %v (except in dev mode)", h.chaincodeID.Name)
}
r.handlers[key] = h
chaincodeLogger.Debugf("registered handler complete for chaincode %s", key)
return nil
}
// Deregister clears references to state associated specified chaincode.
// As part of the cleanup, it closes the handler so it can cleanup any state.
// If the registry does not contain the provided handler, an error is returned.
func (r *HandlerRegistry) Deregister(cname string) error {
chaincodeLogger.Debugf("deregister handler: %s", cname)
r.mutex.Lock()
handler := r.handlers[cname]
delete(r.handlers, cname)
delete(r.launching, cname)
r.mutex.Unlock()
if handler == nil {
return errors.Errorf("could not find handler: %s", cname)
}
handler.Close()
chaincodeLogger.Debugf("deregistered handler with key: %s", cname)
return nil
}