summaryrefslogtreecommitdiff
path: root/extra/hiredis/libev.h
blob: 3246820e9b7099047e87266a570e710946a80df6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
#include <sys/types.h>
#include <ev.h>
#include <hiredis.h>

/* Prototype for the error callback. */
typedef void (redisErrorCallback)(const redisContext*);

typedef struct libevRedisEvents {
    redisContext *context;
    redisErrorCallback *err;
    struct ev_loop *loop;
    ev_io rev, wev;
} libevRedisEvents;

void libevRedisReadEvent(struct ev_loop *loop, ev_io *watcher, int revents) {
    ((void)loop); ((void)revents);
    libevRedisEvents *e = watcher->data;

    if (redisBufferRead(e->context) == REDIS_ERR) {
        redisDisconnect(e->context);
        e->err(e->context);
    } else {
        if (redisProcessCallbacks(e->context) == REDIS_ERR) {
            redisDisconnect(e->context);
            e->err(e->context);
        }
    }
}

void libevRedisWriteEvent(struct ev_loop *loop, ev_io *watcher, int revents) {
    ((void)loop); ((void)revents);
    libevRedisEvents *e = watcher->data;
    int done = 0;

    if (redisBufferWrite(e->context, &done) == REDIS_ERR) {
        redisDisconnect(e->context);
        e->err(e->context);
    } else {
        /* Stop firing the write event when done */
        if (done) {
            ev_io_stop(e->loop,&e->wev);
            ev_io_start(e->loop,&e->rev);
        }
    }
}

void libevRedisCommandCallback(redisContext *c, void *privdata) {
    ((void)c);
    libevRedisEvents *e = privdata;
    ev_io_start(e->loop,&e->wev);
}

void libevRedisDisconnectCallback(redisContext *c, void *privdata) {
    ((void)c);
    libevRedisEvents *e = privdata;
    ev_io_stop(e->loop,&e->rev);
    ev_io_stop(e->loop,&e->wev);
}

void libevRedisFreeCallback(redisContext *c, void *privdata) {
    ((void)c);
    libevRedisEvents *e = privdata;
    free(e);
}

redisContext *libevRedisConnect(struct ev_loop *loop, redisErrorCallback *err, const char *ip, int port) {
    libevRedisEvents *e;
    redisContext *c = redisConnectNonBlock(ip, port, NULL);
    if (c->error != NULL) {
        err(c);
        redisFree(c);
        return NULL;
    }

    /* Create container for context and r/w events */
    e = malloc(sizeof(*e));
    e->context = c;
    e->err = err;
    e->loop = loop;
    e->rev.data = e;
    e->wev.data = e;

    /* Register callbacks */
    redisSetDisconnectCallback(c,libevRedisDisconnectCallback,e);
    redisSetCommandCallback(c,libevRedisCommandCallback,e);
    redisSetFreeCallback(c,libevRedisFreeCallback,e);

    /* Initialize read/write events */
    ev_io_init(&e->rev,libevRedisReadEvent,c->fd,EV_READ);
    ev_io_init(&e->wev,libevRedisWriteEvent,c->fd,EV_WRITE);
    return c;
}