redis/tests/unit/type/list.tcl

1959 lines
68 KiB
Tcl

# check functionality compression of plain and zipped nodes
start_server [list overrides [list save ""] ] {
r config set list-compress-depth 2
r config set list-max-ziplist-size 1
# 3 test to check compression with regular ziplist nodes
# 1. using push + insert
# 2. using push + insert + trim
# 3. using push + insert + set
test {reg node check compression with insert and pop} {
r lpush list1 [string repeat a 500]
r lpush list1 [string repeat b 500]
r lpush list1 [string repeat c 500]
r lpush list1 [string repeat d 500]
r linsert list1 after [string repeat d 500] [string repeat e 500]
r linsert list1 after [string repeat d 500] [string repeat f 500]
r linsert list1 after [string repeat d 500] [string repeat g 500]
r linsert list1 after [string repeat d 500] [string repeat j 500]
assert_equal [r lpop list1] [string repeat d 500]
assert_equal [r lpop list1] [string repeat j 500]
assert_equal [r lpop list1] [string repeat g 500]
assert_equal [r lpop list1] [string repeat f 500]
assert_equal [r lpop list1] [string repeat e 500]
assert_equal [r lpop list1] [string repeat c 500]
assert_equal [r lpop list1] [string repeat b 500]
assert_equal [r lpop list1] [string repeat a 500]
};
test {reg node check compression combined with trim} {
r lpush list2 [string repeat a 500]
r linsert list2 after [string repeat a 500] [string repeat b 500]
r rpush list2 [string repeat c 500]
assert_equal [string repeat b 500] [r lindex list2 1]
r LTRIM list2 1 -1
r llen list2
} {2}
test {reg node check compression with lset} {
r lpush list3 [string repeat a 500]
r LSET list3 0 [string repeat b 500]
assert_equal [string repeat b 500] [r lindex list3 0]
r lpush list3 [string repeat c 500]
r LSET list3 0 [string repeat d 500]
assert_equal [string repeat d 500] [r lindex list3 0]
}
# repeating the 3 tests with plain nodes
# (by adjusting quicklist-packed-threshold)
test {plain node check compression} {
r debug quicklist-packed-threshold 1b
r lpush list4 [string repeat a 500]
r lpush list4 [string repeat b 500]
r lpush list4 [string repeat c 500]
r lpush list4 [string repeat d 500]
r linsert list4 after [string repeat d 500] [string repeat e 500]
r linsert list4 after [string repeat d 500] [string repeat f 500]
r linsert list4 after [string repeat d 500] [string repeat g 500]
r linsert list4 after [string repeat d 500] [string repeat j 500]
assert_equal [r lpop list4] [string repeat d 500]
assert_equal [r lpop list4] [string repeat j 500]
assert_equal [r lpop list4] [string repeat g 500]
assert_equal [r lpop list4] [string repeat f 500]
assert_equal [r lpop list4] [string repeat e 500]
assert_equal [r lpop list4] [string repeat c 500]
assert_equal [r lpop list4] [string repeat b 500]
assert_equal [r lpop list4] [string repeat a 500]
} {} {needs:debug}
test {plain node check compression with ltrim} {
r debug quicklist-packed-threshold 1b
r lpush list5 [string repeat a 500]
r linsert list5 after [string repeat a 500] [string repeat b 500]
r rpush list5 [string repeat c 500]
assert_equal [string repeat b 500] [r lindex list5 1]
r LTRIM list5 1 -1
r llen list5
} {2} {needs:debug}
test {plain node check compression using lset} {
r debug quicklist-packed-threshold 1b
r lpush list6 [string repeat a 500]
r LSET list6 0 [string repeat b 500]
assert_equal [string repeat b 500] [r lindex list6 0]
r lpush list6 [string repeat c 500]
r LSET list6 0 [string repeat d 500]
assert_equal [string repeat d 500] [r lindex list6 0]
} {} {needs:debug}
# revert config for external mode tests.
r config set list-compress-depth 0
}
# check functionality of plain nodes using low packed-threshold
start_server [list overrides [list save ""] ] {
# basic command check for plain nodes - "LPUSH & LPOP"
test {Test LPUSH and LPOP on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r lpush lst 9
r lpush lst xxxxxxxxxx
r lpush lst xxxxxxxxxx
set s0 [s used_memory]
assert {$s0 > 10}
assert {[r llen lst] == 3}
set s0 [r rpop lst]
set s1 [r rpop lst]
assert {$s0 eq "9"}
assert {[r llen lst] == 1}
r lpop lst
assert {[string length $s1] == 10}
# check rdb
r lpush lst xxxxxxxxxx
r lpush lst bb
r debug reload
assert_equal [r rpop lst] "xxxxxxxxxx"
} {} {needs:debug}
# basic command check for plain nodes - "LINDEX & LINSERT"
test {Test LINDEX and LINSERT on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r lpush lst xxxxxxxxxxx
r lpush lst 9
r lpush lst xxxxxxxxxxx
r linsert lst before "9" "8"
assert {[r lindex lst 1] eq "8"}
r linsert lst BEFORE "9" "7"
r linsert lst BEFORE "9" "xxxxxxxxxxx"
assert {[r lindex lst 3] eq "xxxxxxxxxxx"}
} {} {needs:debug}
# basic command check for plain nodes - "LTRIM"
test {Test LTRIM on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r lpush lst1 9
r lpush lst1 xxxxxxxxxxx
r lpush lst1 9
r LTRIM lst1 1 -1
assert_equal [r llen lst1] 2
} {} {needs:debug}
# basic command check for plain nodes - "LREM"
test {Test LREM on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r lpush lst one
r lpush lst xxxxxxxxxxx
set s0 [s used_memory]
assert {$s0 > 10}
r lpush lst 9
r LREM lst -2 "one"
assert_equal [r llen lst] 2
} {} {needs:debug}
# basic command check for plain nodes - "LPOS"
test {Test LPOS on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r RPUSH lst "aa"
r RPUSH lst "bb"
r RPUSH lst "cc"
r LSET lst 0 "xxxxxxxxxxx"
assert_equal [r LPOS lst "xxxxxxxxxxx"] 0
} {} {needs:debug}
# basic command check for plain nodes - "LMOVE"
test {Test LMOVE on plain nodes} {
r flushdb
r debug quicklist-packed-threshold 1b
r RPUSH lst2{t} "aa"
r RPUSH lst2{t} "bb"
r LSET lst2{t} 0 xxxxxxxxxxx
r RPUSH lst2{t} "cc"
r RPUSH lst2{t} "dd"
r LMOVE lst2{t} lst{t} RIGHT LEFT
r LMOVE lst2{t} lst{t} LEFT RIGHT
assert_equal [r llen lst{t}] 2
assert_equal [r llen lst2{t}] 2
assert_equal [r lpop lst2{t}] "bb"
assert_equal [r lpop lst2{t}] "cc"
assert_equal [r lpop lst{t}] "dd"
assert_equal [r lpop lst{t}] "xxxxxxxxxxx"
} {} {needs:debug}
# testing LSET with combinations of node types
# plain->packed , packed->plain, plain->plain, packed->packed
test {Test LSET with packed / plain combinations} {
r debug quicklist-packed-threshold 5b
r RPUSH lst "aa"
r RPUSH lst "bb"
r lset lst 0 [string repeat d 50001]
set s1 [r lpop lst]
assert_equal $s1 [string repeat d 50001]
r RPUSH lst [string repeat f 50001]
r lset lst 0 [string repeat e 50001]
set s1 [r lpop lst]
assert_equal $s1 [string repeat e 50001]
r RPUSH lst [string repeat m 50001]
r lset lst 0 "bb"
set s1 [r lpop lst]
assert_equal $s1 "bb"
r RPUSH lst "bb"
r lset lst 0 "cc"
set s1 [r lpop lst]
assert_equal $s1 "cc"
} {} {needs:debug}
# checking LSET in case ziplist needs to be split
test {Test LSET with packed is split in the middle} {
r flushdb
r debug quicklist-packed-threshold 5b
r RPUSH lst "aa"
r RPUSH lst "bb"
r RPUSH lst "cc"
r RPUSH lst "dd"
r RPUSH lst "ee"
r lset lst 2 [string repeat e 10]
assert_equal [r lpop lst] "aa"
assert_equal [r lpop lst] "bb"
assert_equal [r lpop lst] [string repeat e 10]
assert_equal [r lpop lst] "dd"
assert_equal [r lpop lst] "ee"
} {} {needs:debug}
# repeating "plain check LSET with combinations"
# but now with single item in each ziplist
test {Test LSET with packed consist only one item} {
r flushdb
r config set list-max-ziplist-size 1
r debug quicklist-packed-threshold 1b
r RPUSH lst "aa"
r RPUSH lst "bb"
r lset lst 0 [string repeat d 50001]
set s1 [r lpop lst]
assert_equal $s1 [string repeat d 50001]
r RPUSH lst [string repeat f 50001]
r lset lst 0 [string repeat e 50001]
set s1 [r lpop lst]
assert_equal $s1 [string repeat e 50001]
r RPUSH lst [string repeat m 50001]
r lset lst 0 "bb"
set s1 [r lpop lst]
assert_equal $s1 "bb"
r RPUSH lst "bb"
r lset lst 0 "cc"
set s1 [r lpop lst]
assert_equal $s1 "cc"
} {} {needs:debug}
}
run_solo {list-large-memory} {
start_server [list overrides [list save ""] ] {
# test if the server supports such large configs (avoid 32 bit builds)
catch {
r config set proto-max-bulk-len 10000000000 ;#10gb
r config set client-query-buffer-limit 10000000000 ;#10gb
}
if {[lindex [r config get proto-max-bulk-len] 1] == 10000000000} {
set str_length 5000000000
# repeating all the plain nodes basic checks with 5gb values
test {Test LPUSH and LPOP on plain nodes over 4GB} {
r flushdb
r lpush lst 9
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
set s0 [s used_memory]
assert {$s0 > $str_length}
assert {[r llen lst] == 3}
assert_equal [r rpop lst] "9"
assert_equal [read_big_bulk {r rpop lst}] $str_length
assert {[r llen lst] == 1}
assert_equal [read_big_bulk {r rpop lst}] $str_length
} {} {large-memory}
test {Test LINDEX and LINSERT on plain nodes over 4GB} {
r flushdb
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
r lpush lst 9
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
r linsert lst before "9" "8"
assert_equal [r lindex lst 1] "8"
r LINSERT lst BEFORE "9" "7"
r write "*5\r\n\$7\r\nLINSERT\r\n\$3\r\nlst\r\n\$6\r\nBEFORE\r\n\$3\r\n\"9\"\r\n"
write_big_bulk 10;
assert_equal [read_big_bulk {r rpop lst}] $str_length
} {} {large-memory}
test {Test LTRIM on plain nodes over 4GB} {
r flushdb
r lpush lst 9
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
r lpush lst 9
r LTRIM lst 1 -1
assert_equal [r llen lst] 2
assert_equal [r rpop lst] 9
assert_equal [read_big_bulk {r rpop lst}] $str_length
} {} {large-memory}
test {Test LREM on plain nodes over 4GB} {
r flushdb
r lpush lst one
r write "*3\r\n\$5\r\nLPUSH\r\n\$3\r\nlst\r\n"
write_big_bulk $str_length;
r lpush lst 9
r LREM lst -2 "one"
assert_equal [read_big_bulk {r rpop lst}] $str_length
r llen lst
} {1} {large-memory}
test {Test LSET on plain nodes over 4GB} {
r flushdb
r RPUSH lst "aa"
r RPUSH lst "bb"
r RPUSH lst "cc"
r write "*4\r\n\$4\r\nLSET\r\n\$3\r\nlst\r\n\$1\r\n0\r\n"
write_big_bulk $str_length;
assert_equal [r rpop lst] "cc"
assert_equal [r rpop lst] "bb"
assert_equal [read_big_bulk {r rpop lst}] $str_length
} {} {large-memory}
test {Test LMOVE on plain nodes over 4GB} {
r flushdb
r RPUSH lst2{t} "aa"
r RPUSH lst2{t} "bb"
r write "*4\r\n\$4\r\nLSET\r\n\$7\r\nlst2{t}\r\n\$1\r\n0\r\n"
write_big_bulk $str_length;
r RPUSH lst2{t} "cc"
r RPUSH lst2{t} "dd"
r LMOVE lst2{t} lst{t} RIGHT LEFT
assert_equal [read_big_bulk {r LMOVE lst2{t} lst{t} LEFT RIGHT}] $str_length
assert_equal [r llen lst{t}] 2
assert_equal [r llen lst2{t}] 2
assert_equal [r lpop lst2{t}] "bb"
assert_equal [r lpop lst2{t}] "cc"
assert_equal [r lpop lst{t}] "dd"
assert_equal [read_big_bulk {r rpop lst{t}}] $str_length
} {} {large-memory}
} ;# skip 32bit builds
}
} ;# run_solo
start_server {
tags {"list"}
overrides {
"list-max-ziplist-size" 5
}
} {
source "tests/unit/type/list-common.tcl"
# A helper function to execute either B*POP or BLMPOP* with one input key.
proc bpop_command {rd pop key timeout} {
if {$pop == "BLMPOP_LEFT"} {
$rd blmpop $timeout 1 $key left count 1
} elseif {$pop == "BLMPOP_RIGHT"} {
$rd blmpop $timeout 1 $key right count 1
} else {
$rd $pop $key $timeout
}
}
# A helper function to execute either B*POP or BLMPOP* with two input keys.
proc bpop_command_two_key {rd pop key key2 timeout} {
if {$pop == "BLMPOP_LEFT"} {
$rd blmpop $timeout 2 $key $key2 left count 1
} elseif {$pop == "BLMPOP_RIGHT"} {
$rd blmpop $timeout 2 $key $key2 right count 1
} else {
$rd $pop $key $key2 $timeout
}
}
test {LPOS basic usage} {
r DEL mylist
r RPUSH mylist a b c 1 2 3 c c
assert {[r LPOS mylist a] == 0}
assert {[r LPOS mylist c] == 2}
}
test {LPOS RANK (positive, negative and zero rank) option} {
assert {[r LPOS mylist c RANK 1] == 2}
assert {[r LPOS mylist c RANK 2] == 6}
assert {[r LPOS mylist c RANK 4] eq ""}
assert {[r LPOS mylist c RANK -1] == 7}
assert {[r LPOS mylist c RANK -2] == 6}
assert_error "*RANK can't be zero: use 1 to start from the first match, 2 from the second ... or use negative to start*" {r LPOS mylist c RANK 0}
}
test {LPOS COUNT option} {
assert {[r LPOS mylist c COUNT 0] == {2 6 7}}
assert {[r LPOS mylist c COUNT 1] == {2}}
assert {[r LPOS mylist c COUNT 2] == {2 6}}
assert {[r LPOS mylist c COUNT 100] == {2 6 7}}
}
test {LPOS COUNT + RANK option} {
assert {[r LPOS mylist c COUNT 0 RANK 2] == {6 7}}
assert {[r LPOS mylist c COUNT 2 RANK -1] == {7 6}}
}
test {LPOS non existing key} {
assert {[r LPOS mylistxxx c COUNT 0 RANK 2] eq {}}
}
test {LPOS no match} {
assert {[r LPOS mylist x COUNT 2 RANK -1] eq {}}
assert {[r LPOS mylist x RANK -1] eq {}}
}
test {LPOS MAXLEN} {
assert {[r LPOS mylist a COUNT 0 MAXLEN 1] == {0}}
assert {[r LPOS mylist c COUNT 0 MAXLEN 1] == {}}
assert {[r LPOS mylist c COUNT 0 MAXLEN 3] == {2}}
assert {[r LPOS mylist c COUNT 0 MAXLEN 3 RANK -1] == {7 6}}
assert {[r LPOS mylist c COUNT 0 MAXLEN 7 RANK 2] == {6}}
}
test {LPOS when RANK is greater than matches} {
r DEL mylist
r LPUSH mylist a
assert {[r LPOS mylist b COUNT 10 RANK 5] eq {}}
}
test {LPUSH, RPUSH, LLENGTH, LINDEX, LPOP - ziplist} {
# first lpush then rpush
assert_equal 1 [r lpush myziplist1 aa]
assert_equal 2 [r rpush myziplist1 bb]
assert_equal 3 [r rpush myziplist1 cc]
assert_equal 3 [r llen myziplist1]
assert_equal aa [r lindex myziplist1 0]
assert_equal bb [r lindex myziplist1 1]
assert_equal cc [r lindex myziplist1 2]
assert_equal {} [r lindex myziplist2 3]
assert_equal cc [r rpop myziplist1]
assert_equal aa [r lpop myziplist1]
assert_encoding quicklist myziplist1
# first rpush then lpush
assert_equal 1 [r rpush myziplist2 a]
assert_equal 2 [r lpush myziplist2 b]
assert_equal 3 [r lpush myziplist2 c]
assert_equal 3 [r llen myziplist2]
assert_equal c [r lindex myziplist2 0]
assert_equal b [r lindex myziplist2 1]
assert_equal a [r lindex myziplist2 2]
assert_equal {} [r lindex myziplist2 3]
assert_equal a [r rpop myziplist2]
assert_equal c [r lpop myziplist2]
assert_encoding quicklist myziplist2
}
test {LPUSH, RPUSH, LLENGTH, LINDEX, LPOP - regular list} {
# first lpush then rpush
assert_equal 1 [r lpush mylist1 $largevalue(linkedlist)]
assert_encoding quicklist mylist1
assert_equal 2 [r rpush mylist1 b]
assert_equal 3 [r rpush mylist1 c]
assert_equal 3 [r llen mylist1]
assert_equal $largevalue(linkedlist) [r lindex mylist1 0]
assert_equal b [r lindex mylist1 1]
assert_equal c [r lindex mylist1 2]
assert_equal {} [r lindex mylist1 3]
assert_equal c [r rpop mylist1]
assert_equal $largevalue(linkedlist) [r lpop mylist1]
# first rpush then lpush
assert_equal 1 [r rpush mylist2 $largevalue(linkedlist)]
assert_encoding quicklist mylist2
assert_equal 2 [r lpush mylist2 b]
assert_equal 3 [r lpush mylist2 c]
assert_equal 3 [r llen mylist2]
assert_equal c [r lindex mylist2 0]
assert_equal b [r lindex mylist2 1]
assert_equal $largevalue(linkedlist) [r lindex mylist2 2]
assert_equal {} [r lindex mylist2 3]
assert_equal $largevalue(linkedlist) [r rpop mylist2]
assert_equal c [r lpop mylist2]
}
test "LPOP/RPOP with wrong number of arguments" {
assert_error {*wrong number of arguments for 'lpop' command} {r lpop key 1 1}
assert_error {*wrong number of arguments for 'rpop' command} {r rpop key 2 2}
}
test {RPOP/LPOP with the optional count argument} {
assert_equal 7 [r lpush listcount aa bb cc dd ee ff gg]
assert_equal {gg} [r lpop listcount 1]
assert_equal {ff ee} [r lpop listcount 2]
assert_equal {aa bb} [r rpop listcount 2]
assert_equal {cc} [r rpop listcount 1]
assert_equal {dd} [r rpop listcount 123]
assert_error "*ERR*range*" {r lpop forbarqaz -123}
}
proc verify_resp_response {resp response resp2_response resp3_response} {
if {$resp == 2} {
assert_equal $response $resp2_response
} elseif {$resp == 3} {
assert_equal $response $resp3_response
}
}
foreach resp {3 2} {
if {[lsearch $::denytags "resp3"] >= 0} {
if {$resp == 3} {continue}
} else {
r hello $resp
}
# Make sure we can distinguish between an empty array and a null response
r readraw 1
test "LPOP/RPOP with the count 0 returns an empty array in RESP$resp" {
r lpush listcount zero
assert_equal {*0} [r lpop listcount 0]
assert_equal {*0} [r rpop listcount 0]
}
test "LPOP/RPOP against non existing key in RESP$resp" {
r del non_existing_key
verify_resp_response $resp [r lpop non_existing_key] {$-1} {_}
verify_resp_response $resp [r rpop non_existing_key] {$-1} {_}
}
test "LPOP/RPOP with <count> against non existing key in RESP$resp" {
r del non_existing_key
verify_resp_response $resp [r lpop non_existing_key 0] {*-1} {_}
verify_resp_response $resp [r lpop non_existing_key 1] {*-1} {_}
verify_resp_response $resp [r rpop non_existing_key 0] {*-1} {_}
verify_resp_response $resp [r rpop non_existing_key 1] {*-1} {_}
}
r readraw 0
}
test {Variadic RPUSH/LPUSH} {
r del mylist
assert_equal 4 [r lpush mylist a b c d]
assert_equal 8 [r rpush mylist 0 1 2 3]
assert_equal {d c b a 0 1 2 3} [r lrange mylist 0 -1]
}
test {DEL a list} {
assert_equal 1 [r del mylist2]
assert_equal 0 [r exists mylist2]
assert_equal 0 [r llen mylist2]
}
proc create_list {key entries} {
r del $key
foreach entry $entries { r rpush $key $entry }
assert_encoding quicklist $key
}
foreach {type large} [array get largevalue] {
foreach {pop} {BLPOP BLMPOP_LEFT} {
test "$pop: single existing list - $type" {
set rd [redis_deferring_client]
create_list blist "a b $large c d"
bpop_command $rd $pop blist 1
assert_equal {blist a} [$rd read]
if {$pop == "BLPOP"} {
bpop_command $rd BRPOP blist 1
} else {
bpop_command $rd BLMPOP_RIGHT blist 1
}
assert_equal {blist d} [$rd read]
bpop_command $rd $pop blist 1
assert_equal {blist b} [$rd read]
if {$pop == "BLPOP"} {
bpop_command $rd BRPOP blist 1
} else {
bpop_command $rd BLMPOP_RIGHT blist 1
}
assert_equal {blist c} [$rd read]
assert_equal 1 [r llen blist]
$rd close
}
test "$pop: multiple existing lists - $type" {
set rd [redis_deferring_client]
create_list blist1{t} "a $large c"
create_list blist2{t} "d $large f"
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
assert_equal {blist1{t} a} [$rd read]
if {$pop == "BLPOP"} {
bpop_command_two_key $rd BRPOP blist1{t} blist2{t} 1
} else {
bpop_command_two_key $rd BLMPOP_RIGHT blist1{t} blist2{t} 1
}
assert_equal {blist1{t} c} [$rd read]
assert_equal 1 [r llen blist1{t}]
assert_equal 3 [r llen blist2{t}]
bpop_command_two_key $rd $pop blist2{t} blist1{t} 1
assert_equal {blist2{t} d} [$rd read]
if {$pop == "BLPOP"} {
bpop_command_two_key $rd BRPOP blist2{t} blist1{t} 1
} else {
bpop_command_two_key $rd BLMPOP_RIGHT blist2{t} blist1{t} 1
}
assert_equal {blist2{t} f} [$rd read]
assert_equal 1 [r llen blist1{t}]
assert_equal 1 [r llen blist2{t}]
$rd close
}
test "$pop: second list has an entry - $type" {
set rd [redis_deferring_client]
r del blist1{t}
create_list blist2{t} "d $large f"
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
assert_equal {blist2{t} d} [$rd read]
if {$pop == "BLPOP"} {
bpop_command_two_key $rd BRPOP blist1{t} blist2{t} 1
} else {
bpop_command_two_key $rd BLMPOP_RIGHT blist1{t} blist2{t} 1
}
assert_equal {blist2{t} f} [$rd read]
assert_equal 0 [r llen blist1{t}]
assert_equal 1 [r llen blist2{t}]
$rd close
}
}
test "BRPOPLPUSH - $type" {
r del target{t}
r rpush target{t} bar
set rd [redis_deferring_client]
create_list blist{t} "a b $large c d"
$rd brpoplpush blist{t} target{t} 1
assert_equal d [$rd read]
assert_equal d [r lpop target{t}]
assert_equal "a b $large c" [r lrange blist{t} 0 -1]
$rd close
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "BLMOVE $wherefrom $whereto - $type" {
r del target{t}
r rpush target{t} bar
set rd [redis_deferring_client]
create_list blist{t} "a b $large c d"
$rd blmove blist{t} target{t} $wherefrom $whereto 1
set poppedelement [$rd read]
if {$wherefrom eq "right"} {
assert_equal d $poppedelement
assert_equal "a b $large c" [r lrange blist{t} 0 -1]
} else {
assert_equal a $poppedelement
assert_equal "b $large c d" [r lrange blist{t} 0 -1]
}
if {$whereto eq "right"} {
assert_equal $poppedelement [r rpop target{t}]
} else {
assert_equal $poppedelement [r lpop target{t}]
}
$rd close
}
}
}
}
foreach {pop} {BLPOP BLMPOP_LEFT} {
test "$pop, LPUSH + DEL should not awake blocked client" {
set rd [redis_deferring_client]
r del list
bpop_command $rd $pop list 0
after 100 ;# Make sure rd is blocked before MULTI
wait_for_blocked_client
r multi
r lpush list a
r del list
r exec
r del list
r lpush list b
assert_equal {list b} [$rd read]
$rd close
}
test "$pop, LPUSH + DEL + SET should not awake blocked client" {
set rd [redis_deferring_client]
r del list
bpop_command $rd $pop list 0
after 100 ;# Make sure rd is blocked before MULTI
wait_for_blocked_client
r multi
r lpush list a
r del list
r set list foo
r exec
r del list
r lpush list b
assert_equal {list b} [$rd read]
$rd close
}
}
test "BLPOP with same key multiple times should work (issue #801)" {
set rd [redis_deferring_client]
r del list1{t} list2{t}
# Data arriving after the BLPOP.
$rd blpop list1{t} list2{t} list2{t} list1{t} 0
r lpush list1{t} a
assert_equal [$rd read] {list1{t} a}
$rd blpop list1{t} list2{t} list2{t} list1{t} 0
r lpush list2{t} b
assert_equal [$rd read] {list2{t} b}
# Data already there.
r lpush list1{t} a
r lpush list2{t} b
$rd blpop list1{t} list2{t} list2{t} list1{t} 0
assert_equal [$rd read] {list1{t} a}
$rd blpop list1{t} list2{t} list2{t} list1{t} 0
assert_equal [$rd read] {list2{t} b}
$rd close
}
foreach {pop} {BLPOP BLMPOP_LEFT} {
test "MULTI/EXEC is isolated from the point of view of $pop" {
set rd [redis_deferring_client]
r del list
bpop_command $rd $pop list 0
after 100 ;# Make sure rd is blocked before MULTI
wait_for_blocked_client
r multi
r lpush list a
r lpush list b
r lpush list c
r exec
assert_equal {list c} [$rd read]
$rd close
}
test "$pop with variadic LPUSH" {
set rd [redis_deferring_client]
r del blist
if {$::valgrind} {after 100}
bpop_command $rd $pop blist 0
if {$::valgrind} {after 100}
wait_for_blocked_client
assert_equal 2 [r lpush blist foo bar]
if {$::valgrind} {after 100}
assert_equal {blist bar} [$rd read]
assert_equal foo [lindex [r lrange blist 0 -1] 0]
$rd close
}
}
test "BRPOPLPUSH with zero timeout should block indefinitely" {
set rd [redis_deferring_client]
r del blist{t} target{t}
r rpush target{t} bar
$rd brpoplpush blist{t} target{t} 0
wait_for_blocked_clients_count 1
r rpush blist{t} foo
assert_equal foo [$rd read]
assert_equal {foo bar} [r lrange target{t} 0 -1]
$rd close
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "BLMOVE $wherefrom $whereto with zero timeout should block indefinitely" {
set rd [redis_deferring_client]
r del blist{t} target{t}
r rpush target{t} bar
$rd blmove blist{t} target{t} $wherefrom $whereto 0
wait_for_blocked_clients_count 1
r rpush blist{t} foo
assert_equal foo [$rd read]
if {$whereto eq "right"} {
assert_equal {bar foo} [r lrange target{t} 0 -1]
} else {
assert_equal {foo bar} [r lrange target{t} 0 -1]
}
$rd close
}
}
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "BLMOVE ($wherefrom, $whereto) with a client BLPOPing the target list" {
set rd [redis_deferring_client]
set rd2 [redis_deferring_client]
r del blist{t} target{t}
$rd2 blpop target{t} 0
wait_for_blocked_clients_count 1
$rd blmove blist{t} target{t} $wherefrom $whereto 0
wait_for_blocked_clients_count 2
r rpush blist{t} foo
assert_equal foo [$rd read]
assert_equal {target{t} foo} [$rd2 read]
assert_equal 0 [r exists target{t}]
$rd close
$rd2 close
}
}
}
test "BRPOPLPUSH with wrong source type" {
set rd [redis_deferring_client]
r del blist{t} target{t}
r set blist{t} nolist
$rd brpoplpush blist{t} target{t} 1
assert_error "WRONGTYPE*" {$rd read}
$rd close
}
test "BRPOPLPUSH with wrong destination type" {
set rd [redis_deferring_client]
r del blist{t} target{t}
r set target{t} nolist
r lpush blist{t} foo
$rd brpoplpush blist{t} target{t} 1
assert_error "WRONGTYPE*" {$rd read}
$rd close
set rd [redis_deferring_client]
r del blist{t} target{t}
r set target{t} nolist
$rd brpoplpush blist{t} target{t} 0
wait_for_blocked_clients_count 1
r rpush blist{t} foo
assert_error "WRONGTYPE*" {$rd read}
assert_equal {foo} [r lrange blist{t} 0 -1]
$rd close
}
test "BRPOPLPUSH maintains order of elements after failure" {
set rd [redis_deferring_client]
r del blist{t} target{t}
r set target{t} nolist
$rd brpoplpush blist{t} target{t} 0
r rpush blist{t} a b c
assert_error "WRONGTYPE*" {$rd read}
$rd close
r lrange blist{t} 0 -1
} {a b c}
test "BRPOPLPUSH with multiple blocked clients" {
set rd1 [redis_deferring_client]
set rd2 [redis_deferring_client]
r del blist{t} target1{t} target2{t}
r set target1{t} nolist
$rd1 brpoplpush blist{t} target1{t} 0
$rd2 brpoplpush blist{t} target2{t} 0
r lpush blist{t} foo
assert_error "WRONGTYPE*" {$rd1 read}
assert_equal {foo} [$rd2 read]
assert_equal {foo} [r lrange target2{t} 0 -1]
$rd1 close
$rd2 close
}
test "BLMPOP with multiple blocked clients" {
set rd1 [redis_deferring_client]
set rd2 [redis_deferring_client]
set rd3 [redis_deferring_client]
set rd4 [redis_deferring_client]
r del blist{t} blist2{t}
$rd1 blmpop 0 2 blist{t} blist2{t} left count 1
wait_for_blocked_clients_count 1
$rd2 blmpop 0 2 blist{t} blist2{t} right count 10
wait_for_blocked_clients_count 2
$rd3 blmpop 0 2 blist{t} blist2{t} left count 10
wait_for_blocked_clients_count 3
$rd4 blmpop 0 2 blist{t} blist2{t} right count 1
wait_for_blocked_clients_count 4
r multi
r lpush blist{t} a b c d e
r lpush blist2{t} 1 2 3 4 5
r exec
assert_equal {blist{t} e} [$rd1 read]
assert_equal {blist{t} {a b c d}} [$rd2 read]
assert_equal {blist2{t} {5 4 3 2 1}} [$rd3 read]
r lpush blist2{t} 1 2 3
assert_equal {blist2{t} 1} [$rd4 read]
$rd1 close
$rd2 close
$rd3 close
$rd4 close
}
test "Linked LMOVEs" {
set rd1 [redis_deferring_client]
set rd2 [redis_deferring_client]
r del list1{t} list2{t} list3{t}
$rd1 blmove list1{t} list2{t} right left 0
wait_for_blocked_clients_count 1
$rd2 blmove list2{t} list3{t} left right 0
wait_for_blocked_clients_count 2
r rpush list1{t} foo
assert_equal {} [r lrange list1{t} 0 -1]
assert_equal {} [r lrange list2{t} 0 -1]
assert_equal {foo} [r lrange list3{t} 0 -1]
$rd1 close
$rd2 close
}
test "Circular BRPOPLPUSH" {
set rd1 [redis_deferring_client]
set rd2 [redis_deferring_client]
r del list1{t} list2{t}
$rd1 brpoplpush list1{t} list2{t} 0
wait_for_blocked_clients_count 1
$rd2 brpoplpush list2{t} list1{t} 0
wait_for_blocked_clients_count 2
r rpush list1{t} foo
assert_equal {foo} [r lrange list1{t} 0 -1]
assert_equal {} [r lrange list2{t} 0 -1]
$rd1 close
$rd2 close
}
test "Self-referential BRPOPLPUSH" {
set rd [redis_deferring_client]
r del blist{t}
$rd brpoplpush blist{t} blist{t} 0
wait_for_blocked_client
r rpush blist{t} foo
assert_equal {foo} [r lrange blist{t} 0 -1]
$rd close
}
test "BRPOPLPUSH inside a transaction" {
r del xlist{t} target{t}
r lpush xlist{t} foo
r lpush xlist{t} bar
r multi
r brpoplpush xlist{t} target{t} 0
r brpoplpush xlist{t} target{t} 0
r brpoplpush xlist{t} target{t} 0
r lrange xlist{t} 0 -1
r lrange target{t} 0 -1
r exec
} {foo bar {} {} {bar foo}}
test "PUSH resulting from BRPOPLPUSH affect WATCH" {
set blocked_client [redis_deferring_client]
set watching_client [redis_deferring_client]
r del srclist{t} dstlist{t} somekey{t}
r set somekey{t} somevalue
$blocked_client brpoplpush srclist{t} dstlist{t} 0
wait_for_blocked_client
$watching_client watch dstlist{t}
$watching_client read
$watching_client multi
$watching_client read
$watching_client get somekey{t}
$watching_client read
r lpush srclist{t} element
$watching_client exec
set res [$watching_client read]
$blocked_client close
$watching_client close
set _ $res
} {}
test "BRPOPLPUSH does not affect WATCH while still blocked" {
set blocked_client [redis_deferring_client]
set watching_client [redis_deferring_client]
r del srclist{t} dstlist{t} somekey{t}
r set somekey{t} somevalue
$blocked_client brpoplpush srclist{t} dstlist{t} 0
wait_for_blocked_client
$watching_client watch dstlist{t}
$watching_client read
$watching_client multi
$watching_client read
$watching_client get somekey{t}
$watching_client read
$watching_client exec
# Blocked BLPOPLPUSH may create problems, unblock it.
r lpush srclist{t} element
set res [$watching_client read]
$blocked_client close
$watching_client close
set _ $res
} {somevalue}
test {BRPOPLPUSH timeout} {
set rd [redis_deferring_client]
$rd brpoplpush foo_list{t} bar_list{t} 1
wait_for_blocked_clients_count 1
wait_for_blocked_clients_count 0 500 10
set res [$rd read]
$rd close
set _ $res
} {}
test {SWAPDB awakes blocked client} {
r flushall
r select 1
r rpush k hello
r select 9
set rd [redis_deferring_client]
$rd brpop k 5
wait_for_blocked_clients_count 1
r swapdb 1 9
$rd read
} {k hello} {singledb:skip}
test {SWAPDB wants to wake blocked client, but the key already expired} {
set repl [attach_to_replication_stream]
r flushall
r debug set-active-expire 0
r select 1
r rpush k hello
r pexpire k 100
set rd [redis_deferring_client]
$rd select 9
assert_equal {OK} [$rd read]
$rd client id
set id [$rd read]
$rd brpop k 1
wait_for_blocked_clients_count 1
after 101
r swapdb 1 9
# The SWAPDB command tries to awake the blocked client, but it remains
# blocked because the key is expired. Check that the deferred client is
# still blocked. Then unblock it.
assert_match "*flags=b*" [r client list id $id]
r client unblock $id
assert_equal {} [$rd read]
assert_replication_stream $repl {
{select *}
{flushall}
{select 1}
{rpush k hello}
{pexpireat k *}
{swapdb 1 9}
{select 9}
{del k}
}
close_replication_stream $repl
# Restore server and client state
r debug set-active-expire 1
r select 9
} {OK} {singledb:skip needs:debug}
test {MULTI + LPUSH + EXPIRE + DEBUG SLEEP on blocked client, key already expired} {
set repl [attach_to_replication_stream]
r flushall
r debug set-active-expire 0
set rd [redis_deferring_client]
$rd client id
set id [$rd read]
$rd brpop k 0
wait_for_blocked_clients_count 1
r multi
r rpush k hello
r pexpire k 100
r debug sleep 0.2
r exec
# The EXEC command tries to awake the blocked client, but it remains
# blocked because the key is expired. Check that the deferred client is
# still blocked. Then unblock it.
assert_match "*flags=b*" [r client list id $id]
r client unblock $id
assert_equal {} [$rd read]
assert_replication_stream $repl {
{select *}
{flushall}
{multi}
{rpush k hello}
{pexpireat k *}
{exec}
{del k}
}
close_replication_stream $repl
# Restore server and client state
r debug set-active-expire 1
r select 9
} {OK} {singledb:skip needs:debug}
foreach {pop} {BLPOP BLMPOP_LEFT} {
test "$pop when new key is moved into place" {
set rd [redis_deferring_client]
r del foo{t}
bpop_command $rd $pop foo{t} 0
wait_for_blocked_client
r lpush bob{t} abc def hij
r rename bob{t} foo{t}
set res [$rd read]
$rd close
set _ $res
} {foo{t} hij}
test "$pop when result key is created by SORT..STORE" {
set rd [redis_deferring_client]
# zero out list from previous test without explicit delete
r lpop foo{t}
r lpop foo{t}
r lpop foo{t}
bpop_command $rd $pop foo{t} 5
wait_for_blocked_client
r lpush notfoo{t} hello hola aguacate konichiwa zanzibar
r sort notfoo{t} ALPHA store foo{t}
set res [$rd read]
$rd close
set _ $res
} {foo{t} aguacate}
}
foreach {pop} {BLPOP BRPOP BLMPOP_LEFT BLMPOP_RIGHT} {
test "$pop: with single empty list argument" {
set rd [redis_deferring_client]
r del blist1
bpop_command $rd $pop blist1 1
wait_for_blocked_client
r rpush blist1 foo
assert_equal {blist1 foo} [$rd read]
assert_equal 0 [r exists blist1]
$rd close
}
test "$pop: with negative timeout" {
set rd [redis_deferring_client]
bpop_command $rd $pop blist1 -1
assert_error "ERR *is negative*" {$rd read}
$rd close
}
test "$pop: with non-integer timeout" {
set rd [redis_deferring_client]
r del blist1
bpop_command $rd $pop blist1 0.1
r rpush blist1 foo
assert_equal {blist1 foo} [$rd read]
assert_equal 0 [r exists blist1]
$rd close
}
test "$pop: with zero timeout should block indefinitely" {
# To test this, use a timeout of 0 and wait a second.
# The blocking pop should still be waiting for a push.
set rd [redis_deferring_client]
bpop_command $rd $pop blist1 0
wait_for_blocked_client
after 1000
r rpush blist1 foo
assert_equal {blist1 foo} [$rd read]
$rd close
}
test "$pop: second argument is not a list" {
set rd [redis_deferring_client]
r del blist1{t} blist2{t}
r set blist2{t} nolist{t}
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
$rd $pop blist1{t} blist2{t} 1
assert_error "WRONGTYPE*" {$rd read}
$rd close
}
test "$pop: timeout" {
set rd [redis_deferring_client]
r del blist1{t} blist2{t}
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
wait_for_blocked_client
assert_equal {} [$rd read]
$rd close
}
test "$pop: arguments are empty" {
set rd [redis_deferring_client]
r del blist1{t} blist2{t}
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
wait_for_blocked_client
r rpush blist1{t} foo
assert_equal {blist1{t} foo} [$rd read]
assert_equal 0 [r exists blist1{t}]
assert_equal 0 [r exists blist2{t}]
bpop_command_two_key $rd $pop blist1{t} blist2{t} 1
wait_for_blocked_client
r rpush blist2{t} foo
assert_equal {blist2{t} foo} [$rd read]
assert_equal 0 [r exists blist1{t}]
assert_equal 0 [r exists blist2{t}]
$rd close
}
}
foreach {pop} {BLPOP BLMPOP_LEFT} {
test "$pop inside a transaction" {
r del xlist
r lpush xlist foo
r lpush xlist bar
r multi
bpop_command r $pop xlist 0
bpop_command r $pop xlist 0
bpop_command r $pop xlist 0
r exec
} {{xlist bar} {xlist foo} {}}
}
test {BLMPOP propagate as pop with count command to replica} {
set rd [redis_deferring_client]
set repl [attach_to_replication_stream]
# BLMPOP without being blocked.
r lpush mylist{t} a b c
r rpush mylist2{t} 1 2 3
r blmpop 0 1 mylist{t} left count 1
r blmpop 0 2 mylist{t} mylist2{t} right count 10
r blmpop 0 2 mylist{t} mylist2{t} right count 10
# BLMPOP that gets blocked.
$rd blmpop 0 1 mylist{t} left count 1
wait_for_blocked_client
r lpush mylist{t} a
$rd blmpop 0 2 mylist{t} mylist2{t} left count 5
wait_for_blocked_client
r lpush mylist{t} a b c
$rd blmpop 0 2 mylist{t} mylist2{t} right count 10
wait_for_blocked_client
r rpush mylist2{t} a b c
# Released on timeout.
assert_equal {} [r blmpop 0.01 1 mylist{t} left count 10]
r set foo{t} bar ;# something else to propagate after, so we can make sure the above pop didn't.
$rd close
assert_replication_stream $repl {
{select *}
{lpush mylist{t} a b c}
{rpush mylist2{t} 1 2 3}
{lpop mylist{t} 1}
{rpop mylist{t} 2}
{rpop mylist2{t} 3}
{lpush mylist{t} a}
{lpop mylist{t} 1}
{lpush mylist{t} a b c}
{lpop mylist{t} 3}
{rpush mylist2{t} a b c}
{rpop mylist2{t} 3}
{set foo{t} bar}
}
close_replication_stream $repl
} {} {needs:repl}
test {LPUSHX, RPUSHX - generic} {
r del xlist
assert_equal 0 [r lpushx xlist a]
assert_equal 0 [r llen xlist]
assert_equal 0 [r rpushx xlist a]
assert_equal 0 [r llen xlist]
}
foreach {type large} [array get largevalue] {
test "LPUSHX, RPUSHX - $type" {
create_list xlist "$large c"
assert_equal 3 [r rpushx xlist d]
assert_equal 4 [r lpushx xlist a]
assert_equal 6 [r rpushx xlist 42 x]
assert_equal 9 [r lpushx xlist y3 y2 y1]
assert_equal "y1 y2 y3 a $large c d 42 x" [r lrange xlist 0 -1]
}
test "LINSERT - $type" {
create_list xlist "a $large c d"
assert_equal 5 [r linsert xlist before c zz] "before c"
assert_equal "a $large zz c d" [r lrange xlist 0 10] "lrangeA"
assert_equal 6 [r linsert xlist after c yy] "after c"
assert_equal "a $large zz c yy d" [r lrange xlist 0 10] "lrangeB"
assert_equal 7 [r linsert xlist after d dd] "after d"
assert_equal -1 [r linsert xlist after bad ddd] "after bad"
assert_equal "a $large zz c yy d dd" [r lrange xlist 0 10] "lrangeC"
assert_equal 8 [r linsert xlist before a aa] "before a"
assert_equal -1 [r linsert xlist before bad aaa] "before bad"
assert_equal "aa a $large zz c yy d dd" [r lrange xlist 0 10] "lrangeD"
# check inserting integer encoded value
assert_equal 9 [r linsert xlist before aa 42] "before aa"
assert_equal 42 [r lrange xlist 0 0] "lrangeE"
}
}
test {LINSERT raise error on bad syntax} {
catch {[r linsert xlist aft3r aa 42]} e
set e
} {*ERR*syntax*error*}
foreach {type num} {quicklist 250 quicklist 500} {
proc check_numbered_list_consistency {key} {
set len [r llen $key]
for {set i 0} {$i < $len} {incr i} {
assert_equal $i [r lindex $key $i]
assert_equal [expr $len-1-$i] [r lindex $key [expr (-$i)-1]]
}
}
proc check_random_access_consistency {key} {
set len [r llen $key]
for {set i 0} {$i < $len} {incr i} {
set rint [expr int(rand()*$len)]
assert_equal $rint [r lindex $key $rint]
assert_equal [expr $len-1-$rint] [r lindex $key [expr (-$rint)-1]]
}
}
test "LINDEX consistency test - $type" {
r del mylist
for {set i 0} {$i < $num} {incr i} {
r rpush mylist $i
}
assert_encoding $type mylist
check_numbered_list_consistency mylist
}
test "LINDEX random access - $type" {
assert_encoding $type mylist
check_random_access_consistency mylist
}
test "Check if list is still ok after a DEBUG RELOAD - $type" {
r debug reload
assert_encoding $type mylist
check_numbered_list_consistency mylist
check_random_access_consistency mylist
} {} {needs:debug}
}
test {LLEN against non-list value error} {
r del mylist
r set mylist foobar
assert_error WRONGTYPE* {r llen mylist}
}
test {LLEN against non existing key} {
assert_equal 0 [r llen not-a-key]
}
test {LINDEX against non-list value error} {
assert_error WRONGTYPE* {r lindex mylist 0}
}
test {LINDEX against non existing key} {
assert_equal "" [r lindex not-a-key 10]
}
test {LPUSH against non-list value error} {
assert_error WRONGTYPE* {r lpush mylist 0}
}
test {RPUSH against non-list value error} {
assert_error WRONGTYPE* {r rpush mylist 0}
}
foreach {type large} [array get largevalue] {
test "RPOPLPUSH base case - $type" {
r del mylist1{t} mylist2{t}
create_list mylist1{t} "a $large c d"
assert_equal d [r rpoplpush mylist1{t} mylist2{t}]
assert_equal c [r rpoplpush mylist1{t} mylist2{t}]
assert_equal "a $large" [r lrange mylist1{t} 0 -1]
assert_equal "c d" [r lrange mylist2{t} 0 -1]
assert_encoding quicklist mylist2{t}
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "LMOVE $wherefrom $whereto base case - $type" {
r del mylist1{t} mylist2{t}
if {$wherefrom eq "right"} {
create_list mylist1{t} "c d $large a"
} else {
create_list mylist1{t} "a $large c d"
}
assert_equal a [r lmove mylist1{t} mylist2{t} $wherefrom $whereto]
assert_equal $large [r lmove mylist1{t} mylist2{t} $wherefrom $whereto]
assert_equal "c d" [r lrange mylist1{t} 0 -1]
if {$whereto eq "right"} {
assert_equal "a $large" [r lrange mylist2{t} 0 -1]
} else {
assert_equal "$large a" [r lrange mylist2{t} 0 -1]
}
assert_encoding quicklist mylist2{t}
}
}
}
test "RPOPLPUSH with the same list as src and dst - $type" {
create_list mylist{t} "a $large c"
assert_equal "a $large c" [r lrange mylist{t} 0 -1]
assert_equal c [r rpoplpush mylist{t} mylist{t}]
assert_equal "c a $large" [r lrange mylist{t} 0 -1]
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "LMOVE $wherefrom $whereto with the same list as src and dst - $type" {
if {$wherefrom eq "right"} {
create_list mylist{t} "a $large c"
assert_equal "a $large c" [r lrange mylist{t} 0 -1]
} else {
create_list mylist{t} "c a $large"
assert_equal "c a $large" [r lrange mylist{t} 0 -1]
}
assert_equal c [r lmove mylist{t} mylist{t} $wherefrom $whereto]
if {$whereto eq "right"} {
assert_equal "a $large c" [r lrange mylist{t} 0 -1]
} else {
assert_equal "c a $large" [r lrange mylist{t} 0 -1]
}
}
}
}
foreach {othertype otherlarge} [array get largevalue] {
test "RPOPLPUSH with $type source and existing target $othertype" {
create_list srclist{t} "a b c $large"
create_list dstlist{t} "$otherlarge"
assert_equal $large [r rpoplpush srclist{t} dstlist{t}]
assert_equal c [r rpoplpush srclist{t} dstlist{t}]
assert_equal "a b" [r lrange srclist{t} 0 -1]
assert_equal "c $large $otherlarge" [r lrange dstlist{t} 0 -1]
# When we rpoplpush'ed a large value, dstlist should be
# converted to the same encoding as srclist.
if {$type eq "linkedlist"} {
assert_encoding quicklist dstlist{t}
}
}
foreach wherefrom {left right} {
foreach whereto {left right} {
test "LMOVE $wherefrom $whereto with $type source and existing target $othertype" {
create_list dstlist{t} "$otherlarge"
if {$wherefrom eq "right"} {
create_list srclist{t} "a b c $large"
} else {
create_list srclist{t} "$large c a b"
}
assert_equal $large [r lmove srclist{t} dstlist{t} $wherefrom $whereto]
assert_equal c [r lmove srclist{t} dstlist{t} $wherefrom $whereto]
assert_equal "a b" [r lrange srclist{t} 0 -1]
if {$whereto eq "right"} {
assert_equal "$otherlarge $large c" [r lrange dstlist{t} 0 -1]
} else {
assert_equal "c $large $otherlarge" [r lrange dstlist{t} 0 -1]
}
# When we lmoved a large value, dstlist should be
# converted to the same encoding as srclist.
if {$type eq "linkedlist"} {
assert_encoding quicklist dstlist{t}
}
}
}
}
}
}
test {RPOPLPUSH against non existing key} {
r del srclist{t} dstlist{t}
assert_equal {} [r rpoplpush srclist{t} dstlist{t}]
assert_equal 0 [r exists srclist{t}]
assert_equal 0 [r exists dstlist{t}]
}
test {RPOPLPUSH against non list src key} {
r del srclist{t} dstlist{t}
r set srclist{t} x
assert_error WRONGTYPE* {r rpoplpush srclist{t} dstlist{t}}
assert_type string srclist{t}
assert_equal 0 [r exists newlist{t}]
}
test {RPOPLPUSH against non list dst key} {
create_list srclist{t} {a b c d}
r set dstlist{t} x
assert_error WRONGTYPE* {r rpoplpush srclist{t} dstlist{t}}
assert_type string dstlist{t}
assert_equal {a b c d} [r lrange srclist{t} 0 -1]
}
test {RPOPLPUSH against non existing src key} {
r del srclist{t} dstlist{t}
assert_equal {} [r rpoplpush srclist{t} dstlist{t}]
} {}
foreach {type large} [array get largevalue] {
test "Basic LPOP/RPOP/LMPOP - $type" {
create_list mylist "$large 1 2"
assert_equal $large [r lpop mylist]
assert_equal 2 [r rpop mylist]
assert_equal 1 [r lpop mylist]
assert_equal 0 [r llen mylist]
create_list mylist "$large 1 2"
assert_equal "mylist $large" [r lmpop 1 mylist left count 1]
assert_equal {mylist {2 1}} [r lmpop 2 mylist mylist right count 2]
}
}
test {LPOP/RPOP/LMPOP against empty list} {
r del non-existing-list{t} non-existing-list2{t}
assert_equal {} [r lpop non-existing-list{t}]
assert_equal {} [r rpop non-existing-list2{t}]
assert_equal {} [r lmpop 1 non-existing-list{t} left count 1]
assert_equal {} [r lmpop 1 non-existing-list{t} left count 10]
assert_equal {} [r lmpop 2 non-existing-list{t} non-existing-list2{t} right count 1]
assert_equal {} [r lmpop 2 non-existing-list{t} non-existing-list2{t} right count 10]
}
test {LPOP/RPOP/LMPOP NON-BLOCK or BLOCK against non list value} {
r set notalist{t} foo
assert_error WRONGTYPE* {r lpop notalist{t}}
assert_error WRONGTYPE* {r blpop notalist{t} 0}
assert_error WRONGTYPE* {r rpop notalist{t}}
assert_error WRONGTYPE* {r brpop notalist{t} 0}
r del notalist2{t}
assert_error "WRONGTYPE*" {r lmpop 2 notalist{t} notalist2{t} left count 1}
assert_error "WRONGTYPE*" {r blmpop 0 2 notalist{t} notalist2{t} left count 1}
r del notalist{t}
r set notalist2{t} nolist
assert_error "WRONGTYPE*" {r lmpop 2 notalist{t} notalist2{t} right count 10}
assert_error "WRONGTYPE*" {r blmpop 0 2 notalist{t} notalist2{t} left count 1}
}
foreach {type num} {quicklist 250 quicklist 500} {
test "Mass RPOP/LPOP - $type" {
r del mylist
set sum1 0
for {set i 0} {$i < $num} {incr i} {
r lpush mylist $i
incr sum1 $i
}
assert_encoding $type mylist
set sum2 0
for {set i 0} {$i < [expr $num/2]} {incr i} {
incr sum2 [r lpop mylist]
incr sum2 [r rpop mylist]
}
assert_equal $sum1 $sum2
}
}
test {LMPOP with illegal argument} {
assert_error "ERR wrong number of arguments for 'lmpop' command" {r lmpop}
assert_error "ERR wrong number of arguments for 'lmpop' command" {r lmpop 1}
assert_error "ERR wrong number of arguments for 'lmpop' command" {r lmpop 1 mylist{t}}
assert_error "ERR numkeys*" {r lmpop 0 mylist{t} LEFT}
assert_error "ERR numkeys*" {r lmpop a mylist{t} LEFT}
assert_error "ERR numkeys*" {r lmpop -1 mylist{t} RIGHT}
assert_error "ERR syntax error*" {r lmpop 1 mylist{t} bad_where}
assert_error "ERR syntax error*" {r lmpop 1 mylist{t} LEFT bar_arg}
assert_error "ERR syntax error*" {r lmpop 1 mylist{t} RIGHT LEFT}
assert_error "ERR syntax error*" {r lmpop 1 mylist{t} COUNT}
assert_error "ERR syntax error*" {r lmpop 1 mylist{t} LEFT COUNT 1 COUNT 2}
assert_error "ERR syntax error*" {r lmpop 2 mylist{t} mylist2{t} bad_arg}
assert_error "ERR count*" {r lmpop 1 mylist{t} LEFT COUNT 0}
assert_error "ERR count*" {r lmpop 1 mylist{t} RIGHT COUNT a}
assert_error "ERR count*" {r lmpop 1 mylist{t} LEFT COUNT -1}
assert_error "ERR count*" {r lmpop 2 mylist{t} mylist2{t} RIGHT COUNT -1}
}
test {LMPOP single existing list} {
# Same key multiple times.
create_list mylist{t} "a b c d e f"
assert_equal {mylist{t} {a b}} [r lmpop 2 mylist{t} mylist{t} left count 2]
assert_equal {mylist{t} {f e}} [r lmpop 2 mylist{t} mylist{t} right count 2]
assert_equal 2 [r llen mylist{t}]
# First one exists, second one does not exist.
create_list mylist{t} "a b c d e"
r del mylist2{t}
assert_equal {mylist{t} a} [r lmpop 2 mylist{t} mylist2{t} left count 1]
assert_equal 4 [r llen mylist{t}]
assert_equal {mylist{t} {e d c b}} [r lmpop 2 mylist{t} mylist2{t} right count 10]
assert_equal {} [r lmpop 2 mylist{t} mylist2{t} right count 1]
# First one does not exist, second one exists.
r del mylist{t}
create_list mylist2{t} "1 2 3 4 5"
assert_equal {mylist2{t} 5} [r lmpop 2 mylist{t} mylist2{t} right count 1]
assert_equal 4 [r llen mylist2{t}]
assert_equal {mylist2{t} {1 2 3 4}} [r lmpop 2 mylist{t} mylist2{t} left count 10]
assert_equal 0 [r exists mylist{t} mylist2{t}]
}
test {LMPOP multiple existing lists} {
create_list mylist{t} "a b c d e"
create_list mylist2{t} "1 2 3 4 5"
# Pop up from the first key.
assert_equal {mylist{t} {a b}} [r lmpop 2 mylist{t} mylist2{t} left count 2]
assert_equal 3 [r llen mylist{t}]
assert_equal {mylist{t} {e d c}} [r lmpop 2 mylist{t} mylist2{t} right count 3]
assert_equal 0 [r exists mylist{t}]
# Pop up from the second key.
assert_equal {mylist2{t} {1 2 3}} [r lmpop 2 mylist{t} mylist2{t} left count 3]
assert_equal 2 [r llen mylist2{t}]
assert_equal {mylist2{t} {5 4}} [r lmpop 2 mylist{t} mylist2{t} right count 2]
assert_equal 0 [r exists mylist{t}]
# Pop up all elements.
create_list mylist{t} "a b c"
create_list mylist2{t} "1 2 3"
assert_equal {mylist{t} {a b c}} [r lmpop 2 mylist{t} mylist2{t} left count 10]
assert_equal 0 [r llen mylist{t}]
assert_equal {mylist2{t} {3 2 1}} [r lmpop 2 mylist{t} mylist2{t} right count 10]
assert_equal 0 [r llen mylist2{t}]
assert_equal 0 [r exists mylist{t} mylist2{t}]
}
test {LMPOP propagate as pop with count command to replica} {
set repl [attach_to_replication_stream]
# left/right propagate as lpop/rpop with count
r lpush mylist{t} a b c
# Pop elements from one list.
r lmpop 1 mylist{t} left count 1
r lmpop 1 mylist{t} right count 1
# Now the list have only one element
r lmpop 2 mylist{t} mylist2{t} left count 10
# No elements so we don't propagate.
r lmpop 2 mylist{t} mylist2{t} left count 10
# Pop elements from the second list.
r rpush mylist2{t} 1 2 3
r lmpop 2 mylist{t} mylist2{t} left count 2
r lmpop 2 mylist{t} mylist2{t} right count 1
# Pop all elements.
r rpush mylist{t} a b c
r rpush mylist2{t} 1 2 3
r lmpop 2 mylist{t} mylist2{t} left count 10
r lmpop 2 mylist{t} mylist2{t} right count 10
assert_replication_stream $repl {
{select *}
{lpush mylist{t} a b c}
{lpop mylist{t} 1}
{rpop mylist{t} 1}
{lpop mylist{t} 1}
{rpush mylist2{t} 1 2 3}
{lpop mylist2{t} 2}
{rpop mylist2{t} 1}
{rpush mylist{t} a b c}
{rpush mylist2{t} 1 2 3}
{lpop mylist{t} 3}
{rpop mylist2{t} 3}
}
close_replication_stream $repl
} {} {needs:repl}
foreach {type large} [array get largevalue] {
test "LRANGE basics - $type" {
create_list mylist "$large 1 2 3 4 5 6 7 8 9"
assert_equal {1 2 3 4 5 6 7 8} [r lrange mylist 1 -2]
assert_equal {7 8 9} [r lrange mylist -3 -1]
assert_equal {4} [r lrange mylist 4 4]
}
test "LRANGE inverted indexes - $type" {
create_list mylist "$large 1 2 3 4 5 6 7 8 9"
assert_equal {} [r lrange mylist 6 2]
}
test "LRANGE out of range indexes including the full list - $type" {
create_list mylist "$large 1 2 3"
assert_equal "$large 1 2 3" [r lrange mylist -1000 1000]
}
test "LRANGE out of range negative end index - $type" {
create_list mylist "$large 1 2 3"
assert_equal $large [r lrange mylist 0 -4]
assert_equal {} [r lrange mylist 0 -5]
}
}
test {LRANGE against non existing key} {
assert_equal {} [r lrange nosuchkey 0 1]
}
test {LRANGE with start > end yields an empty array for backward compatibility} {
create_list mylist "1 2 3"
assert_equal {} [r lrange mylist 1 0]
assert_equal {} [r lrange mylist -1 -2]
}
foreach {type large} [array get largevalue] {
proc trim_list {type min max} {
upvar 1 large large
r del mylist
create_list mylist "1 2 3 4 $large"
r ltrim mylist $min $max
r lrange mylist 0 -1
}
test "LTRIM basics - $type" {
assert_equal "1" [trim_list $type 0 0]
assert_equal "1 2" [trim_list $type 0 1]
assert_equal "1 2 3" [trim_list $type 0 2]
assert_equal "2 3" [trim_list $type 1 2]
assert_equal "2 3 4 $large" [trim_list $type 1 -1]
assert_equal "2 3 4" [trim_list $type 1 -2]
assert_equal "4 $large" [trim_list $type -2 -1]
assert_equal "$large" [trim_list $type -1 -1]
assert_equal "1 2 3 4 $large" [trim_list $type -5 -1]
assert_equal "1 2 3 4 $large" [trim_list $type -10 10]
assert_equal "1 2 3 4 $large" [trim_list $type 0 5]
assert_equal "1 2 3 4 $large" [trim_list $type 0 10]
}
test "LTRIM out of range negative end index - $type" {
assert_equal {1} [trim_list $type 0 -5]
assert_equal {} [trim_list $type 0 -6]
}
}
foreach {type large} [array get largevalue] {
test "LSET - $type" {
create_list mylist "99 98 $large 96 95"
r lset mylist 1 foo
r lset mylist -1 bar
assert_equal "99 foo $large 96 bar" [r lrange mylist 0 -1]
}
test "LSET out of range index - $type" {
assert_error ERR*range* {r lset mylist 10 foo}
}
}
test {LSET against non existing key} {
assert_error ERR*key* {r lset nosuchkey 10 foo}
}
test {LSET against non list value} {
r set nolist foobar
assert_error WRONGTYPE* {r lset nolist 0 foo}
}
foreach {type e} [array get largevalue] {
test "LREM remove all the occurrences - $type" {
create_list mylist "$e foo bar foobar foobared zap bar test foo"
assert_equal 2 [r lrem mylist 0 bar]
assert_equal "$e foo foobar foobared zap test foo" [r lrange mylist 0 -1]
}
test "LREM remove the first occurrence - $type" {
assert_equal 1 [r lrem mylist 1 foo]
assert_equal "$e foobar foobared zap test foo" [r lrange mylist 0 -1]
}
test "LREM remove non existing element - $type" {
assert_equal 0 [r lrem mylist 1 nosuchelement]
assert_equal "$e foobar foobared zap test foo" [r lrange mylist 0 -1]
}
test "LREM starting from tail with negative count - $type" {
create_list mylist "$e foo bar foobar foobared zap bar test foo foo"
assert_equal 1 [r lrem mylist -1 bar]
assert_equal "$e foo bar foobar foobared zap test foo foo" [r lrange mylist 0 -1]
}
test "LREM starting from tail with negative count (2) - $type" {
assert_equal 2 [r lrem mylist -2 foo]
assert_equal "$e foo bar foobar foobared zap test" [r lrange mylist 0 -1]
}
test "LREM deleting objects that may be int encoded - $type" {
create_list myotherlist "$e 1 2 3"
assert_equal 1 [r lrem myotherlist 1 2]
assert_equal 3 [r llen myotherlist]
}
}
test "Regression for bug 593 - chaining BRPOPLPUSH with other blocking cmds" {
set rd1 [redis_deferring_client]
set rd2 [redis_deferring_client]
$rd1 brpoplpush a b 0
$rd1 brpoplpush a b 0
$rd2 brpoplpush b c 0
after 1000
r lpush a data
$rd1 close
$rd2 close
r ping
} {PONG}
foreach {pop} {BLPOP BLMPOP_RIGHT} {
test "client unblock tests" {
r del l
set rd [redis_deferring_client]
$rd client id
set id [$rd read]
# test default args
bpop_command $rd $pop l 0
wait_for_blocked_client
r client unblock $id
assert_equal {} [$rd read]
# test with timeout
bpop_command $rd $pop l 0
wait_for_blocked_client
r client unblock $id TIMEOUT
assert_equal {} [$rd read]
# test with error
bpop_command $rd $pop l 0
wait_for_blocked_client
r client unblock $id ERROR
catch {[$rd read]} e
assert_equal $e "UNBLOCKED client unblocked via CLIENT UNBLOCK"
# test with invalid client id
catch {[r client unblock asd]} e
assert_equal $e "ERR value is not an integer or out of range"
# test with non blocked client
set myid [r client id]
catch {[r client unblock $myid]} e
assert_equal $e {invalid command name "0"}
# finally, see the this client and list are still functional
bpop_command $rd $pop l 0
wait_for_blocked_client
r lpush l foo
assert_equal {l foo} [$rd read]
$rd close
}
}
test {List ziplist of various encodings} {
r del k
r lpush k 127 ;# ZIP_INT_8B
r lpush k 32767 ;# ZIP_INT_16B
r lpush k 2147483647 ;# ZIP_INT_32B
r lpush k 9223372036854775808 ;# ZIP_INT_64B
r lpush k 0 ;# ZIP_INT_IMM_MIN
r lpush k 12 ;# ZIP_INT_IMM_MAX
r lpush k [string repeat x 31] ;# ZIP_STR_06B
r lpush k [string repeat x 8191] ;# ZIP_STR_14B
r lpush k [string repeat x 65535] ;# ZIP_STR_32B
set k [r lrange k 0 -1]
set dump [r dump k]
config_set sanitize-dump-payload no mayfail
r restore kk 0 $dump
set kk [r lrange kk 0 -1]
# try some forward and backward searches to make sure all encodings
# can be traversed
assert_equal [r lindex kk 5] {9223372036854775808}
assert_equal [r lindex kk -5] {0}
assert_equal [r lpos kk foo rank 1] {}
assert_equal [r lpos kk foo rank -1] {}
# make sure the values are right
assert_equal $k $kk
assert_equal [lpop k] [string repeat x 65535]
assert_equal [lpop k] [string repeat x 8191]
assert_equal [lpop k] [string repeat x 31]
set _ $k
} {12 0 9223372036854775808 2147483647 32767 127}
test {List ziplist of various encodings - sanitize dump} {
config_set sanitize-dump-payload yes mayfail
r restore kk 0 $dump replace
set k [r lrange k 0 -1]
set kk [r lrange kk 0 -1]
# make sure the values are right
assert_equal $k $kk
assert_equal [lpop k] [string repeat x 65535]
assert_equal [lpop k] [string repeat x 8191]
assert_equal [lpop k] [string repeat x 31]
set _ $k
} {12 0 9223372036854775808 2147483647 32767 127}
}