From 8d5bc445e53ad4114aaac1dd4ff438b638aecede Mon Sep 17 00:00:00 2001 From: "Allen.Dou" Date: Fri, 19 Apr 2013 11:26:43 +0800 Subject: SetKeepAlive Keep client alive even though no command was sent to server for a long time. --- hiredis.h | 2 ++ net.c | 34 ++++++++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+) diff --git a/hiredis.h b/hiredis.h index aadcf35..a861761 100644 --- a/hiredis.h +++ b/hiredis.h @@ -88,6 +88,8 @@ #define REDIS_READER_MAX_BUF (1024*16) /* Default max unused reader buffer. */ +#define REDIS_KEEPALIVE_INTERVAL 15 /* seconds */ + #ifdef __cplusplus extern "C" { #endif diff --git a/net.c b/net.c index a495715..57557cd 100644 --- a/net.c +++ b/net.c @@ -113,6 +113,38 @@ static int redisSetBlocking(redisContext *c, int fd, int blocking) { return REDIS_OK; } +static int redisKeepAlive(redisContext *c,int fd,int interval) { + int val = 1; + + if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &val, sizeof(val)) == -1){ + __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); + return REDIS_ERR; + } + + val = interval; + if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE, &val, sizeof(val)) < 0) { + __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); + return REDIS_ERR; + } + + val = interval/3; + if (val == 0) val = 1; + if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL, &val, sizeof(val)) < 0) { + __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); + return REDIS_ERR; + } + + val = 3; + if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT, &val, sizeof(val)) < 0) { + __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); + return REDIS_ERR; + } + + return REDIS_OK; +} + + + static int redisSetTcpNoDelay(redisContext *c, int fd) { int yes = 1; if (setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &yes, sizeof(yes)) == -1) { @@ -241,6 +273,8 @@ int redisContextConnectTcp(redisContext *c, const char *addr, int port, struct t goto error; if (redisSetTcpNoDelay(c,s) != REDIS_OK) goto error; + if (redisKeepAlive(c,s,REDIS_KEEPALIVE_INTERVAL) != REDIS_OK) + goto error; c->fd = s; c->flags |= REDIS_CONNECTED; -- cgit v1.2.3 From 9dff5105fc01f52437d862bc95d40415f03179f0 Mon Sep 17 00:00:00 2001 From: "Allen.Dou" Date: Tue, 30 Apr 2013 00:11:57 +0800 Subject: Make KeepAlive optional Make Connection KeepAlive being optional instead of default. --- hiredis.c | 7 +++++++ hiredis.h | 1 + net.c | 9 +++------ net.h | 1 + 4 files changed, 12 insertions(+), 6 deletions(-) diff --git a/hiredis.c b/hiredis.c index 958f58b..84baa8c 100644 --- a/hiredis.c +++ b/hiredis.c @@ -1092,6 +1092,13 @@ int redisSetTimeout(redisContext *c, struct timeval tv) { return REDIS_ERR; } +/* Enable connection KeepAlive. */ +int redisEnableKeepAlive(redisContext *c) { + if (redisKeepAlive(c, REDIS_KEEPALIVE_INTERVAL) != REDIS_OK) + return REDIS_ERR; + return REDIS_OK; +} + /* Use this function to handle a read event on the descriptor. It will try * and read some bytes from the socket and feed them to the reply parser. * diff --git a/hiredis.h b/hiredis.h index a861761..42290a1 100644 --- a/hiredis.h +++ b/hiredis.h @@ -179,6 +179,7 @@ redisContext *redisConnectUnix(const char *path); redisContext *redisConnectUnixWithTimeout(const char *path, struct timeval tv); redisContext *redisConnectUnixNonBlock(const char *path); int redisSetTimeout(redisContext *c, struct timeval tv); +int redisEnableKeepAlive(redisContext *c); void redisFree(redisContext *c); int redisBufferRead(redisContext *c); int redisBufferWrite(redisContext *c, int *done); diff --git a/net.c b/net.c index 57557cd..bf571fc 100644 --- a/net.c +++ b/net.c @@ -113,9 +113,10 @@ static int redisSetBlocking(redisContext *c, int fd, int blocking) { return REDIS_OK; } -static int redisKeepAlive(redisContext *c,int fd,int interval) { +int redisKeepAlive(redisContext *c, int interval) { int val = 1; - + int fd = c->fd; + if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &val, sizeof(val)) == -1){ __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); return REDIS_ERR; @@ -143,8 +144,6 @@ static int redisKeepAlive(redisContext *c,int fd,int interval) { return REDIS_OK; } - - static int redisSetTcpNoDelay(redisContext *c, int fd) { int yes = 1; if (setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &yes, sizeof(yes)) == -1) { @@ -273,8 +272,6 @@ int redisContextConnectTcp(redisContext *c, const char *addr, int port, struct t goto error; if (redisSetTcpNoDelay(c,s) != REDIS_OK) goto error; - if (redisKeepAlive(c,s,REDIS_KEEPALIVE_INTERVAL) != REDIS_OK) - goto error; c->fd = s; c->flags |= REDIS_CONNECTED; diff --git a/net.h b/net.h index eb8a0a1..36096fe 100644 --- a/net.h +++ b/net.h @@ -43,5 +43,6 @@ int redisCheckSocketError(redisContext *c, int fd); int redisContextSetTimeout(redisContext *c, struct timeval tv); int redisContextConnectTcp(redisContext *c, const char *addr, int port, struct timeval *timeout); int redisContextConnectUnix(redisContext *c, const char *path, struct timeval *timeout); +int redisKeepAlive(redisContext *c, int interval); #endif -- cgit v1.2.3 From bb3c4c17c45c89d980e2f4f59f59e55d13c77334 Mon Sep 17 00:00:00 2001 From: Pieter Noordhuis Date: Wed, 1 May 2013 09:23:06 -0700 Subject: Make redisKeepAlive work on OSX --- fmacros.h | 4 ++++ net.c | 8 ++++++++ 2 files changed, 12 insertions(+) diff --git a/fmacros.h b/fmacros.h index 21cd9cf..799c12c 100644 --- a/fmacros.h +++ b/fmacros.h @@ -13,4 +13,8 @@ #define _XOPEN_SOURCE #endif +#if __APPLE__ && __MACH__ +#define _OSX +#endif + #endif diff --git a/net.c b/net.c index bf571fc..3c07d68 100644 --- a/net.c +++ b/net.c @@ -122,6 +122,13 @@ int redisKeepAlive(redisContext *c, int interval) { return REDIS_ERR; } +#ifdef _OSX + val = interval; + if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPALIVE, &val, sizeof(val)) < 0) { + __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); + return REDIS_ERR; + } +#else val = interval; if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE, &val, sizeof(val)) < 0) { __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); @@ -140,6 +147,7 @@ int redisKeepAlive(redisContext *c, int interval) { __redisSetError(c,REDIS_ERR_OTHER,strerror(errno)); return REDIS_ERR; } +#endif return REDIS_OK; } -- cgit v1.2.3