From 5a38356cc4133c581f2935b38e12abb2a2037419 Mon Sep 17 00:00:00 2001 From: Pieter Noordhuis Date: Sat, 25 Sep 2010 22:34:22 +0200 Subject: Add example for non-blocking usage of hiredis with libevent --- .gitignore | 1 + Makefile | 3 ++ libevent-example.c | 97 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 101 insertions(+) create mode 100644 libevent-example.c diff --git a/.gitignore b/.gitignore index 0789e9c..30e67d9 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,6 @@ /hiredis-test /hiredis-example +/libevent-example /*.o /*.so /*.dylib diff --git a/Makefile b/Makefile index f9103a5..486f897 100644 --- a/Makefile +++ b/Makefile @@ -58,6 +58,9 @@ hiredis-%: %.o ${DYLIBNAME} test: hiredis-test ./hiredis-test +libevent-example: libevent-example.c ${DYLIBNAME} + $(CC) -o $@ $(CCOPT) $(DEBUG) -L. -lhiredis -levent libevent-example.c + .c.o: $(CC) -c $(CFLAGS) $(DEBUG) $(COMPILE_TIME) $< diff --git a/libevent-example.c b/libevent-example.c new file mode 100644 index 0000000..1f49d29 --- /dev/null +++ b/libevent-example.c @@ -0,0 +1,97 @@ +#include +#include +#include +#include +#include "hiredis.h" + +#define NOT_USED(x) ((void)x) + +/* This struct enables us to pass both the event and the + * redisContext to the read and write handlers. */ +typedef struct redisEvents { + redisContext *context; + struct event read; + struct event write; +} redisEvents; + +void redisLibEventRead(int fd, short event, void *arg) { + NOT_USED(fd); NOT_USED(event); + redisEvents *re = arg; + + /* Always re-schedule read events */ + event_add(&re->read,NULL); + + if (redisBufferRead(re->context) == REDIS_ERR) { + /* Handle error. */ + printf("Read error: %s\n", re->context->error); + } else { + /* Check replies. */ + redisProcessCallbacks(re->context); + } +} + +void redisLibEventWrite(int fd, short event, void *arg) { + NOT_USED(fd); NOT_USED(event); + redisEvents *re = arg; + int done = 0; + + if (redisBufferWrite(re->context, &done) == REDIS_ERR) { + /* Handle error */ + printf("Write error: %s\n", re->context->error); + } else { + /* Schedule write event again when writing is not done. */ + if (!done) { + event_add(&re->write,NULL); + } else { + event_add(&re->read,NULL); + } + } +} + +/* Schedule to be notified on a write event, so the outgoing buffer + * can be flushed to the socket. */ +void redisLibEventOnWrite(redisContext *c, void *privdata) { + NOT_USED(c); + redisEvents *e = privdata; + event_add(&e->write,NULL); +} + +/* Free the redisEvents struct when the context is free'd. */ +void redisLibEventOnFree(redisContext *c, void *privdata) { + NOT_USED(c); + redisEvents *e = privdata; + free(e); +} + +redisContext *redisLibEventConnect(const char *ip, int port) { + redisEvents *e = malloc(sizeof(*e)); + e->context = redisConnectNonBlock(ip, port, NULL); + redisSetCommandCallback(e->context, redisLibEventOnWrite, e); + redisSetFreeCallback(e->context, redisLibEventOnFree, e); + event_set(&e->read, e->context->fd, EV_READ, redisLibEventRead, e); + event_set(&e->write, e->context->fd, EV_WRITE, redisLibEventWrite, e); + return e->context; +} + +void getCallback(redisContext *c, redisReply *reply, void *privdata) { + NOT_USED(c); NOT_USED(privdata); + printf("argv[end-1]: %s\n", reply->reply); + redisFree(c); + exit(0); +} + +int main (int argc, char **argv) { + signal(SIGPIPE, SIG_IGN); + event_init(); + + redisContext *c = redisLibEventConnect("127.0.0.1", 6379); + if (c->error != NULL) { + printf("Connect error: %s\n", c->error); + return 1; + } + + redisCommand(c, "SET key %b", argv[argc-1], strlen(argv[argc-1])); + redisCommandWithCallback(c, getCallback, NULL, "GET key"); + event_dispatch(); + return 0; +} -- cgit v1.2.3