forked from ordishs/go-bitcoin
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathzmq.go
190 lines (156 loc) · 4.57 KB
/
zmq.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
package bitcoin
import (
"context"
"encoding/binary"
"encoding/hex"
"errors"
"fmt"
"log"
"strconv"
"time"
"github.com/go-zeromq/zmq4"
)
var allowedTopics = []string{"hashblock", "hashblock2", "hashtx", "hashtx2", "rawblock", "rawblock2", "rawtx", "rawtx2", "discardedfrommempool", "removedfrommempoolblock", "invalidtx"}
type subscriptionRequest struct {
topic string
ch chan []string
}
// ZMQ struct
type ZMQ struct {
address string
socket zmq4.Socket
connected bool
err error
subscriptions map[string][]chan []string
addSubscription chan subscriptionRequest
removeSubscription chan subscriptionRequest
logger Logger
}
func NewZMQ(host string, port int, optionalLogger ...Logger) *ZMQ {
ctx := context.Background()
return NewZMQWithContext(ctx, host, port, optionalLogger...)
}
func NewZMQWithContext(ctx context.Context, host string, port int, optionalLogger ...Logger) *ZMQ {
zmq := &ZMQ{
address: fmt.Sprintf("tcp://%s:%d", host, port),
subscriptions: make(map[string][]chan []string),
addSubscription: make(chan subscriptionRequest, 10),
removeSubscription: make(chan subscriptionRequest, 10),
logger: &DefaultLogger{},
}
if len(optionalLogger) > 0 {
zmq.logger = optionalLogger[0]
}
go zmq.start(ctx)
return zmq
}
func (zmq *ZMQ) Subscribe(topic string, ch chan []string) error {
if !contains(allowedTopics, topic) {
return fmt.Errorf("topic must be %+v, received %q", allowedTopics, topic)
}
zmq.addSubscription <- subscriptionRequest{
topic: topic,
ch: ch,
}
return nil
}
func (zmq *ZMQ) Unsubscribe(topic string, ch chan []string) error {
if !contains(allowedTopics, topic) {
return fmt.Errorf("topic must be %+v, received %q", allowedTopics, topic)
}
zmq.removeSubscription <- subscriptionRequest{
topic: topic,
ch: ch,
}
return nil
}
func (zmq *ZMQ) start(ctx context.Context) {
for {
zmq.socket = zmq4.NewSub(ctx, zmq4.WithID(zmq4.SocketIdentity("sub")))
defer func() {
if zmq.connected {
zmq.socket.Close()
zmq.connected = false
}
}()
if err := zmq.socket.Dial(zmq.address); err != nil {
zmq.err = err
zmq.logger.Errorf("Could not dial ZMQ at %s: %v", zmq.address, err)
zmq.logger.Infof("Attempting to re-establish ZMQ connection in 10 seconds...")
time.Sleep(10 * time.Second)
continue
}
zmq.logger.Infof("ZMQ: Connecting to %s", zmq.address)
for topic := range zmq.subscriptions {
if err := zmq.socket.SetOption(zmq4.OptionSubscribe, topic); err != nil {
zmq.err = fmt.Errorf("%+v", err)
return
}
zmq.logger.Infof("ZMQ: Subscribed to %s", topic)
}
OUT:
for {
select {
case <-ctx.Done():
zmq.logger.Infof("ZMQ: Context done, exiting")
return
case req := <-zmq.addSubscription:
if err := zmq.socket.SetOption(zmq4.OptionSubscribe, req.topic); err != nil {
zmq.logger.Errorf("ZMQ: Failed to subscribe to %s", req.topic)
} else {
zmq.logger.Infof("ZMQ: Subscribed to %s", req.topic)
}
subscribers := zmq.subscriptions[req.topic]
subscribers = append(subscribers, req.ch)
zmq.subscriptions[req.topic] = subscribers
case req := <-zmq.removeSubscription:
subscribers := zmq.subscriptions[req.topic]
for i, subscriber := range subscribers {
if subscriber == req.ch {
subscribers = append(subscribers[:i], subscribers[i+1:]...)
zmq.logger.Infof("Removed subscription from %s topic", req.topic)
break
}
}
zmq.subscriptions[req.topic] = subscribers
default:
msg, err := zmq.socket.Recv()
if err != nil {
if errors.Is(err, context.Canceled) {
return
}
zmq.logger.Errorf("zmq.socket.Recv() - %v\n", err)
break OUT
} else {
if !zmq.connected {
zmq.connected = true
zmq.logger.Infof("ZMQ: Connection to %s observed\n", zmq.address)
}
subscribers := zmq.subscriptions[string(msg.Frames[0])]
sequence := "N/A"
if len(msg.Frames) > 2 && len(msg.Frames[2]) == 4 {
s := binary.LittleEndian.Uint32(msg.Frames[2])
sequence = strconv.FormatInt(int64(s), 10)
}
for _, subscriber := range subscribers {
subscriber <- []string{string(msg.Frames[0]), hex.EncodeToString(msg.Frames[1]), sequence}
}
}
}
}
if zmq.connected {
zmq.socket.Close()
zmq.connected = false
}
log.Printf("Attempting to re-establish ZMQ connection in 10 seconds...")
time.Sleep(10 * time.Second)
}
}
func contains(s []string, e string) bool {
for _, a := range s {
if a == e {
return true
}
}
return false
}