X-Git-Url: https://git.saurik.com/redis.git/blobdiff_plain/a9b07ac459b435d0ea7ad37da933511f83e05507..dd5fbedf7bb9ac02d14aa9ecaeafb47e48b9a587:/src/scripting.c diff --git a/src/scripting.c b/src/scripting.c index 68f3c946..0b548873 100644 --- a/src/scripting.c +++ b/src/scripting.c @@ -273,13 +273,15 @@ int luaLogCommand(lua_State *lua) { void luaMaskCountHook(lua_State *lua, lua_Debug *ar) { long long elapsed; REDIS_NOTUSED(ar); + REDIS_NOTUSED(lua); elapsed = (ustime()/1000) - server.lua_time_start; - if (elapsed >= server.lua_time_limit) { - redisLog(REDIS_NOTICE,"Lua script aborted for max execution time after %lld milliseconds of running time.",elapsed); - lua_pushstring(lua,"Script aborted for max execution time."); - lua_error(lua); + if (elapsed >= server.lua_time_limit && server.lua_timedout == 0) { + redisLog(REDIS_WARNING,"Lua slow script detected: still in execution after %lld milliseconds. You can shut down the server using the SHUTDOWN command.",elapsed); + server.lua_timedout = 1; } + if (server.lua_timedout) + aeProcessEvents(server.el, AE_FILE_EVENTS|AE_DONT_WAIT); } void luaLoadLib(lua_State *lua, const char *libname, lua_CFunction luafunc) { @@ -500,7 +502,6 @@ void luaSetGlobalArray(lua_State *lua, char *var, robj **elev, int elec) { int luaCreateFunction(redisClient *c, lua_State *lua, char *funcname, robj *body) { sds funcdef = sdsempty(); - lua_pop(lua,1); /* remove the nil from the stack */ funcdef = sdscat(funcdef,"function "); funcdef = sdscatlen(funcdef,funcname,42); funcdef = sdscatlen(funcdef," ()\n",4); @@ -581,12 +582,12 @@ void evalGenericCommand(redisClient *c, int evalsha) { /* Try to lookup the Lua function */ lua_getglobal(lua, funcname); if (lua_isnil(lua,1)) { + lua_pop(lua,1); /* remove the nil from the stack */ /* Function not defined... let's define it if we have the * body of the funciton. If this is an EVALSHA call we can just * return an error. */ if (evalsha) { addReply(c, shared.noscripterr); - lua_pop(lua,1); /* remove the nil from the stack */ return; } if (luaCreateFunction(c,lua,funcname,c->argv[1]) == REDIS_ERR) return; @@ -607,7 +608,7 @@ void evalGenericCommand(redisClient *c, int evalsha) { * is running for too much time. * We set the hook only if the time limit is enabled as the hook will * make the Lua script execution slower. */ - if (server.lua_time_limit > 0) { + if (server.lua_time_limit > 0 && server.masterhost == NULL) { lua_sethook(lua,luaMaskCountHook,LUA_MASKCOUNT,100000); server.lua_time_start = ustime()/1000; } else { @@ -618,6 +619,7 @@ void evalGenericCommand(redisClient *c, int evalsha) { * already defined, we can call it. We have zero arguments and expect * a single return value. */ if (lua_pcall(lua,0,1,0)) { + server.lua_timedout = 0; selectDb(c,server.lua_client->db->id); /* set DB ID from Lua client */ addReplyErrorFormat(c,"Error running script (call to %s): %s\n", funcname, lua_tostring(lua,-1)); @@ -625,6 +627,7 @@ void evalGenericCommand(redisClient *c, int evalsha) { lua_gc(lua,LUA_GCCOLLECT,0); return; } + server.lua_timedout = 0; selectDb(c,server.lua_client->db->id); /* set DB ID from Lua client */ luaReplyToRedisReply(c,lua); lua_gc(lua,LUA_GCSTEP,1); @@ -724,16 +727,22 @@ void scriptCommand(redisClient *c) { addReply(c,shared.czero); } } else if (c->argc == 3 && !strcasecmp(c->argv[1]->ptr,"load")) { - /* We obtain the script SHA1, then check if this function is already - * defined into the Lua state */ char funcname[43]; + sds sha; funcname[0] = 'f'; funcname[1] = '_'; hashScript(funcname+2,c->argv[2]->ptr,sdslen(c->argv[2]->ptr)); - if (luaCreateFunction(c,server.lua,funcname,c->argv[2]) == REDIS_ERR) - return; - addReply(c,shared.ok); + sha = sdsnewlen(funcname+2,40); + if (dictFind(server.lua_scripts,sha) == NULL) { + if (luaCreateFunction(c,server.lua,funcname,c->argv[2]) + == REDIS_ERR) { + sdsfree(sha); + return; + } + } + addReplyBulkCBuffer(c,funcname+2,40); + sdsfree(sha); } else { addReplyError(c, "Unknown SCRIPT subcommand or wrong # of args."); }