Redis模块开发示例

实现一个Redis module,支持两个扩展命令:

1) 可同时对hash的多个field进行incr操作;

2) incrby同时设置一个key的过期时间

在没有module之前,需要借助eval+lua实现相同的功能。有了module,不但可以实现逻辑复杂,且性能高的扩展,同时享受Redis的持久化和容灾能力。

// Redis命令扩展module
#include "redismodule.h"
#include 
#include 
#include 
#include 
#include 

// 带超时的INCRBY命令
// 格式:INCRBYEX KEY SECONDS INCREMENT
//
// 示例:
// ex.incrbyex k1 10 1
// ex.incrbyex k1 10 2
int incrbyex_RedisCommand(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
    // When automatic memory management is enabled:
    // 1) don't need to close open keys
    // 2) don't need to free replies
    // 3) don't need to free RedisModuleString objects
    RedisModule_AutoMemory(ctx); /* Use automatic memory management. */
    if (argc != 4) return RedisModule_WrongArity(ctx);

    RedisModuleKey *key = (RedisModuleKey*)RedisModule_OpenKey(ctx, argv[1],
            REDISMODULE_READ|REDISMODULE_WRITE);
    int type = RedisModule_KeyType(key);
    if (type != REDISMODULE_KEYTYPE_STRING &&
        type != REDISMODULE_KEYTYPE_EMPTY)
    {
        return RedisModule_ReplyWithError(ctx, REDISMODULE_ERRORMSG_WRONGTYPE);
    }

    RedisModuleString* seconds = argv[2];
    RedisModuleString* increment = argv[3];
    long long ll_seconds; // 过期时长
    long long ll_newval; // 新的值
    if (RedisModule_StringToLongLong(seconds, &ll_seconds) != REDISMODULE_OK) {
        return RedisModule_ReplyWithError(ctx, "ERR value is not an integer or out of range");
    }
    if (RedisModule_StringToLongLong(increment, &ll_newval) != REDISMODULE_OK) {
        return RedisModule_ReplyWithError(ctx, "ERR value is not an integer or out of range");
    }

    size_t len;
    char *s = RedisModule_StringDMA(key, &len, REDISMODULE_READ|REDISMODULE_WRITE);
    if (0 == len || NULL == s || s == '\0') {
        // set必须在Expire之前,否则会冲掉Expire的作用,
        // 这也是else分支未用RedisModule_StringSet的原因
        RedisModule_StringSet(key, increment);
        RedisModule_SetExpire(key, ll_seconds*1000); // 以秒为单位,需要乘以1000
    }
    else {
        char* endptr;
        long long ll_oldval = strtoll(s, &endptr, 10); // s不一定是有效的数字,所以需要做检查
        ll_newval = ll_newval + ll_oldval;
        if ((errno == ERANGE && (ll_oldval == LLONG_MAX || ll_oldval == LLONG_MIN))
                || (errno != 0 && ll_oldval == 0)) {
            return RedisModule_ReplyWithError(ctx, "ERR value is not an integer or out of range");
        }
        if (endptr == s || *endptr != '\0') {
            return RedisModule_ReplyWithError(ctx, "ERR value is not an integer or out of range");
        }

        size_t newval_len;
        RedisModuleString* newval = RedisModule_CreateStringFromLongLong(ctx, ll_newval);
        const char* newval_s = RedisModule_StringPtrLen(newval, &newval_len);
        if (newval_len > len)
            RedisModule_StringTruncate(key, newval_len);
        strncpy(s, newval_s, newval_len);
    }

    RedisModule_ReplicateVerbatim(ctx); // 写AOF和复制到slaves
    RedisModule_ReplyWithLongLong(ctx, ll_newval);
    return REDISMODULE_OK;
}

// 实现命令HMINCRBY,同时对HASH的多个field值进行增减操作
// 格式:HMINCRBY KEY FIELD1 VALUE1 FIELD2 VALUE2 FIELD3 VALUE3 ......
//
// 示例:
// hmincrby k1 f1 1 f2 2
// hmincrby k1 f5 1 f6 2 f7 3
int hmincrby_RedisCommand(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
    // When automatic memory management is enabled:
    // 1) don't need to close open keys
    // 2) don't need to free replies
    // 3) don't need to free RedisModuleString objects
    RedisModule_AutoMemory(ctx); /* Use automatic memory management. */
    if (argc < 4 || argc % 2 != 0) return RedisModule_WrongArity(ctx);

    RedisModuleKey *key = (RedisModuleKey*)RedisModule_OpenKey(ctx, argv[1],
            REDISMODULE_READ|REDISMODULE_WRITE);
    int type = RedisModule_KeyType(key);
    if (type != REDISMODULE_KEYTYPE_HASH &&
        type != REDISMODULE_KEYTYPE_EMPTY)
    {
        return RedisModule_ReplyWithError(ctx, REDISMODULE_ERRORMSG_WRONGTYPE);
    }

    const int count = argc/2 - 1; // 键值对个数
    std::vector newval_array(count); // 用来存储新值的数组
    for (int i=2; i::size_type i=0; i

 

你可能感兴趣的:(redis,Redis)