forked from isayme/go-amqp-reconnect
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathrabbitmq.go
221 lines (180 loc) · 4.52 KB
/
rabbitmq.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
package rabbitmq
import (
"fmt"
"os"
"strconv"
"time"
"sync/atomic"
"github.com/streadway/amqp"
)
func getReconnDelay() int {
if os.Getenv("AMQP_RECONN_DELAY_SECONDS") == "" {
return 3
}
delay, err := strconv.Atoi(os.Getenv("AMQP_RECONN_DELAY_SECONDS"))
if err != nil {
fmt.Println("Cannot convert env `AMQP_RECONN_DELAY_SECONDS` to a number, default to 3.")
return 3
}
return delay
}
var delay = getReconnDelay() // reconnect after delay seconds
// Connection amqp.Connection wrapper
type Connection struct {
*amqp.Connection
}
// Channel wrap amqp.Connection.Channel, get a auto reconnect channel
func (c *Connection) Channel() (*Channel, error) {
ch, err := c.Connection.Channel()
if err != nil {
return nil, err
}
channel := &Channel{
Channel: ch,
}
go func() {
for {
reason, ok := <-channel.Channel.NotifyClose(make(chan *amqp.Error))
// exit this goroutine if closed by developer
if !ok || channel.IsClosed() {
debug("channel closed")
channel.Close() // close again, ensure closed flag set when connection closed
break
}
debugf("channel closed, reason: %v", reason)
// reconnect if not closed by developer
for {
// wait 1s for connection reconnect
time.Sleep(time.Duration(delay) * time.Second)
ch, err := c.Connection.Channel()
if err == nil {
debug("channel recreate success")
channel.Channel = ch
break
}
debugf("channel recreate failed, err: %v", err)
}
}
}()
return channel, nil
}
// Dial wrap amqp.Dial, dial and get a reconnect connection
func Dial(url string) (*Connection, error) {
conn, err := amqp.Dial(url)
if err != nil {
return nil, err
}
connection := &Connection{
Connection: conn,
}
go func() {
for {
reason, ok := <-connection.Connection.NotifyClose(make(chan *amqp.Error))
// exit this goroutine if closed by developer
if !ok {
debug("connection closed")
break
}
debugf("connection closed, reason: %v", reason)
// reconnect if not closed by developer
for {
// wait 1s for reconnect
time.Sleep(time.Duration(delay) * time.Second)
conn, err := amqp.Dial(url)
if err == nil {
connection.Connection = conn
debugf("reconnect success")
break
}
debugf("reconnect failed, err: %v", err)
}
}
}()
return connection, nil
}
// DialCluster with reconnect
func DialCluster(urls []string) (*Connection, error) {
nodeSequence := 0
conn, err := amqp.Dial(urls[nodeSequence])
if err != nil {
return nil, err
}
connection := &Connection{
Connection: conn,
}
go func(urls []string, seq *int) {
for {
reason, ok := <-connection.Connection.NotifyClose(make(chan *amqp.Error))
if !ok {
debug("connection closed")
break
}
debugf("connection closed, reason: %v", reason)
// reconnect with another node of cluster
for {
time.Sleep(time.Duration(delay) * time.Second)
newSeq := next(urls, *seq)
*seq = newSeq
conn, err := amqp.Dial(urls[newSeq])
if err == nil {
connection.Connection = conn
debugf("reconnect success")
break
}
debugf("reconnect failed, err: %v", err)
}
}
}(urls, &nodeSequence)
return connection, nil
}
// Next element index of slice
func next(s []string, lastSeq int) int {
length := len(s)
if length == 0 || lastSeq == length-1 {
return 0
} else if lastSeq < length-1 {
return lastSeq + 1
} else {
return -1
}
}
// Channel amqp.Channel wapper
type Channel struct {
*amqp.Channel
closed int32
}
// IsClosed indicate closed by developer
func (ch *Channel) IsClosed() bool {
return (atomic.LoadInt32(&ch.closed) == 1)
}
// Close ensure closed flag set
func (ch *Channel) Close() error {
if ch.IsClosed() {
return amqp.ErrClosed
}
atomic.StoreInt32(&ch.closed, 1)
return ch.Channel.Close()
}
// Consume wrap amqp.Channel.Consume, the returned delivery will end only when channel closed by developer
func (ch *Channel) Consume(queue, consumer string, autoAck, exclusive, noLocal, noWait bool, args amqp.Table) (<-chan amqp.Delivery, error) {
deliveries := make(chan amqp.Delivery)
go func() {
for {
d, err := ch.Channel.Consume(queue, consumer, autoAck, exclusive, noLocal, noWait, args)
if err != nil {
debugf("consume failed, err: %v", err)
time.Sleep(time.Duration(delay) * time.Second)
continue
}
for msg := range d {
deliveries <- msg
}
// sleep before IsClose call. closed flag may not set before sleep.
time.Sleep(time.Duration(delay) * time.Second)
if ch.IsClosed() {
break
}
}
}()
return deliveries, nil
}