mirror of
https://github.com/redis/go-redis.git
synced 2025-07-28 06:42:00 +03:00
multi: fix recovering from bad connection.
This commit is contained in:
29
multi.go
29
multi.go
@ -10,10 +10,10 @@ var errDiscard = errors.New("redis: Discard can be used only inside Exec")
|
||||
|
||||
// Multi implements Redis transactions as described in
|
||||
// http://redis.io/topics/transactions. It's NOT safe for concurrent use
|
||||
// by multiple goroutines, because Exec resets connection state.
|
||||
// by multiple goroutines, because Exec resets list of watched keys.
|
||||
// If you don't need WATCH it is better to use Pipeline.
|
||||
//
|
||||
// TODO(vmihailenco): rename to Tx
|
||||
// TODO(vmihailenco): rename to Tx and rework API
|
||||
type Multi struct {
|
||||
commandable
|
||||
|
||||
@ -34,6 +34,18 @@ func (c *Client) Multi() *Multi {
|
||||
return multi
|
||||
}
|
||||
|
||||
func (c *Multi) putConn(cn *conn, ei error) {
|
||||
var err error
|
||||
if isBadConn(cn, ei) {
|
||||
err = c.base.connPool.Remove(nil) // nil to force removal
|
||||
} else {
|
||||
err = c.base.connPool.Put(cn)
|
||||
}
|
||||
if err != nil {
|
||||
log.Printf("redis: putConn failed: %s", err)
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Multi) process(cmd Cmder) {
|
||||
if c.cmds == nil {
|
||||
c.base.process(cmd)
|
||||
@ -112,15 +124,18 @@ func (c *Multi) Exec(f func() error) ([]Cmder, error) {
|
||||
return []Cmder{}, nil
|
||||
}
|
||||
|
||||
cn, err := c.base.conn()
|
||||
// Strip MULTI and EXEC commands.
|
||||
retCmds := cmds[1 : len(cmds)-1]
|
||||
|
||||
cn, _, err := c.base.conn()
|
||||
if err != nil {
|
||||
setCmdsErr(cmds[1:len(cmds)-1], err)
|
||||
return cmds[1 : len(cmds)-1], err
|
||||
setCmdsErr(retCmds, err)
|
||||
return retCmds, err
|
||||
}
|
||||
|
||||
err = c.execCmds(cn, cmds)
|
||||
c.base.putConn(cn, err)
|
||||
return cmds[1 : len(cmds)-1], err
|
||||
c.putConn(cn, err)
|
||||
return retCmds, err
|
||||
}
|
||||
|
||||
func (c *Multi) execCmds(cn *conn, cmds []Cmder) error {
|
||||
|
Reference in New Issue
Block a user