Delete src/modules/hellotype.c
Some checks failed
CI / test-ubuntu-latest (push) Has been cancelled
CI / test-ubuntu-latest-cmake (push) Has been cancelled
CI / test-sanitizer-address (push) Has been cancelled
CI / test-rdma (push) Has been cancelled
CI / build-debian-old (push) Has been cancelled
CI / build-macos-latest (push) Has been cancelled
CI / build-32bit (push) Has been cancelled
CI / build-libc-malloc (push) Has been cancelled
CI / build-almalinux8-jemalloc (push) Has been cancelled
CI / format-yaml (push) Has been cancelled
Clang Format Check / clang-format-check (push) Has been cancelled
Codecov / code-coverage (push) Has been cancelled
External Server Tests / test-external-standalone (push) Has been cancelled
External Server Tests / test-external-cluster (push) Has been cancelled
External Server Tests / test-external-nodebug (push) Has been cancelled
Spellcheck / Spellcheck (push) Has been cancelled
CodeQL / Analyze (cpp) (push) Has been cancelled
Some checks failed
CI / test-ubuntu-latest (push) Has been cancelled
CI / test-ubuntu-latest-cmake (push) Has been cancelled
CI / test-sanitizer-address (push) Has been cancelled
CI / test-rdma (push) Has been cancelled
CI / build-debian-old (push) Has been cancelled
CI / build-macos-latest (push) Has been cancelled
CI / build-32bit (push) Has been cancelled
CI / build-libc-malloc (push) Has been cancelled
CI / build-almalinux8-jemalloc (push) Has been cancelled
CI / format-yaml (push) Has been cancelled
Clang Format Check / clang-format-check (push) Has been cancelled
Codecov / code-coverage (push) Has been cancelled
External Server Tests / test-external-standalone (push) Has been cancelled
External Server Tests / test-external-cluster (push) Has been cancelled
External Server Tests / test-external-nodebug (push) Has been cancelled
Spellcheck / Spellcheck (push) Has been cancelled
CodeQL / Analyze (cpp) (push) Has been cancelled
This commit is contained in:
parent
fe044a4d63
commit
41464c0cad
@ -1,342 +0,0 @@
|
|||||||
/* This file implements a new module native data type called "HELLOTYPE".
|
|
||||||
* The data structure implemented is a very simple ordered linked list of
|
|
||||||
* 64 bit integers, in order to have something that is real world enough, but
|
|
||||||
* at the same time, extremely simple to understand, to show how the API
|
|
||||||
* works, how a new data type is created, and how to write basic methods
|
|
||||||
* for RDB loading, saving and AOF rewriting.
|
|
||||||
*
|
|
||||||
* -----------------------------------------------------------------------------
|
|
||||||
*
|
|
||||||
* Copyright (c) 2016, Redis Ltd.
|
|
||||||
* All rights reserved.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are met:
|
|
||||||
*
|
|
||||||
* * Redistributions of source code must retain the above copyright notice,
|
|
||||||
* this list of conditions and the following disclaimer.
|
|
||||||
* * Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the distribution.
|
|
||||||
* * Neither the name of Redis nor the names of its contributors may be used
|
|
||||||
* to endorse or promote products derived from this software without
|
|
||||||
* specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
||||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
||||||
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
||||||
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
||||||
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
||||||
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
||||||
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
||||||
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
||||||
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
||||||
* POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "../valkeymodule.h"
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <ctype.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
static ValkeyModuleType *HelloType;
|
|
||||||
|
|
||||||
/* ========================== Internal data structure =======================
|
|
||||||
* This is just a linked list of 64 bit integers where elements are inserted
|
|
||||||
* in-place, so it's ordered. There is no pop/push operation but just insert
|
|
||||||
* because it is enough to show the implementation of new data types without
|
|
||||||
* making things complex. */
|
|
||||||
|
|
||||||
struct HelloTypeNode {
|
|
||||||
int64_t value;
|
|
||||||
struct HelloTypeNode *next;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct HelloTypeObject {
|
|
||||||
struct HelloTypeNode *head;
|
|
||||||
size_t len; /* Number of elements added. */
|
|
||||||
};
|
|
||||||
|
|
||||||
struct HelloTypeObject *createHelloTypeObject(void) {
|
|
||||||
struct HelloTypeObject *o;
|
|
||||||
o = ValkeyModule_Alloc(sizeof(*o));
|
|
||||||
o->head = NULL;
|
|
||||||
o->len = 0;
|
|
||||||
return o;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeInsert(struct HelloTypeObject *o, int64_t ele) {
|
|
||||||
struct HelloTypeNode *next = o->head, *newnode, *prev = NULL;
|
|
||||||
|
|
||||||
while (next && next->value < ele) {
|
|
||||||
prev = next;
|
|
||||||
next = next->next;
|
|
||||||
}
|
|
||||||
newnode = ValkeyModule_Alloc(sizeof(*newnode));
|
|
||||||
newnode->value = ele;
|
|
||||||
newnode->next = next;
|
|
||||||
if (prev) {
|
|
||||||
prev->next = newnode;
|
|
||||||
} else {
|
|
||||||
o->head = newnode;
|
|
||||||
}
|
|
||||||
o->len++;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeReleaseObject(struct HelloTypeObject *o) {
|
|
||||||
struct HelloTypeNode *cur, *next;
|
|
||||||
cur = o->head;
|
|
||||||
while (cur) {
|
|
||||||
next = cur->next;
|
|
||||||
ValkeyModule_Free(cur);
|
|
||||||
cur = next;
|
|
||||||
}
|
|
||||||
ValkeyModule_Free(o);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ========================= "hellotype" type commands ======================= */
|
|
||||||
|
|
||||||
/* HELLOTYPE.INSERT key value */
|
|
||||||
int HelloTypeInsert_ValkeyCommand(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
ValkeyModule_AutoMemory(ctx); /* Use automatic memory management. */
|
|
||||||
|
|
||||||
if (argc != 3) return ValkeyModule_WrongArity(ctx);
|
|
||||||
ValkeyModuleKey *key = ValkeyModule_OpenKey(ctx, argv[1], VALKEYMODULE_READ | VALKEYMODULE_WRITE);
|
|
||||||
int type = ValkeyModule_KeyType(key);
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_EMPTY && ValkeyModule_ModuleTypeGetType(key) != HelloType) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, VALKEYMODULE_ERRORMSG_WRONGTYPE);
|
|
||||||
}
|
|
||||||
|
|
||||||
long long value;
|
|
||||||
if ((ValkeyModule_StringToLongLong(argv[2], &value) != VALKEYMODULE_OK)) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, "ERR invalid value: must be a signed 64 bit integer");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Create an empty value object if the key is currently empty. */
|
|
||||||
struct HelloTypeObject *hto;
|
|
||||||
if (type == VALKEYMODULE_KEYTYPE_EMPTY) {
|
|
||||||
hto = createHelloTypeObject();
|
|
||||||
ValkeyModule_ModuleTypeSetValue(key, HelloType, hto);
|
|
||||||
} else {
|
|
||||||
hto = ValkeyModule_ModuleTypeGetValue(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Insert the new element. */
|
|
||||||
HelloTypeInsert(hto, value);
|
|
||||||
ValkeyModule_SignalKeyAsReady(ctx, argv[1]);
|
|
||||||
|
|
||||||
ValkeyModule_ReplyWithLongLong(ctx, hto->len);
|
|
||||||
ValkeyModule_ReplicateVerbatim(ctx);
|
|
||||||
return VALKEYMODULE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* HELLOTYPE.RANGE key first count */
|
|
||||||
int HelloTypeRange_ValkeyCommand(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
ValkeyModule_AutoMemory(ctx); /* Use automatic memory management. */
|
|
||||||
|
|
||||||
if (argc != 4) return ValkeyModule_WrongArity(ctx);
|
|
||||||
ValkeyModuleKey *key = ValkeyModule_OpenKey(ctx, argv[1], VALKEYMODULE_READ | VALKEYMODULE_WRITE);
|
|
||||||
int type = ValkeyModule_KeyType(key);
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_EMPTY && ValkeyModule_ModuleTypeGetType(key) != HelloType) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, VALKEYMODULE_ERRORMSG_WRONGTYPE);
|
|
||||||
}
|
|
||||||
|
|
||||||
long long first, count;
|
|
||||||
if (ValkeyModule_StringToLongLong(argv[2], &first) != VALKEYMODULE_OK ||
|
|
||||||
ValkeyModule_StringToLongLong(argv[3], &count) != VALKEYMODULE_OK || first < 0 || count < 0) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, "ERR invalid first or count parameters");
|
|
||||||
}
|
|
||||||
|
|
||||||
struct HelloTypeObject *hto = ValkeyModule_ModuleTypeGetValue(key);
|
|
||||||
struct HelloTypeNode *node = hto ? hto->head : NULL;
|
|
||||||
ValkeyModule_ReplyWithArray(ctx, VALKEYMODULE_POSTPONED_LEN);
|
|
||||||
long long arraylen = 0;
|
|
||||||
while (node && count--) {
|
|
||||||
ValkeyModule_ReplyWithLongLong(ctx, node->value);
|
|
||||||
arraylen++;
|
|
||||||
node = node->next;
|
|
||||||
}
|
|
||||||
ValkeyModule_ReplySetArrayLength(ctx, arraylen);
|
|
||||||
return VALKEYMODULE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* HELLOTYPE.LEN key */
|
|
||||||
int HelloTypeLen_ValkeyCommand(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
ValkeyModule_AutoMemory(ctx); /* Use automatic memory management. */
|
|
||||||
|
|
||||||
if (argc != 2) return ValkeyModule_WrongArity(ctx);
|
|
||||||
ValkeyModuleKey *key = ValkeyModule_OpenKey(ctx, argv[1], VALKEYMODULE_READ | VALKEYMODULE_WRITE);
|
|
||||||
int type = ValkeyModule_KeyType(key);
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_EMPTY && ValkeyModule_ModuleTypeGetType(key) != HelloType) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, VALKEYMODULE_ERRORMSG_WRONGTYPE);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct HelloTypeObject *hto = ValkeyModule_ModuleTypeGetValue(key);
|
|
||||||
ValkeyModule_ReplyWithLongLong(ctx, hto ? hto->len : 0);
|
|
||||||
return VALKEYMODULE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ====================== Example of a blocking command ==================== */
|
|
||||||
|
|
||||||
/* Reply callback for blocking command HELLOTYPE.BRANGE, this will get
|
|
||||||
* called when the key we blocked for is ready: we need to check if we
|
|
||||||
* can really serve the client, and reply OK or ERR accordingly. */
|
|
||||||
int HelloBlock_Reply(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
VALKEYMODULE_NOT_USED(argv);
|
|
||||||
VALKEYMODULE_NOT_USED(argc);
|
|
||||||
|
|
||||||
ValkeyModuleString *keyname = ValkeyModule_GetBlockedClientReadyKey(ctx);
|
|
||||||
ValkeyModuleKey *key = ValkeyModule_OpenKey(ctx, keyname, VALKEYMODULE_READ);
|
|
||||||
int type = ValkeyModule_KeyType(key);
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_MODULE || ValkeyModule_ModuleTypeGetType(key) != HelloType) {
|
|
||||||
ValkeyModule_CloseKey(key);
|
|
||||||
return VALKEYMODULE_ERR;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* In case the key is able to serve our blocked client, let's directly
|
|
||||||
* use our original command implementation to make this example simpler. */
|
|
||||||
ValkeyModule_CloseKey(key);
|
|
||||||
return HelloTypeRange_ValkeyCommand(ctx, argv, argc - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Timeout callback for blocking command HELLOTYPE.BRANGE */
|
|
||||||
int HelloBlock_Timeout(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
VALKEYMODULE_NOT_USED(argv);
|
|
||||||
VALKEYMODULE_NOT_USED(argc);
|
|
||||||
return ValkeyModule_ReplyWithSimpleString(ctx, "Request timedout");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Private data freeing callback for HELLOTYPE.BRANGE command. */
|
|
||||||
void HelloBlock_FreeData(ValkeyModuleCtx *ctx, void *privdata) {
|
|
||||||
VALKEYMODULE_NOT_USED(ctx);
|
|
||||||
ValkeyModule_Free(privdata);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* HELLOTYPE.BRANGE key first count timeout -- This is a blocking version of
|
|
||||||
* the RANGE operation, in order to show how to use the API
|
|
||||||
* ValkeyModule_BlockClientOnKeys(). */
|
|
||||||
int HelloTypeBRange_ValkeyCommand(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
if (argc != 5) return ValkeyModule_WrongArity(ctx);
|
|
||||||
ValkeyModule_AutoMemory(ctx); /* Use automatic memory management. */
|
|
||||||
ValkeyModuleKey *key = ValkeyModule_OpenKey(ctx, argv[1], VALKEYMODULE_READ | VALKEYMODULE_WRITE);
|
|
||||||
int type = ValkeyModule_KeyType(key);
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_EMPTY && ValkeyModule_ModuleTypeGetType(key) != HelloType) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, VALKEYMODULE_ERRORMSG_WRONGTYPE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Parse the timeout before even trying to serve the client synchronously,
|
|
||||||
* so that we always fail ASAP on syntax errors. */
|
|
||||||
long long timeout;
|
|
||||||
if (ValkeyModule_StringToLongLong(argv[4], &timeout) != VALKEYMODULE_OK) {
|
|
||||||
return ValkeyModule_ReplyWithError(ctx, "ERR invalid timeout parameter");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Can we serve the reply synchronously? */
|
|
||||||
if (type != VALKEYMODULE_KEYTYPE_EMPTY) {
|
|
||||||
return HelloTypeRange_ValkeyCommand(ctx, argv, argc - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Otherwise let's block on the key. */
|
|
||||||
void *privdata = ValkeyModule_Alloc(100);
|
|
||||||
ValkeyModule_BlockClientOnKeys(ctx, HelloBlock_Reply, HelloBlock_Timeout, HelloBlock_FreeData, timeout, argv + 1, 1,
|
|
||||||
privdata);
|
|
||||||
return VALKEYMODULE_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ========================== "hellotype" type methods ======================= */
|
|
||||||
|
|
||||||
void *HelloTypeRdbLoad(ValkeyModuleIO *rdb, int encver) {
|
|
||||||
if (encver != 0) {
|
|
||||||
/* ValkeyModule_Log("warning","Can't load data with version %d", encver);*/
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
uint64_t elements = ValkeyModule_LoadUnsigned(rdb);
|
|
||||||
struct HelloTypeObject *hto = createHelloTypeObject();
|
|
||||||
while (elements--) {
|
|
||||||
int64_t ele = ValkeyModule_LoadSigned(rdb);
|
|
||||||
HelloTypeInsert(hto, ele);
|
|
||||||
}
|
|
||||||
return hto;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeRdbSave(ValkeyModuleIO *rdb, void *value) {
|
|
||||||
struct HelloTypeObject *hto = value;
|
|
||||||
struct HelloTypeNode *node = hto->head;
|
|
||||||
ValkeyModule_SaveUnsigned(rdb, hto->len);
|
|
||||||
while (node) {
|
|
||||||
ValkeyModule_SaveSigned(rdb, node->value);
|
|
||||||
node = node->next;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeAofRewrite(ValkeyModuleIO *aof, ValkeyModuleString *key, void *value) {
|
|
||||||
struct HelloTypeObject *hto = value;
|
|
||||||
struct HelloTypeNode *node = hto->head;
|
|
||||||
while (node) {
|
|
||||||
ValkeyModule_EmitAOF(aof, "HELLOTYPE.INSERT", "sl", key, node->value);
|
|
||||||
node = node->next;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The goal of this function is to return the amount of memory used by
|
|
||||||
* the HelloType value. */
|
|
||||||
size_t HelloTypeMemUsage(const void *value) {
|
|
||||||
const struct HelloTypeObject *hto = value;
|
|
||||||
struct HelloTypeNode *node = hto->head;
|
|
||||||
return sizeof(*hto) + sizeof(*node) * hto->len;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeFree(void *value) {
|
|
||||||
HelloTypeReleaseObject(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
void HelloTypeDigest(ValkeyModuleDigest *md, void *value) {
|
|
||||||
struct HelloTypeObject *hto = value;
|
|
||||||
struct HelloTypeNode *node = hto->head;
|
|
||||||
while (node) {
|
|
||||||
ValkeyModule_DigestAddLongLong(md, node->value);
|
|
||||||
node = node->next;
|
|
||||||
}
|
|
||||||
ValkeyModule_DigestEndSequence(md);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* This function must be present on each module. It is used in order to
|
|
||||||
* register the commands into the server. */
|
|
||||||
int ValkeyModule_OnLoad(ValkeyModuleCtx *ctx, ValkeyModuleString **argv, int argc) {
|
|
||||||
VALKEYMODULE_NOT_USED(argv);
|
|
||||||
VALKEYMODULE_NOT_USED(argc);
|
|
||||||
|
|
||||||
if (ValkeyModule_Init(ctx, "hellotype", 1, VALKEYMODULE_APIVER_1) == VALKEYMODULE_ERR) return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
ValkeyModuleTypeMethods tm = {.version = VALKEYMODULE_TYPE_METHOD_VERSION,
|
|
||||||
.rdb_load = HelloTypeRdbLoad,
|
|
||||||
.rdb_save = HelloTypeRdbSave,
|
|
||||||
.aof_rewrite = HelloTypeAofRewrite,
|
|
||||||
.mem_usage = HelloTypeMemUsage,
|
|
||||||
.free = HelloTypeFree,
|
|
||||||
.digest = HelloTypeDigest};
|
|
||||||
|
|
||||||
HelloType = ValkeyModule_CreateDataType(ctx, "hellotype", 0, &tm);
|
|
||||||
if (HelloType == NULL) return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
if (ValkeyModule_CreateCommand(ctx, "hellotype.insert", HelloTypeInsert_ValkeyCommand, "write deny-oom", 1, 1, 1) ==
|
|
||||||
VALKEYMODULE_ERR)
|
|
||||||
return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
if (ValkeyModule_CreateCommand(ctx, "hellotype.range", HelloTypeRange_ValkeyCommand, "readonly", 1, 1, 1) ==
|
|
||||||
VALKEYMODULE_ERR)
|
|
||||||
return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
if (ValkeyModule_CreateCommand(ctx, "hellotype.len", HelloTypeLen_ValkeyCommand, "readonly", 1, 1, 1) ==
|
|
||||||
VALKEYMODULE_ERR)
|
|
||||||
return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
if (ValkeyModule_CreateCommand(ctx, "hellotype.brange", HelloTypeBRange_ValkeyCommand, "readonly", 1, 1, 1) ==
|
|
||||||
VALKEYMODULE_ERR)
|
|
||||||
return VALKEYMODULE_ERR;
|
|
||||||
|
|
||||||
return VALKEYMODULE_OK;
|
|
||||||
}
|
|
Loading…
x
Reference in New Issue
Block a user