summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPieter Noordhuis <pcnoordhuis@gmail.com>2010-09-25 22:34:22 +0200
committerPieter Noordhuis <pcnoordhuis@gmail.com>2010-09-25 22:34:22 +0200
commit5a38356cc4133c581f2935b38e12abb2a2037419 (patch)
treed45f9c91205fb05e04d06003a04fb812da450625
parent1c245845eda46512ed3eb5f7e502c3b7f535ac89 (diff)
Add example for non-blocking usage of hiredis with libevent
-rw-r--r--.gitignore1
-rw-r--r--Makefile3
-rw-r--r--libevent-example.c97
3 files changed, 101 insertions, 0 deletions
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 <stdio.h>
+#include <stdlib.h>
+#include <event.h>
+#include <string.h>
+#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;
+}