-
Notifications
You must be signed in to change notification settings - Fork 14
/
node.go
254 lines (225 loc) · 6.11 KB
/
node.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
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
package gorums
import (
"context"
"fmt"
"hash/fnv"
"net"
"sort"
"strconv"
"time"
"google.golang.org/grpc"
"google.golang.org/grpc/metadata"
)
const nilAngleString = "<nil>"
// RawNode encapsulates the state of a node on which a remote procedure call
// can be performed.
//
// This struct is intended to be used by generated code.
// You should use the generated `Node` struct instead.
type RawNode struct {
// Only assigned at creation.
id uint32
addr string
conn *grpc.ClientConn
cancel func()
mgr *RawManager
// the default channel
channel *channel
}
// NewRawNode returns a new node for the provided address.
func NewRawNode(addr string) (*RawNode, error) {
tcpAddr, err := net.ResolveTCPAddr("tcp", addr)
if err != nil {
return nil, err
}
h := fnv.New32a()
_, _ = h.Write([]byte(tcpAddr.String()))
return &RawNode{
id: h.Sum32(),
addr: tcpAddr.String(),
}, nil
}
// NewRawNodeWithID returns a new node for the provided address and id.
func NewRawNodeWithID(addr string, id uint32) (*RawNode, error) {
tcpAddr, err := net.ResolveTCPAddr("tcp", addr)
if err != nil {
return nil, err
}
return &RawNode{
id: id,
addr: tcpAddr.String(),
}, nil
}
// connect to this node and associate it with the manager.
func (n *RawNode) connect(mgr *RawManager) error {
n.mgr = mgr
if n.mgr.opts.noConnect {
return nil
}
n.channel = newChannel(n)
if err := n.channel.connect(); err != nil {
return nodeError{nodeID: n.id, cause: err}
}
return nil
}
// dial the node and close the current connection.
func (n *RawNode) dial() error {
if n.conn != nil {
// close the current connection before dialing again.
n.conn.Close()
}
var err error
n.conn, err = grpc.NewClient(n.addr, n.mgr.opts.grpcDialOpts...)
return err
}
// newContext returns a new context for this node's channel.
// This context is used by the channel implementation to stop
// all goroutines and the NodeStream, when the context is canceled.
//
// This method must be called for each connection to ensure
// fresh contexts. Reusing contexts could result in reusing
// a cancelled context.
func (n *RawNode) newContext() context.Context {
md := n.mgr.opts.metadata.Copy()
if n.mgr.opts.perNodeMD != nil {
md = metadata.Join(md, n.mgr.opts.perNodeMD(n.id))
}
var ctx context.Context
ctx, n.cancel = context.WithCancel(context.Background())
return metadata.NewOutgoingContext(ctx, md)
}
// close this node.
func (n *RawNode) close() error {
// important to cancel first to stop goroutines
n.cancel()
if n.conn == nil {
return nil
}
if err := n.conn.Close(); err != nil {
return nodeError{nodeID: n.id, cause: err}
}
return nil
}
// ID returns the ID of n.
func (n *RawNode) ID() uint32 {
if n != nil {
return n.id
}
return 0
}
// Address returns network address of n.
func (n *RawNode) Address() string {
if n != nil {
return n.addr
}
return nilAngleString
}
// Host returns the network host of n.
func (n *RawNode) Host() string {
if n == nil {
return nilAngleString
}
host, _, _ := net.SplitHostPort(n.addr)
return host
}
// Port returns network port of n.
func (n *RawNode) Port() string {
if n != nil {
_, port, _ := net.SplitHostPort(n.addr)
return port
}
return nilAngleString
}
func (n *RawNode) String() string {
if n != nil {
return fmt.Sprintf("addr: %s", n.addr)
}
return nilAngleString
}
// FullString returns a more descriptive string representation of n that
// includes id, network address and latency information.
func (n *RawNode) FullString() string {
if n != nil {
return fmt.Sprintf("node %d | addr: %s", n.id, n.addr)
}
return nilAngleString
}
// LastErr returns the last error encountered (if any) for this node.
func (n *RawNode) LastErr() error {
return n.channel.lastErr()
}
// Latency returns the latency between the client and this node.
func (n *RawNode) Latency() time.Duration {
return n.channel.channelLatency()
}
type lessFunc func(n1, n2 *RawNode) bool
// MultiSorter implements the Sort interface, sorting the nodes within.
type MultiSorter struct {
nodes []*RawNode
less []lessFunc
}
// Sort sorts the argument slice according to the less functions passed to
// OrderedBy.
func (ms *MultiSorter) Sort(nodes []*RawNode) {
ms.nodes = nodes
sort.Sort(ms)
}
// OrderedBy returns a Sorter that sorts using the less functions, in order.
// Call its Sort method to sort the data.
func OrderedBy(less ...lessFunc) *MultiSorter {
return &MultiSorter{
less: less,
}
}
// Len is part of sort.Interface.
func (ms *MultiSorter) Len() int {
return len(ms.nodes)
}
// Swap is part of sort.Interface.
func (ms *MultiSorter) Swap(i, j int) {
ms.nodes[i], ms.nodes[j] = ms.nodes[j], ms.nodes[i]
}
// Less is part of sort.Interface. It is implemented by looping along the
// less functions until it finds a comparison that is either Less or not
// Less. Note that it can call the less functions twice per call. We
// could change the functions to return -1, 0, 1 and reduce the
// number of calls for greater efficiency: an exercise for the reader.
func (ms *MultiSorter) Less(i, j int) bool {
p, q := ms.nodes[i], ms.nodes[j]
// Try all but the last comparison.
var k int
for k = 0; k < len(ms.less)-1; k++ {
less := ms.less[k]
switch {
case less(p, q):
// p < q, so we have a decision.
return true
case less(q, p):
// p > q, so we have a decision.
return false
}
// p == q; try the next comparison.
}
// All comparisons to here said "equal", so just return whatever
// the final comparison reports.
return ms.less[k](p, q)
}
// ID sorts nodes by their identifier in increasing order.
var ID = func(n1, n2 *RawNode) bool {
return n1.id < n2.id
}
// Port sorts nodes by their port number in increasing order.
// Warning: This function may be removed in the future.
var Port = func(n1, n2 *RawNode) bool {
p1, _ := strconv.Atoi(n1.Port())
p2, _ := strconv.Atoi(n2.Port())
return p1 < p2
}
// LastNodeError sorts nodes by their LastErr() status in increasing order. A
// node with LastErr() != nil is larger than a node with LastErr() == nil.
var LastNodeError = func(n1, n2 *RawNode) bool {
if n1.channel.lastErr() != nil && n2.channel.lastErr() == nil {
return false
}
return true
}