forked from mirror/redis
pubsub: improve API and docs.
This commit is contained in:
parent
2cb6b30541
commit
d7edae84cf
|
@ -165,13 +165,11 @@ func ExampleMulti() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func ExamplePubSub() {
|
func ExamplePubSub() {
|
||||||
pubsub := client.PubSub()
|
pubsub, err := client.Subscribe("mychannel")
|
||||||
defer pubsub.Close()
|
|
||||||
|
|
||||||
err := pubsub.Subscribe("mychannel")
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
err = client.Publish("mychannel", "hello").Err()
|
err = client.Publish("mychannel", "hello").Err()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
60
pubsub.go
60
pubsub.go
|
@ -5,12 +5,20 @@ import (
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Posts a message to the given channel.
|
||||||
|
func (c *Client) Publish(channel, message string) *IntCmd {
|
||||||
|
req := NewIntCmd("PUBLISH", channel, message)
|
||||||
|
c.Process(req)
|
||||||
|
return req
|
||||||
|
}
|
||||||
|
|
||||||
// PubSub implements Pub/Sub commands as described in
|
// PubSub implements Pub/Sub commands as described in
|
||||||
// http://redis.io/topics/pubsub.
|
// http://redis.io/topics/pubsub.
|
||||||
type PubSub struct {
|
type PubSub struct {
|
||||||
*baseClient
|
*baseClient
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Deprecated. Use Subscribe/PSubscribe instead.
|
||||||
func (c *Client) PubSub() *PubSub {
|
func (c *Client) PubSub() *PubSub {
|
||||||
return &PubSub{
|
return &PubSub{
|
||||||
baseClient: &baseClient{
|
baseClient: &baseClient{
|
||||||
|
@ -20,10 +28,16 @@ func (c *Client) PubSub() *PubSub {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Publish(channel, message string) *IntCmd {
|
// Subscribes the client to the specified channels.
|
||||||
req := NewIntCmd("PUBLISH", channel, message)
|
func (c *Client) Subscribe(channels ...string) (*PubSub, error) {
|
||||||
c.Process(req)
|
pubsub := c.PubSub()
|
||||||
return req
|
return pubsub, pubsub.Subscribe(channels...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Subscribes the client to the given patterns.
|
||||||
|
func (c *Client) PSubscribe(channels ...string) (*PubSub, error) {
|
||||||
|
pubsub := c.PubSub()
|
||||||
|
return pubsub, pubsub.PSubscribe(channels...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *PubSub) Ping(payload string) error {
|
func (c *PubSub) Ping(payload string) error {
|
||||||
|
@ -40,6 +54,20 @@ func (c *PubSub) Ping(payload string) error {
|
||||||
return cn.writeCmds(cmd)
|
return cn.writeCmds(cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Message received after a successful subscription to channel.
|
||||||
|
type Subscription struct {
|
||||||
|
// Can be "subscribe", "unsubscribe", "psubscribe" or "punsubscribe".
|
||||||
|
Kind string
|
||||||
|
// Channel name we have subscribed to.
|
||||||
|
Channel string
|
||||||
|
// Number of channels we are currently subscribed to.
|
||||||
|
Count int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Subscription) String() string {
|
||||||
|
return fmt.Sprintf("%s: %s", m.Kind, m.Channel)
|
||||||
|
}
|
||||||
|
|
||||||
// Message received as result of a PUBLISH command issued by another client.
|
// Message received as result of a PUBLISH command issued by another client.
|
||||||
type Message struct {
|
type Message struct {
|
||||||
Channel string
|
Channel string
|
||||||
|
@ -74,20 +102,8 @@ func (p *Pong) String() string {
|
||||||
return "Pong"
|
return "Pong"
|
||||||
}
|
}
|
||||||
|
|
||||||
// Message received after a successful subscription to channel.
|
// Returns a message as a Subscription, Message, PMessage, Pong or
|
||||||
type Subscription struct {
|
// error. See PubSub example for details.
|
||||||
// Can be "subscribe", "unsubscribe", "psubscribe" or "punsubscribe".
|
|
||||||
Kind string
|
|
||||||
// Channel name we have subscribed to.
|
|
||||||
Channel string
|
|
||||||
// Number of channels we are currently subscribed to.
|
|
||||||
Count int
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *Subscription) String() string {
|
|
||||||
return fmt.Sprintf("%s: %s", m.Kind, m.Channel)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *PubSub) Receive() (interface{}, error) {
|
func (c *PubSub) Receive() (interface{}, error) {
|
||||||
return c.ReceiveTimeout(0)
|
return c.ReceiveTimeout(0)
|
||||||
}
|
}
|
||||||
|
@ -120,6 +136,8 @@ func newMessage(reply []interface{}) (interface{}, error) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReceiveTimeout acts like Receive but returns an error if message
|
||||||
|
// is not received in time.
|
||||||
func (c *PubSub) ReceiveTimeout(timeout time.Duration) (interface{}, error) {
|
func (c *PubSub) ReceiveTimeout(timeout time.Duration) (interface{}, error) {
|
||||||
cn, err := c.conn()
|
cn, err := c.conn()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -149,18 +167,24 @@ func (c *PubSub) subscribe(cmd string, channels ...string) error {
|
||||||
return cn.writeCmds(req)
|
return cn.writeCmds(req)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Subscribes the client to the specified channels.
|
||||||
func (c *PubSub) Subscribe(channels ...string) error {
|
func (c *PubSub) Subscribe(channels ...string) error {
|
||||||
return c.subscribe("SUBSCRIBE", channels...)
|
return c.subscribe("SUBSCRIBE", channels...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Subscribes the client to the given patterns.
|
||||||
func (c *PubSub) PSubscribe(patterns ...string) error {
|
func (c *PubSub) PSubscribe(patterns ...string) error {
|
||||||
return c.subscribe("PSUBSCRIBE", patterns...)
|
return c.subscribe("PSUBSCRIBE", patterns...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Unsubscribes the client from the given channels, or from all of
|
||||||
|
// them if none is given.
|
||||||
func (c *PubSub) Unsubscribe(channels ...string) error {
|
func (c *PubSub) Unsubscribe(channels ...string) error {
|
||||||
return c.subscribe("UNSUBSCRIBE", channels...)
|
return c.subscribe("UNSUBSCRIBE", channels...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Unsubscribes the client from the given patterns, or from all of
|
||||||
|
// them if none is given.
|
||||||
func (c *PubSub) PUnsubscribe(patterns ...string) error {
|
func (c *PubSub) PUnsubscribe(patterns ...string) error {
|
||||||
return c.subscribe("PUNSUBSCRIBE", patterns...)
|
return c.subscribe("PUNSUBSCRIBE", patterns...)
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,27 +12,26 @@ import (
|
||||||
|
|
||||||
var _ = Describe("PubSub", func() {
|
var _ = Describe("PubSub", func() {
|
||||||
var client *redis.Client
|
var client *redis.Client
|
||||||
var pubsub *redis.PubSub
|
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
client = redis.NewClient(&redis.Options{
|
client = redis.NewClient(&redis.Options{
|
||||||
Addr: redisAddr,
|
Addr: redisAddr,
|
||||||
})
|
})
|
||||||
Expect(client.FlushDb().Err()).NotTo(HaveOccurred())
|
Expect(client.FlushDb().Err()).NotTo(HaveOccurred())
|
||||||
pubsub = client.PubSub()
|
|
||||||
})
|
})
|
||||||
|
|
||||||
AfterEach(func() {
|
AfterEach(func() {
|
||||||
Expect(pubsub.Close()).NotTo(HaveOccurred())
|
|
||||||
Expect(client.Close()).NotTo(HaveOccurred())
|
Expect(client.Close()).NotTo(HaveOccurred())
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should support pattern matching", func() {
|
It("should support pattern matching", func() {
|
||||||
Expect(pubsub.PSubscribe("mychannel*")).NotTo(HaveOccurred())
|
pubsub, err := client.PSubscribe("mychannel*")
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
pub := client.Publish("mychannel1", "hello")
|
n, err := client.Publish("mychannel1", "hello").Result()
|
||||||
Expect(pub.Err()).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(pub.Val()).To(Equal(int64(1)))
|
Expect(n).To(Equal(int64(1)))
|
||||||
|
|
||||||
Expect(pubsub.PUnsubscribe("mychannel*")).NotTo(HaveOccurred())
|
Expect(pubsub.PUnsubscribe("mychannel*")).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
@ -75,7 +74,9 @@ var _ = Describe("PubSub", func() {
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(channels).To(BeEmpty())
|
Expect(channels).To(BeEmpty())
|
||||||
|
|
||||||
Expect(pubsub.Subscribe("mychannel", "mychannel2")).NotTo(HaveOccurred())
|
pubsub, err := client.Subscribe("mychannel", "mychannel2")
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
channels, err = client.PubSubChannels("mychannel*").Result()
|
channels, err = client.PubSubChannels("mychannel*").Result()
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
@ -91,7 +92,9 @@ var _ = Describe("PubSub", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should return the numbers of subscribers", func() {
|
It("should return the numbers of subscribers", func() {
|
||||||
Expect(pubsub.Subscribe("mychannel", "mychannel2")).NotTo(HaveOccurred())
|
pubsub, err := client.Subscribe("mychannel", "mychannel2")
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
channels, err := client.PubSubNumSub("mychannel", "mychannel2", "mychannel3").Result()
|
channels, err := client.PubSubNumSub("mychannel", "mychannel2", "mychannel3").Result()
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
@ -107,7 +110,9 @@ var _ = Describe("PubSub", func() {
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(num).To(Equal(int64(0)))
|
Expect(num).To(Equal(int64(0)))
|
||||||
|
|
||||||
Expect(pubsub.PSubscribe("*")).NotTo(HaveOccurred())
|
pubsub, err := client.PSubscribe("*")
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
num, err = client.PubSubNumPat().Result()
|
num, err = client.PubSubNumPat().Result()
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
@ -115,15 +120,17 @@ var _ = Describe("PubSub", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should pub/sub", func() {
|
It("should pub/sub", func() {
|
||||||
Expect(pubsub.Subscribe("mychannel", "mychannel2")).NotTo(HaveOccurred())
|
pubsub, err := client.Subscribe("mychannel", "mychannel2")
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
pub := client.Publish("mychannel", "hello")
|
n, err := client.Publish("mychannel", "hello").Result()
|
||||||
Expect(pub.Err()).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(pub.Val()).To(Equal(int64(1)))
|
Expect(n).To(Equal(int64(1)))
|
||||||
|
|
||||||
pub = client.Publish("mychannel2", "hello2")
|
n, err = client.Publish("mychannel2", "hello2").Result()
|
||||||
Expect(pub.Err()).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(pub.Val()).To(Equal(int64(1)))
|
Expect(n).To(Equal(int64(1)))
|
||||||
|
|
||||||
Expect(pubsub.Unsubscribe("mychannel", "mychannel2")).NotTo(HaveOccurred())
|
Expect(pubsub.Unsubscribe("mychannel", "mychannel2")).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
@ -187,8 +194,9 @@ var _ = Describe("PubSub", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should ping/pong", func() {
|
It("should ping/pong", func() {
|
||||||
err := pubsub.Subscribe("mychannel")
|
pubsub, err := client.Subscribe("mychannel")
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
_, err = pubsub.ReceiveTimeout(time.Second)
|
_, err = pubsub.ReceiveTimeout(time.Second)
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
@ -203,8 +211,9 @@ var _ = Describe("PubSub", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should ping/pong with payload", func() {
|
It("should ping/pong with payload", func() {
|
||||||
err := pubsub.Subscribe("mychannel")
|
pubsub, err := client.Subscribe("mychannel")
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
defer pubsub.Close()
|
||||||
|
|
||||||
_, err = pubsub.ReceiveTimeout(time.Second)
|
_, err = pubsub.ReceiveTimeout(time.Second)
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
|
Loading…
Reference in New Issue