mirror of
https://github.com/fluencelabs/redis
synced 2025-04-02 15:51:05 +00:00
hiredis library updated.
This version of hiredis merges modifications of the Redis fork with latest changes in the hiredis repository. The same version was pushed on the hiredis repository and will probably merged into the master branch in short time.
This commit is contained in:
parent
cada7f9671
commit
d6704c9bd0
29
deps/hiredis/README.md
vendored
29
deps/hiredis/README.md
vendored
@ -73,7 +73,7 @@ convert it to the protocol used to communicate with Redis.
|
|||||||
One or more spaces separates arguments, so you can use the specifiers
|
One or more spaces separates arguments, so you can use the specifiers
|
||||||
anywhere in an argument:
|
anywhere in an argument:
|
||||||
|
|
||||||
reply = redisCommand("SET key:%s %s", myid, value);
|
reply = redisCommand(context, "SET key:%s %s", myid, value);
|
||||||
|
|
||||||
### Using replies
|
### Using replies
|
||||||
|
|
||||||
@ -320,6 +320,10 @@ The reply parsing API consists of the following functions:
|
|||||||
int redisReaderFeed(redisReader *reader, const char *buf, size_t len);
|
int redisReaderFeed(redisReader *reader, const char *buf, size_t len);
|
||||||
int redisReaderGetReply(redisReader *reader, void **reply);
|
int redisReaderGetReply(redisReader *reader, void **reply);
|
||||||
|
|
||||||
|
The same set of functions are used internally by hiredis when creating a
|
||||||
|
normal Redis context, the above API just exposes it to the user for a direct
|
||||||
|
usage.
|
||||||
|
|
||||||
### Usage
|
### Usage
|
||||||
|
|
||||||
The function `redisReaderCreate` creates a `redisReader` structure that holds a
|
The function `redisReaderCreate` creates a `redisReader` structure that holds a
|
||||||
@ -346,6 +350,29 @@ immediately after creating the `redisReader`.
|
|||||||
For example, [hiredis-rb](https://github.com/pietern/hiredis-rb/blob/master/ext/hiredis_ext/reader.c)
|
For example, [hiredis-rb](https://github.com/pietern/hiredis-rb/blob/master/ext/hiredis_ext/reader.c)
|
||||||
uses customized reply object functions to create Ruby objects.
|
uses customized reply object functions to create Ruby objects.
|
||||||
|
|
||||||
|
### Reader max buffer
|
||||||
|
|
||||||
|
Both when using the Reader API directly or when using it indirectly via a
|
||||||
|
normal Redis context, the redisReader structure uses a buffer in order to
|
||||||
|
accumulate data from the server.
|
||||||
|
Usually this buffer is destroyed when it is empty and is larger than 16
|
||||||
|
kb in order to avoid wasting memory in unused buffers
|
||||||
|
|
||||||
|
However when working with very big payloads destroying the buffer may slow
|
||||||
|
down performances considerably, so it is possible to modify the max size of
|
||||||
|
an idle buffer changing the value of the `maxbuf` field of the reader structure
|
||||||
|
to the desired value. The special value of 0 means that there is no maximum
|
||||||
|
value for an idle buffer, so the buffer will never get freed.
|
||||||
|
|
||||||
|
For instance if you have a normal Redis context you can set the maximum idle
|
||||||
|
buffer to zero (unlimited) just with:
|
||||||
|
|
||||||
|
context->reader->maxbuf = 0;
|
||||||
|
|
||||||
|
This should be done only in order to maximize performances when working with
|
||||||
|
large payloads. The context should be set back to `REDIS_READER_MAX_BUF` again
|
||||||
|
as soon as possible in order to prevent allocation of useless memory.
|
||||||
|
|
||||||
## AUTHORS
|
## AUTHORS
|
||||||
|
|
||||||
Hiredis was written by Salvatore Sanfilippo (antirez at gmail) and
|
Hiredis was written by Salvatore Sanfilippo (antirez at gmail) and
|
||||||
|
38
deps/hiredis/async.c
vendored
38
deps/hiredis/async.c
vendored
@ -373,6 +373,11 @@ void redisProcessCallbacks(redisAsyncContext *ac) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* If monitor mode, repush callback */
|
||||||
|
if(c->flags & REDIS_MONITORING) {
|
||||||
|
__redisPushCallback(&ac->replies,&cb);
|
||||||
|
}
|
||||||
|
|
||||||
/* When the connection is not being disconnected, simply stop
|
/* When the connection is not being disconnected, simply stop
|
||||||
* trying to get replies and wait for the next loop tick. */
|
* trying to get replies and wait for the next loop tick. */
|
||||||
break;
|
break;
|
||||||
@ -381,21 +386,30 @@ void redisProcessCallbacks(redisAsyncContext *ac) {
|
|||||||
/* Even if the context is subscribed, pending regular callbacks will
|
/* Even if the context is subscribed, pending regular callbacks will
|
||||||
* get a reply before pub/sub messages arrive. */
|
* get a reply before pub/sub messages arrive. */
|
||||||
if (__redisShiftCallback(&ac->replies,&cb) != REDIS_OK) {
|
if (__redisShiftCallback(&ac->replies,&cb) != REDIS_OK) {
|
||||||
/* A spontaneous reply in a not-subscribed context can only be the
|
/*
|
||||||
* error reply that is sent when a new connection exceeds the
|
* A spontaneous reply in a not-subscribed context can be the error
|
||||||
* maximum number of allowed connections on the server side. This
|
* reply that is sent when a new connection exceeds the maximum
|
||||||
* is seen as an error instead of a regular reply because the
|
* number of allowed connections on the server side.
|
||||||
* server closes the connection after sending it. To prevent the
|
*
|
||||||
* error from being overwritten by an EOF error the connection is
|
* This is seen as an error instead of a regular reply because the
|
||||||
* closed here. See issue #43. */
|
* server closes the connection after sending it.
|
||||||
if ( !(c->flags & REDIS_SUBSCRIBED) && ((redisReply*)reply)->type == REDIS_REPLY_ERROR ) {
|
*
|
||||||
|
* To prevent the error from being overwritten by an EOF error the
|
||||||
|
* connection is closed here. See issue #43.
|
||||||
|
*
|
||||||
|
* Another possibility is that the server is loading its dataset.
|
||||||
|
* In this case we also want to close the connection, and have the
|
||||||
|
* user wait until the server is ready to take our request.
|
||||||
|
*/
|
||||||
|
if (((redisReply*)reply)->type == REDIS_REPLY_ERROR) {
|
||||||
c->err = REDIS_ERR_OTHER;
|
c->err = REDIS_ERR_OTHER;
|
||||||
snprintf(c->errstr,sizeof(c->errstr),"%s",((redisReply*)reply)->str);
|
snprintf(c->errstr,sizeof(c->errstr),"%s",((redisReply*)reply)->str);
|
||||||
__redisAsyncDisconnect(ac);
|
__redisAsyncDisconnect(ac);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
/* No more regular callbacks and no errors, the context *must* be subscribed. */
|
/* No more regular callbacks and no errors, the context *must* be subscribed or monitoring. */
|
||||||
assert(c->flags & REDIS_SUBSCRIBED);
|
assert((c->flags & REDIS_SUBSCRIBED || c->flags & REDIS_MONITORING));
|
||||||
|
if(c->flags & REDIS_SUBSCRIBED)
|
||||||
__redisGetSubscribeCallback(ac,reply,&cb);
|
__redisGetSubscribeCallback(ac,reply,&cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -557,6 +571,10 @@ static int __redisAsyncCommand(redisAsyncContext *ac, redisCallbackFn *fn, void
|
|||||||
/* (P)UNSUBSCRIBE does not have its own response: every channel or
|
/* (P)UNSUBSCRIBE does not have its own response: every channel or
|
||||||
* pattern that is unsubscribed will receive a message. This means we
|
* pattern that is unsubscribed will receive a message. This means we
|
||||||
* should not append a callback function for this command. */
|
* should not append a callback function for this command. */
|
||||||
|
} else if(strncasecmp(cstr,"monitor\r\n",9) == 0) {
|
||||||
|
/* Set monitor flag and push callback */
|
||||||
|
c->flags |= REDIS_MONITORING;
|
||||||
|
__redisPushCallback(&ac->replies,&cb);
|
||||||
} else {
|
} else {
|
||||||
if (c->flags & REDIS_SUBSCRIBED)
|
if (c->flags & REDIS_SUBSCRIBED)
|
||||||
/* This will likely result in an error reply, but it needs to be
|
/* This will likely result in an error reply, but it needs to be
|
||||||
|
7
deps/hiredis/hiredis.c
vendored
7
deps/hiredis/hiredis.c
vendored
@ -446,7 +446,7 @@ static int processMultiBulkItem(redisReader *r) {
|
|||||||
long elements;
|
long elements;
|
||||||
int root = 0;
|
int root = 0;
|
||||||
|
|
||||||
/* Set error for nested multi bulks with depth > 2 */
|
/* Set error for nested multi bulks with depth > 7 */
|
||||||
if (r->ridx == 8) {
|
if (r->ridx == 8) {
|
||||||
__redisReaderSetError(r,REDIS_ERR_PROTOCOL,
|
__redisReaderSetError(r,REDIS_ERR_PROTOCOL,
|
||||||
"No support for nested multi bulk replies with depth > 7");
|
"No support for nested multi bulk replies with depth > 7");
|
||||||
@ -564,6 +564,7 @@ redisReader *redisReaderCreate(void) {
|
|||||||
r->errstr[0] = '\0';
|
r->errstr[0] = '\0';
|
||||||
r->fn = &defaultFunctions;
|
r->fn = &defaultFunctions;
|
||||||
r->buf = sdsempty();
|
r->buf = sdsempty();
|
||||||
|
r->maxbuf = REDIS_READER_MAX_BUF;
|
||||||
if (r->buf == NULL) {
|
if (r->buf == NULL) {
|
||||||
free(r);
|
free(r);
|
||||||
return NULL;
|
return NULL;
|
||||||
@ -590,9 +591,8 @@ int redisReaderFeed(redisReader *r, const char *buf, size_t len) {
|
|||||||
|
|
||||||
/* Copy the provided buffer. */
|
/* Copy the provided buffer. */
|
||||||
if (buf != NULL && len >= 1) {
|
if (buf != NULL && len >= 1) {
|
||||||
#if 0
|
|
||||||
/* Destroy internal buffer when it is empty and is quite large. */
|
/* Destroy internal buffer when it is empty and is quite large. */
|
||||||
if (r->len == 0 && sdsavail(r->buf) > 16*1024) {
|
if (r->len == 0 && r->maxbuf != 0 && sdsavail(r->buf) > r->maxbuf) {
|
||||||
sdsfree(r->buf);
|
sdsfree(r->buf);
|
||||||
r->buf = sdsempty();
|
r->buf = sdsempty();
|
||||||
r->pos = 0;
|
r->pos = 0;
|
||||||
@ -600,7 +600,6 @@ int redisReaderFeed(redisReader *r, const char *buf, size_t len) {
|
|||||||
/* r->buf should not be NULL since we just free'd a larger one. */
|
/* r->buf should not be NULL since we just free'd a larger one. */
|
||||||
assert(r->buf != NULL);
|
assert(r->buf != NULL);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
newbuf = sdscatlen(r->buf,buf,len);
|
newbuf = sdscatlen(r->buf,buf,len);
|
||||||
if (newbuf == NULL) {
|
if (newbuf == NULL) {
|
||||||
|
6
deps/hiredis/hiredis.h
vendored
6
deps/hiredis/hiredis.h
vendored
@ -76,6 +76,9 @@
|
|||||||
/* Flag that is set when the async context has one or more subscriptions. */
|
/* Flag that is set when the async context has one or more subscriptions. */
|
||||||
#define REDIS_SUBSCRIBED 0x20
|
#define REDIS_SUBSCRIBED 0x20
|
||||||
|
|
||||||
|
/* Flag that is set when monitor mode is active */
|
||||||
|
#define REDIS_MONITORING 0x40
|
||||||
|
|
||||||
#define REDIS_REPLY_STRING 1
|
#define REDIS_REPLY_STRING 1
|
||||||
#define REDIS_REPLY_ARRAY 2
|
#define REDIS_REPLY_ARRAY 2
|
||||||
#define REDIS_REPLY_INTEGER 3
|
#define REDIS_REPLY_INTEGER 3
|
||||||
@ -83,6 +86,8 @@
|
|||||||
#define REDIS_REPLY_STATUS 5
|
#define REDIS_REPLY_STATUS 5
|
||||||
#define REDIS_REPLY_ERROR 6
|
#define REDIS_REPLY_ERROR 6
|
||||||
|
|
||||||
|
#define REDIS_READER_MAX_BUF (1024*16) /* Default max unused reader buffer. */
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
@ -122,6 +127,7 @@ typedef struct redisReader {
|
|||||||
char *buf; /* Read buffer */
|
char *buf; /* Read buffer */
|
||||||
size_t pos; /* Buffer cursor */
|
size_t pos; /* Buffer cursor */
|
||||||
size_t len; /* Buffer length */
|
size_t len; /* Buffer length */
|
||||||
|
size_t maxbuf; /* Max length of unused buffer */
|
||||||
|
|
||||||
redisReadTask rstack[9];
|
redisReadTask rstack[9];
|
||||||
int ridx; /* Index of current read task */
|
int ridx; /* Index of current read task */
|
||||||
|
40
deps/hiredis/net.c
vendored
40
deps/hiredis/net.c
vendored
@ -45,6 +45,8 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <poll.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
#include "net.h"
|
#include "net.h"
|
||||||
#include "sds.h"
|
#include "sds.h"
|
||||||
@ -121,28 +123,38 @@ static int redisSetTcpNoDelay(redisContext *c, int fd) {
|
|||||||
return REDIS_OK;
|
return REDIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define __MAX_MSEC (((LONG_MAX) - 999) / 1000)
|
||||||
|
|
||||||
static int redisContextWaitReady(redisContext *c, int fd, const struct timeval *timeout) {
|
static int redisContextWaitReady(redisContext *c, int fd, const struct timeval *timeout) {
|
||||||
struct timeval to;
|
struct pollfd wfd[1];
|
||||||
struct timeval *toptr = NULL;
|
long msec;
|
||||||
fd_set wfd;
|
|
||||||
|
msec = -1;
|
||||||
|
wfd[0].fd = fd;
|
||||||
|
wfd[0].events = POLLOUT;
|
||||||
|
|
||||||
/* Only use timeout when not NULL. */
|
/* Only use timeout when not NULL. */
|
||||||
if (timeout != NULL) {
|
if (timeout != NULL) {
|
||||||
to = *timeout;
|
if (timeout->tv_usec > 1000000 || timeout->tv_sec > __MAX_MSEC) {
|
||||||
toptr = &to;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (errno == EINPROGRESS) {
|
|
||||||
FD_ZERO(&wfd);
|
|
||||||
FD_SET(fd, &wfd);
|
|
||||||
|
|
||||||
if (select(FD_SETSIZE, NULL, &wfd, NULL, toptr) == -1) {
|
|
||||||
__redisSetErrorFromErrno(c,REDIS_ERR_IO,"select(2)");
|
|
||||||
close(fd);
|
close(fd);
|
||||||
return REDIS_ERR;
|
return REDIS_ERR;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!FD_ISSET(fd, &wfd)) {
|
msec = (timeout->tv_sec * 1000) + ((timeout->tv_usec + 999) / 1000);
|
||||||
|
|
||||||
|
if (msec < 0 || msec > INT_MAX) {
|
||||||
|
msec = INT_MAX;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errno == EINPROGRESS) {
|
||||||
|
int res;
|
||||||
|
|
||||||
|
if ((res = poll(wfd, 1, msec)) == -1) {
|
||||||
|
__redisSetErrorFromErrno(c, REDIS_ERR_IO, "poll(2)");
|
||||||
|
close(fd);
|
||||||
|
return REDIS_ERR;
|
||||||
|
} else if (res == 0) {
|
||||||
errno = ETIMEDOUT;
|
errno = ETIMEDOUT;
|
||||||
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
|
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
|
||||||
close(fd);
|
close(fd);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user