mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-23 00:28:26 -05:00
0a82fe8447
Adds RM_SetCommandInfo, allowing modules to provide the following command info: * summary * complexity * since * history * hints * arity * key specs * args This information affects the output of `COMMAND`, `COMMAND INFO` and `COMMAND DOCS`, Cluster, ACL and is used to filter commands with the wrong number of arguments before the call reaches the module code. The recently added API functions for key specs (never released) are removed. A minimalist example would look like so: ```c RedisModuleCommand *mycmd = RedisModule_GetCommand(ctx,"mymodule.mycommand"); RedisModuleCommandInfo mycmd_info = { .version = REDISMODULE_COMMAND_INFO_VERSION, .arity = -5, .summary = "some description", }; if (RedisModule_SetCommandInfo(mycmd, &mycmd_info) == REDISMODULE_ERR) return REDISMODULE_ERR; ```` Notes: * All the provided information (including strings) is copied, not keeping references to the API input data. * The version field is actually a static struct that contains the sizes of the the structs used in arrays, so we can extend these in the future and old version will still be able to take the part they can support.
158 lines
6.0 KiB
C
158 lines
6.0 KiB
C
#include "redismodule.h"
|
|
|
|
#define UNUSED(V) ((void) V)
|
|
|
|
int cmd_xadd(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
|
|
UNUSED(argv);
|
|
UNUSED(argc);
|
|
RedisModule_ReplyWithSimpleString(ctx, "OK");
|
|
return REDISMODULE_OK;
|
|
}
|
|
|
|
int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
|
|
REDISMODULE_NOT_USED(argv);
|
|
REDISMODULE_NOT_USED(argc);
|
|
if (RedisModule_Init(ctx, "cmdintrospection", 1, REDISMODULE_APIVER_1) == REDISMODULE_ERR)
|
|
return REDISMODULE_ERR;
|
|
|
|
if (RedisModule_CreateCommand(ctx,"cmdintrospection.xadd",cmd_xadd,"write deny-oom random fast",0,0,0) == REDISMODULE_ERR)
|
|
return REDISMODULE_ERR;
|
|
|
|
RedisModuleCommand *xadd = RedisModule_GetCommand(ctx,"cmdintrospection.xadd");
|
|
|
|
RedisModuleCommandInfo info = {
|
|
.version = REDISMODULE_COMMAND_INFO_VERSION,
|
|
.arity = -5,
|
|
.summary = "Appends a new entry to a stream",
|
|
.since = "5.0.0",
|
|
.complexity = "O(1) when adding a new entry, O(N) when trimming where N being the number of entries evicted.",
|
|
.tips = "nondeterministic_output",
|
|
.history = (RedisModuleCommandHistoryEntry[]){
|
|
/* NOTE: All versions specified should be the module's versions, not
|
|
* Redis'! We use Redis versions in this example for the purpose of
|
|
* testing (comparing the output with the output of the vanilla
|
|
* XADD). */
|
|
{"6.2.0", "Added the `NOMKSTREAM` option, `MINID` trimming strategy and the `LIMIT` option."},
|
|
{"7.0.0", "Added support for the `<ms>-*` explicit ID form."},
|
|
{0}
|
|
},
|
|
.key_specs = (RedisModuleCommandKeySpec[]){
|
|
{
|
|
.notes = "UPDATE instead of INSERT because of the optional trimming feature",
|
|
.flags = REDISMODULE_CMD_KEY_RW | REDISMODULE_CMD_KEY_UPDATE,
|
|
.begin_search_type = REDISMODULE_KSPEC_BS_INDEX,
|
|
.bs.index.pos = 1,
|
|
.find_keys_type = REDISMODULE_KSPEC_FK_RANGE,
|
|
.fk.range = {0,1,0}
|
|
},
|
|
{0}
|
|
},
|
|
.args = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "key",
|
|
.type = REDISMODULE_ARG_TYPE_KEY,
|
|
.key_spec_index = 0
|
|
},
|
|
{
|
|
.name = "nomkstream",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "NOMKSTREAM",
|
|
.since = "6.2.0",
|
|
.flags = REDISMODULE_CMD_ARG_OPTIONAL
|
|
},
|
|
{
|
|
.name = "trim",
|
|
.type = REDISMODULE_ARG_TYPE_BLOCK,
|
|
.flags = REDISMODULE_CMD_ARG_OPTIONAL,
|
|
.subargs = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "strategy",
|
|
.type = REDISMODULE_ARG_TYPE_ONEOF,
|
|
.subargs = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "maxlen",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "MAXLEN",
|
|
},
|
|
{
|
|
.name = "minid",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "MINID",
|
|
.since = "6.2.0",
|
|
},
|
|
{0}
|
|
}
|
|
},
|
|
{
|
|
.name = "operator",
|
|
.type = REDISMODULE_ARG_TYPE_ONEOF,
|
|
.flags = REDISMODULE_CMD_ARG_OPTIONAL,
|
|
.subargs = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "equal",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "="
|
|
},
|
|
{
|
|
.name = "approximately",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "~"
|
|
},
|
|
{0}
|
|
}
|
|
},
|
|
{
|
|
.name = "threshold",
|
|
.type = REDISMODULE_ARG_TYPE_STRING,
|
|
},
|
|
{
|
|
.name = "count",
|
|
.type = REDISMODULE_ARG_TYPE_INTEGER,
|
|
.token = "LIMIT",
|
|
.since = "6.2.0",
|
|
.flags = REDISMODULE_CMD_ARG_OPTIONAL
|
|
},
|
|
{0}
|
|
}
|
|
},
|
|
{
|
|
.name = "id_or_auto",
|
|
.type = REDISMODULE_ARG_TYPE_ONEOF,
|
|
.subargs = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "auto_id",
|
|
.type = REDISMODULE_ARG_TYPE_PURE_TOKEN,
|
|
.token = "*"
|
|
},
|
|
{
|
|
.name = "id",
|
|
.type = REDISMODULE_ARG_TYPE_STRING,
|
|
},
|
|
{0}
|
|
}
|
|
},
|
|
{
|
|
.name = "field_value",
|
|
.type = REDISMODULE_ARG_TYPE_BLOCK,
|
|
.flags = REDISMODULE_CMD_ARG_MULTIPLE,
|
|
.subargs = (RedisModuleCommandArg[]){
|
|
{
|
|
.name = "field",
|
|
.type = REDISMODULE_ARG_TYPE_STRING,
|
|
},
|
|
{
|
|
.name = "value",
|
|
.type = REDISMODULE_ARG_TYPE_STRING,
|
|
},
|
|
{0}
|
|
}
|
|
},
|
|
{0}
|
|
}
|
|
};
|
|
if (RedisModule_SetCommandInfo(xadd, &info) == REDISMODULE_ERR)
|
|
return REDISMODULE_ERR;
|
|
|
|
return REDISMODULE_OK;
|
|
}
|