diff options
author | Mark Nunberg <mnunberg@haskalah.org> | 2019-04-11 15:08:27 -0400 |
---|---|---|
committer | Mark Nunberg <mnunberg@haskalah.org> | 2019-04-11 15:08:27 -0400 |
commit | 5aa2397f9ec494380774fb9155446f14ebbf3d1f (patch) | |
tree | a032bb704b4c22db1aae60b494ed51680b1b3dc0 | |
parent | dc3c6ce85cf17e1da653a4a17bf3cca69ed02732 (diff) |
fix blocking timeouts on SSL reads/writes
-rw-r--r-- | sslio.c | 25 |
1 files changed, 24 insertions, 1 deletions
@@ -4,6 +4,7 @@ #include <assert.h> #ifdef HIREDIS_SSL #include <pthread.h> +#include <errno.h> void __redisSetError(redisContext *c, int type, const char *str); @@ -184,6 +185,28 @@ int redisSslRead(redisContext *c, char *buf, size_t bufcap) { return -1; } else { int err = SSL_get_error(c->ssl->ssl, nread); + if (c->flags & REDIS_BLOCK) { + /** + * In blocking mode, we should never end up in a situation where + * we get an error without it being an actual error, except + * in the case of EINTR, which can be spuriously received from + * debuggers or whatever. + */ + if (errno == EINTR) { + return 0; + } else { + const char *msg = NULL; + if (errno == EAGAIN) { + msg = "Timed out"; + } + __redisSetError(c, REDIS_ERR_IO, msg); + return -1; + } + } + + /** + * We can very well get an EWOULDBLOCK/EAGAIN, however + */ if (maybeCheckWant(c->ssl, err)) { return 0; } else { @@ -203,7 +226,7 @@ int redisSslWrite(redisContext *c) { c->ssl->lastLen = len; int err = SSL_get_error(c->ssl->ssl, rv); - if (maybeCheckWant(c->ssl, err)) { + if ((c->flags & REDIS_BLOCK) == 0 && maybeCheckWant(c->ssl, err)) { return 0; } else { __redisSetError(c, REDIS_ERR_IO, NULL); |