pubsub: improve API and docs.

This commit is contained in:
Vladimir Mihailenco 2015-07-11 13:42:44 +03:00
parent 2cb6b30541
commit d7edae84cf
3 changed files with 72 additions and 41 deletions

View File

@ -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 {

View File

@ -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...)
} }

View File

@ -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())