讓redis支持動態庫
原文 http://wendal.net/2013/10/20.html
So, let’s put some fun in redis
redis的命令列表是寫死在代碼里面,編譯后就沒法修改(跟nginx一個德行)
硬編碼一些額外的方法到redis中
最初寫的一個hello world級的測試命令
void wendalCommand(redisClient *c) { struct timeval tv; // 從timeCommand拷貝的... addReplyMultiBulkLen(c,3); // 共返回3個結果 addReplyBulkCString(c, "Hi,Wendal"); // 輸出個字符串啦 gettimeofday(&tv,NULL); // 獲取時間, 也是從timeCommand拷貝的 addReplyBulkLongLong(c,tv.tv_sec); // 不解釋了 addReplyBulkLongLong(c,tv.tv_usec); // 不解釋了 }
然后在redisCommandTable中聲明一個新的命令,就搞定了
{"wendal", wendalCommand,1,"rR",0,NULL,0,0,0,0,0}
做得更靈活?上動態庫,哈哈
so,我聲明了2個新的命令, 加載/卸載動態庫
{"loadlib",loadlibCommand,2,"wmaR",0,NULL,0,0,0,0,0}, {"unloadlib",unloadlibCommand,2,"wmaR",0,NULL,0,0,0,0,0}
其中, 加載動態庫,就是接受一個路徑,并執行其redis_lib_init方法
// 簡化版的loadlibCommand void loadlibCommand(redisClient *c) { void *handle; char *error; int (*redis_lib_init)(redisClient*, dict*); int re; handle = dlopen(c->argv[1]->ptr, RTLD_LAZY); if (!handle) { addReplyError(c, dlerror()); return; } redis_lib_init = dlsym(handle, "redis_lib_init"); // 查找redis_lib_init if ((error = dlerror()) != NULL) { dlclose(handle); addReplyError(c, error); return; } re = redis_lib_init(c, server.commands); // 執行之, 返回0就成功 if (re) { dlclose(handle); addReplyError(c, "lib init error"); return; } addReply(c,shared.ok); server.dirty ++; // 使dirty自增,這樣就能同步到slave了 }
redis動態庫示例
禁用flushdb命令
#include <redis.h> static void* flushdb; extern int redis_lib_init(redisClient *c) { flushdb = dictFetchValue(server.commands, sdsnew("FLUSHDB")); // 命令列表就是server.commands中,一個dict if (!flushdb) return 1; // 找不到? 被其他庫刪掉了? dictDeleteNoFree(server.commands, sdsnew("FLUSHDB")); // 刪除但不要執行free return 0; } extern void redis_lib_depose(redisClient *c) { dictAdd(server.commands, sdsnew("FLUSHDB"), flushdb); // 卸載的時候就賦值回去,呵呵 return; }
項目地址
本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!