mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-24 17:17:51 -05:00
3c6f9ac37c
ALL comment fixes
807 lines
22 KiB
Tcl
807 lines
22 KiB
Tcl
start_server {tags {"basic"}} {
|
|
test {DEL all keys to start with a clean DB} {
|
|
foreach key [r keys *] {r del $key}
|
|
r dbsize
|
|
} {0}
|
|
|
|
test {SET and GET an item} {
|
|
r set x foobar
|
|
r get x
|
|
} {foobar}
|
|
|
|
test {SET and GET an empty item} {
|
|
r set x {}
|
|
r get x
|
|
} {}
|
|
|
|
test {DEL against a single item} {
|
|
r del x
|
|
r get x
|
|
} {}
|
|
|
|
test {Vararg DEL} {
|
|
r set foo1 a
|
|
r set foo2 b
|
|
r set foo3 c
|
|
list [r del foo1 foo2 foo3 foo4] [r mget foo1 foo2 foo3]
|
|
} {3 {{} {} {}}}
|
|
|
|
test {KEYS with pattern} {
|
|
foreach key {key_x key_y key_z foo_a foo_b foo_c} {
|
|
r set $key hello
|
|
}
|
|
lsort [r keys foo*]
|
|
} {foo_a foo_b foo_c}
|
|
|
|
test {KEYS to get all keys} {
|
|
lsort [r keys *]
|
|
} {foo_a foo_b foo_c key_x key_y key_z}
|
|
|
|
test {DBSIZE} {
|
|
r dbsize
|
|
} {6}
|
|
|
|
test {DEL all keys} {
|
|
foreach key [r keys *] {r del $key}
|
|
r dbsize
|
|
} {0}
|
|
|
|
test {Very big payload in GET/SET} {
|
|
set buf [string repeat "abcd" 1000000]
|
|
r set foo $buf
|
|
r get foo
|
|
} [string repeat "abcd" 1000000]
|
|
|
|
tags {"slow"} {
|
|
test {Very big payload random access} {
|
|
set err {}
|
|
array set payload {}
|
|
for {set j 0} {$j < 100} {incr j} {
|
|
set size [expr 1+[randomInt 100000]]
|
|
set buf [string repeat "pl-$j" $size]
|
|
set payload($j) $buf
|
|
r set bigpayload_$j $buf
|
|
}
|
|
for {set j 0} {$j < 1000} {incr j} {
|
|
set index [randomInt 100]
|
|
set buf [r get bigpayload_$index]
|
|
if {$buf != $payload($index)} {
|
|
set err "Values differ: I set '$payload($index)' but I read back '$buf'"
|
|
break
|
|
}
|
|
}
|
|
unset payload
|
|
set _ $err
|
|
} {}
|
|
|
|
test {SET 10000 numeric keys and access all them in reverse order} {
|
|
set err {}
|
|
for {set x 0} {$x < 10000} {incr x} {
|
|
r set $x $x
|
|
}
|
|
set sum 0
|
|
for {set x 9999} {$x >= 0} {incr x -1} {
|
|
set val [r get $x]
|
|
if {$val ne $x} {
|
|
set err "Element at position $x is $val instead of $x"
|
|
break
|
|
}
|
|
}
|
|
set _ $err
|
|
} {}
|
|
|
|
test {DBSIZE should be 10101 now} {
|
|
r dbsize
|
|
} {10101}
|
|
}
|
|
|
|
test {INCR against non existing key} {
|
|
set res {}
|
|
append res [r incr novar]
|
|
append res [r get novar]
|
|
} {11}
|
|
|
|
test {INCR against key created by incr itself} {
|
|
r incr novar
|
|
} {2}
|
|
|
|
test {INCR against key originally set with SET} {
|
|
r set novar 100
|
|
r incr novar
|
|
} {101}
|
|
|
|
test {INCR over 32bit value} {
|
|
r set novar 17179869184
|
|
r incr novar
|
|
} {17179869185}
|
|
|
|
test {INCRBY over 32bit value with over 32bit increment} {
|
|
r set novar 17179869184
|
|
r incrby novar 17179869184
|
|
} {34359738368}
|
|
|
|
test {INCR fails against key with spaces (left)} {
|
|
r set novar " 11"
|
|
catch {r incr novar} err
|
|
format $err
|
|
} {ERR*}
|
|
|
|
test {INCR fails against key with spaces (right)} {
|
|
r set novar "11 "
|
|
catch {r incr novar} err
|
|
format $err
|
|
} {ERR*}
|
|
|
|
test {INCR fails against key with spaces (both)} {
|
|
r set novar " 11 "
|
|
catch {r incr novar} err
|
|
format $err
|
|
} {ERR*}
|
|
|
|
test {INCR fails against a key holding a list} {
|
|
r rpush mylist 1
|
|
catch {r incr mylist} err
|
|
r rpop mylist
|
|
format $err
|
|
} {WRONGTYPE*}
|
|
|
|
test {DECRBY over 32bit value with over 32bit increment, negative res} {
|
|
r set novar 17179869184
|
|
r decrby novar 17179869185
|
|
} {-1}
|
|
|
|
test {INCR uses shared objects in the 0-9999 range} {
|
|
r set foo -1
|
|
r incr foo
|
|
assert {[r object refcount foo] > 1}
|
|
r set foo 9998
|
|
r incr foo
|
|
assert {[r object refcount foo] > 1}
|
|
r incr foo
|
|
assert {[r object refcount foo] == 1}
|
|
}
|
|
|
|
test {INCR can modify objects in-place} {
|
|
r set foo 20000
|
|
r incr foo
|
|
assert {[r object refcount foo] == 1}
|
|
set old [lindex [split [r debug object foo]] 1]
|
|
r incr foo
|
|
set new [lindex [split [r debug object foo]] 1]
|
|
assert {[string range $old 0 2] eq "at:"}
|
|
assert {[string range $new 0 2] eq "at:"}
|
|
assert {$old eq $new}
|
|
}
|
|
|
|
test {INCRBYFLOAT against non existing key} {
|
|
r del novar
|
|
list [roundFloat [r incrbyfloat novar 1]] \
|
|
[roundFloat [r get novar]] \
|
|
[roundFloat [r incrbyfloat novar 0.25]] \
|
|
[roundFloat [r get novar]]
|
|
} {1 1 1.25 1.25}
|
|
|
|
test {INCRBYFLOAT against key originally set with SET} {
|
|
r set novar 1.5
|
|
roundFloat [r incrbyfloat novar 1.5]
|
|
} {3}
|
|
|
|
test {INCRBYFLOAT over 32bit value} {
|
|
r set novar 17179869184
|
|
r incrbyfloat novar 1.5
|
|
} {17179869185.5}
|
|
|
|
test {INCRBYFLOAT over 32bit value with over 32bit increment} {
|
|
r set novar 17179869184
|
|
r incrbyfloat novar 17179869184
|
|
} {34359738368}
|
|
|
|
test {INCRBYFLOAT fails against key with spaces (left)} {
|
|
set err {}
|
|
r set novar " 11"
|
|
catch {r incrbyfloat novar 1.0} err
|
|
format $err
|
|
} {ERR*valid*}
|
|
|
|
test {INCRBYFLOAT fails against key with spaces (right)} {
|
|
set err {}
|
|
r set novar "11 "
|
|
catch {r incrbyfloat novar 1.0} err
|
|
format $err
|
|
} {ERR*valid*}
|
|
|
|
test {INCRBYFLOAT fails against key with spaces (both)} {
|
|
set err {}
|
|
r set novar " 11 "
|
|
catch {r incrbyfloat novar 1.0} err
|
|
format $err
|
|
} {ERR*valid*}
|
|
|
|
test {INCRBYFLOAT fails against a key holding a list} {
|
|
r del mylist
|
|
set err {}
|
|
r rpush mylist 1
|
|
catch {r incrbyfloat mylist 1.0} err
|
|
r del mylist
|
|
format $err
|
|
} {WRONGTYPE*}
|
|
|
|
test {INCRBYFLOAT does not allow NaN or Infinity} {
|
|
r set foo 0
|
|
set err {}
|
|
catch {r incrbyfloat foo +inf} err
|
|
set err
|
|
# p.s. no way I can force NaN to test it from the API because
|
|
# there is no way to increment / decrement by infinity nor to
|
|
# perform divisions.
|
|
} {ERR*would produce*}
|
|
|
|
test {INCRBYFLOAT decrement} {
|
|
r set foo 1
|
|
roundFloat [r incrbyfloat foo -1.1]
|
|
} {-0.1}
|
|
|
|
test "SETNX target key missing" {
|
|
r del novar
|
|
assert_equal 1 [r setnx novar foobared]
|
|
assert_equal "foobared" [r get novar]
|
|
}
|
|
|
|
test "SETNX target key exists" {
|
|
r set novar foobared
|
|
assert_equal 0 [r setnx novar blabla]
|
|
assert_equal "foobared" [r get novar]
|
|
}
|
|
|
|
test "SETNX against not-expired volatile key" {
|
|
r set x 10
|
|
r expire x 10000
|
|
assert_equal 0 [r setnx x 20]
|
|
assert_equal 10 [r get x]
|
|
}
|
|
|
|
test "SETNX against expired volatile key" {
|
|
# Make it very unlikely for the key this test uses to be expired by the
|
|
# active expiry cycle. This is tightly coupled to the implementation of
|
|
# active expiry and dbAdd() but currently the only way to test that
|
|
# SETNX expires a key when it should have been.
|
|
for {set x 0} {$x < 9999} {incr x} {
|
|
r setex key-$x 3600 value
|
|
}
|
|
|
|
# This will be one of 10000 expiring keys. A cycle is executed every
|
|
# 100ms, sampling 10 keys for being expired or not. This key will be
|
|
# expired for at most 1s when we wait 2s, resulting in a total sample
|
|
# of 100 keys. The probability of the success of this test being a
|
|
# false positive is therefore approx. 1%.
|
|
r set x 10
|
|
r expire x 1
|
|
|
|
# Wait for the key to expire
|
|
after 2000
|
|
|
|
assert_equal 1 [r setnx x 20]
|
|
assert_equal 20 [r get x]
|
|
}
|
|
|
|
test "DEL against expired key" {
|
|
r debug set-active-expire 0
|
|
r setex keyExpire 1 valExpire
|
|
after 1100
|
|
assert_equal 0 [r del keyExpire]
|
|
r debug set-active-expire 1
|
|
}
|
|
|
|
test {EXISTS} {
|
|
set res {}
|
|
r set newkey test
|
|
append res [r exists newkey]
|
|
r del newkey
|
|
append res [r exists newkey]
|
|
} {10}
|
|
|
|
test {Zero length value in key. SET/GET/EXISTS} {
|
|
r set emptykey {}
|
|
set res [r get emptykey]
|
|
append res [r exists emptykey]
|
|
r del emptykey
|
|
append res [r exists emptykey]
|
|
} {10}
|
|
|
|
test {Commands pipelining} {
|
|
set fd [r channel]
|
|
puts -nonewline $fd "SET k1 xyzk\r\nGET k1\r\nPING\r\n"
|
|
flush $fd
|
|
set res {}
|
|
append res [string match OK* [r read]]
|
|
append res [r read]
|
|
append res [string match PONG* [r read]]
|
|
format $res
|
|
} {1xyzk1}
|
|
|
|
test {Non existing command} {
|
|
catch {r foobaredcommand} err
|
|
string match ERR* $err
|
|
} {1}
|
|
|
|
test {RENAME basic usage} {
|
|
r set mykey hello
|
|
r rename mykey mykey1
|
|
r rename mykey1 mykey2
|
|
r get mykey2
|
|
} {hello}
|
|
|
|
test {RENAME source key should no longer exist} {
|
|
r exists mykey
|
|
} {0}
|
|
|
|
test {RENAME against already existing key} {
|
|
r set mykey a
|
|
r set mykey2 b
|
|
r rename mykey2 mykey
|
|
set res [r get mykey]
|
|
append res [r exists mykey2]
|
|
} {b0}
|
|
|
|
test {RENAMENX basic usage} {
|
|
r del mykey
|
|
r del mykey2
|
|
r set mykey foobar
|
|
r renamenx mykey mykey2
|
|
set res [r get mykey2]
|
|
append res [r exists mykey]
|
|
} {foobar0}
|
|
|
|
test {RENAMENX against already existing key} {
|
|
r set mykey foo
|
|
r set mykey2 bar
|
|
r renamenx mykey mykey2
|
|
} {0}
|
|
|
|
test {RENAMENX against already existing key (2)} {
|
|
set res [r get mykey]
|
|
append res [r get mykey2]
|
|
} {foobar}
|
|
|
|
test {RENAME against non existing source key} {
|
|
catch {r rename nokey foobar} err
|
|
format $err
|
|
} {ERR*}
|
|
|
|
test {RENAME where source and dest key is the same} {
|
|
catch {r rename mykey mykey} err
|
|
format $err
|
|
} {ERR*}
|
|
|
|
test {RENAME with volatile key, should move the TTL as well} {
|
|
r del mykey mykey2
|
|
r set mykey foo
|
|
r expire mykey 100
|
|
assert {[r ttl mykey] > 95 && [r ttl mykey] <= 100}
|
|
r rename mykey mykey2
|
|
assert {[r ttl mykey2] > 95 && [r ttl mykey2] <= 100}
|
|
}
|
|
|
|
test {RENAME with volatile key, should not inherit TTL of target key} {
|
|
r del mykey mykey2
|
|
r set mykey foo
|
|
r set mykey2 bar
|
|
r expire mykey2 100
|
|
assert {[r ttl mykey] == -1 && [r ttl mykey2] > 0}
|
|
r rename mykey mykey2
|
|
r ttl mykey2
|
|
} {-1}
|
|
|
|
test {DEL all keys again (DB 0)} {
|
|
foreach key [r keys *] {
|
|
r del $key
|
|
}
|
|
r dbsize
|
|
} {0}
|
|
|
|
test {DEL all keys again (DB 1)} {
|
|
r select 10
|
|
foreach key [r keys *] {
|
|
r del $key
|
|
}
|
|
set res [r dbsize]
|
|
r select 9
|
|
format $res
|
|
} {0}
|
|
|
|
test {MOVE basic usage} {
|
|
r set mykey foobar
|
|
r move mykey 10
|
|
set res {}
|
|
lappend res [r exists mykey]
|
|
lappend res [r dbsize]
|
|
r select 10
|
|
lappend res [r get mykey]
|
|
lappend res [r dbsize]
|
|
r select 9
|
|
format $res
|
|
} [list 0 0 foobar 1]
|
|
|
|
test {MOVE against key existing in the target DB} {
|
|
r set mykey hello
|
|
r move mykey 10
|
|
} {0}
|
|
|
|
test {MOVE against non-integer DB (#1428)} {
|
|
r set mykey hello
|
|
catch {r move mykey notanumber} e
|
|
set e
|
|
} {*ERR*index out of range}
|
|
|
|
test {SET/GET keys in different DBs} {
|
|
r set a hello
|
|
r set b world
|
|
r select 10
|
|
r set a foo
|
|
r set b bared
|
|
r select 9
|
|
set res {}
|
|
lappend res [r get a]
|
|
lappend res [r get b]
|
|
r select 10
|
|
lappend res [r get a]
|
|
lappend res [r get b]
|
|
r select 9
|
|
format $res
|
|
} {hello world foo bared}
|
|
|
|
test {MGET} {
|
|
r flushdb
|
|
r set foo BAR
|
|
r set bar FOO
|
|
r mget foo bar
|
|
} {BAR FOO}
|
|
|
|
test {MGET against non existing key} {
|
|
r mget foo baazz bar
|
|
} {BAR {} FOO}
|
|
|
|
test {MGET against non-string key} {
|
|
r sadd myset ciao
|
|
r sadd myset bau
|
|
r mget foo baazz bar myset
|
|
} {BAR {} FOO {}}
|
|
|
|
test {RANDOMKEY} {
|
|
r flushdb
|
|
r set foo x
|
|
r set bar y
|
|
set foo_seen 0
|
|
set bar_seen 0
|
|
for {set i 0} {$i < 100} {incr i} {
|
|
set rkey [r randomkey]
|
|
if {$rkey eq {foo}} {
|
|
set foo_seen 1
|
|
}
|
|
if {$rkey eq {bar}} {
|
|
set bar_seen 1
|
|
}
|
|
}
|
|
list $foo_seen $bar_seen
|
|
} {1 1}
|
|
|
|
test {RANDOMKEY against empty DB} {
|
|
r flushdb
|
|
r randomkey
|
|
} {}
|
|
|
|
test {RANDOMKEY regression 1} {
|
|
r flushdb
|
|
r set x 10
|
|
r del x
|
|
r randomkey
|
|
} {}
|
|
|
|
test {GETSET (set new value)} {
|
|
list [r getset foo xyz] [r get foo]
|
|
} {{} xyz}
|
|
|
|
test {GETSET (replace old value)} {
|
|
r set foo bar
|
|
list [r getset foo xyz] [r get foo]
|
|
} {bar xyz}
|
|
|
|
test {MSET base case} {
|
|
r mset x 10 y "foo bar" z "x x x x x x x\n\n\r\n"
|
|
r mget x y z
|
|
} [list 10 {foo bar} "x x x x x x x\n\n\r\n"]
|
|
|
|
test {MSET wrong number of args} {
|
|
catch {r mset x 10 y "foo bar" z} err
|
|
format $err
|
|
} {*wrong number*}
|
|
|
|
test {MSETNX with already existent key} {
|
|
list [r msetnx x1 xxx y2 yyy x 20] [r exists x1] [r exists y2]
|
|
} {0 0 0}
|
|
|
|
test {MSETNX with not existing keys} {
|
|
list [r msetnx x1 xxx y2 yyy] [r get x1] [r get y2]
|
|
} {1 xxx yyy}
|
|
|
|
test "STRLEN against non-existing key" {
|
|
assert_equal 0 [r strlen notakey]
|
|
}
|
|
|
|
test "STRLEN against integer-encoded value" {
|
|
r set myinteger -555
|
|
assert_equal 4 [r strlen myinteger]
|
|
}
|
|
|
|
test "STRLEN against plain string" {
|
|
r set mystring "foozzz0123456789 baz"
|
|
assert_equal 20 [r strlen mystring]
|
|
}
|
|
|
|
test "SETBIT against non-existing key" {
|
|
r del mykey
|
|
assert_equal 0 [r setbit mykey 1 1]
|
|
assert_equal [binary format B* 01000000] [r get mykey]
|
|
}
|
|
|
|
test "SETBIT against string-encoded key" {
|
|
# Ascii "@" is integer 64 = 01 00 00 00
|
|
r set mykey "@"
|
|
|
|
assert_equal 0 [r setbit mykey 2 1]
|
|
assert_equal [binary format B* 01100000] [r get mykey]
|
|
assert_equal 1 [r setbit mykey 1 0]
|
|
assert_equal [binary format B* 00100000] [r get mykey]
|
|
}
|
|
|
|
test "SETBIT against integer-encoded key" {
|
|
# Ascii "1" is integer 49 = 00 11 00 01
|
|
r set mykey 1
|
|
assert_encoding int mykey
|
|
|
|
assert_equal 0 [r setbit mykey 6 1]
|
|
assert_equal [binary format B* 00110011] [r get mykey]
|
|
assert_equal 1 [r setbit mykey 2 0]
|
|
assert_equal [binary format B* 00010011] [r get mykey]
|
|
}
|
|
|
|
test "SETBIT against key with wrong type" {
|
|
r del mykey
|
|
r lpush mykey "foo"
|
|
assert_error "WRONGTYPE*" {r setbit mykey 0 1}
|
|
}
|
|
|
|
test "SETBIT with out of range bit offset" {
|
|
r del mykey
|
|
assert_error "*out of range*" {r setbit mykey [expr 4*1024*1024*1024] 1}
|
|
assert_error "*out of range*" {r setbit mykey -1 1}
|
|
}
|
|
|
|
test "SETBIT with non-bit argument" {
|
|
r del mykey
|
|
assert_error "*out of range*" {r setbit mykey 0 -1}
|
|
assert_error "*out of range*" {r setbit mykey 0 2}
|
|
assert_error "*out of range*" {r setbit mykey 0 10}
|
|
assert_error "*out of range*" {r setbit mykey 0 20}
|
|
}
|
|
|
|
test "SETBIT fuzzing" {
|
|
set str ""
|
|
set len [expr 256*8]
|
|
r del mykey
|
|
|
|
for {set i 0} {$i < 2000} {incr i} {
|
|
set bitnum [randomInt $len]
|
|
set bitval [randomInt 2]
|
|
set fmt [format "%%-%ds%%d%%-s" $bitnum]
|
|
set head [string range $str 0 $bitnum-1]
|
|
set tail [string range $str $bitnum+1 end]
|
|
set str [string map {" " 0} [format $fmt $head $bitval $tail]]
|
|
|
|
r setbit mykey $bitnum $bitval
|
|
assert_equal [binary format B* $str] [r get mykey]
|
|
}
|
|
}
|
|
|
|
test "GETBIT against non-existing key" {
|
|
r del mykey
|
|
assert_equal 0 [r getbit mykey 0]
|
|
}
|
|
|
|
test "GETBIT against string-encoded key" {
|
|
# Single byte with 2nd and 3rd bit set
|
|
r set mykey "`"
|
|
|
|
# In-range
|
|
assert_equal 0 [r getbit mykey 0]
|
|
assert_equal 1 [r getbit mykey 1]
|
|
assert_equal 1 [r getbit mykey 2]
|
|
assert_equal 0 [r getbit mykey 3]
|
|
|
|
# Out-range
|
|
assert_equal 0 [r getbit mykey 8]
|
|
assert_equal 0 [r getbit mykey 100]
|
|
assert_equal 0 [r getbit mykey 10000]
|
|
}
|
|
|
|
test "GETBIT against integer-encoded key" {
|
|
r set mykey 1
|
|
assert_encoding int mykey
|
|
|
|
# Ascii "1" is integer 49 = 00 11 00 01
|
|
assert_equal 0 [r getbit mykey 0]
|
|
assert_equal 0 [r getbit mykey 1]
|
|
assert_equal 1 [r getbit mykey 2]
|
|
assert_equal 1 [r getbit mykey 3]
|
|
|
|
# Out-range
|
|
assert_equal 0 [r getbit mykey 8]
|
|
assert_equal 0 [r getbit mykey 100]
|
|
assert_equal 0 [r getbit mykey 10000]
|
|
}
|
|
|
|
test "SETRANGE against non-existing key" {
|
|
r del mykey
|
|
assert_equal 3 [r setrange mykey 0 foo]
|
|
assert_equal "foo" [r get mykey]
|
|
|
|
r del mykey
|
|
assert_equal 0 [r setrange mykey 0 ""]
|
|
assert_equal 0 [r exists mykey]
|
|
|
|
r del mykey
|
|
assert_equal 4 [r setrange mykey 1 foo]
|
|
assert_equal "\000foo" [r get mykey]
|
|
}
|
|
|
|
test "SETRANGE against string-encoded key" {
|
|
r set mykey "foo"
|
|
assert_equal 3 [r setrange mykey 0 b]
|
|
assert_equal "boo" [r get mykey]
|
|
|
|
r set mykey "foo"
|
|
assert_equal 3 [r setrange mykey 0 ""]
|
|
assert_equal "foo" [r get mykey]
|
|
|
|
r set mykey "foo"
|
|
assert_equal 3 [r setrange mykey 1 b]
|
|
assert_equal "fbo" [r get mykey]
|
|
|
|
r set mykey "foo"
|
|
assert_equal 7 [r setrange mykey 4 bar]
|
|
assert_equal "foo\000bar" [r get mykey]
|
|
}
|
|
|
|
test "SETRANGE against integer-encoded key" {
|
|
r set mykey 1234
|
|
assert_encoding int mykey
|
|
assert_equal 4 [r setrange mykey 0 2]
|
|
assert_encoding raw mykey
|
|
assert_equal 2234 [r get mykey]
|
|
|
|
# Shouldn't change encoding when nothing is set
|
|
r set mykey 1234
|
|
assert_encoding int mykey
|
|
assert_equal 4 [r setrange mykey 0 ""]
|
|
assert_encoding int mykey
|
|
assert_equal 1234 [r get mykey]
|
|
|
|
r set mykey 1234
|
|
assert_encoding int mykey
|
|
assert_equal 4 [r setrange mykey 1 3]
|
|
assert_encoding raw mykey
|
|
assert_equal 1334 [r get mykey]
|
|
|
|
r set mykey 1234
|
|
assert_encoding int mykey
|
|
assert_equal 6 [r setrange mykey 5 2]
|
|
assert_encoding raw mykey
|
|
assert_equal "1234\0002" [r get mykey]
|
|
}
|
|
|
|
test "SETRANGE against key with wrong type" {
|
|
r del mykey
|
|
r lpush mykey "foo"
|
|
assert_error "WRONGTYPE*" {r setrange mykey 0 bar}
|
|
}
|
|
|
|
test "SETRANGE with out of range offset" {
|
|
r del mykey
|
|
assert_error "*maximum allowed size*" {r setrange mykey [expr 512*1024*1024-4] world}
|
|
|
|
r set mykey "hello"
|
|
assert_error "*out of range*" {r setrange mykey -1 world}
|
|
assert_error "*maximum allowed size*" {r setrange mykey [expr 512*1024*1024-4] world}
|
|
}
|
|
|
|
test "GETRANGE against non-existing key" {
|
|
r del mykey
|
|
assert_equal "" [r getrange mykey 0 -1]
|
|
}
|
|
|
|
test "GETRANGE against string value" {
|
|
r set mykey "Hello World"
|
|
assert_equal "Hell" [r getrange mykey 0 3]
|
|
assert_equal "Hello World" [r getrange mykey 0 -1]
|
|
assert_equal "orld" [r getrange mykey -4 -1]
|
|
assert_equal "" [r getrange mykey 5 3]
|
|
assert_equal " World" [r getrange mykey 5 5000]
|
|
assert_equal "Hello World" [r getrange mykey -5000 10000]
|
|
}
|
|
|
|
test "GETRANGE against integer-encoded value" {
|
|
r set mykey 1234
|
|
assert_equal "123" [r getrange mykey 0 2]
|
|
assert_equal "1234" [r getrange mykey 0 -1]
|
|
assert_equal "234" [r getrange mykey -3 -1]
|
|
assert_equal "" [r getrange mykey 5 3]
|
|
assert_equal "4" [r getrange mykey 3 5000]
|
|
assert_equal "1234" [r getrange mykey -5000 10000]
|
|
}
|
|
|
|
test "GETRANGE fuzzing" {
|
|
for {set i 0} {$i < 1000} {incr i} {
|
|
r set bin [set bin [randstring 0 1024 binary]]
|
|
set _start [set start [randomInt 1500]]
|
|
set _end [set end [randomInt 1500]]
|
|
if {$_start < 0} {set _start "end-[abs($_start)-1]"}
|
|
if {$_end < 0} {set _end "end-[abs($_end)-1]"}
|
|
assert_equal [string range $bin $_start $_end] [r getrange bin $start $end]
|
|
}
|
|
}
|
|
|
|
test {Extended SET can detect syntax errors} {
|
|
set e {}
|
|
catch {r set foo bar non-existing-option} e
|
|
set e
|
|
} {*syntax*}
|
|
|
|
test {Extended SET NX option} {
|
|
r del foo
|
|
set v1 [r set foo 1 nx]
|
|
set v2 [r set foo 2 nx]
|
|
list $v1 $v2 [r get foo]
|
|
} {OK {} 1}
|
|
|
|
test {Extended SET XX option} {
|
|
r del foo
|
|
set v1 [r set foo 1 xx]
|
|
r set foo bar
|
|
set v2 [r set foo 2 xx]
|
|
list $v1 $v2 [r get foo]
|
|
} {{} OK 2}
|
|
|
|
test {Extended SET EX option} {
|
|
r del foo
|
|
r set foo bar ex 10
|
|
set ttl [r ttl foo]
|
|
assert {$ttl <= 10 && $ttl > 5}
|
|
}
|
|
|
|
test {Extended SET PX option} {
|
|
r del foo
|
|
r set foo bar px 10000
|
|
set ttl [r ttl foo]
|
|
assert {$ttl <= 10 && $ttl > 5}
|
|
}
|
|
|
|
test {Extended SET using multiple options at once} {
|
|
r set foo val
|
|
assert {[r set foo bar xx px 10000] eq {OK}}
|
|
set ttl [r ttl foo]
|
|
assert {$ttl <= 10 && $ttl > 5}
|
|
}
|
|
|
|
test {KEYS * two times with long key, Github issue #1208} {
|
|
r flushdb
|
|
r set dlskeriewrioeuwqoirueioqwrueoqwrueqw test
|
|
r keys *
|
|
r keys *
|
|
} {dlskeriewrioeuwqoirueioqwrueoqwrueqw}
|
|
|
|
test {GETRANGE with huge ranges, Github issue #1844} {
|
|
r set foo bar
|
|
r getrange foo 0 4294967297
|
|
} {bar}
|
|
}
|