mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 16:18:28 -05:00
Handle BRPOPLPUSH inside a transaction.
This commit is contained in:
parent
ba3b474111
commit
7c25a43adc
62
src/t_list.c
62
src/t_list.c
@ -790,23 +790,28 @@ int handleClientsWaitingListPush(redisClient *c, robj *key, robj *ele) {
|
||||
receiver = ln->value;
|
||||
|
||||
if (receiver->bstate.target == NULL) {
|
||||
addReplyMultiBulkLen(receiver,2);
|
||||
addReplyBulk(receiver,key);
|
||||
addReplyBulk(receiver,ele);
|
||||
/* BRPOP/BLPOP return a multi-bulk with the name
|
||||
* of the popped list */
|
||||
addReplyMultiBulkLen(receiver,2);
|
||||
addReplyBulk(receiver,key);
|
||||
addReplyBulk(receiver,ele);
|
||||
}
|
||||
else {
|
||||
robj *dobj = lookupKeyWrite(receiver->db,receiver->bstate.target);
|
||||
if (dobj && checkType(receiver,dobj,REDIS_LIST)) return 0;
|
||||
/* BRPOPLPUSH */
|
||||
robj *dobj = lookupKeyWrite(receiver->db,receiver->bstate.target);
|
||||
if (dobj && checkType(receiver,dobj,REDIS_LIST)) return 0;
|
||||
|
||||
addReplyBulk(receiver,ele);
|
||||
addReplyBulk(receiver,ele);
|
||||
|
||||
/* Create the list if the key does not exist */
|
||||
if (!dobj) {
|
||||
dobj = createZiplistObject();
|
||||
dbAdd(receiver->db,receiver->bstate.target,dobj);
|
||||
}
|
||||
if (!handleClientsWaitingListPush(receiver, receiver->bstate.target, ele)) {
|
||||
/* Create the list if the key does not exist */
|
||||
if (!dobj) {
|
||||
dobj = createZiplistObject();
|
||||
dbAdd(receiver->db, receiver->bstate.target, dobj);
|
||||
}
|
||||
|
||||
listTypePush(dobj,ele,REDIS_HEAD);
|
||||
listTypePush(dobj, ele, REDIS_HEAD);
|
||||
}
|
||||
}
|
||||
|
||||
unblockClientWaitingData(receiver);
|
||||
@ -880,13 +885,13 @@ int checkTimeout(redisClient *c, robj *object, time_t *timeout) {
|
||||
long long lltimeout;
|
||||
|
||||
if (getLongLongFromObject(object, &lltimeout) != REDIS_OK) {
|
||||
addReplyError(c, "timeout is not an integer");
|
||||
return REDIS_ERR;
|
||||
addReplyError(c, "timeout is not an integer");
|
||||
return REDIS_ERR;
|
||||
}
|
||||
|
||||
|
||||
if (lltimeout < 0) {
|
||||
addReplyError(c, "timeout is negative");
|
||||
return REDIS_ERR;
|
||||
addReplyError(c, "timeout is negative");
|
||||
return REDIS_ERR;
|
||||
}
|
||||
|
||||
*timeout = lltimeout;
|
||||
@ -911,20 +916,27 @@ void brpoplpushCommand(redisClient *c) {
|
||||
|
||||
robj *key = lookupKeyWrite(c->db, c->argv[1]);
|
||||
|
||||
|
||||
if (key == NULL) {
|
||||
// block
|
||||
if (c->flags & REDIS_MULTI) {
|
||||
addReply(c,shared.nullmultibulk);
|
||||
|
||||
/* Blocking against an empty list in a multi state
|
||||
* returns immediately. */
|
||||
addReply(c, shared.nullmultibulk);
|
||||
} else {
|
||||
if (timeout > 0) timeout += time(NULL);
|
||||
|
||||
/* The list is empty and the client blocks. */
|
||||
blockForKeys(c, c->argv + 1, 1, timeout, c->argv[2]);
|
||||
}
|
||||
} else if (key->type != REDIS_LIST) {
|
||||
addReply(c, shared.wrongtypeerr);
|
||||
} else {
|
||||
// The list exists and has elements.
|
||||
redisAssert(listTypeLength(key) > 0);
|
||||
rpoplpushCommand(c);
|
||||
if (key->type != REDIS_LIST) {
|
||||
addReply(c, shared.wrongtypeerr);
|
||||
} else {
|
||||
|
||||
/* The list exists and has elements, so
|
||||
* the regular rpoplpushCommand is executed. */
|
||||
redisAssert(listTypeLength(key) > 0);
|
||||
rpoplpushCommand(c);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -152,6 +152,19 @@ start_server {
|
||||
assert_equal {foo} [r lrange target 0 -1]
|
||||
}
|
||||
|
||||
test "BRPOPLPUSH with a client BLPOPing the target list" {
|
||||
set rd [redis_deferring_client]
|
||||
set rd2 [redis_deferring_client]
|
||||
r del blist target
|
||||
$rd2 blpop target 0
|
||||
$rd brpoplpush blist target 0
|
||||
after 1000
|
||||
r rpush blist foo
|
||||
assert_equal foo [$rd read]
|
||||
assert_equal {target foo} [$rd2 read]
|
||||
assert_equal 0 [r exists target]
|
||||
}
|
||||
|
||||
test "BRPOPLPUSH with wrong source type" {
|
||||
set rd [redis_deferring_client]
|
||||
r del blist target
|
||||
@ -178,7 +191,7 @@ start_server {
|
||||
assert_equal {foo} [r lrange blist 0 -1]
|
||||
}
|
||||
|
||||
test {BRPOPLPUSH inside a transaction} {
|
||||
test "BRPOPLPUSH inside a transaction" {
|
||||
r del xlist target
|
||||
r lpush xlist foo
|
||||
r lpush xlist bar
|
||||
|
Loading…
Reference in New Issue
Block a user