#include "redismodule.h" #include #include /* ZSET.REM key element * * Removes an occurrence of an element from a sorted set. Replies with the * number of removed elements (0 or 1). */ int zset_rem(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) { if (argc != 3) return RedisModule_WrongArity(ctx); RedisModule_AutoMemory(ctx); int keymode = REDISMODULE_READ | REDISMODULE_WRITE; RedisModuleKey *key = RedisModule_OpenKey(ctx, argv[1], keymode); int deleted; if (RedisModule_ZsetRem(key, argv[2], &deleted) == REDISMODULE_OK) return RedisModule_ReplyWithLongLong(ctx, deleted); else return RedisModule_ReplyWithError(ctx, "ERR ZsetRem failed"); } /* ZSET.ADD key score member * * Adds a specified member with the specified score to the sorted * set stored at key. */ int zset_add(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) { if (argc != 4) return RedisModule_WrongArity(ctx); RedisModule_AutoMemory(ctx); int keymode = REDISMODULE_READ | REDISMODULE_WRITE; RedisModuleKey *key = RedisModule_OpenKey(ctx, argv[1], keymode); size_t len; double score; char *endptr; const char *str = RedisModule_StringPtrLen(argv[2], &len); score = strtod(str, &endptr); if (*endptr != '\0' || errno == ERANGE) return RedisModule_ReplyWithError(ctx, "value is not a valid float"); if (RedisModule_ZsetAdd(key, score, argv[3], NULL) == REDISMODULE_OK) return RedisModule_ReplyWithSimpleString(ctx, "OK"); else return RedisModule_ReplyWithError(ctx, "ERR ZsetAdd failed"); } /* ZSET.INCRBY key member increment * * Increments the score stored at member in the sorted set stored at key by increment. * Replies with the new score of this element. */ int zset_incrby(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) { if (argc != 4) return RedisModule_WrongArity(ctx); RedisModule_AutoMemory(ctx); int keymode = REDISMODULE_READ | REDISMODULE_WRITE; RedisModuleKey *key = RedisModule_OpenKey(ctx, argv[1], keymode); size_t len; double score, newscore; char *endptr; const char *str = RedisModule_StringPtrLen(argv[3], &len); score = strtod(str, &endptr); if (*endptr != '\0' || errno == ERANGE) return RedisModule_ReplyWithError(ctx, "value is not a valid float"); if (RedisModule_ZsetIncrby(key, score, argv[2], NULL, &newscore) == REDISMODULE_OK) return RedisModule_ReplyWithDouble(ctx, newscore); else return RedisModule_ReplyWithError(ctx, "ERR ZsetIncrby failed"); } /* Structure to hold data for the delall scan callback */ typedef struct { RedisModuleCtx *ctx; RedisModuleString **keys_to_delete; size_t keys_capacity; size_t keys_count; } zset_delall_data; /* Callback function for scanning keys and collecting zset keys to delete */ void zset_delall_callback(RedisModuleCtx *ctx, RedisModuleString *keyname, RedisModuleKey *key, void *privdata) { zset_delall_data *data = privdata; int was_opened = 0; /* Open the key if it wasn't already opened */ if (!key) { key = RedisModule_OpenKey(ctx, keyname, REDISMODULE_READ); was_opened = 1; } /* Check if the key is a zset and add it to the list */ if (RedisModule_KeyType(key) == REDISMODULE_KEYTYPE_ZSET) { /* Expand the array if needed */ if (data->keys_count >= data->keys_capacity) { data->keys_capacity = data->keys_capacity ? data->keys_capacity * 2 : 16; data->keys_to_delete = RedisModule_Realloc(data->keys_to_delete, data->keys_capacity * sizeof(RedisModuleString*)); } /* Store the key name (retain it so it doesn't get freed) */ data->keys_to_delete[data->keys_count] = keyname; RedisModule_RetainString(ctx, keyname); data->keys_count++; } /* Close the key if we opened it */ if (was_opened) { RedisModule_CloseKey(key); } } /* ZSET.DELALL * * Iterates through the keyspace and deletes all keys of type "zset". * Returns the number of deleted keys. */ int zset_delall(RedisModuleCtx *ctx, REDISMODULE_ATTR_UNUSED RedisModuleString **argv, int argc) { if (argc != 1) return RedisModule_WrongArity(ctx); RedisModule_AutoMemory(ctx); zset_delall_data data = { .ctx = ctx, .keys_to_delete = NULL, .keys_capacity = 0, .keys_count = 0 }; /* Create a scan cursor and iterate through all keys */ RedisModuleScanCursor *cursor = RedisModule_ScanCursorCreate(); while (RedisModule_Scan(ctx, cursor, zset_delall_callback, &data)); RedisModule_ScanCursorDestroy(cursor); /* Delete all the collected zset keys after scan is complete */ size_t deleted_count = 0; for (size_t i = 0; i < data.keys_count; i++) { RedisModuleCallReply *reply = RedisModule_Call(ctx, "DEL", "s!", data.keys_to_delete[i]); if (reply && RedisModule_CallReplyType(reply) == REDISMODULE_REPLY_INTEGER) { long long del_result = RedisModule_CallReplyInteger(reply); if (del_result > 0) { deleted_count++; } } if (reply) { RedisModule_FreeCallReply(reply); } RedisModule_FreeString(ctx, data.keys_to_delete[i]); } /* Free the keys array */ if (data.keys_to_delete) { RedisModule_Free(data.keys_to_delete); } return RedisModule_ReplyWithLongLong(ctx, deleted_count); } int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) { REDISMODULE_NOT_USED(argv); REDISMODULE_NOT_USED(argc); if (RedisModule_Init(ctx, "zset", 1, REDISMODULE_APIVER_1) == REDISMODULE_ERR) return REDISMODULE_ERR; if (RedisModule_CreateCommand(ctx, "zset.rem", zset_rem, "write", 1, 1, 1) == REDISMODULE_ERR) return REDISMODULE_ERR; if (RedisModule_CreateCommand(ctx, "zset.add", zset_add, "write", 1, 1, 1) == REDISMODULE_ERR) return REDISMODULE_ERR; if (RedisModule_CreateCommand(ctx, "zset.incrby", zset_incrby, "write", 1, 1, 1) == REDISMODULE_ERR) return REDISMODULE_ERR; if (RedisModule_CreateCommand(ctx, "zset.delall", zset_delall, "write touches-arbitrary-keys", 0, 0, 0) == REDISMODULE_ERR) return REDISMODULE_ERR; return REDISMODULE_OK; }