-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathredisconn_pubsub.go
89 lines (80 loc) · 2.03 KB
/
redisconn_pubsub.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
package redisconn
import (
"fmt"
"github.com/go-redis/redis"
)
type RedisPubSubService interface {
Publish(channel string, message interface{}) error
Subscribe(channels ...string) ([]*redis.PubSub, error)
SubscribeWith(callback func(message *redis.Message, err error), close bool, channels ...string) error
Unsubscribe(channels ...string) error
Close() error
}
func NewRedisPubSub(redisConn *redis.Client) RedisPubSubService {
s := &RedisPubSubClient{
redisConn: redisConn,
subscriptionMap: make(map[string]*redis.PubSub),
}
return s
}
func (c *RedisPubSubClient) Publish(channel string, message interface{}) error {
return c.redisConn.Publish(channel, message).Err()
}
func (c *RedisPubSubClient) Subscribe(channels ...string) ([]*redis.PubSub, error) {
pubSubs := []*redis.PubSub{}
for _, channel := range channels {
pubsub := c.redisConn.Subscribe(channel)
c.subscriptionMap[channel] = pubsub
pubSubs = append(pubSubs, pubsub)
}
return pubSubs, nil
}
func (c *RedisPubSubClient) SubscribeWith(callback func(message *redis.Message, err error), close bool, channels ...string) error {
subs, err := c.Subscribe(channels...)
if err != nil {
return err
}
go func() {
for _, sub := range subs {
go func(ps *redis.PubSub) {
for {
msg, err := ps.ReceiveMessage()
callback(msg, err)
}
}(sub)
}
}()
if !close {
return nil
}
err = c.Unsubscribe(channels...)
if err != nil {
return err
}
err = c.Close()
return err
}
func (c *RedisPubSubClient) Unsubscribe(channels ...string) error {
for _, channel := range channels {
pubsub, ok := c.subscriptionMap[channel]
if !ok {
return fmt.Errorf("not subscribed to channel %s", channel)
}
err := pubsub.Unsubscribe(channel)
if err != nil {
return err
}
delete(c.subscriptionMap, channel)
}
return nil
}
func (c *RedisPubSubClient) Close() error {
for channel, pubsub := range c.subscriptionMap {
err := pubsub.Unsubscribe(channel)
if err != nil {
return err
}
delete(c.subscriptionMap, channel)
}
return c.redisConn.Close()
}