mirror of https://github.com/go-redis/redis.git
Merge pull request #120 from go-redis/feature/ring-pipeline
Add RingPipeline.
This commit is contained in:
commit
a972489416
|
@ -20,48 +20,42 @@ func (c *ClusterClient) Pipeline() *ClusterPipeline {
|
||||||
return pipe
|
return pipe
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *ClusterPipeline) process(cmd Cmder) {
|
func (pipe *ClusterPipeline) process(cmd Cmder) {
|
||||||
c.cmds = append(c.cmds, cmd)
|
pipe.cmds = append(pipe.cmds, cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close marks the pipeline as closed
|
// Discard resets the pipeline and discards queued commands.
|
||||||
func (c *ClusterPipeline) Close() error {
|
func (pipe *ClusterPipeline) Discard() error {
|
||||||
c.closed = true
|
if pipe.closed {
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Discard resets the pipeline and discards queued commands
|
|
||||||
func (c *ClusterPipeline) Discard() error {
|
|
||||||
if c.closed {
|
|
||||||
return errClosed
|
return errClosed
|
||||||
}
|
}
|
||||||
c.cmds = c.cmds[:0]
|
pipe.cmds = pipe.cmds[:0]
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *ClusterPipeline) Exec() (cmds []Cmder, retErr error) {
|
func (pipe *ClusterPipeline) Exec() (cmds []Cmder, retErr error) {
|
||||||
if c.closed {
|
if pipe.closed {
|
||||||
return nil, errClosed
|
return nil, errClosed
|
||||||
}
|
}
|
||||||
if len(c.cmds) == 0 {
|
if len(pipe.cmds) == 0 {
|
||||||
return []Cmder{}, nil
|
return []Cmder{}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
cmds = c.cmds
|
cmds = pipe.cmds
|
||||||
c.cmds = make([]Cmder, 0, 10)
|
pipe.cmds = make([]Cmder, 0, 10)
|
||||||
|
|
||||||
cmdsMap := make(map[string][]Cmder)
|
cmdsMap := make(map[string][]Cmder)
|
||||||
for _, cmd := range cmds {
|
for _, cmd := range cmds {
|
||||||
slot := hashSlot(cmd.clusterKey())
|
slot := hashSlot(cmd.clusterKey())
|
||||||
addr := c.cluster.slotMasterAddr(slot)
|
addr := pipe.cluster.slotMasterAddr(slot)
|
||||||
cmdsMap[addr] = append(cmdsMap[addr], cmd)
|
cmdsMap[addr] = append(cmdsMap[addr], cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
for attempt := 0; attempt <= c.cluster.opt.getMaxRedirects(); attempt++ {
|
for attempt := 0; attempt <= pipe.cluster.opt.getMaxRedirects(); attempt++ {
|
||||||
failedCmds := make(map[string][]Cmder)
|
failedCmds := make(map[string][]Cmder)
|
||||||
|
|
||||||
for addr, cmds := range cmdsMap {
|
for addr, cmds := range cmdsMap {
|
||||||
client, err := c.cluster.getClient(addr)
|
client, err := pipe.cluster.getClient(addr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
setCmdsErr(cmds, err)
|
setCmdsErr(cmds, err)
|
||||||
retErr = err
|
retErr = err
|
||||||
|
@ -75,7 +69,7 @@ func (c *ClusterPipeline) Exec() (cmds []Cmder, retErr error) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
failedCmds, err = c.execClusterCmds(cn, cmds, failedCmds)
|
failedCmds, err = pipe.execClusterCmds(cn, cmds, failedCmds)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
retErr = err
|
retErr = err
|
||||||
}
|
}
|
||||||
|
@ -88,7 +82,14 @@ func (c *ClusterPipeline) Exec() (cmds []Cmder, retErr error) {
|
||||||
return cmds, retErr
|
return cmds, retErr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *ClusterPipeline) execClusterCmds(
|
// Close marks the pipeline as closed
|
||||||
|
func (pipe *ClusterPipeline) Close() error {
|
||||||
|
pipe.Discard()
|
||||||
|
pipe.closed = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pipe *ClusterPipeline) execClusterCmds(
|
||||||
cn *conn, cmds []Cmder, failedCmds map[string][]Cmder,
|
cn *conn, cmds []Cmder, failedCmds map[string][]Cmder,
|
||||||
) (map[string][]Cmder, error) {
|
) (map[string][]Cmder, error) {
|
||||||
if err := cn.writeCmds(cmds...); err != nil {
|
if err := cn.writeCmds(cmds...); err != nil {
|
||||||
|
@ -107,7 +108,7 @@ func (c *ClusterPipeline) execClusterCmds(
|
||||||
failedCmds[""] = append(failedCmds[""], cmds[i:]...)
|
failedCmds[""] = append(failedCmds[""], cmds[i:]...)
|
||||||
break
|
break
|
||||||
} else if moved, ask, addr := isMovedError(err); moved {
|
} else if moved, ask, addr := isMovedError(err); moved {
|
||||||
c.cluster.lazyReloadSlots()
|
pipe.cluster.lazyReloadSlots()
|
||||||
cmd.reset()
|
cmd.reset()
|
||||||
failedCmds[addr] = append(failedCmds[addr], cmd)
|
failedCmds[addr] = append(failedCmds[addr], cmd)
|
||||||
} else if ask {
|
} else if ask {
|
||||||
|
|
91
pipeline.go
91
pipeline.go
|
@ -1,102 +1,113 @@
|
||||||
package redis
|
package redis
|
||||||
|
|
||||||
// Not thread-safe.
|
// Pipeline implements pipelining as described in
|
||||||
|
// http://redis.io/topics/pipelining.
|
||||||
|
//
|
||||||
|
// Pipeline is not thread-safe.
|
||||||
type Pipeline struct {
|
type Pipeline struct {
|
||||||
commandable
|
commandable
|
||||||
|
|
||||||
cmds []Cmder
|
|
||||||
client *baseClient
|
client *baseClient
|
||||||
|
|
||||||
|
cmds []Cmder
|
||||||
closed bool
|
closed bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Pipeline() *Pipeline {
|
func (c *Client) Pipeline() *Pipeline {
|
||||||
pipe := &Pipeline{
|
pipe := &Pipeline{
|
||||||
client: &baseClient{
|
client: c.baseClient,
|
||||||
opt: c.opt,
|
|
||||||
connPool: c.connPool,
|
|
||||||
},
|
|
||||||
cmds: make([]Cmder, 0, 10),
|
cmds: make([]Cmder, 0, 10),
|
||||||
}
|
}
|
||||||
pipe.commandable.process = pipe.process
|
pipe.commandable.process = pipe.process
|
||||||
return pipe
|
return pipe
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Pipelined(f func(*Pipeline) error) ([]Cmder, error) {
|
func (c *Client) Pipelined(fn func(*Pipeline) error) ([]Cmder, error) {
|
||||||
pc := c.Pipeline()
|
pipe := c.Pipeline()
|
||||||
if err := f(pc); err != nil {
|
if err := fn(pipe); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
cmds, err := pc.Exec()
|
cmds, err := pipe.Exec()
|
||||||
pc.Close()
|
pipe.Close()
|
||||||
return cmds, err
|
return cmds, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Pipeline) process(cmd Cmder) {
|
func (pipe *Pipeline) process(cmd Cmder) {
|
||||||
c.cmds = append(c.cmds, cmd)
|
pipe.cmds = append(pipe.cmds, cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Pipeline) Close() error {
|
func (pipe *Pipeline) Close() error {
|
||||||
c.closed = true
|
pipe.Discard()
|
||||||
|
pipe.closed = true
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Pipeline) Discard() error {
|
// Discard resets the pipeline and discards queued commands.
|
||||||
if c.closed {
|
func (pipe *Pipeline) Discard() error {
|
||||||
|
if pipe.closed {
|
||||||
return errClosed
|
return errClosed
|
||||||
}
|
}
|
||||||
c.cmds = c.cmds[:0]
|
pipe.cmds = pipe.cmds[:0]
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Exec always returns list of commands and error of the first failed
|
// Exec always returns list of commands and error of the first failed
|
||||||
// command if any.
|
// command if any.
|
||||||
func (c *Pipeline) Exec() (cmds []Cmder, retErr error) {
|
func (pipe *Pipeline) Exec() (cmds []Cmder, retErr error) {
|
||||||
if c.closed {
|
if pipe.closed {
|
||||||
return nil, errClosed
|
return nil, errClosed
|
||||||
}
|
}
|
||||||
if len(c.cmds) == 0 {
|
if len(pipe.cmds) == 0 {
|
||||||
return c.cmds, nil
|
return pipe.cmds, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
cmds = c.cmds
|
cmds = pipe.cmds
|
||||||
c.cmds = make([]Cmder, 0, 0)
|
pipe.cmds = make([]Cmder, 0, 10)
|
||||||
|
|
||||||
for i := 0; i <= c.client.opt.MaxRetries; i++ {
|
failedCmds := cmds
|
||||||
if i > 0 {
|
for i := 0; i <= pipe.client.opt.MaxRetries; i++ {
|
||||||
resetCmds(cmds)
|
cn, err := pipe.client.conn()
|
||||||
}
|
|
||||||
|
|
||||||
cn, err := c.client.conn()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
setCmdsErr(cmds, err)
|
setCmdsErr(failedCmds, err)
|
||||||
return cmds, err
|
return cmds, err
|
||||||
}
|
}
|
||||||
|
|
||||||
retErr = c.execCmds(cn, cmds)
|
if i > 0 {
|
||||||
c.client.putConn(cn, err)
|
resetCmds(failedCmds)
|
||||||
if shouldRetry(err) {
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
|
failedCmds, err = execCmds(cn, failedCmds)
|
||||||
|
pipe.client.putConn(cn, err)
|
||||||
|
if err != nil && retErr == nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
|
if len(failedCmds) == 0 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return cmds, retErr
|
return cmds, retErr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Pipeline) execCmds(cn *conn, cmds []Cmder) error {
|
func execCmds(cn *conn, cmds []Cmder) ([]Cmder, error) {
|
||||||
if err := cn.writeCmds(cmds...); err != nil {
|
if err := cn.writeCmds(cmds...); err != nil {
|
||||||
setCmdsErr(cmds, err)
|
setCmdsErr(cmds, err)
|
||||||
return err
|
return cmds, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var firstCmdErr error
|
var firstCmdErr error
|
||||||
|
var failedCmds []Cmder
|
||||||
for _, cmd := range cmds {
|
for _, cmd := range cmds {
|
||||||
err := cmd.parseReply(cn.rd)
|
err := cmd.parseReply(cn.rd)
|
||||||
if err != nil && firstCmdErr == nil {
|
if err == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if firstCmdErr == nil {
|
||||||
firstCmdErr = err
|
firstCmdErr = err
|
||||||
}
|
}
|
||||||
|
if shouldRetry(err) {
|
||||||
|
failedCmds = append(failedCmds, cmd)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return firstCmdErr
|
return failedCmds, firstCmdErr
|
||||||
}
|
}
|
||||||
|
|
117
ring.go
117
ring.go
|
@ -25,6 +25,8 @@ type RingOptions struct {
|
||||||
DB int64
|
DB int64
|
||||||
Password string
|
Password string
|
||||||
|
|
||||||
|
MaxRetries int
|
||||||
|
|
||||||
DialTimeout time.Duration
|
DialTimeout time.Duration
|
||||||
ReadTimeout time.Duration
|
ReadTimeout time.Duration
|
||||||
WriteTimeout time.Duration
|
WriteTimeout time.Duration
|
||||||
|
@ -105,6 +107,7 @@ func (shard *ringShard) Vote(up bool) bool {
|
||||||
type Ring struct {
|
type Ring struct {
|
||||||
commandable
|
commandable
|
||||||
|
|
||||||
|
opt *RingOptions
|
||||||
nreplicas int
|
nreplicas int
|
||||||
|
|
||||||
mx sync.RWMutex
|
mx sync.RWMutex
|
||||||
|
@ -117,7 +120,9 @@ type Ring struct {
|
||||||
func NewRing(opt *RingOptions) *Ring {
|
func NewRing(opt *RingOptions) *Ring {
|
||||||
const nreplicas = 100
|
const nreplicas = 100
|
||||||
ring := &Ring{
|
ring := &Ring{
|
||||||
|
opt: opt,
|
||||||
nreplicas: nreplicas,
|
nreplicas: nreplicas,
|
||||||
|
|
||||||
hash: consistenthash.New(nreplicas, nil),
|
hash: consistenthash.New(nreplicas, nil),
|
||||||
shards: make(map[string]*ringShard),
|
shards: make(map[string]*ringShard),
|
||||||
}
|
}
|
||||||
|
@ -235,3 +240,115 @@ func (ring *Ring) Close() (retErr error) {
|
||||||
|
|
||||||
return retErr
|
return retErr
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RingPipeline creates a new pipeline which is able to execute commands
|
||||||
|
// against multiple shards.
|
||||||
|
type RingPipeline struct {
|
||||||
|
commandable
|
||||||
|
|
||||||
|
ring *Ring
|
||||||
|
|
||||||
|
cmds []Cmder
|
||||||
|
closed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ring *Ring) Pipeline() *RingPipeline {
|
||||||
|
pipe := &RingPipeline{
|
||||||
|
ring: ring,
|
||||||
|
cmds: make([]Cmder, 0, 10),
|
||||||
|
}
|
||||||
|
pipe.commandable.process = pipe.process
|
||||||
|
return pipe
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ring *Ring) Pipelined(fn func(*RingPipeline) error) ([]Cmder, error) {
|
||||||
|
pipe := ring.Pipeline()
|
||||||
|
if err := fn(pipe); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
cmds, err := pipe.Exec()
|
||||||
|
pipe.Close()
|
||||||
|
return cmds, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pipe *RingPipeline) process(cmd Cmder) {
|
||||||
|
pipe.cmds = append(pipe.cmds, cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Discard resets the pipeline and discards queued commands.
|
||||||
|
func (pipe *RingPipeline) Discard() error {
|
||||||
|
if pipe.closed {
|
||||||
|
return errClosed
|
||||||
|
}
|
||||||
|
pipe.cmds = pipe.cmds[:0]
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exec always returns list of commands and error of the first failed
|
||||||
|
// command if any.
|
||||||
|
func (pipe *RingPipeline) Exec() (cmds []Cmder, retErr error) {
|
||||||
|
if pipe.closed {
|
||||||
|
return nil, errClosed
|
||||||
|
}
|
||||||
|
if len(pipe.cmds) == 0 {
|
||||||
|
return pipe.cmds, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
cmds = pipe.cmds
|
||||||
|
pipe.cmds = make([]Cmder, 0, 10)
|
||||||
|
|
||||||
|
cmdsMap := make(map[string][]Cmder)
|
||||||
|
for _, cmd := range cmds {
|
||||||
|
name := pipe.ring.hash.Get(cmd.clusterKey())
|
||||||
|
cmdsMap[name] = append(cmdsMap[name], cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := 0; i <= pipe.ring.opt.MaxRetries; i++ {
|
||||||
|
failedCmdsMap := make(map[string][]Cmder)
|
||||||
|
|
||||||
|
for name, cmds := range cmdsMap {
|
||||||
|
client, err := pipe.ring.getClient(name)
|
||||||
|
if err != nil {
|
||||||
|
setCmdsErr(cmds, err)
|
||||||
|
if retErr == nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
cn, err := client.conn()
|
||||||
|
if err != nil {
|
||||||
|
setCmdsErr(cmds, err)
|
||||||
|
if retErr == nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if i > 0 {
|
||||||
|
resetCmds(cmds)
|
||||||
|
}
|
||||||
|
failedCmds, err := execCmds(cn, cmds)
|
||||||
|
client.putConn(cn, err)
|
||||||
|
if err != nil && retErr == nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
|
if len(failedCmds) > 0 {
|
||||||
|
failedCmdsMap[name] = failedCmds
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(failedCmdsMap) == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
cmdsMap = failedCmdsMap
|
||||||
|
}
|
||||||
|
|
||||||
|
return cmds, retErr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pipe *RingPipeline) Close() error {
|
||||||
|
pipe.Discard()
|
||||||
|
pipe.closed = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
21
ring_test.go
21
ring_test.go
|
@ -81,4 +81,25 @@ var _ = Describe("Redis ring", func() {
|
||||||
// RingShard2 should have its keys.
|
// RingShard2 should have its keys.
|
||||||
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
|
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("supports pipelining", func() {
|
||||||
|
pipe := ring.Pipeline()
|
||||||
|
for i := 0; i < 100; i++ {
|
||||||
|
err := pipe.Set(fmt.Sprintf("key%d", i), "value", 0).Err()
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
}
|
||||||
|
cmds, err := pipe.Exec()
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
Expect(cmds).To(HaveLen(100))
|
||||||
|
Expect(pipe.Close()).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
for _, cmd := range cmds {
|
||||||
|
Expect(cmd.Err()).NotTo(HaveOccurred())
|
||||||
|
Expect(cmd.(*redis.StatusCmd).Val()).To(Equal("OK"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Both shards should have some keys now.
|
||||||
|
Expect(ringShard1.Info().Val()).To(ContainSubstring("keys=57"))
|
||||||
|
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in New Issue