mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 08:08:53 -05:00
Add script tests to cover keys with expiration time set (#10096)
This commit adds some tests that the test cases will access the keys with expiration time set in the script call. There was no test case for this part before. See #10080 Also there is a test will cover #1525. we block the time so that the key can not expire in the middle of the script execution. Other changes: 1. Delete `evalTimeSnapshot` and just use `scriptTimeSnapshot` in it's place. 2. Some cleanups to scripting.tcl. 3. better names for tests that run in a loop to make them distinctable
This commit is contained in:
parent
6790d848c5
commit
e22146b07a
2
src/db.c
2
src/db.c
@ -1559,7 +1559,7 @@ int keyIsExpired(redisDb *db, robj *key) {
|
|||||||
* script execution, making propagation to slaves / AOF consistent.
|
* script execution, making propagation to slaves / AOF consistent.
|
||||||
* See issue #1525 on Github for more information. */
|
* See issue #1525 on Github for more information. */
|
||||||
if (server.script_caller) {
|
if (server.script_caller) {
|
||||||
now = evalTimeSnapshot();
|
now = scriptTimeSnapshot();
|
||||||
}
|
}
|
||||||
/* If we are in the middle of a command execution, we still want to use
|
/* If we are in the middle of a command execution, we still want to use
|
||||||
* a reference time that does not change: in that case we just use the
|
* a reference time that does not change: in that case we just use the
|
||||||
|
@ -567,12 +567,6 @@ unsigned long evalScriptsMemory() {
|
|||||||
dictSlots(lctx.lua_scripts) * sizeof(dictEntry*);
|
dictSlots(lctx.lua_scripts) * sizeof(dictEntry*);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Returns the time when the script invocation started */
|
|
||||||
mstime_t evalTimeSnapshot() {
|
|
||||||
return scriptTimeSnapshot();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* ---------------------------------------------------------------------------
|
/* ---------------------------------------------------------------------------
|
||||||
* LDB: Redis Lua debugging facilities
|
* LDB: Redis Lua debugging facilities
|
||||||
* ------------------------------------------------------------------------- */
|
* ------------------------------------------------------------------------- */
|
||||||
|
@ -421,5 +421,3 @@ long long scriptRunDuration() {
|
|||||||
serverAssert(scriptIsRunning());
|
serverAssert(scriptIsRunning());
|
||||||
return elapsedMs(curr_run_ctx->start_time);
|
return elapsedMs(curr_run_ctx->start_time);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -3032,7 +3032,6 @@ void sha1hex(char *digest, char *script, size_t len);
|
|||||||
unsigned long evalMemory();
|
unsigned long evalMemory();
|
||||||
dict* evalScriptsDict();
|
dict* evalScriptsDict();
|
||||||
unsigned long evalScriptsMemory();
|
unsigned long evalScriptsMemory();
|
||||||
mstime_t evalTimeSnapshot();
|
|
||||||
|
|
||||||
/* Blocked clients */
|
/* Blocked clients */
|
||||||
void processUnblockedClients(void);
|
void processUnblockedClients(void);
|
||||||
|
@ -676,10 +676,29 @@ start_server {tags {"scripting"}} {
|
|||||||
local a = {}
|
local a = {}
|
||||||
for i=1,7999 do
|
for i=1,7999 do
|
||||||
a[i] = 1
|
a[i] = 1
|
||||||
end
|
end
|
||||||
return redis.call("lpush", "l", unpack(a))
|
return redis.call("lpush", "l", unpack(a))
|
||||||
} 0
|
} 0
|
||||||
} {7999}
|
} {7999}
|
||||||
|
|
||||||
|
test "Script read key with expiration set" {
|
||||||
|
r SET key value EX 10
|
||||||
|
assert_equal [run_script {
|
||||||
|
if redis.call("EXISTS", "key") then
|
||||||
|
return redis.call("GET", "key")
|
||||||
|
else
|
||||||
|
return redis.call("EXISTS", "key")
|
||||||
|
end
|
||||||
|
} 0] "value"
|
||||||
|
}
|
||||||
|
|
||||||
|
test "Script del key with expiration set" {
|
||||||
|
r SET key value EX 10
|
||||||
|
assert_equal [run_script {
|
||||||
|
redis.call("DEL", "key")
|
||||||
|
return redis.call("EXISTS", "key")
|
||||||
|
} 0] 0
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# Start a new server since the last test in this stanza will kill the
|
# Start a new server since the last test in this stanza will kill the
|
||||||
@ -702,7 +721,7 @@ start_server {tags {"scripting"}} {
|
|||||||
set rd [redis_deferring_client]
|
set rd [redis_deferring_client]
|
||||||
r config set lua-time-limit 10
|
r config set lua-time-limit 10
|
||||||
run_script_on_connection $rd {local f = function() while 1 do redis.call('ping') end end while 1 do pcall(f) end} 0
|
run_script_on_connection $rd {local f = function() while 1 do redis.call('ping') end end while 1 do pcall(f) end} 0
|
||||||
|
|
||||||
wait_for_condition 50 100 {
|
wait_for_condition 50 100 {
|
||||||
[catch {r ping} e] == 1
|
[catch {r ping} e] == 1
|
||||||
} else {
|
} else {
|
||||||
@ -723,7 +742,7 @@ start_server {tags {"scripting"}} {
|
|||||||
catch {$rd read} res
|
catch {$rd read} res
|
||||||
$rd close
|
$rd close
|
||||||
|
|
||||||
assert_match {*killed by user*} $res
|
assert_match {*killed by user*} $res
|
||||||
}
|
}
|
||||||
|
|
||||||
test {Timedout script does not cause a false dead client} {
|
test {Timedout script does not cause a false dead client} {
|
||||||
@ -769,7 +788,7 @@ start_server {tags {"scripting"}} {
|
|||||||
assert_match {*killed by user*} $res
|
assert_match {*killed by user*} $res
|
||||||
|
|
||||||
set res [$rd read]
|
set res [$rd read]
|
||||||
assert_match {*PONG*} $res
|
assert_match {*PONG*} $res
|
||||||
|
|
||||||
$rd close
|
$rd close
|
||||||
}
|
}
|
||||||
@ -1051,17 +1070,19 @@ start_server {tags {"scripting needs:debug external:skip"}} {
|
|||||||
}
|
}
|
||||||
} ;# is_eval
|
} ;# is_eval
|
||||||
|
|
||||||
start_server {tags {"scripting resp3 needs:debug"}} {
|
start_server {tags {"scripting needs:debug"}} {
|
||||||
r debug set-disable-deny-scripts 1
|
r debug set-disable-deny-scripts 1
|
||||||
|
|
||||||
for {set i 2} {$i <= 3} {incr i} {
|
for {set i 2} {$i <= 3} {incr i} {
|
||||||
for {set client_proto 2} {$client_proto <= 3} {incr client_proto} {
|
for {set client_proto 2} {$client_proto <= 3} {incr client_proto} {
|
||||||
|
set extra "RESP$i/$client_proto"
|
||||||
r hello $client_proto
|
r hello $client_proto
|
||||||
r readraw 1
|
r readraw 1
|
||||||
|
|
||||||
test {test resp3 big number protocol parsing} {
|
test "test $extra big number protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'bignum')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'bignum')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {$37}
|
assert_equal $ret {$37}
|
||||||
assert_equal [r read] {1234567999999999999999999999999999999}
|
assert_equal [r read] {1234567999999999999999999999999999999}
|
||||||
} else {
|
} else {
|
||||||
@ -1069,10 +1090,10 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 malformed big number protocol parsing} {
|
test "test $extra malformed big number protocol parsing" {
|
||||||
set ret [r eval "return {big_number='123\\r\\n123'}" 0]
|
set ret [run_script "return {big_number='123\\r\\n123'}" 0]
|
||||||
if {$client_proto == 2} {
|
if {$client_proto == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {$8}
|
assert_equal $ret {$8}
|
||||||
assert_equal [r read] {123 123}
|
assert_equal [r read] {123 123}
|
||||||
} else {
|
} else {
|
||||||
@ -1080,10 +1101,10 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 map protocol parsing} {
|
test "test $extra map protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'map')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'map')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {*6}
|
assert_equal $ret {*6}
|
||||||
} else {
|
} else {
|
||||||
assert_equal $ret {%3}
|
assert_equal $ret {%3}
|
||||||
@ -1093,10 +1114,10 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 set protocol parsing} {
|
test "test $extra set protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'set')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'set')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {*3}
|
assert_equal $ret {*3}
|
||||||
} else {
|
} else {
|
||||||
assert_equal $ret {~3}
|
assert_equal $ret {~3}
|
||||||
@ -1106,10 +1127,10 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 double protocol parsing} {
|
test "test $extra double protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'double')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'double')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {$5}
|
assert_equal $ret {$5}
|
||||||
assert_equal [r read] {3.141}
|
assert_equal [r read] {3.141}
|
||||||
} else {
|
} else {
|
||||||
@ -1117,7 +1138,7 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 null protocol parsing} {
|
test "test $extra null protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'null')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'null')" 0]
|
||||||
if {$client_proto == 2} {
|
if {$client_proto == 2} {
|
||||||
# null is a special case in which a Lua client format does not effect the reply to the client
|
# null is a special case in which a Lua client format does not effect the reply to the client
|
||||||
@ -1127,10 +1148,10 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
} {}
|
} {}
|
||||||
|
|
||||||
test {test resp3 verbatim protocol parsing} {
|
test "test $extra verbatim protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'verbatim')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'verbatim')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {$25}
|
assert_equal $ret {$25}
|
||||||
assert_equal [r read] {This is a verbatim}
|
assert_equal [r read] {This is a verbatim}
|
||||||
assert_equal [r read] {string}
|
assert_equal [r read] {string}
|
||||||
@ -1141,20 +1162,20 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 true protocol parsing} {
|
test "test $extra true protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'true')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'true')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {:1}
|
assert_equal $ret {:1}
|
||||||
} else {
|
} else {
|
||||||
assert_equal $ret {#t}
|
assert_equal $ret {#t}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test resp3 false protocol parsing} {
|
test "test $extra false protocol parsing" {
|
||||||
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'false')" 0]
|
set ret [run_script "redis.setresp($i);return redis.call('debug', 'protocol', 'false')" 0]
|
||||||
if {$client_proto == 2 || $i == 2} {
|
if {$client_proto == 2 || $i == 2} {
|
||||||
# if either Lua or the clien is RESP2 the reply will be RESP2
|
# if either Lua or the client is RESP2 the reply will be RESP2
|
||||||
assert_equal $ret {:0}
|
assert_equal $ret {:0}
|
||||||
} else {
|
} else {
|
||||||
assert_equal $ret {#f}
|
assert_equal $ret {#f}
|
||||||
@ -1172,6 +1193,29 @@ start_server {tags {"scripting resp3 needs:debug"}} {
|
|||||||
run_script "redis.setresp(3);return redis.call('debug', 'protocol', 'attrib')" 0
|
run_script "redis.setresp(3);return redis.call('debug', 'protocol', 'attrib')" 0
|
||||||
} {Some real reply following the attribute}
|
} {Some real reply following the attribute}
|
||||||
|
|
||||||
|
test "Script block the time during execution" {
|
||||||
|
assert_equal [run_script {
|
||||||
|
redis.call("SET", "key", "value", "PX", "1")
|
||||||
|
redis.call("DEBUG", "SLEEP", 0.01)
|
||||||
|
return redis.call("EXISTS", "key")
|
||||||
|
} 0] 1
|
||||||
|
|
||||||
|
assert_equal 0 [r EXISTS key]
|
||||||
|
}
|
||||||
|
|
||||||
|
test "Script delete the expired key" {
|
||||||
|
r DEBUG set-active-expire 0
|
||||||
|
r SET key value PX 1
|
||||||
|
after 2
|
||||||
|
|
||||||
|
# use DEBUG OBJECT to make sure it doesn't error (means the key still exists)
|
||||||
|
r DEBUG OBJECT key
|
||||||
|
|
||||||
|
assert_equal [run_script "return redis.call('EXISTS', 'key')" 0] 0
|
||||||
|
assert_equal 0 [r EXISTS key]
|
||||||
|
r DEBUG set-active-expire 1
|
||||||
|
}
|
||||||
|
|
||||||
r debug set-disable-deny-scripts 0
|
r debug set-disable-deny-scripts 0
|
||||||
}
|
}
|
||||||
} ;# foreach is_eval
|
} ;# foreach is_eval
|
||||||
|
Loading…
Reference in New Issue
Block a user