1*572c4311Sfengbojiang /* Timer API example -- Register and handle timer events
2*572c4311Sfengbojiang  *
3*572c4311Sfengbojiang  * -----------------------------------------------------------------------------
4*572c4311Sfengbojiang  *
5*572c4311Sfengbojiang  * Copyright (c) 2018, Salvatore Sanfilippo <antirez at gmail dot com>
6*572c4311Sfengbojiang  * All rights reserved.
7*572c4311Sfengbojiang  *
8*572c4311Sfengbojiang  * Redistribution and use in source and binary forms, with or without
9*572c4311Sfengbojiang  * modification, are permitted provided that the following conditions are met:
10*572c4311Sfengbojiang  *
11*572c4311Sfengbojiang  *   * Redistributions of source code must retain the above copyright notice,
12*572c4311Sfengbojiang  *     this list of conditions and the following disclaimer.
13*572c4311Sfengbojiang  *   * Redistributions in binary form must reproduce the above copyright
14*572c4311Sfengbojiang  *     notice, this list of conditions and the following disclaimer in the
15*572c4311Sfengbojiang  *     documentation and/or other materials provided with the distribution.
16*572c4311Sfengbojiang  *   * Neither the name of Redis nor the names of its contributors may be used
17*572c4311Sfengbojiang  *     to endorse or promote products derived from this software without
18*572c4311Sfengbojiang  *     specific prior written permission.
19*572c4311Sfengbojiang  *
20*572c4311Sfengbojiang  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21*572c4311Sfengbojiang  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22*572c4311Sfengbojiang  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23*572c4311Sfengbojiang  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
24*572c4311Sfengbojiang  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25*572c4311Sfengbojiang  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26*572c4311Sfengbojiang  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27*572c4311Sfengbojiang  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28*572c4311Sfengbojiang  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29*572c4311Sfengbojiang  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30*572c4311Sfengbojiang  * POSSIBILITY OF SUCH DAMAGE.
31*572c4311Sfengbojiang  */
32*572c4311Sfengbojiang 
33*572c4311Sfengbojiang #define REDISMODULE_EXPERIMENTAL_API
34*572c4311Sfengbojiang #include "../redismodule.h"
35*572c4311Sfengbojiang #include <stdio.h>
36*572c4311Sfengbojiang #include <stdlib.h>
37*572c4311Sfengbojiang #include <ctype.h>
38*572c4311Sfengbojiang #include <string.h>
39*572c4311Sfengbojiang 
40*572c4311Sfengbojiang /* Timer callback. */
timerHandler(RedisModuleCtx * ctx,void * data)41*572c4311Sfengbojiang void timerHandler(RedisModuleCtx *ctx, void *data) {
42*572c4311Sfengbojiang     REDISMODULE_NOT_USED(ctx);
43*572c4311Sfengbojiang     printf("Fired %s!\n", data);
44*572c4311Sfengbojiang     RedisModule_Free(data);
45*572c4311Sfengbojiang }
46*572c4311Sfengbojiang 
47*572c4311Sfengbojiang /* HELLOTIMER.TIMER*/
TimerCommand_RedisCommand(RedisModuleCtx * ctx,RedisModuleString ** argv,int argc)48*572c4311Sfengbojiang int TimerCommand_RedisCommand(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
49*572c4311Sfengbojiang     REDISMODULE_NOT_USED(argv);
50*572c4311Sfengbojiang     REDISMODULE_NOT_USED(argc);
51*572c4311Sfengbojiang 
52*572c4311Sfengbojiang     for (int j = 0; j < 10; j++) {
53*572c4311Sfengbojiang         int delay = rand() % 5000;
54*572c4311Sfengbojiang         char *buf = RedisModule_Alloc(256);
55*572c4311Sfengbojiang         snprintf(buf,256,"After %d", delay);
56*572c4311Sfengbojiang         RedisModuleTimerID tid = RedisModule_CreateTimer(ctx,delay,timerHandler,buf);
57*572c4311Sfengbojiang         REDISMODULE_NOT_USED(tid);
58*572c4311Sfengbojiang     }
59*572c4311Sfengbojiang     return RedisModule_ReplyWithSimpleString(ctx, "OK");
60*572c4311Sfengbojiang }
61*572c4311Sfengbojiang 
62*572c4311Sfengbojiang /* This function must be present on each Redis module. It is used in order to
63*572c4311Sfengbojiang  * register the commands into the Redis server. */
RedisModule_OnLoad(RedisModuleCtx * ctx,RedisModuleString ** argv,int argc)64*572c4311Sfengbojiang int RedisModule_OnLoad(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
65*572c4311Sfengbojiang     REDISMODULE_NOT_USED(argv);
66*572c4311Sfengbojiang     REDISMODULE_NOT_USED(argc);
67*572c4311Sfengbojiang 
68*572c4311Sfengbojiang     if (RedisModule_Init(ctx,"hellotimer",1,REDISMODULE_APIVER_1)
69*572c4311Sfengbojiang         == REDISMODULE_ERR) return REDISMODULE_ERR;
70*572c4311Sfengbojiang 
71*572c4311Sfengbojiang     if (RedisModule_CreateCommand(ctx,"hellotimer.timer",
72*572c4311Sfengbojiang         TimerCommand_RedisCommand,"readonly",0,0,0) == REDISMODULE_ERR)
73*572c4311Sfengbojiang         return REDISMODULE_ERR;
74*572c4311Sfengbojiang 
75*572c4311Sfengbojiang     return REDISMODULE_OK;
76*572c4311Sfengbojiang }
77