Skip to content

Commit 68b29e1

Browse files
Add timeout support to libhv adapter. (#1109)
Add timeout support to libhv adapter. See: #904
1 parent 722e340 commit 68b29e1

File tree

2 files changed

+81
-12
lines changed

2 files changed

+81
-12
lines changed

adapters/libhv.h

+66-12
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,11 @@
55
#include "../hiredis.h"
66
#include "../async.h"
77

8+
typedef struct redisLibhvEvents {
9+
hio_t *io;
10+
htimer_t *timer;
11+
} redisLibhvEvents;
12+
813
static void redisLibhvHandleEvents(hio_t* io) {
914
redisAsyncContext* context = (redisAsyncContext*)hevent_userdata(io);
1015
int events = hio_events(io);
@@ -18,51 +23,100 @@ static void redisLibhvHandleEvents(hio_t* io) {
1823
}
1924

2025
static void redisLibhvAddRead(void *privdata) {
21-
hio_t* io = (hio_t*)privdata;
22-
hio_add(io, redisLibhvHandleEvents, HV_READ);
26+
redisLibhvEvents* events = (redisLibhvEvents*)privdata;
27+
hio_add(events->io, redisLibhvHandleEvents, HV_READ);
2328
}
2429

2530
static void redisLibhvDelRead(void *privdata) {
26-
hio_t* io = (hio_t*)privdata;
27-
hio_del(io, HV_READ);
31+
redisLibhvEvents* events = (redisLibhvEvents*)privdata;
32+
hio_del(events->io, HV_READ);
2833
}
2934

3035
static void redisLibhvAddWrite(void *privdata) {
31-
hio_t* io = (hio_t*)privdata;
32-
hio_add(io, redisLibhvHandleEvents, HV_WRITE);
36+
redisLibhvEvents* events = (redisLibhvEvents*)privdata;
37+
hio_add(events->io, redisLibhvHandleEvents, HV_WRITE);
3338
}
3439

3540
static void redisLibhvDelWrite(void *privdata) {
36-
hio_t* io = (hio_t*)privdata;
37-
hio_del(io, HV_WRITE);
41+
redisLibhvEvents* events = (redisLibhvEvents*)privdata;
42+
hio_del(events->io, HV_WRITE);
3843
}
3944

4045
static void redisLibhvCleanup(void *privdata) {
41-
hio_t* io = (hio_t*)privdata;
42-
hio_close(io);
43-
hevent_set_userdata(io, NULL);
46+
redisLibhvEvents* events = (redisLibhvEvents*)privdata;
47+
48+
if (events->timer)
49+
htimer_del(events->timer);
50+
51+
hio_close(events->io);
52+
hevent_set_userdata(events->io, NULL);
53+
54+
hi_free(events);
55+
}
56+
57+
static void redisLibhvTimeout(htimer_t* timer) {
58+
hio_t* io = (hio_t*)hevent_userdata(timer);
59+
redisAsyncHandleTimeout(hevent_userdata(io));
60+
}
61+
62+
static void redisLibhvSetTimeout(void *privdata, struct timeval tv) {
63+
redisLibhvEvents* events;
64+
uint32_t millis;
65+
hloop_t* loop;
66+
67+
events = (redisLibhvEvents*)privdata;
68+
millis = tv.tv_sec * 1000 + tv.tv_usec / 1000;
69+
70+
if (millis == 0) {
71+
/* Libhv disallows zero'd timers so treat this as a delete or NO OP */
72+
if (events->timer) {
73+
htimer_del(events->timer);
74+
events->timer = NULL;
75+
}
76+
} else if (events->timer == NULL) {
77+
/* Add new timer */
78+
loop = hevent_loop(events->io);
79+
events->timer = htimer_add(loop, redisLibhvTimeout, millis, 1);
80+
hevent_set_userdata(events->timer, events->io);
81+
} else {
82+
/* Update existing timer */
83+
htimer_reset(events->timer, millis);
84+
}
4485
}
4586

4687
static int redisLibhvAttach(redisAsyncContext* ac, hloop_t* loop) {
4788
redisContext *c = &(ac->c);
89+
redisLibhvEvents *events;
4890
hio_t* io = NULL;
4991

5092
if (ac->ev.data != NULL) {
5193
return REDIS_ERR;
5294
}
5395

96+
/* Create container struct to keep track of our io and any timer */
97+
events = hi_malloc(sizeof(*events));
98+
if (events == NULL) {
99+
return REDIS_ERR;
100+
}
101+
54102
io = hio_get(loop, c->fd);
55103
if (io == NULL) {
104+
hi_free(events);
56105
return REDIS_ERR;
57106
}
107+
58108
hevent_set_userdata(io, ac);
59109

110+
events->io = io;
111+
events->timer = NULL;
112+
60113
ac->ev.addRead = redisLibhvAddRead;
61114
ac->ev.delRead = redisLibhvDelRead;
62115
ac->ev.addWrite = redisLibhvAddWrite;
63116
ac->ev.delWrite = redisLibhvDelWrite;
64117
ac->ev.cleanup = redisLibhvCleanup;
65-
ac->ev.data = io;
118+
ac->ev.scheduleTimer = redisLibhvSetTimeout;
119+
ac->ev.data = events;
66120

67121
return REDIS_OK;
68122
}

examples/example-libhv.c

+15
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,18 @@ void getCallback(redisAsyncContext *c, void *r, void *privdata) {
1616
redisAsyncDisconnect(c);
1717
}
1818

19+
void debugCallback(redisAsyncContext *c, void *r, void *privdata) {
20+
(void)privdata;
21+
redisReply *reply = r;
22+
23+
if (reply == NULL) {
24+
printf("`DEBUG SLEEP` error: %s\n", c->errstr ? c->errstr : "unknown error");
25+
return;
26+
}
27+
28+
redisAsyncDisconnect(c);
29+
}
30+
1931
void connectCallback(const redisAsyncContext *c, int status) {
2032
if (status != REDIS_OK) {
2133
printf("Error: %s\n", c->errstr);
@@ -46,10 +58,13 @@ int main (int argc, char **argv) {
4658

4759
hloop_t* loop = hloop_new(HLOOP_FLAG_QUIT_WHEN_NO_ACTIVE_EVENTS);
4860
redisLibhvAttach(c, loop);
61+
redisAsyncSetTimeout(c, (struct timeval){.tv_sec = 0, .tv_usec = 500000});
4962
redisAsyncSetConnectCallback(c,connectCallback);
5063
redisAsyncSetDisconnectCallback(c,disconnectCallback);
5164
redisAsyncCommand(c, NULL, NULL, "SET key %b", argv[argc-1], strlen(argv[argc-1]));
5265
redisAsyncCommand(c, getCallback, (char*)"end-1", "GET key");
66+
redisAsyncCommand(c, debugCallback, NULL, "DEBUG SLEEP %d", 1);
5367
hloop_run(loop);
68+
hloop_free(&loop);
5469
return 0;
5570
}

0 commit comments

Comments
 (0)