如何确定在Redis中执行Lua脚本的执行时间?

我有一个小 Lua 脚本要在 Redis 中运行,我想知道它的执行时间。由于 Redis 及其 Lua 实现的性质,我不能在脚本的起始点/返回点使用 TIME 函数,并在返回中包含此信息以供处理 (参见 http://redis.io/commands/eval - Scripts as pure functions)。这会导致错误:(error) ERR Error running script (call to f_a49ed2fea72f1f529843d6024d1515e76e69bcbd): Write commands not allowed after non deterministic commands。我搜索了一些我可以调用的函数/调用,可以返回上次运行脚本的执行时间,但目前还没有找到任何东西。我正在使用 PHP 和 Predis 库。虽然我可以从 PHP 端检查执行时间,但我希望消除传输开销并找出 Lua 脚本将阻止对数据库的访问多长时间。如果我不需要更改在 Redis 中存储的数据,我已成功返回了时间,而这些时间约为 PHP 报告的时间的 1/10。如何确定 Redis 中 Lua 脚本的执行时间,而不是通过 PHP?

点赞
用户1114486
用户1114486

你可以通过将 slowlog-log-slower-than 参数更改为 0 来激活 Redis 慢日志功能。它将记录所有命令的执行时间(包括 Lua 脚本和任何执行时间)。

慢日志保存在内存队列中,您必须定期转储才能收集数据。根据流量的大小,您可能需要增加 slowlog-max-len 以确保捕获您感兴趣的执行时间。

您可以使用 slowlog get 命令 来转储慢日志。由您来过滤掉不需要的结果。据我所知,在数据收集时没有过滤 Lua 统计信息的可能性。

2012-11-05 16:15:56
用户70405
用户70405

如果在构建 Redis 时启用了 os 模块,则可以在脚本中使用 os.clock()。

https://github.com/antirez/redis/blob/unstable/src/scripting.c 的第 484 行将 #if 0 更改为 #if 1

void luaLoadLibraries(lua_State *lua) {
    luaLoadLib(lua, "", luaopen_base);
    luaLoadLib(lua, LUA_TABLIBNAME, luaopen_table);
    luaLoadLib(lua, LUA_STRLIBNAME, luaopen_string);
    luaLoadLib(lua, LUA_MATHLIBNAME, luaopen_math);
    luaLoadLib(lua, LUA_DBLIBNAME, luaopen_debug);
    luaLoadLib(lua, "cjson", luaopen_cjson);
    luaLoadLib(lua, "struct", luaopen_struct);
    luaLoadLib(lua, "cmsgpack", luaopen_cmsgpack);

#if 1 /* Stuff that we don't load currently, for sandboxing concerns. */
    luaLoadLib(lua, LUA_LOADLIBNAME, luaopen_package);
    luaLoadLib(lua, LUA_OSLIBNAME, luaopen_os);
#endif
}
2014-03-18 12:04:13
用户3160475
用户3160475

编辑:rld 已过时。

您可以使用 rld(https://github.com/RedisLabs/redis-lua-debugger)并将其打印到日志中-日志具有时间信息(尽管自然地,rld会使您的脚本运行更慢)。

2015-04-07 14:43:52
用户3160475
用户3160475

从Redis v3.2开始,您可以选择通过调用新的redis.replicate_commands()API来复制命令而不是脚本。这将允许您在执行写操作的脚本中调用TIME,而不触发不确定性错误,等等。

2016-02-05 13:29:41