scan.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. #include "redismodule.h"
  2. #include <string.h>
  3. #include <assert.h>
  4. #include <unistd.h>
  5. typedef struct {
  6. size_t nkeys;
  7. } scan_strings_pd;
  8. void scan_strings_callback(RedisModuleCtx *ctx, RedisModuleString* keyname, RedisModuleKey* key, void *privdata) {
  9. scan_strings_pd* pd = privdata;
  10. int was_opened = 0;
  11. if (!key) {
  12. key = RedisModule_OpenKey(ctx, keyname, REDISMODULE_READ);
  13. was_opened = 1;
  14. }
  15. if (RedisModule_KeyType(key) == REDISMODULE_KEYTYPE_STRING) {
  16. size_t len;
  17. char * data = RedisModule_StringDMA(key, &len, REDISMODULE_READ);
  18. RedisModule_ReplyWithArray(ctx, 2);
  19. RedisModule_ReplyWithString(ctx, keyname);
  20. RedisModule_ReplyWithStringBuffer(ctx, data, len);
  21. pd->nkeys++;
  22. }
  23. if (was_opened)
  24. RedisModule_CloseKey(key);
  25. }
  26. int scan_strings(RedisModuleCtx *ctx, RedisModuleString **argv, int argc)
  27. {
  28. REDISMODULE_NOT_USED(argv);
  29. REDISMODULE_NOT_USED(argc);
  30. scan_strings_pd pd = {
  31. .nkeys = 0,
  32. };
  33. RedisModule_ReplyWithArray(ctx, REDISMODULE_POSTPONED_LEN);
  34. RedisModuleScanCursor* cursor = RedisModule_ScanCursorCreate();
  35. while(RedisModule_Scan(ctx, cursor, scan_strings_callback, &pd));
  36. RedisModule_ScanCursorDestroy(cursor);
  37. RedisModule_ReplySetArrayLength(ctx, pd.nkeys);
  38. return REDISMODULE_OK;
  39. }
  40. typedef struct {
  41. RedisModuleCtx *ctx;
  42. size_t nreplies;
  43. } scan_key_pd;
  44. void scan_key_callback(RedisModuleKey *key, RedisModuleString* field, RedisModuleString* value, void *privdata) {
  45. REDISMODULE_NOT_USED(key);
  46. scan_key_pd* pd = privdata;
  47. RedisModule_ReplyWithArray(pd->ctx, 2);
  48. size_t fieldCStrLen;
  49. // The implementation of RedisModuleString is robj with lots of encodings.
  50. // We want to make sure the robj that passes to this callback in
  51. // String encoded, this is why we use RedisModule_StringPtrLen and
  52. // RedisModule_ReplyWithStringBuffer instead of directly use
  53. // RedisModule_ReplyWithString.
  54. const char* fieldCStr = RedisModule_StringPtrLen(field, &fieldCStrLen);
  55. RedisModule_ReplyWithStringBuffer(pd->ctx, fieldCStr, fieldCStrLen);
  56. if(value){
  57. size_t valueCStrLen;
  58. const char* valueCStr = RedisModule_StringPtrLen(value, &valueCStrLen);
  59. RedisModule_ReplyWithStringBuffer(pd->ctx, valueCStr, valueCStrLen);
  60. } else {
  61. RedisModule_ReplyWithNull(pd->ctx);
  62. }
  63. pd->nreplies++;
  64. }
  65. int scan_key(RedisModuleCtx *ctx, RedisModuleString **argv, int argc)
  66. {
  67. if (argc != 2) {
  68. RedisModule_WrongArity(ctx);
  69. return REDISMODULE_OK;
  70. }
  71. scan_key_pd pd = {
  72. .ctx = ctx,
  73. .nreplies = 0,
  74. };
  75. RedisModuleKey *key = RedisModule_OpenKey(ctx, argv[1], REDISMODULE_READ);
  76. if (!key) {
  77. RedisModule_ReplyWithError(ctx, "not found");
  78. return REDISMODULE_OK;
  79. }
  80. RedisModule_ReplyWithArray(ctx, REDISMODULE_POSTPONED_ARRAY_LEN);
  81. RedisModuleScanCursor* cursor = RedisModule_ScanCursorCreate();
  82. while(RedisModule_ScanKey(key, cursor, scan_key_callback, &pd));
  83. RedisModule_ScanCursorDestroy(cursor);
  84. RedisModule_ReplySetArrayLength(ctx, pd.nreplies);
  85. RedisModule_CloseKey(key);
  86. return REDISMODULE_OK;
  87. }
  88. int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
  89. REDISMODULE_NOT_USED(argv);
  90. REDISMODULE_NOT_USED(argc);
  91. if (RedisModule_Init(ctx, "scan", 1, REDISMODULE_APIVER_1)== REDISMODULE_ERR)
  92. return REDISMODULE_ERR;
  93. if (RedisModule_CreateCommand(ctx, "scan.scan_strings", scan_strings, "", 0, 0, 0) == REDISMODULE_ERR)
  94. return REDISMODULE_ERR;
  95. if (RedisModule_CreateCommand(ctx, "scan.scan_key", scan_key, "", 0, 0, 0) == REDISMODULE_ERR)
  96. return REDISMODULE_ERR;
  97. return REDISMODULE_OK;
  98. }