mirror of
https://github.com/mochi-mqtt/server.git
synced 2025-09-26 20:21:12 +08:00
180 lines
3.0 KiB
Go
180 lines
3.0 KiB
Go
// Commands from https://redis.io/commands#transactions
|
|
|
|
package miniredis
|
|
|
|
import (
|
|
"github.com/alicebob/miniredis/v2/server"
|
|
)
|
|
|
|
// commandsTransaction handles MULTI &c.
|
|
func commandsTransaction(m *Miniredis) {
|
|
m.srv.Register("DISCARD", m.cmdDiscard)
|
|
m.srv.Register("EXEC", m.cmdExec)
|
|
m.srv.Register("MULTI", m.cmdMulti)
|
|
m.srv.Register("UNWATCH", m.cmdUnwatch)
|
|
m.srv.Register("WATCH", m.cmdWatch)
|
|
}
|
|
|
|
// MULTI
|
|
func (m *Miniredis) cmdMulti(c *server.Peer, cmd string, args []string) {
|
|
if len(args) != 0 {
|
|
c.WriteError(errWrongNumber(cmd))
|
|
return
|
|
}
|
|
if !m.handleAuth(c) {
|
|
return
|
|
}
|
|
if m.checkPubsub(c, cmd) {
|
|
return
|
|
}
|
|
|
|
ctx := getCtx(c)
|
|
if ctx.nested {
|
|
c.WriteError(msgNotFromScripts)
|
|
return
|
|
}
|
|
if inTx(ctx) {
|
|
c.WriteError("ERR MULTI calls can not be nested")
|
|
return
|
|
}
|
|
|
|
startTx(ctx)
|
|
|
|
c.WriteOK()
|
|
}
|
|
|
|
// EXEC
|
|
func (m *Miniredis) cmdExec(c *server.Peer, cmd string, args []string) {
|
|
if len(args) != 0 {
|
|
setDirty(c)
|
|
c.WriteError(errWrongNumber(cmd))
|
|
return
|
|
}
|
|
if !m.handleAuth(c) {
|
|
return
|
|
}
|
|
if m.checkPubsub(c, cmd) {
|
|
return
|
|
}
|
|
|
|
ctx := getCtx(c)
|
|
if ctx.nested {
|
|
c.WriteError(msgNotFromScripts)
|
|
return
|
|
}
|
|
if !inTx(ctx) {
|
|
c.WriteError("ERR EXEC without MULTI")
|
|
return
|
|
}
|
|
|
|
if ctx.dirtyTransaction {
|
|
c.WriteError("EXECABORT Transaction discarded because of previous errors.")
|
|
// a failed EXEC finishes the tx
|
|
stopTx(ctx)
|
|
return
|
|
}
|
|
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
|
|
// Check WATCHed keys.
|
|
for t, version := range ctx.watch {
|
|
if m.db(t.db).keyVersion[t.key] > version {
|
|
// Abort! Abort!
|
|
stopTx(ctx)
|
|
c.WriteLen(-1)
|
|
return
|
|
}
|
|
}
|
|
|
|
c.WriteLen(len(ctx.transaction))
|
|
for _, cb := range ctx.transaction {
|
|
cb(c, ctx)
|
|
}
|
|
// wake up anyone who waits on anything.
|
|
m.signal.Broadcast()
|
|
|
|
stopTx(ctx)
|
|
}
|
|
|
|
// DISCARD
|
|
func (m *Miniredis) cmdDiscard(c *server.Peer, cmd string, args []string) {
|
|
if len(args) != 0 {
|
|
setDirty(c)
|
|
c.WriteError(errWrongNumber(cmd))
|
|
return
|
|
}
|
|
if !m.handleAuth(c) {
|
|
return
|
|
}
|
|
if m.checkPubsub(c, cmd) {
|
|
return
|
|
}
|
|
|
|
ctx := getCtx(c)
|
|
if !inTx(ctx) {
|
|
c.WriteError("ERR DISCARD without MULTI")
|
|
return
|
|
}
|
|
|
|
stopTx(ctx)
|
|
c.WriteOK()
|
|
}
|
|
|
|
// WATCH
|
|
func (m *Miniredis) cmdWatch(c *server.Peer, cmd string, args []string) {
|
|
if len(args) == 0 {
|
|
setDirty(c)
|
|
c.WriteError(errWrongNumber(cmd))
|
|
return
|
|
}
|
|
if !m.handleAuth(c) {
|
|
return
|
|
}
|
|
if m.checkPubsub(c, cmd) {
|
|
return
|
|
}
|
|
|
|
ctx := getCtx(c)
|
|
if ctx.nested {
|
|
c.WriteError(msgNotFromScripts)
|
|
return
|
|
}
|
|
if inTx(ctx) {
|
|
c.WriteError("ERR WATCH in MULTI")
|
|
return
|
|
}
|
|
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
db := m.db(ctx.selectedDB)
|
|
|
|
for _, key := range args {
|
|
watch(db, ctx, key)
|
|
}
|
|
c.WriteOK()
|
|
}
|
|
|
|
// UNWATCH
|
|
func (m *Miniredis) cmdUnwatch(c *server.Peer, cmd string, args []string) {
|
|
if len(args) != 0 {
|
|
setDirty(c)
|
|
c.WriteError(errWrongNumber(cmd))
|
|
return
|
|
}
|
|
if !m.handleAuth(c) {
|
|
return
|
|
}
|
|
if m.checkPubsub(c, cmd) {
|
|
return
|
|
}
|
|
|
|
// Doesn't matter if UNWATCH is in a TX or not. Looks like a Redis bug to me.
|
|
unwatch(getCtx(c))
|
|
|
|
withTx(m, c, func(c *server.Peer, ctx *connCtx) {
|
|
// Do nothing if it's called in a transaction.
|
|
c.WriteOK()
|
|
})
|
|
}
|