mirror of
https://github.com/fluencelabs/redis
synced 2025-03-19 17:10:50 +00:00
Modules: handle the conflict of registering commands
This commit is contained in:
parent
474adba9fa
commit
cb9dde3280
49
src/module.c
49
src/module.c
@ -615,7 +615,7 @@ int RM_CreateCommand(RedisModuleCtx *ctx, const char *name, RedisModuleCmdFunc c
|
|||||||
sds cmdname = sdsnew(name);
|
sds cmdname = sdsnew(name);
|
||||||
|
|
||||||
/* Check if the command name is busy. */
|
/* Check if the command name is busy. */
|
||||||
if (lookupCommand((char*)name) != NULL) {
|
if (lookupCommand(cmdname) != NULL) {
|
||||||
sdsfree(cmdname);
|
sdsfree(cmdname);
|
||||||
return REDISMODULE_ERR;
|
return REDISMODULE_ERR;
|
||||||
}
|
}
|
||||||
@ -3661,6 +3661,28 @@ void moduleFreeModuleStructure(struct RedisModule *module) {
|
|||||||
zfree(module);
|
zfree(module);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void moduleUnregisterCommands(struct RedisModule *module) {
|
||||||
|
/* Unregister all the commands registered by this module. */
|
||||||
|
dictIterator *di = dictGetSafeIterator(server.commands);
|
||||||
|
dictEntry *de;
|
||||||
|
while ((de = dictNext(di)) != NULL) {
|
||||||
|
struct redisCommand *cmd = dictGetVal(de);
|
||||||
|
if (cmd->proc == RedisModuleCommandDispatcher) {
|
||||||
|
RedisModuleCommandProxy *cp =
|
||||||
|
(void*)(unsigned long)cmd->getkeys_proc;
|
||||||
|
sds cmdname = cp->rediscmd->name;
|
||||||
|
if (cp->module == module) {
|
||||||
|
dictDelete(server.commands,cmdname);
|
||||||
|
dictDelete(server.orig_commands,cmdname);
|
||||||
|
sdsfree(cmdname);
|
||||||
|
zfree(cp->rediscmd);
|
||||||
|
zfree(cp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dictReleaseIterator(di);
|
||||||
|
}
|
||||||
|
|
||||||
/* Load a module and initialize it. On success C_OK is returned, otherwise
|
/* Load a module and initialize it. On success C_OK is returned, otherwise
|
||||||
* C_ERR is returned. */
|
* C_ERR is returned. */
|
||||||
int moduleLoad(const char *path, void **module_argv, int module_argc) {
|
int moduleLoad(const char *path, void **module_argv, int module_argc) {
|
||||||
@ -3681,7 +3703,10 @@ int moduleLoad(const char *path, void **module_argv, int module_argc) {
|
|||||||
return C_ERR;
|
return C_ERR;
|
||||||
}
|
}
|
||||||
if (onload((void*)&ctx,module_argv,module_argc) == REDISMODULE_ERR) {
|
if (onload((void*)&ctx,module_argv,module_argc) == REDISMODULE_ERR) {
|
||||||
if (ctx.module) moduleFreeModuleStructure(ctx.module);
|
if (ctx.module) {
|
||||||
|
moduleUnregisterCommands(ctx.module);
|
||||||
|
moduleFreeModuleStructure(ctx.module);
|
||||||
|
}
|
||||||
dlclose(handle);
|
dlclose(handle);
|
||||||
serverLog(LL_WARNING,
|
serverLog(LL_WARNING,
|
||||||
"Module %s initialization failed. Module not loaded",path);
|
"Module %s initialization failed. Module not loaded",path);
|
||||||
@ -3715,25 +3740,7 @@ int moduleUnload(sds name) {
|
|||||||
return REDISMODULE_ERR;
|
return REDISMODULE_ERR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Unregister all the commands registered by this module. */
|
moduleUnregisterCommands(module);
|
||||||
dictIterator *di = dictGetSafeIterator(server.commands);
|
|
||||||
dictEntry *de;
|
|
||||||
while ((de = dictNext(di)) != NULL) {
|
|
||||||
struct redisCommand *cmd = dictGetVal(de);
|
|
||||||
if (cmd->proc == RedisModuleCommandDispatcher) {
|
|
||||||
RedisModuleCommandProxy *cp =
|
|
||||||
(void*)(unsigned long)cmd->getkeys_proc;
|
|
||||||
sds cmdname = cp->rediscmd->name;
|
|
||||||
if (cp->module == module) {
|
|
||||||
dictDelete(server.commands,cmdname);
|
|
||||||
dictDelete(server.orig_commands,cmdname);
|
|
||||||
sdsfree(cmdname);
|
|
||||||
zfree(cp->rediscmd);
|
|
||||||
zfree(cp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dictReleaseIterator(di);
|
|
||||||
|
|
||||||
/* Unregister all the hooks. TODO: Yet no hooks support here. */
|
/* Unregister all the hooks. TODO: Yet no hooks support here. */
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user