mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 16:18:28 -05:00
Merge remote-tracking branch 'origin/unstable' into unstable
This commit is contained in:
commit
0f9997845a
@ -22,6 +22,12 @@ but it is better to keep them in mind:
|
|||||||
starting with '#'. All the major clients should be already fixed to work
|
starting with '#'. All the major clients should be already fixed to work
|
||||||
with the new INFO format.
|
with the new INFO format.
|
||||||
|
|
||||||
|
Also the following redis.conf and CONFIG GET / SET parameters changed name:
|
||||||
|
|
||||||
|
* hash-max-zipmap-entries, now replaced by hash-max-ziplist-entries
|
||||||
|
* hash-max-zipmap-value, now replaced by hash-max-ziplist-value
|
||||||
|
* glueoutputbuf was no completely removed as it does not make sense
|
||||||
|
|
||||||
---------
|
---------
|
||||||
CHANGELOG
|
CHANGELOG
|
||||||
---------
|
---------
|
||||||
|
44
redis.conf
44
redis.conf
@ -93,6 +93,21 @@ save 900 1
|
|||||||
save 300 10
|
save 300 10
|
||||||
save 60 10000
|
save 60 10000
|
||||||
|
|
||||||
|
# By default Redis will stop accepting writes if RDB snapshots are enabled
|
||||||
|
# (at least one save point) and the latest background save failed.
|
||||||
|
# This will make the user aware (in an hard way) that data is not persisting
|
||||||
|
# on disk properly, otherwise chances are that no one will notice and some
|
||||||
|
# distater will happen.
|
||||||
|
#
|
||||||
|
# If the background saving process will start working again Redis will
|
||||||
|
# automatically allow writes again.
|
||||||
|
#
|
||||||
|
# However if you have setup your proper monitoring of the Redis server
|
||||||
|
# and persistence, you may want to disable this feature so that Redis will
|
||||||
|
# continue to work as usually even if there are problems with disk,
|
||||||
|
# permissions, and so forth.
|
||||||
|
stop-writes-on-bgsave-error yes
|
||||||
|
|
||||||
# Compress string objects using LZF when dump .rdb databases?
|
# Compress string objects using LZF when dump .rdb databases?
|
||||||
# For default that's set to 'yes' as it's almost always a win.
|
# For default that's set to 'yes' as it's almost always a win.
|
||||||
# If you want to save some CPU in the saving child set it to 'no' but
|
# If you want to save some CPU in the saving child set it to 'no' but
|
||||||
@ -141,6 +156,22 @@ dir ./
|
|||||||
#
|
#
|
||||||
slave-serve-stale-data yes
|
slave-serve-stale-data yes
|
||||||
|
|
||||||
|
# You can configure a slave instance to accept writes or not. Writing against
|
||||||
|
# a slave instance may be useful to store some ephemeral data (because data
|
||||||
|
# written on a slave will be easily deleted after resync with the master) but
|
||||||
|
# may also cause problems if clients are writing to it because of a
|
||||||
|
# misconfiguration.
|
||||||
|
#
|
||||||
|
# Since Redis 2.6 by default slaves are read-only.
|
||||||
|
#
|
||||||
|
# Note: read only slaves are not designed to be exposed to untrusted clients
|
||||||
|
# on the internet. It's just a protection layer against misuse of the instance.
|
||||||
|
# Still a read only slave exports by default all the administrative commands
|
||||||
|
# such as CONFIG, DEBUG, and so forth. To a limited extend you can improve
|
||||||
|
# security of read only slaves using 'rename-command' to shadow all the
|
||||||
|
# administrative / dangerous commands.
|
||||||
|
slave-read-only yes
|
||||||
|
|
||||||
# Slaves send PINGs to server in a predefined interval. It's possible to change
|
# Slaves send PINGs to server in a predefined interval. It's possible to change
|
||||||
# this interval with the repl_ping_slave_period option. The default value is 10
|
# this interval with the repl_ping_slave_period option. The default value is 10
|
||||||
# seconds.
|
# seconds.
|
||||||
@ -231,7 +262,7 @@ slave-serve-stale-data yes
|
|||||||
# volatile-lru -> remove the key with an expire set using an LRU algorithm
|
# volatile-lru -> remove the key with an expire set using an LRU algorithm
|
||||||
# allkeys-lru -> remove any key accordingly to the LRU algorithm
|
# allkeys-lru -> remove any key accordingly to the LRU algorithm
|
||||||
# volatile-random -> remove a random key with an expire set
|
# volatile-random -> remove a random key with an expire set
|
||||||
# allkeys->random -> remove a random key, any key
|
# allkeys-random -> remove a random key, any key
|
||||||
# volatile-ttl -> remove the key with the nearest expire time (minor TTL)
|
# volatile-ttl -> remove the key with the nearest expire time (minor TTL)
|
||||||
# noeviction -> don't expire at all, just return an error on write operations
|
# noeviction -> don't expire at all, just return an error on write operations
|
||||||
#
|
#
|
||||||
@ -406,12 +437,11 @@ slowlog-max-len 1024
|
|||||||
|
|
||||||
############################### ADVANCED CONFIG ###############################
|
############################### ADVANCED CONFIG ###############################
|
||||||
|
|
||||||
# Hashes are encoded in a special way (much more memory efficient) when they
|
# Hashes are encoded using a memory efficient data structure when they have a
|
||||||
# have at max a given number of elements, and the biggest element does not
|
# small number of entries, and the biggest entry does not exceed a given
|
||||||
# exceed a given threshold. You can configure this limits with the following
|
# threshold. These thresholds can be configured using the following directives.
|
||||||
# configuration directives.
|
hash-max-ziplist-entries 512
|
||||||
hash-max-zipmap-entries 512
|
hash-max-ziplist-value 64
|
||||||
hash-max-zipmap-value 64
|
|
||||||
|
|
||||||
# Similarly to hashes, small lists are also encoded in a special way in order
|
# Similarly to hashes, small lists are also encoded in a special way in order
|
||||||
# to save a lot of space. The special representation is only used when
|
# to save a lot of space. The special representation is only used when
|
||||||
|
63
src/Makefile
63
src/Makefile
@ -73,7 +73,7 @@ QUIET_CC = @printf ' %b %b\n' $(CCCOLOR)CC$(ENDCOLOR) $(SRCCOLOR)$@$(ENDCOLOR
|
|||||||
QUIET_LINK = @printf ' %b %b\n' $(LINKCOLOR)LINK$(ENDCOLOR) $(BINCOLOR)$@$(ENDCOLOR);
|
QUIET_LINK = @printf ' %b %b\n' $(LINKCOLOR)LINK$(ENDCOLOR) $(BINCOLOR)$@$(ENDCOLOR);
|
||||||
endif
|
endif
|
||||||
|
|
||||||
OBJ = adlist.o ae.o anet.o dict.o redis.o sds.o zmalloc.o lzf_c.o lzf_d.o pqsort.o zipmap.o sha1.o ziplist.o release.o networking.o util.o object.o db.o replication.o rdb.o t_string.o t_list.o t_set.o t_zset.o t_hash.o config.o aof.o pubsub.o multi.o debug.o sort.o intset.o syncio.o cluster.o crc16.o endianconv.o slowlog.o scripting.o bio.o rio.o rand.o
|
OBJ = adlist.o ae.o anet.o dict.o redis.o sds.o zmalloc.o lzf_c.o lzf_d.o pqsort.o zipmap.o sha1.o ziplist.o release.o networking.o util.o object.o db.o replication.o rdb.o t_string.o t_list.o t_set.o t_zset.o t_hash.o config.o aof.o pubsub.o multi.o debug.o sort.o intset.o syncio.o cluster.o crc16.o endianconv.o slowlog.o scripting.o bio.o rio.o rand.o memtest.o
|
||||||
BENCHOBJ = ae.o anet.o redis-benchmark.o sds.o adlist.o zmalloc.o
|
BENCHOBJ = ae.o anet.o redis-benchmark.o sds.o adlist.o zmalloc.o
|
||||||
CLIOBJ = anet.o sds.o adlist.o redis-cli.o zmalloc.o release.o
|
CLIOBJ = anet.o sds.o adlist.o redis-cli.o zmalloc.o release.o
|
||||||
CHECKDUMPOBJ = redis-check-dump.o lzf_c.o lzf_d.o
|
CHECKDUMPOBJ = redis-check-dump.o lzf_c.o lzf_d.o
|
||||||
@ -98,35 +98,39 @@ ae_kqueue.o: ae_kqueue.c
|
|||||||
ae_select.o: ae_select.c
|
ae_select.o: ae_select.c
|
||||||
anet.o: anet.c fmacros.h anet.h
|
anet.o: anet.c fmacros.h anet.h
|
||||||
aof.o: aof.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
aof.o: aof.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h bio.h
|
||||||
bio.o: bio.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
bio.o: bio.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h bio.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h bio.h
|
||||||
cluster.o: cluster.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
cluster.o: cluster.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
|
rio.h
|
||||||
config.o: config.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
config.o: config.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
crc16.o: crc16.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
crc16.o: crc16.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
db.o: db.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
db.o: db.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
debug.o: debug.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
debug.o: debug.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h sha1.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h sha1.h
|
||||||
dict.o: dict.c fmacros.h dict.h zmalloc.h
|
dict.o: dict.c fmacros.h dict.h zmalloc.h
|
||||||
endianconv.o: endianconv.c
|
endianconv.o: endianconv.c
|
||||||
intset.o: intset.c intset.h zmalloc.h endianconv.h
|
intset.o: intset.c intset.h zmalloc.h endianconv.h
|
||||||
lzf_c.o: lzf_c.c lzfP.h
|
lzf_c.o: lzf_c.c lzfP.h
|
||||||
lzf_d.o: lzf_d.c lzfP.h
|
lzf_d.o: lzf_d.c lzfP.h
|
||||||
multi.o: multi.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
multi.o: multi.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
networking.o: networking.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
networking.o: networking.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
|
rio.h
|
||||||
object.o: object.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
object.o: object.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
pqsort.o: pqsort.c
|
pqsort.o: pqsort.c
|
||||||
pubsub.o: pubsub.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
pubsub.o: pubsub.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
|
rand.o: rand.c
|
||||||
rdb.o: rdb.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
rdb.o: rdb.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h lzf.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h lzf.h \
|
||||||
|
zipmap.h
|
||||||
redis-benchmark.o: redis-benchmark.c fmacros.h ae.h \
|
redis-benchmark.o: redis-benchmark.c fmacros.h ae.h \
|
||||||
../deps/hiredis/hiredis.h sds.h adlist.h zmalloc.h
|
../deps/hiredis/hiredis.h sds.h adlist.h zmalloc.h
|
||||||
redis-check-aof.o: redis-check-aof.c fmacros.h config.h
|
redis-check-aof.o: redis-check-aof.c fmacros.h config.h
|
||||||
@ -134,34 +138,37 @@ redis-check-dump.o: redis-check-dump.c lzf.h
|
|||||||
redis-cli.o: redis-cli.c fmacros.h version.h ../deps/hiredis/hiredis.h \
|
redis-cli.o: redis-cli.c fmacros.h version.h ../deps/hiredis/hiredis.h \
|
||||||
sds.h zmalloc.h ../deps/linenoise/linenoise.h help.h
|
sds.h zmalloc.h ../deps/linenoise/linenoise.h help.h
|
||||||
redis.o: redis.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
redis.o: redis.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h slowlog.h \
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h \
|
||||||
bio.h asciilogo.h
|
slowlog.h bio.h asciilogo.h
|
||||||
release.o: release.c release.h
|
release.o: release.c release.h
|
||||||
replication.o: replication.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
replication.o: replication.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
|
rio.h
|
||||||
|
rio.o: rio.c fmacros.h rio.h sds.h util.h
|
||||||
scripting.o: scripting.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
scripting.o: scripting.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h \
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
sha1.h rand.h
|
rio.h sha1.h rand.h
|
||||||
rio.o: rio.c sds.h
|
|
||||||
sds.o: sds.c sds.h zmalloc.h
|
sds.o: sds.c sds.h zmalloc.h
|
||||||
sha1.o: sha1.c sha1.h config.h
|
sha1.o: sha1.c sha1.h config.h
|
||||||
slowlog.o: slowlog.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
slowlog.o: slowlog.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h \
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
slowlog.h
|
rio.h slowlog.h
|
||||||
sort.o: sort.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
sort.o: sort.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h pqsort.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h \
|
||||||
|
pqsort.h
|
||||||
syncio.o: syncio.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
syncio.o: syncio.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
t_hash.o: t_hash.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
t_hash.o: t_hash.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
t_list.o: t_list.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
t_list.o: t_list.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
t_set.o: t_set.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
t_set.o: t_set.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
t_string.o: t_string.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
t_string.o: t_string.c redis.h fmacros.h config.h ae.h sds.h dict.h \
|
||||||
adlist.h zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
adlist.h zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h \
|
||||||
|
rio.h
|
||||||
t_zset.o: t_zset.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
t_zset.o: t_zset.c redis.h fmacros.h config.h ae.h sds.h dict.h adlist.h \
|
||||||
zmalloc.h anet.h zipmap.h ziplist.h intset.h version.h util.h
|
zmalloc.h anet.h ziplist.h intset.h version.h util.h rdb.h rio.h
|
||||||
util.o: util.c fmacros.h util.h
|
util.o: util.c fmacros.h util.h
|
||||||
ziplist.o: ziplist.c zmalloc.h util.h ziplist.h endianconv.h
|
ziplist.o: ziplist.c zmalloc.h util.h ziplist.h endianconv.h
|
||||||
zipmap.o: zipmap.c zmalloc.h endianconv.h
|
zipmap.o: zipmap.c zmalloc.h endianconv.h
|
||||||
|
16
src/adlist.c
16
src/adlist.c
@ -323,3 +323,19 @@ listNode *listIndex(list *list, long index) {
|
|||||||
}
|
}
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Rotate the list removing the tail node and inserting it to the head. */
|
||||||
|
void listRotate(list *list) {
|
||||||
|
listNode *tail = list->tail;
|
||||||
|
|
||||||
|
if (listLength(list) <= 1) return;
|
||||||
|
|
||||||
|
/* Detatch current tail */
|
||||||
|
list->tail = tail->prev;
|
||||||
|
list->tail->next = NULL;
|
||||||
|
/* Move it as head */
|
||||||
|
list->head->prev = tail;
|
||||||
|
tail->prev = NULL;
|
||||||
|
tail->next = list->head;
|
||||||
|
list->head = tail;
|
||||||
|
}
|
||||||
|
@ -84,6 +84,7 @@ listNode *listSearchKey(list *list, void *key);
|
|||||||
listNode *listIndex(list *list, long index);
|
listNode *listIndex(list *list, long index);
|
||||||
void listRewind(list *list, listIter *li);
|
void listRewind(list *list, listIter *li);
|
||||||
void listRewindTail(list *list, listIter *li);
|
void listRewindTail(list *list, listIter *li);
|
||||||
|
void listRotate(list *list);
|
||||||
|
|
||||||
/* Directions for iterators */
|
/* Directions for iterators */
|
||||||
#define AL_START_HEAD 0
|
#define AL_START_HEAD 0
|
||||||
|
@ -353,7 +353,12 @@ int anetPeerToString(int fd, char *ip, int *port) {
|
|||||||
struct sockaddr_in sa;
|
struct sockaddr_in sa;
|
||||||
socklen_t salen = sizeof(sa);
|
socklen_t salen = sizeof(sa);
|
||||||
|
|
||||||
if (getpeername(fd,(struct sockaddr*)&sa,&salen) == -1) return -1;
|
if (getpeername(fd,(struct sockaddr*)&sa,&salen) == -1) {
|
||||||
|
*port = 0;
|
||||||
|
ip[0] = '?';
|
||||||
|
ip[1] = '\0';
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
if (ip) strcpy(ip,inet_ntoa(sa.sin_addr));
|
if (ip) strcpy(ip,inet_ntoa(sa.sin_addr));
|
||||||
if (port) *port = ntohs(sa.sin_port);
|
if (port) *port = ntohs(sa.sin_port);
|
||||||
return 0;
|
return 0;
|
||||||
|
85
src/aof.c
85
src/aof.c
@ -287,6 +287,7 @@ struct redisClient *createFakeClient(void) {
|
|||||||
selectDb(c,0);
|
selectDb(c,0);
|
||||||
c->fd = -1;
|
c->fd = -1;
|
||||||
c->querybuf = sdsempty();
|
c->querybuf = sdsempty();
|
||||||
|
c->querybuf_peak = 0;
|
||||||
c->argc = 0;
|
c->argc = 0;
|
||||||
c->argv = NULL;
|
c->argv = NULL;
|
||||||
c->bufpos = 0;
|
c->bufpos = 0;
|
||||||
@ -609,53 +610,61 @@ int rewriteSortedSetObject(rio *r, robj *key, robj *o) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Write either the key or the value of the currently selected item of an hash.
|
||||||
|
* The 'hi' argument passes a valid Redis hash iterator.
|
||||||
|
* The 'what' filed specifies if to write a key or a value and can be
|
||||||
|
* either REDIS_HASH_KEY or REDIS_HASH_VALUE.
|
||||||
|
*
|
||||||
|
* The function returns 0 on error, non-zero on success. */
|
||||||
|
static int rioWriteHashIteratorCursor(rio *r, hashTypeIterator *hi, int what) {
|
||||||
|
if (hi->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
unsigned char *vstr = NULL;
|
||||||
|
unsigned int vlen = UINT_MAX;
|
||||||
|
long long vll = LLONG_MAX;
|
||||||
|
|
||||||
|
hashTypeCurrentFromZiplist(hi, what, &vstr, &vlen, &vll);
|
||||||
|
if (vstr) {
|
||||||
|
return rioWriteBulkString(r, (char*)vstr, vlen);
|
||||||
|
} else {
|
||||||
|
return rioWriteBulkLongLong(r, vll);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
|
robj *value;
|
||||||
|
|
||||||
|
hashTypeCurrentFromHashTable(hi, what, &value);
|
||||||
|
return rioWriteBulkObject(r, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* Emit the commands needed to rebuild a hash object.
|
/* Emit the commands needed to rebuild a hash object.
|
||||||
* The function returns 0 on error, 1 on success. */
|
* The function returns 0 on error, 1 on success. */
|
||||||
int rewriteHashObject(rio *r, robj *key, robj *o) {
|
int rewriteHashObject(rio *r, robj *key, robj *o) {
|
||||||
|
hashTypeIterator *hi;
|
||||||
long long count = 0, items = hashTypeLength(o);
|
long long count = 0, items = hashTypeLength(o);
|
||||||
|
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
hi = hashTypeInitIterator(o);
|
||||||
unsigned char *p = zipmapRewind(o->ptr);
|
while (hashTypeNext(hi) != REDIS_ERR) {
|
||||||
unsigned char *field, *val;
|
if (count == 0) {
|
||||||
unsigned int flen, vlen;
|
int cmd_items = (items > REDIS_AOF_REWRITE_ITEMS_PER_CMD) ?
|
||||||
|
REDIS_AOF_REWRITE_ITEMS_PER_CMD : items;
|
||||||
|
|
||||||
while((p = zipmapNext(p,&field,&flen,&val,&vlen)) != NULL) {
|
if (rioWriteBulkCount(r,'*',2+cmd_items*2) == 0) return 0;
|
||||||
if (count == 0) {
|
if (rioWriteBulkString(r,"HMSET",5) == 0) return 0;
|
||||||
int cmd_items = (items > REDIS_AOF_REWRITE_ITEMS_PER_CMD) ?
|
if (rioWriteBulkObject(r,key) == 0) return 0;
|
||||||
REDIS_AOF_REWRITE_ITEMS_PER_CMD : items;
|
|
||||||
|
|
||||||
if (rioWriteBulkCount(r,'*',2+cmd_items*2) == 0) return 0;
|
|
||||||
if (rioWriteBulkString(r,"HMSET",5) == 0) return 0;
|
|
||||||
if (rioWriteBulkObject(r,key) == 0) return 0;
|
|
||||||
}
|
|
||||||
if (rioWriteBulkString(r,(char*)field,flen) == 0) return 0;
|
|
||||||
if (rioWriteBulkString(r,(char*)val,vlen) == 0) return 0;
|
|
||||||
if (++count == REDIS_AOF_REWRITE_ITEMS_PER_CMD) count = 0;
|
|
||||||
items--;
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
dictIterator *di = dictGetIterator(o->ptr);
|
|
||||||
dictEntry *de;
|
|
||||||
|
|
||||||
while((de = dictNext(di)) != NULL) {
|
if (rioWriteHashIteratorCursor(r, hi, REDIS_HASH_KEY) == 0) return 0;
|
||||||
robj *field = dictGetKey(de);
|
if (rioWriteHashIteratorCursor(r, hi, REDIS_HASH_VALUE) == 0) return 0;
|
||||||
robj *val = dictGetVal(de);
|
if (++count == REDIS_AOF_REWRITE_ITEMS_PER_CMD) count = 0;
|
||||||
|
items--;
|
||||||
if (count == 0) {
|
|
||||||
int cmd_items = (items > REDIS_AOF_REWRITE_ITEMS_PER_CMD) ?
|
|
||||||
REDIS_AOF_REWRITE_ITEMS_PER_CMD : items;
|
|
||||||
|
|
||||||
if (rioWriteBulkCount(r,'*',2+cmd_items*2) == 0) return 0;
|
|
||||||
if (rioWriteBulkString(r,"HMSET",5) == 0) return 0;
|
|
||||||
if (rioWriteBulkObject(r,key) == 0) return 0;
|
|
||||||
}
|
|
||||||
if (rioWriteBulkObject(r,field) == 0) return 0;
|
|
||||||
if (rioWriteBulkObject(r,val) == 0) return 0;
|
|
||||||
if (++count == REDIS_AOF_REWRITE_ITEMS_PER_CMD) count = 0;
|
|
||||||
items--;
|
|
||||||
}
|
|
||||||
dictReleaseIterator(di);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hashTypeReleaseIterator(hi);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,20 +19,6 @@ int clusterAddSlot(clusterNode *n, int slot);
|
|||||||
* Initialization
|
* Initialization
|
||||||
* -------------------------------------------------------------------------- */
|
* -------------------------------------------------------------------------- */
|
||||||
|
|
||||||
void clusterGetRandomName(char *p) {
|
|
||||||
FILE *fp = fopen("/dev/urandom","r");
|
|
||||||
char *charset = "0123456789abcdef";
|
|
||||||
int j;
|
|
||||||
|
|
||||||
if (fp == NULL || fread(p,REDIS_CLUSTER_NAMELEN,1,fp) == 0) {
|
|
||||||
for (j = 0; j < REDIS_CLUSTER_NAMELEN; j++)
|
|
||||||
p[j] = rand();
|
|
||||||
}
|
|
||||||
for (j = 0; j < REDIS_CLUSTER_NAMELEN; j++)
|
|
||||||
p[j] = charset[p[j] & 0x0F];
|
|
||||||
fclose(fp);
|
|
||||||
}
|
|
||||||
|
|
||||||
int clusterLoadConfig(char *filename) {
|
int clusterLoadConfig(char *filename) {
|
||||||
FILE *fp = fopen(filename,"r");
|
FILE *fp = fopen(filename,"r");
|
||||||
char *line;
|
char *line;
|
||||||
@ -304,7 +290,7 @@ clusterNode *createClusterNode(char *nodename, int flags) {
|
|||||||
if (nodename)
|
if (nodename)
|
||||||
memcpy(node->name, nodename, REDIS_CLUSTER_NAMELEN);
|
memcpy(node->name, nodename, REDIS_CLUSTER_NAMELEN);
|
||||||
else
|
else
|
||||||
clusterGetRandomName(node->name);
|
getRandomHexChars(node->name, REDIS_CLUSTER_NAMELEN);
|
||||||
node->flags = flags;
|
node->flags = flags;
|
||||||
memset(node->slots,0,sizeof(node->slots));
|
memset(node->slots,0,sizeof(node->slots));
|
||||||
node->numslaves = 0;
|
node->numslaves = 0;
|
||||||
|
256
src/config.c
256
src/config.c
@ -202,8 +202,10 @@ void loadServerConfigFromString(char *config) {
|
|||||||
if ((server.repl_serve_stale_data = yesnotoi(argv[1])) == -1) {
|
if ((server.repl_serve_stale_data = yesnotoi(argv[1])) == -1) {
|
||||||
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
||||||
}
|
}
|
||||||
} else if (!strcasecmp(argv[0],"glueoutputbuf")) {
|
} else if (!strcasecmp(argv[0],"slave-read-only") && argc == 2) {
|
||||||
redisLog(REDIS_WARNING, "Deprecated configuration directive: \"%s\"", argv[0]);
|
if ((server.repl_slave_ro = yesnotoi(argv[1])) == -1) {
|
||||||
|
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
||||||
|
}
|
||||||
} else if (!strcasecmp(argv[0],"rdbcompression") && argc == 2) {
|
} else if (!strcasecmp(argv[0],"rdbcompression") && argc == 2) {
|
||||||
if ((server.rdb_compression = yesnotoi(argv[1])) == -1) {
|
if ((server.rdb_compression = yesnotoi(argv[1])) == -1) {
|
||||||
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
||||||
@ -262,10 +264,10 @@ void loadServerConfigFromString(char *config) {
|
|||||||
} else if (!strcasecmp(argv[0],"dbfilename") && argc == 2) {
|
} else if (!strcasecmp(argv[0],"dbfilename") && argc == 2) {
|
||||||
zfree(server.rdb_filename);
|
zfree(server.rdb_filename);
|
||||||
server.rdb_filename = zstrdup(argv[1]);
|
server.rdb_filename = zstrdup(argv[1]);
|
||||||
} else if (!strcasecmp(argv[0],"hash-max-zipmap-entries") && argc == 2) {
|
} else if (!strcasecmp(argv[0],"hash-max-ziplist-entries") && argc == 2) {
|
||||||
server.hash_max_zipmap_entries = memtoll(argv[1], NULL);
|
server.hash_max_ziplist_entries = memtoll(argv[1], NULL);
|
||||||
} else if (!strcasecmp(argv[0],"hash-max-zipmap-value") && argc == 2) {
|
} else if (!strcasecmp(argv[0],"hash-max-ziplist-value") && argc == 2) {
|
||||||
server.hash_max_zipmap_value = memtoll(argv[1], NULL);
|
server.hash_max_ziplist_value = memtoll(argv[1], NULL);
|
||||||
} else if (!strcasecmp(argv[0],"list-max-ziplist-entries") && argc == 2){
|
} else if (!strcasecmp(argv[0],"list-max-ziplist-entries") && argc == 2){
|
||||||
server.list_max_ziplist_entries = memtoll(argv[1], NULL);
|
server.list_max_ziplist_entries = memtoll(argv[1], NULL);
|
||||||
} else if (!strcasecmp(argv[0],"list-max-ziplist-value") && argc == 2) {
|
} else if (!strcasecmp(argv[0],"list-max-ziplist-value") && argc == 2) {
|
||||||
@ -336,6 +338,11 @@ void loadServerConfigFromString(char *config) {
|
|||||||
server.client_obuf_limits[class].hard_limit_bytes = hard;
|
server.client_obuf_limits[class].hard_limit_bytes = hard;
|
||||||
server.client_obuf_limits[class].soft_limit_bytes = soft;
|
server.client_obuf_limits[class].soft_limit_bytes = soft;
|
||||||
server.client_obuf_limits[class].soft_limit_seconds = soft_seconds;
|
server.client_obuf_limits[class].soft_limit_seconds = soft_seconds;
|
||||||
|
} else if (!strcasecmp(argv[0],"stop-writes-on-bgsave-error") &&
|
||||||
|
argc == 2) {
|
||||||
|
if ((server.stop_writes_on_bgsave_err = yesnotoi(argv[1])) == -1) {
|
||||||
|
err = "argument must be 'yes' or 'no'"; goto loaderr;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
err = "Bad directive or wrong number of arguments"; goto loaderr;
|
err = "Bad directive or wrong number of arguments"; goto loaderr;
|
||||||
}
|
}
|
||||||
@ -511,17 +518,22 @@ void configSetCommand(redisClient *c) {
|
|||||||
|
|
||||||
if (yn == -1) goto badfmt;
|
if (yn == -1) goto badfmt;
|
||||||
server.repl_serve_stale_data = yn;
|
server.repl_serve_stale_data = yn;
|
||||||
|
} else if (!strcasecmp(c->argv[2]->ptr,"slave-read-only")) {
|
||||||
|
int yn = yesnotoi(o->ptr);
|
||||||
|
|
||||||
|
if (yn == -1) goto badfmt;
|
||||||
|
server.repl_slave_ro = yn;
|
||||||
} else if (!strcasecmp(c->argv[2]->ptr,"dir")) {
|
} else if (!strcasecmp(c->argv[2]->ptr,"dir")) {
|
||||||
if (chdir((char*)o->ptr) == -1) {
|
if (chdir((char*)o->ptr) == -1) {
|
||||||
addReplyErrorFormat(c,"Changing directory: %s", strerror(errno));
|
addReplyErrorFormat(c,"Changing directory: %s", strerror(errno));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else if (!strcasecmp(c->argv[2]->ptr,"hash-max-zipmap-entries")) {
|
} else if (!strcasecmp(c->argv[2]->ptr,"hash-max-ziplist-entries")) {
|
||||||
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
||||||
server.hash_max_zipmap_entries = ll;
|
server.hash_max_ziplist_entries = ll;
|
||||||
} else if (!strcasecmp(c->argv[2]->ptr,"hash-max-zipmap-value")) {
|
} else if (!strcasecmp(c->argv[2]->ptr,"hash-max-ziplist-value")) {
|
||||||
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
||||||
server.hash_max_zipmap_value = ll;
|
server.hash_max_ziplist_value = ll;
|
||||||
} else if (!strcasecmp(c->argv[2]->ptr,"list-max-ziplist-entries")) {
|
} else if (!strcasecmp(c->argv[2]->ptr,"list-max-ziplist-entries")) {
|
||||||
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll < 0) goto badfmt;
|
||||||
server.list_max_ziplist_entries = ll;
|
server.list_max_ziplist_entries = ll;
|
||||||
@ -604,7 +616,17 @@ void configSetCommand(redisClient *c) {
|
|||||||
server.client_obuf_limits[class].soft_limit_seconds = soft_seconds;
|
server.client_obuf_limits[class].soft_limit_seconds = soft_seconds;
|
||||||
}
|
}
|
||||||
sdsfreesplitres(v,vlen);
|
sdsfreesplitres(v,vlen);
|
||||||
|
} else if (!strcasecmp(c->argv[2]->ptr,"stop-writes-on-bgsave-error")) {
|
||||||
|
int yn = yesnotoi(o->ptr);
|
||||||
|
|
||||||
|
if (yn == -1) goto badfmt;
|
||||||
|
server.stop_writes_on_bgsave_err = yn;
|
||||||
|
} else if (!strcasecmp(c->argv[2]->ptr,"repl-ping-slave-period")) {
|
||||||
|
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll <= 0) goto badfmt;
|
||||||
|
server.repl_ping_slave_period = ll;
|
||||||
|
} else if (!strcasecmp(c->argv[2]->ptr,"repl-timeout")) {
|
||||||
|
if (getLongLongFromObject(o,&ll) == REDIS_ERR || ll <= 0) goto badfmt;
|
||||||
|
server.repl_timeout = ll;
|
||||||
} else {
|
} else {
|
||||||
addReplyErrorFormat(c,"Unsupported CONFIG parameter: %s",
|
addReplyErrorFormat(c,"Unsupported CONFIG parameter: %s",
|
||||||
(char*)c->argv[2]->ptr);
|
(char*)c->argv[2]->ptr);
|
||||||
@ -619,6 +641,31 @@ badfmt: /* Bad format errors */
|
|||||||
(char*)c->argv[2]->ptr);
|
(char*)c->argv[2]->ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define config_get_string_field(_name,_var) do { \
|
||||||
|
if (stringmatch(pattern,_name,0)) { \
|
||||||
|
addReplyBulkCString(c,_name); \
|
||||||
|
addReplyBulkCString(c,_var ? _var : ""); \
|
||||||
|
matches++; \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
|
#define config_get_bool_field(_name,_var) do { \
|
||||||
|
if (stringmatch(pattern,_name,0)) { \
|
||||||
|
addReplyBulkCString(c,_name); \
|
||||||
|
addReplyBulkCString(c,_var ? "yes" : "no"); \
|
||||||
|
matches++; \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
|
#define config_get_numerical_field(_name,_var) do { \
|
||||||
|
if (stringmatch(pattern,_name,0)) { \
|
||||||
|
ll2string(buf,sizeof(buf),_var); \
|
||||||
|
addReplyBulkCString(c,_name); \
|
||||||
|
addReplyBulkCString(c,buf); \
|
||||||
|
matches++; \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
void configGetCommand(redisClient *c) {
|
void configGetCommand(redisClient *c) {
|
||||||
robj *o = c->argv[2];
|
robj *o = c->argv[2];
|
||||||
void *replylen = addDeferredMultiBulkLength(c);
|
void *replylen = addDeferredMultiBulkLength(c);
|
||||||
@ -627,6 +674,68 @@ void configGetCommand(redisClient *c) {
|
|||||||
int matches = 0;
|
int matches = 0;
|
||||||
redisAssertWithInfo(c,o,o->encoding == REDIS_ENCODING_RAW);
|
redisAssertWithInfo(c,o,o->encoding == REDIS_ENCODING_RAW);
|
||||||
|
|
||||||
|
/* String values */
|
||||||
|
config_get_string_field("dbfilename",server.rdb_filename);
|
||||||
|
config_get_string_field("requirepass",server.requirepass);
|
||||||
|
config_get_string_field("masterauth",server.requirepass);
|
||||||
|
config_get_string_field("bind",server.bindaddr);
|
||||||
|
config_get_string_field("unixsocket",server.unixsocket);
|
||||||
|
config_get_string_field("logfile",server.logfile);
|
||||||
|
config_get_string_field("pidfile",server.pidfile);
|
||||||
|
|
||||||
|
/* Numerical values */
|
||||||
|
config_get_numerical_field("maxmemory",server.maxmemory);
|
||||||
|
config_get_numerical_field("maxmemory-samples",server.maxmemory_samples);
|
||||||
|
config_get_numerical_field("timeout",server.maxidletime);
|
||||||
|
config_get_numerical_field("auto-aof-rewrite-percentage",
|
||||||
|
server.aof_rewrite_perc);
|
||||||
|
config_get_numerical_field("auto-aof-rewrite-min-size",
|
||||||
|
server.aof_rewrite_min_size);
|
||||||
|
config_get_numerical_field("hash-max-ziplist-entries",
|
||||||
|
server.hash_max_ziplist_entries);
|
||||||
|
config_get_numerical_field("hash-max-ziplist-value",
|
||||||
|
server.hash_max_ziplist_value);
|
||||||
|
config_get_numerical_field("list-max-ziplist-entries",
|
||||||
|
server.list_max_ziplist_entries);
|
||||||
|
config_get_numerical_field("list-max-ziplist-value",
|
||||||
|
server.list_max_ziplist_value);
|
||||||
|
config_get_numerical_field("set-max-intset-entries",
|
||||||
|
server.set_max_intset_entries);
|
||||||
|
config_get_numerical_field("zset-max-ziplist-entries",
|
||||||
|
server.zset_max_ziplist_entries);
|
||||||
|
config_get_numerical_field("zset-max-ziplist-value",
|
||||||
|
server.zset_max_ziplist_value);
|
||||||
|
config_get_numerical_field("lua-time-limit",server.lua_time_limit);
|
||||||
|
config_get_numerical_field("slowlog-log-slower-than",
|
||||||
|
server.slowlog_log_slower_than);
|
||||||
|
config_get_numerical_field("slowlog-max-len",
|
||||||
|
server.slowlog_max_len);
|
||||||
|
config_get_numerical_field("port",server.port);
|
||||||
|
config_get_numerical_field("databases",server.dbnum);
|
||||||
|
config_get_numerical_field("repl-ping-slave-period",server.repl_ping_slave_period);
|
||||||
|
config_get_numerical_field("repl-timeout",server.repl_timeout);
|
||||||
|
config_get_numerical_field("maxclients",server.maxclients);
|
||||||
|
|
||||||
|
/* Bool (yes/no) values */
|
||||||
|
config_get_bool_field("no-appendfsync-on-rewrite",
|
||||||
|
server.aof_no_fsync_on_rewrite);
|
||||||
|
config_get_bool_field("slave-serve-stale-data",
|
||||||
|
server.repl_serve_stale_data);
|
||||||
|
config_get_bool_field("slave-read-only",
|
||||||
|
server.repl_slave_ro);
|
||||||
|
config_get_bool_field("stop-writes-on-bgsave-error",
|
||||||
|
server.stop_writes_on_bgsave_err);
|
||||||
|
config_get_bool_field("daemonize", server.daemonize);
|
||||||
|
config_get_bool_field("rdbcompression", server.rdb_compression);
|
||||||
|
config_get_bool_field("activerehashing", server.activerehashing);
|
||||||
|
|
||||||
|
/* Everything we can't handle with macros follows. */
|
||||||
|
|
||||||
|
if (stringmatch(pattern,"appendonly",0)) {
|
||||||
|
addReplyBulkCString(c,"appendonly");
|
||||||
|
addReplyBulkCString(c,server.aof_state == REDIS_AOF_OFF ? "no" : "yes");
|
||||||
|
matches++;
|
||||||
|
}
|
||||||
if (stringmatch(pattern,"dir",0)) {
|
if (stringmatch(pattern,"dir",0)) {
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
|
|
||||||
@ -637,27 +746,6 @@ void configGetCommand(redisClient *c) {
|
|||||||
addReplyBulkCString(c,buf);
|
addReplyBulkCString(c,buf);
|
||||||
matches++;
|
matches++;
|
||||||
}
|
}
|
||||||
if (stringmatch(pattern,"dbfilename",0)) {
|
|
||||||
addReplyBulkCString(c,"dbfilename");
|
|
||||||
addReplyBulkCString(c,server.rdb_filename);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"requirepass",0)) {
|
|
||||||
addReplyBulkCString(c,"requirepass");
|
|
||||||
addReplyBulkCString(c,server.requirepass);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"masterauth",0)) {
|
|
||||||
addReplyBulkCString(c,"masterauth");
|
|
||||||
addReplyBulkCString(c,server.masterauth);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"maxmemory",0)) {
|
|
||||||
ll2string(buf,sizeof(buf),server.maxmemory);
|
|
||||||
addReplyBulkCString(c,"maxmemory");
|
|
||||||
addReplyBulkCString(c,buf);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"maxmemory-policy",0)) {
|
if (stringmatch(pattern,"maxmemory-policy",0)) {
|
||||||
char *s;
|
char *s;
|
||||||
|
|
||||||
@ -674,28 +762,6 @@ void configGetCommand(redisClient *c) {
|
|||||||
addReplyBulkCString(c,s);
|
addReplyBulkCString(c,s);
|
||||||
matches++;
|
matches++;
|
||||||
}
|
}
|
||||||
if (stringmatch(pattern,"maxmemory-samples",0)) {
|
|
||||||
ll2string(buf,sizeof(buf),server.maxmemory_samples);
|
|
||||||
addReplyBulkCString(c,"maxmemory-samples");
|
|
||||||
addReplyBulkCString(c,buf);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"timeout",0)) {
|
|
||||||
ll2string(buf,sizeof(buf),server.maxidletime);
|
|
||||||
addReplyBulkCString(c,"timeout");
|
|
||||||
addReplyBulkCString(c,buf);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"appendonly",0)) {
|
|
||||||
addReplyBulkCString(c,"appendonly");
|
|
||||||
addReplyBulkCString(c,server.aof_state == REDIS_AOF_OFF ? "no" : "yes");
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"no-appendfsync-on-rewrite",0)) {
|
|
||||||
addReplyBulkCString(c,"no-appendfsync-on-rewrite");
|
|
||||||
addReplyBulkCString(c,server.aof_no_fsync_on_rewrite ? "yes" : "no");
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"appendfsync",0)) {
|
if (stringmatch(pattern,"appendfsync",0)) {
|
||||||
char *policy;
|
char *policy;
|
||||||
|
|
||||||
@ -725,71 +791,6 @@ void configGetCommand(redisClient *c) {
|
|||||||
sdsfree(buf);
|
sdsfree(buf);
|
||||||
matches++;
|
matches++;
|
||||||
}
|
}
|
||||||
if (stringmatch(pattern,"auto-aof-rewrite-percentage",0)) {
|
|
||||||
addReplyBulkCString(c,"auto-aof-rewrite-percentage");
|
|
||||||
addReplyBulkLongLong(c,server.aof_rewrite_perc);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"auto-aof-rewrite-min-size",0)) {
|
|
||||||
addReplyBulkCString(c,"auto-aof-rewrite-min-size");
|
|
||||||
addReplyBulkLongLong(c,server.aof_rewrite_min_size);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"slave-serve-stale-data",0)) {
|
|
||||||
addReplyBulkCString(c,"slave-serve-stale-data");
|
|
||||||
addReplyBulkCString(c,server.repl_serve_stale_data ? "yes" : "no");
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"hash-max-zipmap-entries",0)) {
|
|
||||||
addReplyBulkCString(c,"hash-max-zipmap-entries");
|
|
||||||
addReplyBulkLongLong(c,server.hash_max_zipmap_entries);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"hash-max-zipmap-value",0)) {
|
|
||||||
addReplyBulkCString(c,"hash-max-zipmap-value");
|
|
||||||
addReplyBulkLongLong(c,server.hash_max_zipmap_value);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"list-max-ziplist-entries",0)) {
|
|
||||||
addReplyBulkCString(c,"list-max-ziplist-entries");
|
|
||||||
addReplyBulkLongLong(c,server.list_max_ziplist_entries);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"list-max-ziplist-value",0)) {
|
|
||||||
addReplyBulkCString(c,"list-max-ziplist-value");
|
|
||||||
addReplyBulkLongLong(c,server.list_max_ziplist_value);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"set-max-intset-entries",0)) {
|
|
||||||
addReplyBulkCString(c,"set-max-intset-entries");
|
|
||||||
addReplyBulkLongLong(c,server.set_max_intset_entries);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"zset-max-ziplist-entries",0)) {
|
|
||||||
addReplyBulkCString(c,"zset-max-ziplist-entries");
|
|
||||||
addReplyBulkLongLong(c,server.zset_max_ziplist_entries);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"zset-max-ziplist-value",0)) {
|
|
||||||
addReplyBulkCString(c,"zset-max-ziplist-value");
|
|
||||||
addReplyBulkLongLong(c,server.zset_max_ziplist_value);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"lua-time-limit",0)) {
|
|
||||||
addReplyBulkCString(c,"lua-time-limit");
|
|
||||||
addReplyBulkLongLong(c,server.lua_time_limit);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"slowlog-log-slower-than",0)) {
|
|
||||||
addReplyBulkCString(c,"slowlog-log-slower-than");
|
|
||||||
addReplyBulkLongLong(c,server.slowlog_log_slower_than);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"slowlog-max-len",0)) {
|
|
||||||
addReplyBulkCString(c,"slowlog-max-len");
|
|
||||||
addReplyBulkLongLong(c,server.slowlog_max_len);
|
|
||||||
matches++;
|
|
||||||
}
|
|
||||||
if (stringmatch(pattern,"loglevel",0)) {
|
if (stringmatch(pattern,"loglevel",0)) {
|
||||||
char *s;
|
char *s;
|
||||||
|
|
||||||
@ -822,6 +823,25 @@ void configGetCommand(redisClient *c) {
|
|||||||
sdsfree(buf);
|
sdsfree(buf);
|
||||||
matches++;
|
matches++;
|
||||||
}
|
}
|
||||||
|
if (stringmatch(pattern,"unixsocketperm",0)) {
|
||||||
|
char buf[32];
|
||||||
|
snprintf(buf,sizeof(buf),"%o",server.unixsocketperm);
|
||||||
|
addReplyBulkCString(c,"unixsocketperm");
|
||||||
|
addReplyBulkCString(c,buf);
|
||||||
|
matches++;
|
||||||
|
}
|
||||||
|
if (stringmatch(pattern,"slaveof",0)) {
|
||||||
|
char buf[256];
|
||||||
|
|
||||||
|
addReplyBulkCString(c,"slaveof");
|
||||||
|
if (server.masterhost)
|
||||||
|
snprintf(buf,sizeof(buf),"%s %d",
|
||||||
|
server.masterhost, server.masterport);
|
||||||
|
else
|
||||||
|
buf[0] = '\0';
|
||||||
|
addReplyBulkCString(c,buf);
|
||||||
|
matches++;
|
||||||
|
}
|
||||||
setDeferredMultiBulkLength(c,replylen,matches*2);
|
setDeferredMultiBulkLength(c,replylen,matches*2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -636,8 +636,9 @@ void sigsegvHandler(int sig, siginfo_t *info, void *secret) {
|
|||||||
|
|
||||||
redisLog(REDIS_WARNING,
|
redisLog(REDIS_WARNING,
|
||||||
"\n=== REDIS BUG REPORT END. Make sure to include from START to END. ===\n\n"
|
"\n=== REDIS BUG REPORT END. Make sure to include from START to END. ===\n\n"
|
||||||
" Please report the crash opening an issue on github:\n\n"
|
" Please report the crash opening an issue on github:\n\n"
|
||||||
" http://github.com/antirez/redis/issues\n\n"
|
" http://github.com/antirez/redis/issues\n\n"
|
||||||
|
" Suspect RAM error? Use redis-server --test-memory to veryfy it.\n\n"
|
||||||
);
|
);
|
||||||
/* free(messages); Don't call free() with possibly corrupted memory. */
|
/* free(messages); Don't call free() with possibly corrupted memory. */
|
||||||
if (server.daemonize) unlink(server.pidfile);
|
if (server.daemonize) unlink(server.pidfile);
|
||||||
|
225
src/memtest.c
Normal file
225
src/memtest.c
Normal file
@ -0,0 +1,225 @@
|
|||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <assert.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <termios.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
|
||||||
|
#if (ULONG_MAX == 4294967295UL)
|
||||||
|
#define MEMTEST_32BIT
|
||||||
|
#elif (ULONG_MAX == 18446744073709551615ULL)
|
||||||
|
#define MEMTEST_64BIT
|
||||||
|
#else
|
||||||
|
#error "ULONG_MAX value not supported."
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef MEMTEST_32BIT
|
||||||
|
#define ULONG_ONEZERO 0xaaaaaaaaaaaaaaaaUL
|
||||||
|
#define ULONG_ZEROONE 0x5555555555555555UL
|
||||||
|
#else
|
||||||
|
#define ULONG_ONEZERO 0xaaaaaaaaUL
|
||||||
|
#define ULONG_ZEROONE 0x55555555UL
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static struct winsize ws;
|
||||||
|
size_t progress_printed; /* Printed chars in screen-wide progress bar. */
|
||||||
|
size_t progress_full; /* How many chars to write to fill the progress bar. */
|
||||||
|
|
||||||
|
void memtest_progress_start(char *title, int pass) {
|
||||||
|
int j;
|
||||||
|
|
||||||
|
printf("\x1b[H\x1b[2J"); /* Cursor home, clear screen. */
|
||||||
|
/* Fill with dots. */
|
||||||
|
for (j = 0; j < ws.ws_col*(ws.ws_row-2); j++) printf(".");
|
||||||
|
printf("Please keep the test running several minutes per GB of memory.\n");
|
||||||
|
printf("Also check http://www.memtest86.com/ and http://pyropus.ca/software/memtester/");
|
||||||
|
printf("\x1b[H\x1b[2K"); /* Cursor home, clear current line. */
|
||||||
|
printf("%s [%d]\n", title, pass); /* Print title. */
|
||||||
|
progress_printed = 0;
|
||||||
|
progress_full = ws.ws_col*(ws.ws_row-3);
|
||||||
|
fflush(stdout);
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest_progress_end(void) {
|
||||||
|
printf("\x1b[H\x1b[2J"); /* Cursor home, clear screen. */
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest_progress_step(size_t curr, size_t size, char c) {
|
||||||
|
size_t chars = (curr*progress_full)/size, j;
|
||||||
|
|
||||||
|
for (j = 0; j < chars-progress_printed; j++) {
|
||||||
|
printf("%c",c);
|
||||||
|
progress_printed++;
|
||||||
|
}
|
||||||
|
fflush(stdout);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test that addressing is fine. Every location is populated with its own
|
||||||
|
* address, and finally verified. This test is very fast but may detect
|
||||||
|
* ASAP big issues with the memory subsystem. */
|
||||||
|
void memtest_addressing(unsigned long *l, size_t bytes) {
|
||||||
|
unsigned long words = bytes/sizeof(unsigned long);
|
||||||
|
unsigned long j, *p;
|
||||||
|
|
||||||
|
/* Fill */
|
||||||
|
p = l;
|
||||||
|
for (j = 0; j < words; j++) {
|
||||||
|
*p = (unsigned long)p;
|
||||||
|
p++;
|
||||||
|
if ((j & 0xffff) == 0) memtest_progress_step(j,words*2,'A');
|
||||||
|
}
|
||||||
|
/* Test */
|
||||||
|
p = l;
|
||||||
|
for (j = 0; j < words; j++) {
|
||||||
|
if (*p != (unsigned long)p) {
|
||||||
|
printf("\n*** MEMORY ADDRESSING ERROR: %p contains %lu\n",
|
||||||
|
(void*) p, *p);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
p++;
|
||||||
|
if ((j & 0xffff) == 0) memtest_progress_step(j+words,words*2,'A');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fill words stepping a single page at every write, so we continue to
|
||||||
|
* touch all the pages in the smallest amount of time reducing the
|
||||||
|
* effectiveness of caches, and making it hard for the OS to transfer
|
||||||
|
* pages on the swap. */
|
||||||
|
void memtest_fill_random(unsigned long *l, size_t bytes) {
|
||||||
|
unsigned long step = 4096/sizeof(unsigned long);
|
||||||
|
unsigned long words = bytes/sizeof(unsigned long)/2;
|
||||||
|
unsigned long iwords = words/step; /* words per iteration */
|
||||||
|
unsigned long off, w, *l1, *l2;
|
||||||
|
|
||||||
|
assert((bytes & 4095) == 0);
|
||||||
|
for (off = 0; off < step; off++) {
|
||||||
|
l1 = l+off;
|
||||||
|
l2 = l1+words;
|
||||||
|
for (w = 0; w < iwords; w++) {
|
||||||
|
#ifdef MEMTEST_32BIT
|
||||||
|
*l1 = *l2 = ((unsigned long) (rand()&0xffff)) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 16);
|
||||||
|
#else
|
||||||
|
*l1 = *l2 = ((unsigned long) (rand()&0xffff)) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 16) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 32) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 48);
|
||||||
|
#endif
|
||||||
|
l1 += step;
|
||||||
|
l2 += step;
|
||||||
|
if ((w & 0xffff) == 0)
|
||||||
|
memtest_progress_step(w+iwords*off,words,'R');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Like memtest_fill_random() but uses the two specified values to fill
|
||||||
|
* memory, in an alternated way (v1|v2|v1|v2|...) */
|
||||||
|
void memtest_fill_value(unsigned long *l, size_t bytes, unsigned long v1,
|
||||||
|
unsigned long v2, char sym)
|
||||||
|
{
|
||||||
|
unsigned long step = 4096/sizeof(unsigned long);
|
||||||
|
unsigned long words = bytes/sizeof(unsigned long)/2;
|
||||||
|
unsigned long iwords = words/step; /* words per iteration */
|
||||||
|
unsigned long off, w, *l1, *l2, v;
|
||||||
|
|
||||||
|
assert((bytes & 4095) == 0);
|
||||||
|
for (off = 0; off < step; off++) {
|
||||||
|
l1 = l+off;
|
||||||
|
l2 = l1+words;
|
||||||
|
v = (off & 1) ? v2 : v1;
|
||||||
|
for (w = 0; w < iwords; w++) {
|
||||||
|
#ifdef MEMTEST_32BIT
|
||||||
|
*l1 = *l2 = ((unsigned long) (rand()&0xffff)) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 16);
|
||||||
|
#else
|
||||||
|
*l1 = *l2 = ((unsigned long) (rand()&0xffff)) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 16) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 32) |
|
||||||
|
(((unsigned long) (rand()&0xffff)) << 48);
|
||||||
|
#endif
|
||||||
|
l1 += step;
|
||||||
|
l2 += step;
|
||||||
|
if ((w & 0xffff) == 0)
|
||||||
|
memtest_progress_step(w+iwords*off,words,sym);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest_compare(unsigned long *l, size_t bytes) {
|
||||||
|
unsigned long words = bytes/sizeof(unsigned long)/2;
|
||||||
|
unsigned long w, *l1, *l2;
|
||||||
|
|
||||||
|
assert((bytes & 4095) == 0);
|
||||||
|
l1 = l;
|
||||||
|
l2 = l1+words;
|
||||||
|
for (w = 0; w < words; w++) {
|
||||||
|
if (*l1 != *l2) {
|
||||||
|
printf("\n*** MEMORY ERROR DETECTED: %p != %p (%lu vs %lu)\n",
|
||||||
|
(void*)l1, (void*)l2, *l1, *l2);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
l1 ++;
|
||||||
|
l2 ++;
|
||||||
|
if ((w & 0xffff) == 0) memtest_progress_step(w,words,'=');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest_compare_times(unsigned long *m, size_t bytes, int pass, int times) {
|
||||||
|
int j;
|
||||||
|
|
||||||
|
for (j = 0; j < times; j++) {
|
||||||
|
memtest_progress_start("Compare",pass);
|
||||||
|
memtest_compare(m,bytes);
|
||||||
|
memtest_progress_end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest_test(size_t megabytes, int passes) {
|
||||||
|
size_t bytes = megabytes*1024*1024;
|
||||||
|
unsigned long *m = malloc(bytes);
|
||||||
|
int pass = 0;
|
||||||
|
|
||||||
|
if (m == NULL) {
|
||||||
|
fprintf(stderr,"Unable to allocate %zu megabytes: %s",
|
||||||
|
megabytes, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
while (pass != passes) {
|
||||||
|
pass++;
|
||||||
|
|
||||||
|
memtest_progress_start("Addressing test",pass);
|
||||||
|
memtest_addressing(m,bytes);
|
||||||
|
memtest_progress_end();
|
||||||
|
|
||||||
|
memtest_progress_start("Random fill",pass);
|
||||||
|
memtest_fill_random(m,bytes);
|
||||||
|
memtest_progress_end();
|
||||||
|
memtest_compare_times(m,bytes,pass,4);
|
||||||
|
|
||||||
|
memtest_progress_start("Solid fill",pass);
|
||||||
|
memtest_fill_value(m,bytes,0,(unsigned long)-1,'S');
|
||||||
|
memtest_progress_end();
|
||||||
|
memtest_compare_times(m,bytes,pass,4);
|
||||||
|
|
||||||
|
memtest_progress_start("Checkerboard fill",pass);
|
||||||
|
memtest_fill_value(m,bytes,ULONG_ONEZERO,ULONG_ZEROONE,'C');
|
||||||
|
memtest_progress_end();
|
||||||
|
memtest_compare_times(m,bytes,pass,4);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void memtest(size_t megabytes, int passes) {
|
||||||
|
if (ioctl(1, TIOCGWINSZ, &ws) == -1) {
|
||||||
|
ws.ws_col = 80;
|
||||||
|
ws.ws_row = 20;
|
||||||
|
}
|
||||||
|
memtest_test(megabytes,passes);
|
||||||
|
printf("\nYour memory passed this test.\n");
|
||||||
|
printf("Please if you are still in doubt use the following two tools:\n");
|
||||||
|
printf("1) memtest86: http://www.memtest86.com/\n");
|
||||||
|
printf("2) memtester: http://pyropus.ca/software/memtester/\n");
|
||||||
|
exit(0);
|
||||||
|
}
|
13
src/multi.c
13
src/multi.c
@ -40,6 +40,13 @@ void queueMultiCommand(redisClient *c) {
|
|||||||
c->mstate.count++;
|
c->mstate.count++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void discardTransaction(redisClient *c) {
|
||||||
|
freeClientMultiState(c);
|
||||||
|
initClientMultiState(c);
|
||||||
|
c->flags &= ~(REDIS_MULTI|REDIS_DIRTY_CAS);;
|
||||||
|
unwatchAllKeys(c);
|
||||||
|
}
|
||||||
|
|
||||||
void multiCommand(redisClient *c) {
|
void multiCommand(redisClient *c) {
|
||||||
if (c->flags & REDIS_MULTI) {
|
if (c->flags & REDIS_MULTI) {
|
||||||
addReplyError(c,"MULTI calls can not be nested");
|
addReplyError(c,"MULTI calls can not be nested");
|
||||||
@ -54,11 +61,7 @@ void discardCommand(redisClient *c) {
|
|||||||
addReplyError(c,"DISCARD without MULTI");
|
addReplyError(c,"DISCARD without MULTI");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
discardTransaction(c);
|
||||||
freeClientMultiState(c);
|
|
||||||
initClientMultiState(c);
|
|
||||||
c->flags &= ~(REDIS_MULTI|REDIS_DIRTY_CAS);;
|
|
||||||
unwatchAllKeys(c);
|
|
||||||
addReply(c,shared.ok);
|
addReply(c,shared.ok);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,7 +22,6 @@ int listMatchObjects(void *a, void *b) {
|
|||||||
|
|
||||||
redisClient *createClient(int fd) {
|
redisClient *createClient(int fd) {
|
||||||
redisClient *c = zmalloc(sizeof(redisClient));
|
redisClient *c = zmalloc(sizeof(redisClient));
|
||||||
c->bufpos = 0;
|
|
||||||
|
|
||||||
/* passing -1 as fd it is possible to create a non connected client.
|
/* passing -1 as fd it is possible to create a non connected client.
|
||||||
* This is useful since all the Redis commands needs to be executed
|
* This is useful since all the Redis commands needs to be executed
|
||||||
@ -42,7 +41,9 @@ redisClient *createClient(int fd) {
|
|||||||
|
|
||||||
selectDb(c,0);
|
selectDb(c,0);
|
||||||
c->fd = fd;
|
c->fd = fd;
|
||||||
|
c->bufpos = 0;
|
||||||
c->querybuf = sdsempty();
|
c->querybuf = sdsempty();
|
||||||
|
c->querybuf_peak = 0;
|
||||||
c->reqtype = 0;
|
c->reqtype = 0;
|
||||||
c->argc = 0;
|
c->argc = 0;
|
||||||
c->argv = NULL;
|
c->argv = NULL;
|
||||||
@ -51,7 +52,7 @@ redisClient *createClient(int fd) {
|
|||||||
c->bulklen = -1;
|
c->bulklen = -1;
|
||||||
c->sentlen = 0;
|
c->sentlen = 0;
|
||||||
c->flags = 0;
|
c->flags = 0;
|
||||||
c->lastinteraction = time(NULL);
|
c->ctime = c->lastinteraction = time(NULL);
|
||||||
c->authenticated = 0;
|
c->authenticated = 0;
|
||||||
c->replstate = REDIS_REPL_NONE;
|
c->replstate = REDIS_REPL_NONE;
|
||||||
c->reply = listCreate();
|
c->reply = listCreate();
|
||||||
@ -751,34 +752,6 @@ void resetClient(redisClient *c) {
|
|||||||
if (!(c->flags & REDIS_MULTI)) c->flags &= (~REDIS_ASKING);
|
if (!(c->flags & REDIS_MULTI)) c->flags &= (~REDIS_ASKING);
|
||||||
}
|
}
|
||||||
|
|
||||||
void closeTimedoutClients(void) {
|
|
||||||
redisClient *c;
|
|
||||||
listNode *ln;
|
|
||||||
time_t now = time(NULL);
|
|
||||||
listIter li;
|
|
||||||
|
|
||||||
listRewind(server.clients,&li);
|
|
||||||
while ((ln = listNext(&li)) != NULL) {
|
|
||||||
c = listNodeValue(ln);
|
|
||||||
if (server.maxidletime &&
|
|
||||||
!(c->flags & REDIS_SLAVE) && /* no timeout for slaves */
|
|
||||||
!(c->flags & REDIS_MASTER) && /* no timeout for masters */
|
|
||||||
!(c->flags & REDIS_BLOCKED) && /* no timeout for BLPOP */
|
|
||||||
dictSize(c->pubsub_channels) == 0 && /* no timeout for pubsub */
|
|
||||||
listLength(c->pubsub_patterns) == 0 &&
|
|
||||||
(now - c->lastinteraction > server.maxidletime))
|
|
||||||
{
|
|
||||||
redisLog(REDIS_VERBOSE,"Closing idle client");
|
|
||||||
freeClient(c);
|
|
||||||
} else if (c->flags & REDIS_BLOCKED) {
|
|
||||||
if (c->bpop.timeout != 0 && c->bpop.timeout < now) {
|
|
||||||
addReply(c,shared.nullmultibulk);
|
|
||||||
unblockClientWaitingData(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int processInlineBuffer(redisClient *c) {
|
int processInlineBuffer(redisClient *c) {
|
||||||
char *newline = strstr(c->querybuf,"\r\n");
|
char *newline = strstr(c->querybuf,"\r\n");
|
||||||
int argc, j;
|
int argc, j;
|
||||||
@ -1026,6 +999,7 @@ void readQueryFromClient(aeEventLoop *el, int fd, void *privdata, int mask) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
qblen = sdslen(c->querybuf);
|
qblen = sdslen(c->querybuf);
|
||||||
|
if (c->querybuf_peak < qblen) c->querybuf_peak = qblen;
|
||||||
c->querybuf = sdsMakeRoomFor(c->querybuf, readlen);
|
c->querybuf = sdsMakeRoomFor(c->querybuf, readlen);
|
||||||
nread = read(fd, c->querybuf+qblen, readlen);
|
nread = read(fd, c->querybuf+qblen, readlen);
|
||||||
if (nread == -1) {
|
if (nread == -1) {
|
||||||
@ -1087,11 +1061,7 @@ sds getClientInfoString(redisClient *client) {
|
|||||||
time_t now = time(NULL);
|
time_t now = time(NULL);
|
||||||
int emask;
|
int emask;
|
||||||
|
|
||||||
if (anetPeerToString(client->fd,ip,&port) == -1) {
|
anetPeerToString(client->fd,ip,&port);
|
||||||
ip[0] = '?';
|
|
||||||
ip[1] = '\0';
|
|
||||||
port = 0;
|
|
||||||
}
|
|
||||||
p = flags;
|
p = flags;
|
||||||
if (client->flags & REDIS_SLAVE) {
|
if (client->flags & REDIS_SLAVE) {
|
||||||
if (client->flags & REDIS_MONITOR)
|
if (client->flags & REDIS_MONITOR)
|
||||||
@ -1115,14 +1085,16 @@ sds getClientInfoString(redisClient *client) {
|
|||||||
if (emask & AE_WRITABLE) *p++ = 'w';
|
if (emask & AE_WRITABLE) *p++ = 'w';
|
||||||
*p = '\0';
|
*p = '\0';
|
||||||
return sdscatprintf(sdsempty(),
|
return sdscatprintf(sdsempty(),
|
||||||
"addr=%s:%d fd=%d idle=%ld flags=%s db=%d sub=%d psub=%d qbuf=%lu obl=%lu oll=%lu omem=%lu events=%s cmd=%s",
|
"addr=%s:%d fd=%d age=%ld idle=%ld flags=%s db=%d sub=%d psub=%d qbuf=%lu qbuf-free=%lu obl=%lu oll=%lu omem=%lu events=%s cmd=%s",
|
||||||
ip,port,client->fd,
|
ip,port,client->fd,
|
||||||
|
(long)(now - client->ctime),
|
||||||
(long)(now - client->lastinteraction),
|
(long)(now - client->lastinteraction),
|
||||||
flags,
|
flags,
|
||||||
client->db->id,
|
client->db->id,
|
||||||
(int) dictSize(client->pubsub_channels),
|
(int) dictSize(client->pubsub_channels),
|
||||||
(int) listLength(client->pubsub_patterns),
|
(int) listLength(client->pubsub_patterns),
|
||||||
(unsigned long) sdslen(client->querybuf),
|
(unsigned long) sdslen(client->querybuf),
|
||||||
|
(unsigned long) sdsavail(client->querybuf),
|
||||||
(unsigned long) client->bufpos,
|
(unsigned long) client->bufpos,
|
||||||
(unsigned long) listLength(client->reply),
|
(unsigned long) listLength(client->reply),
|
||||||
getClientOutputBufferMemoryUsage(client),
|
getClientOutputBufferMemoryUsage(client),
|
||||||
|
23
src/object.c
23
src/object.c
@ -56,7 +56,16 @@ robj *createStringObjectFromLongDouble(long double value) {
|
|||||||
* that is "non surprising" for the user (that is, most small decimal
|
* that is "non surprising" for the user (that is, most small decimal
|
||||||
* numbers will be represented in a way that when converted back into
|
* numbers will be represented in a way that when converted back into
|
||||||
* a string are exactly the same as what the user typed.) */
|
* a string are exactly the same as what the user typed.) */
|
||||||
len = snprintf(buf,sizeof(buf),"%.17Lg", value);
|
len = snprintf(buf,sizeof(buf),"%.17Lf", value);
|
||||||
|
/* Now remove trailing zeroes after the '.' */
|
||||||
|
if (strchr(buf,'.') != NULL) {
|
||||||
|
char *p = buf+len-1;
|
||||||
|
while(*p == '0') {
|
||||||
|
p--;
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
if (*p == '.') len--;
|
||||||
|
}
|
||||||
return createStringObject(buf,len);
|
return createStringObject(buf,len);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,12 +104,9 @@ robj *createIntsetObject(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
robj *createHashObject(void) {
|
robj *createHashObject(void) {
|
||||||
/* All the Hashes start as zipmaps. Will be automatically converted
|
unsigned char *zl = ziplistNew();
|
||||||
* into hash tables if there are enough elements or big elements
|
robj *o = createObject(REDIS_HASH, zl);
|
||||||
* inside. */
|
o->encoding = REDIS_ENCODING_ZIPLIST;
|
||||||
unsigned char *zm = zipmapNew();
|
|
||||||
robj *o = createObject(REDIS_HASH,zm);
|
|
||||||
o->encoding = REDIS_ENCODING_ZIPMAP;
|
|
||||||
return o;
|
return o;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -176,7 +182,7 @@ void freeHashObject(robj *o) {
|
|||||||
case REDIS_ENCODING_HT:
|
case REDIS_ENCODING_HT:
|
||||||
dictRelease((dict*) o->ptr);
|
dictRelease((dict*) o->ptr);
|
||||||
break;
|
break;
|
||||||
case REDIS_ENCODING_ZIPMAP:
|
case REDIS_ENCODING_ZIPLIST:
|
||||||
zfree(o->ptr);
|
zfree(o->ptr);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -492,7 +498,6 @@ char *strEncoding(int encoding) {
|
|||||||
case REDIS_ENCODING_RAW: return "raw";
|
case REDIS_ENCODING_RAW: return "raw";
|
||||||
case REDIS_ENCODING_INT: return "int";
|
case REDIS_ENCODING_INT: return "int";
|
||||||
case REDIS_ENCODING_HT: return "hashtable";
|
case REDIS_ENCODING_HT: return "hashtable";
|
||||||
case REDIS_ENCODING_ZIPMAP: return "zipmap";
|
|
||||||
case REDIS_ENCODING_LINKEDLIST: return "linkedlist";
|
case REDIS_ENCODING_LINKEDLIST: return "linkedlist";
|
||||||
case REDIS_ENCODING_ZIPLIST: return "ziplist";
|
case REDIS_ENCODING_ZIPLIST: return "ziplist";
|
||||||
case REDIS_ENCODING_INTSET: return "intset";
|
case REDIS_ENCODING_INTSET: return "intset";
|
||||||
|
158
src/rdb.c
158
src/rdb.c
@ -1,5 +1,6 @@
|
|||||||
#include "redis.h"
|
#include "redis.h"
|
||||||
#include "lzf.h" /* LZF compression library */
|
#include "lzf.h" /* LZF compression library */
|
||||||
|
#include "zipmap.h"
|
||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
@ -424,8 +425,8 @@ int rdbSaveObjectType(rio *rdb, robj *o) {
|
|||||||
else
|
else
|
||||||
redisPanic("Unknown sorted set encoding");
|
redisPanic("Unknown sorted set encoding");
|
||||||
case REDIS_HASH:
|
case REDIS_HASH:
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP)
|
if (o->encoding == REDIS_ENCODING_ZIPLIST)
|
||||||
return rdbSaveType(rdb,REDIS_RDB_TYPE_HASH_ZIPMAP);
|
return rdbSaveType(rdb,REDIS_RDB_TYPE_HASH_ZIPLIST);
|
||||||
else if (o->encoding == REDIS_ENCODING_HT)
|
else if (o->encoding == REDIS_ENCODING_HT)
|
||||||
return rdbSaveType(rdb,REDIS_RDB_TYPE_HASH);
|
return rdbSaveType(rdb,REDIS_RDB_TYPE_HASH);
|
||||||
else
|
else
|
||||||
@ -530,12 +531,13 @@ int rdbSaveObject(rio *rdb, robj *o) {
|
|||||||
}
|
}
|
||||||
} else if (o->type == REDIS_HASH) {
|
} else if (o->type == REDIS_HASH) {
|
||||||
/* Save a hash value */
|
/* Save a hash value */
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
size_t l = zipmapBlobLen((unsigned char*)o->ptr);
|
size_t l = ziplistBlobLen((unsigned char*)o->ptr);
|
||||||
|
|
||||||
if ((n = rdbSaveRawString(rdb,o->ptr,l)) == -1) return -1;
|
if ((n = rdbSaveRawString(rdb,o->ptr,l)) == -1) return -1;
|
||||||
nwritten += n;
|
nwritten += n;
|
||||||
} else {
|
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
dictIterator *di = dictGetIterator(o->ptr);
|
dictIterator *di = dictGetIterator(o->ptr);
|
||||||
dictEntry *de;
|
dictEntry *de;
|
||||||
|
|
||||||
@ -552,7 +554,11 @@ int rdbSaveObject(rio *rdb, robj *o) {
|
|||||||
nwritten += n;
|
nwritten += n;
|
||||||
}
|
}
|
||||||
dictReleaseIterator(di);
|
dictReleaseIterator(di);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
redisPanic("Unknown object type");
|
redisPanic("Unknown object type");
|
||||||
}
|
}
|
||||||
@ -610,7 +616,7 @@ int rdbSave(char *filename) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
rioInitWithFile(&rdb,fp);
|
rioInitWithFile(&rdb,fp);
|
||||||
if (rdbWriteRaw(&rdb,"REDIS0003",9) == -1) goto werr;
|
if (rdbWriteRaw(&rdb,"REDIS0004",9) == -1) goto werr;
|
||||||
|
|
||||||
for (j = 0; j < server.dbnum; j++) {
|
for (j = 0; j < server.dbnum; j++) {
|
||||||
redisDb *db = server.db+j;
|
redisDb *db = server.db+j;
|
||||||
@ -656,6 +662,7 @@ int rdbSave(char *filename) {
|
|||||||
redisLog(REDIS_NOTICE,"DB saved on disk");
|
redisLog(REDIS_NOTICE,"DB saved on disk");
|
||||||
server.dirty = 0;
|
server.dirty = 0;
|
||||||
server.lastsave = time(NULL);
|
server.lastsave = time(NULL);
|
||||||
|
server.lastbgsave_status = REDIS_OK;
|
||||||
return REDIS_OK;
|
return REDIS_OK;
|
||||||
|
|
||||||
werr:
|
werr:
|
||||||
@ -824,55 +831,74 @@ robj *rdbLoadObject(int rdbtype, rio *rdb) {
|
|||||||
maxelelen <= server.zset_max_ziplist_value)
|
maxelelen <= server.zset_max_ziplist_value)
|
||||||
zsetConvert(o,REDIS_ENCODING_ZIPLIST);
|
zsetConvert(o,REDIS_ENCODING_ZIPLIST);
|
||||||
} else if (rdbtype == REDIS_RDB_TYPE_HASH) {
|
} else if (rdbtype == REDIS_RDB_TYPE_HASH) {
|
||||||
size_t hashlen;
|
size_t len;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
len = rdbLoadLen(rdb, NULL);
|
||||||
|
if (len == REDIS_RDB_LENERR) return NULL;
|
||||||
|
|
||||||
if ((hashlen = rdbLoadLen(rdb,NULL)) == REDIS_RDB_LENERR) return NULL;
|
|
||||||
o = createHashObject();
|
o = createHashObject();
|
||||||
|
|
||||||
/* Too many entries? Use an hash table. */
|
/* Too many entries? Use an hash table. */
|
||||||
if (hashlen > server.hash_max_zipmap_entries)
|
if (len > server.hash_max_ziplist_entries)
|
||||||
convertToRealHash(o);
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
/* Load every key/value, then set it into the zipmap or hash
|
|
||||||
* table, as needed. */
|
|
||||||
while(hashlen--) {
|
|
||||||
robj *key, *val;
|
|
||||||
|
|
||||||
if ((key = rdbLoadEncodedStringObject(rdb)) == NULL) return NULL;
|
/* Load every field and value into the ziplist */
|
||||||
if ((val = rdbLoadEncodedStringObject(rdb)) == NULL) return NULL;
|
while (o->encoding == REDIS_ENCODING_ZIPLIST && len > 0) {
|
||||||
/* If we are using a zipmap and there are too big values
|
robj *field, *value;
|
||||||
* the object is converted to real hash table encoding. */
|
|
||||||
if (o->encoding != REDIS_ENCODING_HT &&
|
len--;
|
||||||
((key->encoding == REDIS_ENCODING_RAW &&
|
/* Load raw strings */
|
||||||
sdslen(key->ptr) > server.hash_max_zipmap_value) ||
|
field = rdbLoadStringObject(rdb);
|
||||||
(val->encoding == REDIS_ENCODING_RAW &&
|
if (field == NULL) return NULL;
|
||||||
sdslen(val->ptr) > server.hash_max_zipmap_value)))
|
redisAssert(field->encoding == REDIS_ENCODING_RAW);
|
||||||
|
value = rdbLoadStringObject(rdb);
|
||||||
|
if (value == NULL) return NULL;
|
||||||
|
redisAssert(field->encoding == REDIS_ENCODING_RAW);
|
||||||
|
|
||||||
|
/* Add pair to ziplist */
|
||||||
|
o->ptr = ziplistPush(o->ptr, field->ptr, sdslen(field->ptr), ZIPLIST_TAIL);
|
||||||
|
o->ptr = ziplistPush(o->ptr, value->ptr, sdslen(value->ptr), ZIPLIST_TAIL);
|
||||||
|
/* Convert to hash table if size threshold is exceeded */
|
||||||
|
if (sdslen(field->ptr) > server.hash_max_ziplist_value ||
|
||||||
|
sdslen(value->ptr) > server.hash_max_ziplist_value)
|
||||||
{
|
{
|
||||||
convertToRealHash(o);
|
decrRefCount(field);
|
||||||
}
|
decrRefCount(value);
|
||||||
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
break;
|
||||||
unsigned char *zm = o->ptr;
|
|
||||||
robj *deckey, *decval;
|
|
||||||
|
|
||||||
/* We need raw string objects to add them to the zipmap */
|
|
||||||
deckey = getDecodedObject(key);
|
|
||||||
decval = getDecodedObject(val);
|
|
||||||
zm = zipmapSet(zm,deckey->ptr,sdslen(deckey->ptr),
|
|
||||||
decval->ptr,sdslen(decval->ptr),NULL);
|
|
||||||
o->ptr = zm;
|
|
||||||
decrRefCount(deckey);
|
|
||||||
decrRefCount(decval);
|
|
||||||
decrRefCount(key);
|
|
||||||
decrRefCount(val);
|
|
||||||
} else {
|
|
||||||
key = tryObjectEncoding(key);
|
|
||||||
val = tryObjectEncoding(val);
|
|
||||||
dictAdd((dict*)o->ptr,key,val);
|
|
||||||
}
|
}
|
||||||
|
decrRefCount(field);
|
||||||
|
decrRefCount(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Load remaining fields and values into the hash table */
|
||||||
|
while (o->encoding == REDIS_ENCODING_HT && len > 0) {
|
||||||
|
robj *field, *value;
|
||||||
|
|
||||||
|
len--;
|
||||||
|
/* Load encoded strings */
|
||||||
|
field = rdbLoadEncodedStringObject(rdb);
|
||||||
|
if (field == NULL) return NULL;
|
||||||
|
value = rdbLoadEncodedStringObject(rdb);
|
||||||
|
if (value == NULL) return NULL;
|
||||||
|
|
||||||
|
field = tryObjectEncoding(field);
|
||||||
|
value = tryObjectEncoding(value);
|
||||||
|
|
||||||
|
/* Add pair to hash table */
|
||||||
|
ret = dictAdd((dict*)o->ptr, field, value);
|
||||||
|
redisAssert(ret == REDIS_OK);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* All pairs should be read by now */
|
||||||
|
redisAssert(len == 0);
|
||||||
|
|
||||||
} else if (rdbtype == REDIS_RDB_TYPE_HASH_ZIPMAP ||
|
} else if (rdbtype == REDIS_RDB_TYPE_HASH_ZIPMAP ||
|
||||||
rdbtype == REDIS_RDB_TYPE_LIST_ZIPLIST ||
|
rdbtype == REDIS_RDB_TYPE_LIST_ZIPLIST ||
|
||||||
rdbtype == REDIS_RDB_TYPE_SET_INTSET ||
|
rdbtype == REDIS_RDB_TYPE_SET_INTSET ||
|
||||||
rdbtype == REDIS_RDB_TYPE_ZSET_ZIPLIST)
|
rdbtype == REDIS_RDB_TYPE_ZSET_ZIPLIST ||
|
||||||
|
rdbtype == REDIS_RDB_TYPE_HASH_ZIPLIST)
|
||||||
{
|
{
|
||||||
robj *aux = rdbLoadStringObject(rdb);
|
robj *aux = rdbLoadStringObject(rdb);
|
||||||
|
|
||||||
@ -890,10 +916,33 @@ robj *rdbLoadObject(int rdbtype, rio *rdb) {
|
|||||||
* converted. */
|
* converted. */
|
||||||
switch(rdbtype) {
|
switch(rdbtype) {
|
||||||
case REDIS_RDB_TYPE_HASH_ZIPMAP:
|
case REDIS_RDB_TYPE_HASH_ZIPMAP:
|
||||||
o->type = REDIS_HASH;
|
/* Convert to ziplist encoded hash. This must be deprecated
|
||||||
o->encoding = REDIS_ENCODING_ZIPMAP;
|
* when loading dumps created by Redis 2.4 gets deprecated. */
|
||||||
if (zipmapLen(o->ptr) > server.hash_max_zipmap_entries)
|
{
|
||||||
convertToRealHash(o);
|
unsigned char *zl = ziplistNew();
|
||||||
|
unsigned char *zi = zipmapRewind(o->ptr);
|
||||||
|
unsigned char *fstr, *vstr;
|
||||||
|
unsigned int flen, vlen;
|
||||||
|
unsigned int maxlen = 0;
|
||||||
|
|
||||||
|
while ((zi = zipmapNext(zi, &fstr, &flen, &vstr, &vlen)) != NULL) {
|
||||||
|
if (flen > maxlen) maxlen = flen;
|
||||||
|
if (vlen > maxlen) maxlen = vlen;
|
||||||
|
zl = ziplistPush(zl, fstr, flen, ZIPLIST_TAIL);
|
||||||
|
zl = ziplistPush(zl, vstr, vlen, ZIPLIST_TAIL);
|
||||||
|
}
|
||||||
|
|
||||||
|
zfree(o->ptr);
|
||||||
|
o->ptr = zl;
|
||||||
|
o->type = REDIS_HASH;
|
||||||
|
o->encoding = REDIS_ENCODING_ZIPLIST;
|
||||||
|
|
||||||
|
if (hashTypeLength(o) > server.hash_max_ziplist_entries ||
|
||||||
|
maxlen > server.hash_max_ziplist_value)
|
||||||
|
{
|
||||||
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case REDIS_RDB_TYPE_LIST_ZIPLIST:
|
case REDIS_RDB_TYPE_LIST_ZIPLIST:
|
||||||
o->type = REDIS_LIST;
|
o->type = REDIS_LIST;
|
||||||
@ -913,6 +962,12 @@ robj *rdbLoadObject(int rdbtype, rio *rdb) {
|
|||||||
if (zsetLength(o) > server.zset_max_ziplist_entries)
|
if (zsetLength(o) > server.zset_max_ziplist_entries)
|
||||||
zsetConvert(o,REDIS_ENCODING_SKIPLIST);
|
zsetConvert(o,REDIS_ENCODING_SKIPLIST);
|
||||||
break;
|
break;
|
||||||
|
case REDIS_RDB_TYPE_HASH_ZIPLIST:
|
||||||
|
o->type = REDIS_HASH;
|
||||||
|
o->encoding = REDIS_ENCODING_ZIPLIST;
|
||||||
|
if (hashTypeLength(o) > server.hash_max_ziplist_entries)
|
||||||
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
redisPanic("Unknown encoding");
|
redisPanic("Unknown encoding");
|
||||||
break;
|
break;
|
||||||
@ -973,7 +1028,7 @@ int rdbLoad(char *filename) {
|
|||||||
return REDIS_ERR;
|
return REDIS_ERR;
|
||||||
}
|
}
|
||||||
rdbver = atoi(buf+5);
|
rdbver = atoi(buf+5);
|
||||||
if (rdbver < 1 || rdbver > 3) {
|
if (rdbver < 1 || rdbver > 4) {
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
redisLog(REDIS_WARNING,"Can't handle RDB format version %d",rdbver);
|
redisLog(REDIS_WARNING,"Can't handle RDB format version %d",rdbver);
|
||||||
errno = EINVAL;
|
errno = EINVAL;
|
||||||
@ -1061,12 +1116,15 @@ void backgroundSaveDoneHandler(int exitcode, int bysignal) {
|
|||||||
"Background saving terminated with success");
|
"Background saving terminated with success");
|
||||||
server.dirty = server.dirty - server.dirty_before_bgsave;
|
server.dirty = server.dirty - server.dirty_before_bgsave;
|
||||||
server.lastsave = time(NULL);
|
server.lastsave = time(NULL);
|
||||||
|
server.lastbgsave_status = REDIS_OK;
|
||||||
} else if (!bysignal && exitcode != 0) {
|
} else if (!bysignal && exitcode != 0) {
|
||||||
redisLog(REDIS_WARNING, "Background saving error");
|
redisLog(REDIS_WARNING, "Background saving error");
|
||||||
|
server.lastbgsave_status = REDIS_ERR;
|
||||||
} else {
|
} else {
|
||||||
redisLog(REDIS_WARNING,
|
redisLog(REDIS_WARNING,
|
||||||
"Background saving terminated by signal %d", bysignal);
|
"Background saving terminated by signal %d", bysignal);
|
||||||
rdbRemoveTempFile(server.rdb_child_pid);
|
rdbRemoveTempFile(server.rdb_child_pid);
|
||||||
|
server.lastbgsave_status = REDIS_ERR;
|
||||||
}
|
}
|
||||||
server.rdb_child_pid = -1;
|
server.rdb_child_pid = -1;
|
||||||
/* Possibly there are slaves waiting for a BGSAVE in order to be served
|
/* Possibly there are slaves waiting for a BGSAVE in order to be served
|
||||||
|
@ -47,6 +47,7 @@
|
|||||||
#define REDIS_RDB_TYPE_LIST_ZIPLIST 10
|
#define REDIS_RDB_TYPE_LIST_ZIPLIST 10
|
||||||
#define REDIS_RDB_TYPE_SET_INTSET 11
|
#define REDIS_RDB_TYPE_SET_INTSET 11
|
||||||
#define REDIS_RDB_TYPE_ZSET_ZIPLIST 12
|
#define REDIS_RDB_TYPE_ZSET_ZIPLIST 12
|
||||||
|
#define REDIS_RDB_TYPE_HASH_ZIPLIST 13
|
||||||
|
|
||||||
/* Test if a type is an object type. */
|
/* Test if a type is an object type. */
|
||||||
#define rdbIsObjectType(t) ((t >= 0 && t <= 4) || (t >= 9 && t <= 12))
|
#define rdbIsObjectType(t) ((t >= 0 && t <= 4) || (t >= 9 && t <= 12))
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
#define REDIS_LIST_ZIPLIST 10
|
#define REDIS_LIST_ZIPLIST 10
|
||||||
#define REDIS_SET_INTSET 11
|
#define REDIS_SET_INTSET 11
|
||||||
#define REDIS_ZSET_ZIPLIST 12
|
#define REDIS_ZSET_ZIPLIST 12
|
||||||
|
#define REDIS_HASH_ZIPLIST 13
|
||||||
|
|
||||||
/* Objects encoding. Some kind of objects like Strings and Hashes can be
|
/* Objects encoding. Some kind of objects like Strings and Hashes can be
|
||||||
* internally represented in multiple ways. The 'encoding' field of the object
|
* internally represented in multiple ways. The 'encoding' field of the object
|
||||||
@ -136,7 +137,7 @@ int processHeader() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dump_version = (int)strtol(buf + 5, NULL, 10);
|
dump_version = (int)strtol(buf + 5, NULL, 10);
|
||||||
if (dump_version < 1 || dump_version > 2) {
|
if (dump_version < 1 || dump_version > 4) {
|
||||||
ERROR("Unknown RDB format version: %d\n", dump_version);
|
ERROR("Unknown RDB format version: %d\n", dump_version);
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
@ -384,6 +385,7 @@ int loadPair(entry *e) {
|
|||||||
case REDIS_LIST_ZIPLIST:
|
case REDIS_LIST_ZIPLIST:
|
||||||
case REDIS_SET_INTSET:
|
case REDIS_SET_INTSET:
|
||||||
case REDIS_ZSET_ZIPLIST:
|
case REDIS_ZSET_ZIPLIST:
|
||||||
|
case REDIS_HASH_ZIPLIST:
|
||||||
if (!processStringObject(NULL)) {
|
if (!processStringObject(NULL)) {
|
||||||
SHIFT_ERROR(offset, "Error reading entry value");
|
SHIFT_ERROR(offset, "Error reading entry value");
|
||||||
return 0;
|
return 0;
|
||||||
|
123
src/redis-cli.c
123
src/redis-cli.c
@ -49,6 +49,10 @@
|
|||||||
|
|
||||||
#define REDIS_NOTUSED(V) ((void) V)
|
#define REDIS_NOTUSED(V) ((void) V)
|
||||||
|
|
||||||
|
#define OUTPUT_STANDARD 0
|
||||||
|
#define OUTPUT_RAW 1
|
||||||
|
#define OUTPUT_CSV 2
|
||||||
|
|
||||||
static redisContext *context;
|
static redisContext *context;
|
||||||
static struct config {
|
static struct config {
|
||||||
char *hostip;
|
char *hostip;
|
||||||
@ -64,9 +68,10 @@ static struct config {
|
|||||||
int latency_mode;
|
int latency_mode;
|
||||||
int cluster_mode;
|
int cluster_mode;
|
||||||
int cluster_reissue_command;
|
int cluster_reissue_command;
|
||||||
|
int slave_mode;
|
||||||
int stdinarg; /* get last arg from stdin. (-x option) */
|
int stdinarg; /* get last arg from stdin. (-x option) */
|
||||||
char *auth;
|
char *auth;
|
||||||
int raw_output; /* output mode per command */
|
int output; /* output mode, see OUTPUT_* defines */
|
||||||
sds mb_delim;
|
sds mb_delim;
|
||||||
char prompt[128];
|
char prompt[128];
|
||||||
char *eval;
|
char *eval;
|
||||||
@ -434,10 +439,46 @@ static sds cliFormatReplyRaw(redisReply *r) {
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static sds cliFormatReplyCSV(redisReply *r) {
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
sds out = sdsempty();
|
||||||
|
switch (r->type) {
|
||||||
|
case REDIS_REPLY_ERROR:
|
||||||
|
out = sdscat(out,"ERROR,");
|
||||||
|
out = sdscatrepr(out,r->str,strlen(r->str));
|
||||||
|
break;
|
||||||
|
case REDIS_REPLY_STATUS:
|
||||||
|
out = sdscatrepr(out,r->str,r->len);
|
||||||
|
break;
|
||||||
|
case REDIS_REPLY_INTEGER:
|
||||||
|
out = sdscatprintf(out,"%lld",r->integer);
|
||||||
|
break;
|
||||||
|
case REDIS_REPLY_STRING:
|
||||||
|
out = sdscatrepr(out,r->str,r->len);
|
||||||
|
break;
|
||||||
|
case REDIS_REPLY_NIL:
|
||||||
|
out = sdscat(out,"NIL\n");
|
||||||
|
break;
|
||||||
|
case REDIS_REPLY_ARRAY:
|
||||||
|
for (i = 0; i < r->elements; i++) {
|
||||||
|
sds tmp = cliFormatReplyCSV(r->element[i]);
|
||||||
|
out = sdscatlen(out,tmp,sdslen(tmp));
|
||||||
|
if (i != r->elements-1) out = sdscat(out,",");
|
||||||
|
sdsfree(tmp);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
fprintf(stderr,"Unknown reply type: %d\n", r->type);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
static int cliReadReply(int output_raw_strings) {
|
static int cliReadReply(int output_raw_strings) {
|
||||||
void *_reply;
|
void *_reply;
|
||||||
redisReply *reply;
|
redisReply *reply;
|
||||||
sds out;
|
sds out = NULL;
|
||||||
int output = 1;
|
int output = 1;
|
||||||
|
|
||||||
if (redisGetReply(context,&_reply) != REDIS_OK) {
|
if (redisGetReply(context,&_reply) != REDIS_OK) {
|
||||||
@ -457,7 +498,8 @@ static int cliReadReply(int output_raw_strings) {
|
|||||||
|
|
||||||
reply = (redisReply*)_reply;
|
reply = (redisReply*)_reply;
|
||||||
|
|
||||||
/* Check if we need to connect to a different node and reissue the request. */
|
/* Check if we need to connect to a different node and reissue the
|
||||||
|
* request. */
|
||||||
if (config.cluster_mode && reply->type == REDIS_REPLY_ERROR &&
|
if (config.cluster_mode && reply->type == REDIS_REPLY_ERROR &&
|
||||||
(!strncmp(reply->str,"MOVED",5) || !strcmp(reply->str,"ASK")))
|
(!strncmp(reply->str,"MOVED",5) || !strcmp(reply->str,"ASK")))
|
||||||
{
|
{
|
||||||
@ -489,11 +531,14 @@ static int cliReadReply(int output_raw_strings) {
|
|||||||
if (output_raw_strings) {
|
if (output_raw_strings) {
|
||||||
out = cliFormatReplyRaw(reply);
|
out = cliFormatReplyRaw(reply);
|
||||||
} else {
|
} else {
|
||||||
if (config.raw_output) {
|
if (config.output == OUTPUT_RAW) {
|
||||||
out = cliFormatReplyRaw(reply);
|
out = cliFormatReplyRaw(reply);
|
||||||
out = sdscat(out,"\n");
|
out = sdscat(out,"\n");
|
||||||
} else {
|
} else if (config.output == OUTPUT_STANDARD) {
|
||||||
out = cliFormatReplyTTY(reply,"");
|
out = cliFormatReplyTTY(reply,"");
|
||||||
|
} else if (config.output == OUTPUT_CSV) {
|
||||||
|
out = cliFormatReplyCSV(reply);
|
||||||
|
out = sdscat(out,"\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fwrite(out,sdslen(out),1,stdout);
|
fwrite(out,sdslen(out),1,stdout);
|
||||||
@ -545,7 +590,7 @@ static int cliSendCommand(int argc, char **argv, int repeat) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (config.pubsub_mode) {
|
if (config.pubsub_mode) {
|
||||||
if (!config.raw_output)
|
if (config.output != OUTPUT_RAW)
|
||||||
printf("Reading messages... (press Ctrl-C to quit)\n");
|
printf("Reading messages... (press Ctrl-C to quit)\n");
|
||||||
while (1) {
|
while (1) {
|
||||||
if (cliReadReply(output_raw) != REDIS_OK) exit(1);
|
if (cliReadReply(output_raw) != REDIS_OK) exit(1);
|
||||||
@ -603,9 +648,13 @@ static int parseOptions(int argc, char **argv) {
|
|||||||
} else if (!strcmp(argv[i],"-a") && !lastarg) {
|
} else if (!strcmp(argv[i],"-a") && !lastarg) {
|
||||||
config.auth = argv[++i];
|
config.auth = argv[++i];
|
||||||
} else if (!strcmp(argv[i],"--raw")) {
|
} else if (!strcmp(argv[i],"--raw")) {
|
||||||
config.raw_output = 1;
|
config.output = OUTPUT_RAW;
|
||||||
|
} else if (!strcmp(argv[i],"--csv")) {
|
||||||
|
config.output = OUTPUT_CSV;
|
||||||
} else if (!strcmp(argv[i],"--latency")) {
|
} else if (!strcmp(argv[i],"--latency")) {
|
||||||
config.latency_mode = 1;
|
config.latency_mode = 1;
|
||||||
|
} else if (!strcmp(argv[i],"--slave")) {
|
||||||
|
config.slave_mode = 1;
|
||||||
} else if (!strcmp(argv[i],"--eval") && !lastarg) {
|
} else if (!strcmp(argv[i],"--eval") && !lastarg) {
|
||||||
config.eval = argv[++i];
|
config.eval = argv[++i];
|
||||||
} else if (!strcmp(argv[i],"-c")) {
|
} else if (!strcmp(argv[i],"-c")) {
|
||||||
@ -661,6 +710,7 @@ static void usage() {
|
|||||||
" -c Enable cluster mode (follow -ASK and -MOVED redirections)\n"
|
" -c Enable cluster mode (follow -ASK and -MOVED redirections)\n"
|
||||||
" --raw Use raw formatting for replies (default when STDOUT is not a tty)\n"
|
" --raw Use raw formatting for replies (default when STDOUT is not a tty)\n"
|
||||||
" --latency Enter a special mode continuously sampling latency.\n"
|
" --latency Enter a special mode continuously sampling latency.\n"
|
||||||
|
" --slave Simulate a slave showing commands received from the master.\n"
|
||||||
" --eval <file> Send an EVAL command using the Lua script at <file>.\n"
|
" --eval <file> Send an EVAL command using the Lua script at <file>.\n"
|
||||||
" --help Output this help and exit\n"
|
" --help Output this help and exit\n"
|
||||||
" --version Output version and exit\n"
|
" --version Output version and exit\n"
|
||||||
@ -866,6 +916,54 @@ static void latencyMode(void) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void slaveMode(void) {
|
||||||
|
/* To start we need to send the SYNC command and return the payload.
|
||||||
|
* The hiredis client lib does not understand this part of the protocol
|
||||||
|
* and we don't want to mess with its buffers, so everything is performed
|
||||||
|
* using direct low-level I/O. */
|
||||||
|
int fd = context->fd;
|
||||||
|
char buf[1024], *p;
|
||||||
|
ssize_t nread;
|
||||||
|
unsigned long long payload;
|
||||||
|
|
||||||
|
/* Send the SYNC command. */
|
||||||
|
if (write(fd,"SYNC\r\n",6) != 6) {
|
||||||
|
fprintf(stderr,"Error writing to master\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read $<payload>\r\n, making sure to read just up to "\n" */
|
||||||
|
p = buf;
|
||||||
|
while(1) {
|
||||||
|
nread = read(fd,p,1);
|
||||||
|
if (nread <= 0) {
|
||||||
|
fprintf(stderr,"Error reading bulk length while SYNCing\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
if (*p == '\n') break;
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
*p = '\0';
|
||||||
|
payload = strtoull(buf+1,NULL,10);
|
||||||
|
fprintf(stderr,"SYNC with master, discarding %lld bytes of bulk tranfer...\n",
|
||||||
|
payload);
|
||||||
|
|
||||||
|
/* Discard the payload. */
|
||||||
|
while(payload) {
|
||||||
|
nread = read(fd,buf,(payload > sizeof(buf)) ? sizeof(buf) : payload);
|
||||||
|
if (nread <= 0) {
|
||||||
|
fprintf(stderr,"Error reading RDB payload while SYNCing\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
payload -= nread;
|
||||||
|
}
|
||||||
|
fprintf(stderr,"SYNC done. Logging commands from master.\n");
|
||||||
|
|
||||||
|
/* Now we can use the hiredis to read the incoming protocol. */
|
||||||
|
config.output = OUTPUT_CSV;
|
||||||
|
while (cliReadReply(0) == REDIS_OK);
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
int firstarg;
|
int firstarg;
|
||||||
|
|
||||||
@ -884,7 +982,10 @@ int main(int argc, char **argv) {
|
|||||||
config.stdinarg = 0;
|
config.stdinarg = 0;
|
||||||
config.auth = NULL;
|
config.auth = NULL;
|
||||||
config.eval = NULL;
|
config.eval = NULL;
|
||||||
config.raw_output = !isatty(fileno(stdout)) && (getenv("FAKETTY") == NULL);
|
if (!isatty(fileno(stdout)) && (getenv("FAKETTY") == NULL))
|
||||||
|
config.output = OUTPUT_RAW;
|
||||||
|
else
|
||||||
|
config.output = OUTPUT_STANDARD;
|
||||||
config.mb_delim = sdsnew("\n");
|
config.mb_delim = sdsnew("\n");
|
||||||
cliInitHelp();
|
cliInitHelp();
|
||||||
|
|
||||||
@ -898,6 +999,12 @@ int main(int argc, char **argv) {
|
|||||||
latencyMode();
|
latencyMode();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Start in slave mode if appropriate */
|
||||||
|
if (config.slave_mode) {
|
||||||
|
cliConnect(0);
|
||||||
|
slaveMode();
|
||||||
|
}
|
||||||
|
|
||||||
/* Start interactive mode when no command is provided */
|
/* Start interactive mode when no command is provided */
|
||||||
if (argc == 0 && !config.eval) {
|
if (argc == 0 && !config.eval) {
|
||||||
/* Note that in repl mode we don't abort on connection error.
|
/* Note that in repl mode we don't abort on connection error.
|
||||||
|
306
src/redis.c
306
src/redis.c
@ -211,7 +211,7 @@ struct redisCommand redisCommandTable[] = {
|
|||||||
{"lastsave",lastsaveCommand,1,"r",0,NULL,0,0,0,0,0},
|
{"lastsave",lastsaveCommand,1,"r",0,NULL,0,0,0,0,0},
|
||||||
{"type",typeCommand,2,"r",0,NULL,1,1,1,0,0},
|
{"type",typeCommand,2,"r",0,NULL,1,1,1,0,0},
|
||||||
{"multi",multiCommand,1,"rs",0,NULL,0,0,0,0,0},
|
{"multi",multiCommand,1,"rs",0,NULL,0,0,0,0,0},
|
||||||
{"exec",execCommand,1,"wms",0,NULL,0,0,0,0,0},
|
{"exec",execCommand,1,"s",0,NULL,0,0,0,0,0},
|
||||||
{"discard",discardCommand,1,"rs",0,NULL,0,0,0,0,0},
|
{"discard",discardCommand,1,"rs",0,NULL,0,0,0,0,0},
|
||||||
{"sync",syncCommand,1,"ars",0,NULL,0,0,0,0,0},
|
{"sync",syncCommand,1,"ars",0,NULL,0,0,0,0,0},
|
||||||
{"flushdb",flushdbCommand,1,"w",0,NULL,0,0,0,0,0},
|
{"flushdb",flushdbCommand,1,"w",0,NULL,0,0,0,0,0},
|
||||||
@ -223,13 +223,13 @@ struct redisCommand redisCommandTable[] = {
|
|||||||
{"pttl",pttlCommand,2,"r",0,NULL,1,1,1,0,0},
|
{"pttl",pttlCommand,2,"r",0,NULL,1,1,1,0,0},
|
||||||
{"persist",persistCommand,2,"w",0,NULL,1,1,1,0,0},
|
{"persist",persistCommand,2,"w",0,NULL,1,1,1,0,0},
|
||||||
{"slaveof",slaveofCommand,3,"aws",0,NULL,0,0,0,0,0},
|
{"slaveof",slaveofCommand,3,"aws",0,NULL,0,0,0,0,0},
|
||||||
{"debug",debugCommand,-2,"aws",0,NULL,0,0,0,0,0},
|
{"debug",debugCommand,-2,"as",0,NULL,0,0,0,0,0},
|
||||||
{"config",configCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
{"config",configCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
||||||
{"subscribe",subscribeCommand,-2,"rps",0,NULL,0,0,0,0,0},
|
{"subscribe",subscribeCommand,-2,"rps",0,NULL,0,0,0,0,0},
|
||||||
{"unsubscribe",unsubscribeCommand,-1,"rps",0,NULL,0,0,0,0,0},
|
{"unsubscribe",unsubscribeCommand,-1,"rps",0,NULL,0,0,0,0,0},
|
||||||
{"psubscribe",psubscribeCommand,-2,"rps",0,NULL,0,0,0,0,0},
|
{"psubscribe",psubscribeCommand,-2,"rps",0,NULL,0,0,0,0,0},
|
||||||
{"punsubscribe",punsubscribeCommand,-1,"rps",0,NULL,0,0,0,0,0},
|
{"punsubscribe",punsubscribeCommand,-1,"rps",0,NULL,0,0,0,0,0},
|
||||||
{"publish",publishCommand,3,"rpf",0,NULL,0,0,0,0,0},
|
{"publish",publishCommand,3,"pf",0,NULL,0,0,0,0,0},
|
||||||
{"watch",watchCommand,-2,"rs",0,noPreloadGetKeys,1,-1,1,0,0},
|
{"watch",watchCommand,-2,"rs",0,noPreloadGetKeys,1,-1,1,0,0},
|
||||||
{"unwatch",unwatchCommand,1,"rs",0,NULL,0,0,0,0,0},
|
{"unwatch",unwatchCommand,1,"rs",0,NULL,0,0,0,0,0},
|
||||||
{"cluster",clusterCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
{"cluster",clusterCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
||||||
@ -239,10 +239,11 @@ struct redisCommand redisCommandTable[] = {
|
|||||||
{"dump",dumpCommand,2,"ar",0,NULL,1,1,1,0,0},
|
{"dump",dumpCommand,2,"ar",0,NULL,1,1,1,0,0},
|
||||||
{"object",objectCommand,-2,"r",0,NULL,2,2,2,0,0},
|
{"object",objectCommand,-2,"r",0,NULL,2,2,2,0,0},
|
||||||
{"client",clientCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
{"client",clientCommand,-2,"ar",0,NULL,0,0,0,0,0},
|
||||||
{"eval",evalCommand,-3,"wms",0,zunionInterGetKeys,0,0,0,0,0},
|
{"eval",evalCommand,-3,"s",0,zunionInterGetKeys,0,0,0,0,0},
|
||||||
{"evalsha",evalShaCommand,-3,"wms",0,zunionInterGetKeys,0,0,0,0,0},
|
{"evalsha",evalShaCommand,-3,"s",0,zunionInterGetKeys,0,0,0,0,0},
|
||||||
{"slowlog",slowlogCommand,-2,"r",0,NULL,0,0,0,0,0},
|
{"slowlog",slowlogCommand,-2,"r",0,NULL,0,0,0,0,0},
|
||||||
{"script",scriptCommand,-2,"ras",0,NULL,0,0,0,0,0}
|
{"script",scriptCommand,-2,"ras",0,NULL,0,0,0,0,0},
|
||||||
|
{"time",timeCommand,1,"rR",0,NULL,0,0,0,0,0}
|
||||||
};
|
};
|
||||||
|
|
||||||
/*============================ Utility functions ============================ */
|
/*============================ Utility functions ============================ */
|
||||||
@ -615,6 +616,110 @@ void updateLRUClock(void) {
|
|||||||
REDIS_LRU_CLOCK_MAX;
|
REDIS_LRU_CLOCK_MAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Add a sample to the operations per second array of samples. */
|
||||||
|
void trackOperationsPerSecond(void) {
|
||||||
|
long long t = mstime() - server.ops_sec_last_sample_time;
|
||||||
|
long long ops = server.stat_numcommands - server.ops_sec_last_sample_ops;
|
||||||
|
long long ops_sec;
|
||||||
|
|
||||||
|
ops_sec = t > 0 ? (ops*1000/t) : 0;
|
||||||
|
|
||||||
|
server.ops_sec_samples[server.ops_sec_idx] = ops_sec;
|
||||||
|
server.ops_sec_idx = (server.ops_sec_idx+1) % REDIS_OPS_SEC_SAMPLES;
|
||||||
|
server.ops_sec_last_sample_time = mstime();
|
||||||
|
server.ops_sec_last_sample_ops = server.stat_numcommands;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the mean of all the samples. */
|
||||||
|
long long getOperationsPerSecond(void) {
|
||||||
|
int j;
|
||||||
|
long long sum = 0;
|
||||||
|
|
||||||
|
for (j = 0; j < REDIS_OPS_SEC_SAMPLES; j++)
|
||||||
|
sum += server.ops_sec_samples[j];
|
||||||
|
return sum / REDIS_OPS_SEC_SAMPLES;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check for timeouts. Returns non-zero if the client was terminated */
|
||||||
|
int clientsCronHandleTimeout(redisClient *c) {
|
||||||
|
time_t now = server.unixtime;
|
||||||
|
|
||||||
|
if (server.maxidletime &&
|
||||||
|
!(c->flags & REDIS_SLAVE) && /* no timeout for slaves */
|
||||||
|
!(c->flags & REDIS_MASTER) && /* no timeout for masters */
|
||||||
|
!(c->flags & REDIS_BLOCKED) && /* no timeout for BLPOP */
|
||||||
|
dictSize(c->pubsub_channels) == 0 && /* no timeout for pubsub */
|
||||||
|
listLength(c->pubsub_patterns) == 0 &&
|
||||||
|
(now - c->lastinteraction > server.maxidletime))
|
||||||
|
{
|
||||||
|
redisLog(REDIS_VERBOSE,"Closing idle client");
|
||||||
|
freeClient(c);
|
||||||
|
return 1;
|
||||||
|
} else if (c->flags & REDIS_BLOCKED) {
|
||||||
|
if (c->bpop.timeout != 0 && c->bpop.timeout < now) {
|
||||||
|
addReply(c,shared.nullmultibulk);
|
||||||
|
unblockClientWaitingData(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The client query buffer is an sds.c string that can end with a lot of
|
||||||
|
* free space not used, this function reclaims space if needed.
|
||||||
|
*
|
||||||
|
* The funciton always returns 0 as it never terminates the client. */
|
||||||
|
int clientsCronResizeQueryBuffer(redisClient *c) {
|
||||||
|
size_t querybuf_size = sdsAllocSize(c->querybuf);
|
||||||
|
time_t idletime = server.unixtime - c->lastinteraction;
|
||||||
|
|
||||||
|
/* There are two conditions to resize the query buffer:
|
||||||
|
* 1) Query buffer is > BIG_ARG and too big for latest peak.
|
||||||
|
* 2) Client is inactive and the buffer is bigger than 1k. */
|
||||||
|
if (((querybuf_size > REDIS_MBULK_BIG_ARG) &&
|
||||||
|
(querybuf_size/(c->querybuf_peak+1)) > 2) ||
|
||||||
|
(querybuf_size > 1024 && idletime > 2))
|
||||||
|
{
|
||||||
|
/* Only resize the query buffer if it is actually wasting space. */
|
||||||
|
if (sdsavail(c->querybuf) > 1024) {
|
||||||
|
c->querybuf = sdsRemoveFreeSpace(c->querybuf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* Reset the peak again to capture the peak memory usage in the next
|
||||||
|
* cycle. */
|
||||||
|
c->querybuf_peak = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void clientsCron(void) {
|
||||||
|
/* Make sure to process at least 1/100 of clients per call.
|
||||||
|
* Since this function is called 10 times per second we are sure that
|
||||||
|
* in the worst case we process all the clients in 10 seconds.
|
||||||
|
* In normal conditions (a reasonable number of clients) we process
|
||||||
|
* all the clients in a shorter time. */
|
||||||
|
int numclients = listLength(server.clients);
|
||||||
|
int iterations = numclients/100;
|
||||||
|
|
||||||
|
if (iterations < 50)
|
||||||
|
iterations = (numclients < 50) ? numclients : 50;
|
||||||
|
while(listLength(server.clients) && iterations--) {
|
||||||
|
redisClient *c;
|
||||||
|
listNode *head;
|
||||||
|
|
||||||
|
/* Rotate the list, take the current head, process.
|
||||||
|
* This way if the client must be removed from the list it's the
|
||||||
|
* first element and we don't incur into O(N) computation. */
|
||||||
|
listRotate(server.clients);
|
||||||
|
head = listFirst(server.clients);
|
||||||
|
c = listNodeValue(head);
|
||||||
|
/* The following functions do different service checks on the client.
|
||||||
|
* The protocol is that they return non-zero if the client was
|
||||||
|
* terminated. */
|
||||||
|
if (clientsCronHandleTimeout(c)) continue;
|
||||||
|
if (clientsCronResizeQueryBuffer(c)) continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
|
int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
|
||||||
int j, loops = server.cronloops;
|
int j, loops = server.cronloops;
|
||||||
REDIS_NOTUSED(eventLoop);
|
REDIS_NOTUSED(eventLoop);
|
||||||
@ -627,6 +732,8 @@ int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
|
|||||||
* To access a global var is faster than calling time(NULL) */
|
* To access a global var is faster than calling time(NULL) */
|
||||||
server.unixtime = time(NULL);
|
server.unixtime = time(NULL);
|
||||||
|
|
||||||
|
trackOperationsPerSecond();
|
||||||
|
|
||||||
/* We have just 22 bits per object for LRU information.
|
/* We have just 22 bits per object for LRU information.
|
||||||
* So we use an (eventually wrapping) LRU clock with 10 seconds resolution.
|
* So we use an (eventually wrapping) LRU clock with 10 seconds resolution.
|
||||||
* 2^22 bits with 10 seconds resoluton is more or less 1.5 years.
|
* 2^22 bits with 10 seconds resoluton is more or less 1.5 years.
|
||||||
@ -684,9 +791,8 @@ int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
|
|||||||
zmalloc_used_memory());
|
zmalloc_used_memory());
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Close connections of timedout clients */
|
/* We need to do a few operations on clients asynchronously. */
|
||||||
if ((server.maxidletime && !(loops % 100)) || server.bpop_blocked_clients)
|
clientsCron();
|
||||||
closeTimedoutClients();
|
|
||||||
|
|
||||||
/* Start a scheduled AOF rewrite if this was requested by the user while
|
/* Start a scheduled AOF rewrite if this was requested by the user while
|
||||||
* a BGSAVE was in progress. */
|
* a BGSAVE was in progress. */
|
||||||
@ -832,6 +938,12 @@ void createSharedObjects(void) {
|
|||||||
"-LOADING Redis is loading the dataset in memory\r\n"));
|
"-LOADING Redis is loading the dataset in memory\r\n"));
|
||||||
shared.slowscripterr = createObject(REDIS_STRING,sdsnew(
|
shared.slowscripterr = createObject(REDIS_STRING,sdsnew(
|
||||||
"-BUSY Redis is busy running a script. You can only call SCRIPT KILL or SHUTDOWN NOSAVE.\r\n"));
|
"-BUSY Redis is busy running a script. You can only call SCRIPT KILL or SHUTDOWN NOSAVE.\r\n"));
|
||||||
|
shared.bgsaveerr = createObject(REDIS_STRING,sdsnew(
|
||||||
|
"-MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk. Commands that may modify the data set are disabled. Please check Redis logs for details about the error.\r\n"));
|
||||||
|
shared.roslaveerr = createObject(REDIS_STRING,sdsnew(
|
||||||
|
"-READONLY You can't write against a read only slave.\r\n"));
|
||||||
|
shared.oomerr = createObject(REDIS_STRING,sdsnew(
|
||||||
|
"-OOM command not allowed when used memory > 'maxmemory'.\r\n"));
|
||||||
shared.space = createObject(REDIS_STRING,sdsnew(" "));
|
shared.space = createObject(REDIS_STRING,sdsnew(" "));
|
||||||
shared.colon = createObject(REDIS_STRING,sdsnew(":"));
|
shared.colon = createObject(REDIS_STRING,sdsnew(":"));
|
||||||
shared.plus = createObject(REDIS_STRING,sdsnew("+"));
|
shared.plus = createObject(REDIS_STRING,sdsnew("+"));
|
||||||
@ -852,6 +964,8 @@ void createSharedObjects(void) {
|
|||||||
shared.psubscribebulk = createStringObject("$10\r\npsubscribe\r\n",17);
|
shared.psubscribebulk = createStringObject("$10\r\npsubscribe\r\n",17);
|
||||||
shared.punsubscribebulk = createStringObject("$12\r\npunsubscribe\r\n",19);
|
shared.punsubscribebulk = createStringObject("$12\r\npunsubscribe\r\n",19);
|
||||||
shared.del = createStringObject("DEL",3);
|
shared.del = createStringObject("DEL",3);
|
||||||
|
shared.rpop = createStringObject("RPOP",4);
|
||||||
|
shared.lpop = createStringObject("LPOP",4);
|
||||||
for (j = 0; j < REDIS_SHARED_INTEGERS; j++) {
|
for (j = 0; j < REDIS_SHARED_INTEGERS; j++) {
|
||||||
shared.integers[j] = createObject(REDIS_STRING,(void*)(long)j);
|
shared.integers[j] = createObject(REDIS_STRING,(void*)(long)j);
|
||||||
shared.integers[j]->encoding = REDIS_ENCODING_INT;
|
shared.integers[j]->encoding = REDIS_ENCODING_INT;
|
||||||
@ -865,6 +979,8 @@ void createSharedObjects(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void initServerConfig() {
|
void initServerConfig() {
|
||||||
|
getRandomHexChars(server.runid,REDIS_RUN_ID_SIZE);
|
||||||
|
server.runid[REDIS_RUN_ID_SIZE] = '\0';
|
||||||
server.arch_bits = (sizeof(long) == 8) ? 64 : 32;
|
server.arch_bits = (sizeof(long) == 8) ? 64 : 32;
|
||||||
server.port = REDIS_SERVERPORT;
|
server.port = REDIS_SERVERPORT;
|
||||||
server.bindaddr = NULL;
|
server.bindaddr = NULL;
|
||||||
@ -905,8 +1021,8 @@ void initServerConfig() {
|
|||||||
server.maxmemory = 0;
|
server.maxmemory = 0;
|
||||||
server.maxmemory_policy = REDIS_MAXMEMORY_VOLATILE_LRU;
|
server.maxmemory_policy = REDIS_MAXMEMORY_VOLATILE_LRU;
|
||||||
server.maxmemory_samples = 3;
|
server.maxmemory_samples = 3;
|
||||||
server.hash_max_zipmap_entries = REDIS_HASH_MAX_ZIPMAP_ENTRIES;
|
server.hash_max_ziplist_entries = REDIS_HASH_MAX_ZIPLIST_ENTRIES;
|
||||||
server.hash_max_zipmap_value = REDIS_HASH_MAX_ZIPMAP_VALUE;
|
server.hash_max_ziplist_value = REDIS_HASH_MAX_ZIPLIST_VALUE;
|
||||||
server.list_max_ziplist_entries = REDIS_LIST_MAX_ZIPLIST_ENTRIES;
|
server.list_max_ziplist_entries = REDIS_LIST_MAX_ZIPLIST_ENTRIES;
|
||||||
server.list_max_ziplist_value = REDIS_LIST_MAX_ZIPLIST_VALUE;
|
server.list_max_ziplist_value = REDIS_LIST_MAX_ZIPLIST_VALUE;
|
||||||
server.set_max_intset_entries = REDIS_SET_MAX_INTSET_ENTRIES;
|
server.set_max_intset_entries = REDIS_SET_MAX_INTSET_ENTRIES;
|
||||||
@ -936,6 +1052,7 @@ void initServerConfig() {
|
|||||||
server.repl_state = REDIS_REPL_NONE;
|
server.repl_state = REDIS_REPL_NONE;
|
||||||
server.repl_syncio_timeout = REDIS_REPL_SYNCIO_TIMEOUT;
|
server.repl_syncio_timeout = REDIS_REPL_SYNCIO_TIMEOUT;
|
||||||
server.repl_serve_stale_data = 1;
|
server.repl_serve_stale_data = 1;
|
||||||
|
server.repl_slave_ro = 1;
|
||||||
server.repl_down_since = -1;
|
server.repl_down_since = -1;
|
||||||
|
|
||||||
/* Client output buffer limits */
|
/* Client output buffer limits */
|
||||||
@ -962,6 +1079,7 @@ void initServerConfig() {
|
|||||||
populateCommandTable();
|
populateCommandTable();
|
||||||
server.delCommand = lookupCommandByCString("del");
|
server.delCommand = lookupCommandByCString("del");
|
||||||
server.multiCommand = lookupCommandByCString("multi");
|
server.multiCommand = lookupCommandByCString("multi");
|
||||||
|
server.lpushCommand = lookupCommandByCString("lpush");
|
||||||
|
|
||||||
/* Slow log */
|
/* Slow log */
|
||||||
server.slowlog_log_slower_than = REDIS_SLOWLOG_LOG_SLOWER_THAN;
|
server.slowlog_log_slower_than = REDIS_SLOWLOG_LOG_SLOWER_THAN;
|
||||||
@ -1083,7 +1201,13 @@ void initServer() {
|
|||||||
server.stat_peak_memory = 0;
|
server.stat_peak_memory = 0;
|
||||||
server.stat_fork_time = 0;
|
server.stat_fork_time = 0;
|
||||||
server.stat_rejected_conn = 0;
|
server.stat_rejected_conn = 0;
|
||||||
|
memset(server.ops_sec_samples,0,sizeof(server.ops_sec_samples));
|
||||||
|
server.ops_sec_idx = 0;
|
||||||
|
server.ops_sec_last_sample_time = mstime();
|
||||||
|
server.ops_sec_last_sample_ops = 0;
|
||||||
server.unixtime = time(NULL);
|
server.unixtime = time(NULL);
|
||||||
|
server.lastbgsave_status = REDIS_OK;
|
||||||
|
server.stop_writes_on_bgsave_err = 1;
|
||||||
aeCreateTimeEvent(server.el, 1, serverCron, NULL, NULL);
|
aeCreateTimeEvent(server.el, 1, serverCron, NULL, NULL);
|
||||||
if (server.ipfd > 0 && aeCreateFileEvent(server.el,server.ipfd,AE_READABLE,
|
if (server.ipfd > 0 && aeCreateFileEvent(server.el,server.ipfd,AE_READABLE,
|
||||||
acceptTcpHandler,NULL) == AE_ERR) oom("creating file event");
|
acceptTcpHandler,NULL) == AE_ERR) oom("creating file event");
|
||||||
@ -1160,6 +1284,43 @@ void resetCommandTableStats(void) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ========================== Redis OP Array API ============================ */
|
||||||
|
|
||||||
|
void redisOpArrayInit(redisOpArray *oa) {
|
||||||
|
oa->ops = NULL;
|
||||||
|
oa->numops = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int redisOpArrayAppend(redisOpArray *oa, struct redisCommand *cmd, int dbid,
|
||||||
|
robj **argv, int argc, int target)
|
||||||
|
{
|
||||||
|
redisOp *op;
|
||||||
|
|
||||||
|
oa->ops = zrealloc(oa->ops,sizeof(redisOp)*(oa->numops+1));
|
||||||
|
op = oa->ops+oa->numops;
|
||||||
|
op->cmd = cmd;
|
||||||
|
op->dbid = dbid;
|
||||||
|
op->argv = argv;
|
||||||
|
op->argc = argc;
|
||||||
|
op->target = target;
|
||||||
|
oa->numops++;
|
||||||
|
return oa->numops;
|
||||||
|
}
|
||||||
|
|
||||||
|
void redisOpArrayFree(redisOpArray *oa) {
|
||||||
|
while(oa->numops) {
|
||||||
|
int j;
|
||||||
|
redisOp *op;
|
||||||
|
|
||||||
|
oa->numops--;
|
||||||
|
op = oa->ops+oa->numops;
|
||||||
|
for (j = 0; j < op->argc; j++)
|
||||||
|
decrRefCount(op->argv[j]);
|
||||||
|
zfree(op->argv);
|
||||||
|
}
|
||||||
|
zfree(oa->ops);
|
||||||
|
}
|
||||||
|
|
||||||
/* ====================== Commands lookup and execution ===================== */
|
/* ====================== Commands lookup and execution ===================== */
|
||||||
|
|
||||||
struct redisCommand *lookupCommand(sds name) {
|
struct redisCommand *lookupCommand(sds name) {
|
||||||
@ -1175,10 +1336,42 @@ struct redisCommand *lookupCommandByCString(char *s) {
|
|||||||
return cmd;
|
return cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Propagate the specified command (in the context of the specified database id)
|
||||||
|
* to AOF, Slaves and Monitors.
|
||||||
|
*
|
||||||
|
* flags are an xor between:
|
||||||
|
* + REDIS_PROPAGATE_NONE (no propagation of command at all)
|
||||||
|
* + REDIS_PROPAGATE_AOF (propagate into the AOF file if is enabled)
|
||||||
|
* + REDIS_PROPAGATE_REPL (propagate into the replication link)
|
||||||
|
*/
|
||||||
|
void propagate(struct redisCommand *cmd, int dbid, robj **argv, int argc,
|
||||||
|
int flags)
|
||||||
|
{
|
||||||
|
if (server.aof_state != REDIS_AOF_OFF && flags & REDIS_PROPAGATE_AOF)
|
||||||
|
feedAppendOnlyFile(cmd,dbid,argv,argc);
|
||||||
|
if (flags & REDIS_PROPAGATE_REPL && listLength(server.slaves))
|
||||||
|
replicationFeedSlaves(server.slaves,dbid,argv,argc);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Used inside commands to schedule the propagation of additional commands
|
||||||
|
* after the current command is propagated to AOF / Replication. */
|
||||||
|
void alsoPropagate(struct redisCommand *cmd, int dbid, robj **argv, int argc,
|
||||||
|
int target)
|
||||||
|
{
|
||||||
|
redisOpArrayAppend(&server.also_propagate,cmd,dbid,argv,argc,target);
|
||||||
|
}
|
||||||
|
|
||||||
/* Call() is the core of Redis execution of a command */
|
/* Call() is the core of Redis execution of a command */
|
||||||
void call(redisClient *c, int flags) {
|
void call(redisClient *c, int flags) {
|
||||||
long long dirty, start = ustime(), duration;
|
long long dirty, start = ustime(), duration;
|
||||||
|
|
||||||
|
/* Sent the command to clients in MONITOR mode, only if the commands are
|
||||||
|
* not geneated from reading an AOF. */
|
||||||
|
if (listLength(server.monitors) && !server.loading)
|
||||||
|
replicationFeedMonitors(c,server.monitors,c->db->id,c->argv,c->argc);
|
||||||
|
|
||||||
|
/* Call the command. */
|
||||||
|
redisOpArrayInit(&server.also_propagate);
|
||||||
dirty = server.dirty;
|
dirty = server.dirty;
|
||||||
c->cmd->proc(c);
|
c->cmd->proc(c);
|
||||||
dirty = server.dirty-dirty;
|
dirty = server.dirty-dirty;
|
||||||
@ -1189,20 +1382,37 @@ void call(redisClient *c, int flags) {
|
|||||||
if (server.loading && c->flags & REDIS_LUA_CLIENT)
|
if (server.loading && c->flags & REDIS_LUA_CLIENT)
|
||||||
flags &= ~(REDIS_CALL_SLOWLOG | REDIS_CALL_STATS);
|
flags &= ~(REDIS_CALL_SLOWLOG | REDIS_CALL_STATS);
|
||||||
|
|
||||||
|
/* Log the command into the Slow log if needed, and populate the
|
||||||
|
* per-command statistics that we show in INFO commandstats. */
|
||||||
if (flags & REDIS_CALL_SLOWLOG)
|
if (flags & REDIS_CALL_SLOWLOG)
|
||||||
slowlogPushEntryIfNeeded(c->argv,c->argc,duration);
|
slowlogPushEntryIfNeeded(c->argv,c->argc,duration);
|
||||||
if (flags & REDIS_CALL_STATS) {
|
if (flags & REDIS_CALL_STATS) {
|
||||||
c->cmd->microseconds += duration;
|
c->cmd->microseconds += duration;
|
||||||
c->cmd->calls++;
|
c->cmd->calls++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Propagate the command into the AOF and replication link */
|
||||||
if (flags & REDIS_CALL_PROPAGATE) {
|
if (flags & REDIS_CALL_PROPAGATE) {
|
||||||
if (server.aof_state != REDIS_AOF_OFF && dirty > 0)
|
int flags = REDIS_PROPAGATE_NONE;
|
||||||
feedAppendOnlyFile(c->cmd,c->db->id,c->argv,c->argc);
|
|
||||||
if ((dirty > 0 || c->cmd->flags & REDIS_CMD_FORCE_REPLICATION) &&
|
if (c->cmd->flags & REDIS_CMD_FORCE_REPLICATION)
|
||||||
listLength(server.slaves))
|
flags |= REDIS_PROPAGATE_REPL;
|
||||||
replicationFeedSlaves(server.slaves,c->db->id,c->argv,c->argc);
|
if (dirty)
|
||||||
if (listLength(server.monitors))
|
flags |= (REDIS_PROPAGATE_REPL | REDIS_PROPAGATE_AOF);
|
||||||
replicationFeedMonitors(server.monitors,c->db->id,c->argv,c->argc);
|
if (flags != REDIS_PROPAGATE_NONE)
|
||||||
|
propagate(c->cmd,c->db->id,c->argv,c->argc,flags);
|
||||||
|
}
|
||||||
|
/* Commands such as LPUSH or BRPOPLPUSH may propagate an additional
|
||||||
|
* PUSH command. */
|
||||||
|
if (server.also_propagate.numops) {
|
||||||
|
int j;
|
||||||
|
redisOp *rop;
|
||||||
|
|
||||||
|
for (j = 0; j < server.also_propagate.numops; j++) {
|
||||||
|
rop = &server.also_propagate.ops[j];
|
||||||
|
propagate(rop->cmd, rop->dbid, rop->argv, rop->argc, rop->target);
|
||||||
|
}
|
||||||
|
redisOpArrayFree(&server.also_propagate);
|
||||||
}
|
}
|
||||||
server.stat_numcommands++;
|
server.stat_numcommands++;
|
||||||
}
|
}
|
||||||
@ -1278,12 +1488,31 @@ int processCommand(redisClient *c) {
|
|||||||
if (server.maxmemory) {
|
if (server.maxmemory) {
|
||||||
int retval = freeMemoryIfNeeded();
|
int retval = freeMemoryIfNeeded();
|
||||||
if ((c->cmd->flags & REDIS_CMD_DENYOOM) && retval == REDIS_ERR) {
|
if ((c->cmd->flags & REDIS_CMD_DENYOOM) && retval == REDIS_ERR) {
|
||||||
addReplyError(c,
|
addReply(c, shared.oomerr);
|
||||||
"command not allowed when used memory > 'maxmemory'");
|
|
||||||
return REDIS_OK;
|
return REDIS_OK;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Don't accept write commands if there are problems persisting on disk. */
|
||||||
|
if (server.stop_writes_on_bgsave_err &&
|
||||||
|
server.saveparamslen > 0
|
||||||
|
&& server.lastbgsave_status == REDIS_ERR &&
|
||||||
|
c->cmd->flags & REDIS_CMD_WRITE)
|
||||||
|
{
|
||||||
|
addReply(c, shared.bgsaveerr);
|
||||||
|
return REDIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Don't accept wirte commands if this is a read only slave. But
|
||||||
|
* accept write commands if this is our master. */
|
||||||
|
if (server.masterhost && server.repl_slave_ro &&
|
||||||
|
!(c->flags & REDIS_MASTER) &&
|
||||||
|
c->cmd->flags & REDIS_CMD_WRITE)
|
||||||
|
{
|
||||||
|
addReply(c, shared.roslaveerr);
|
||||||
|
return REDIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
/* Only allow SUBSCRIBE and UNSUBSCRIBE in the context of Pub/Sub */
|
/* Only allow SUBSCRIBE and UNSUBSCRIBE in the context of Pub/Sub */
|
||||||
if ((dictSize(c->pubsub_channels) > 0 || listLength(c->pubsub_patterns) > 0)
|
if ((dictSize(c->pubsub_channels) > 0 || listLength(c->pubsub_patterns) > 0)
|
||||||
&&
|
&&
|
||||||
@ -1416,6 +1645,17 @@ void echoCommand(redisClient *c) {
|
|||||||
addReplyBulk(c,c->argv[1]);
|
addReplyBulk(c,c->argv[1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void timeCommand(redisClient *c) {
|
||||||
|
struct timeval tv;
|
||||||
|
|
||||||
|
/* gettimeofday() can only fail if &tv is a bad addresss so we
|
||||||
|
* don't check for errors. */
|
||||||
|
gettimeofday(&tv,NULL);
|
||||||
|
addReplyMultiBulkLen(c,2);
|
||||||
|
addReplyBulkLongLong(c,tv.tv_sec);
|
||||||
|
addReplyBulkLongLong(c,tv.tv_usec);
|
||||||
|
}
|
||||||
|
|
||||||
/* Convert an amount of bytes into a human readable string in the form
|
/* Convert an amount of bytes into a human readable string in the form
|
||||||
* of 100B, 2G, 100M, 4K, and so forth. */
|
* of 100B, 2G, 100M, 4K, and so forth. */
|
||||||
void bytesToHuman(char *s, unsigned long long n) {
|
void bytesToHuman(char *s, unsigned long long n) {
|
||||||
@ -1470,6 +1710,7 @@ sds genRedisInfoString(char *section) {
|
|||||||
"multiplexing_api:%s\r\n"
|
"multiplexing_api:%s\r\n"
|
||||||
"gcc_version:%d.%d.%d\r\n"
|
"gcc_version:%d.%d.%d\r\n"
|
||||||
"process_id:%ld\r\n"
|
"process_id:%ld\r\n"
|
||||||
|
"run_id:%s\r\n"
|
||||||
"tcp_port:%d\r\n"
|
"tcp_port:%d\r\n"
|
||||||
"uptime_in_seconds:%ld\r\n"
|
"uptime_in_seconds:%ld\r\n"
|
||||||
"uptime_in_days:%ld\r\n"
|
"uptime_in_days:%ld\r\n"
|
||||||
@ -1485,6 +1726,7 @@ sds genRedisInfoString(char *section) {
|
|||||||
0,0,0,
|
0,0,0,
|
||||||
#endif
|
#endif
|
||||||
(long) getpid(),
|
(long) getpid(),
|
||||||
|
server.runid,
|
||||||
server.port,
|
server.port,
|
||||||
uptime,
|
uptime,
|
||||||
uptime/(3600*24),
|
uptime/(3600*24),
|
||||||
@ -1544,12 +1786,14 @@ sds genRedisInfoString(char *section) {
|
|||||||
"changes_since_last_save:%lld\r\n"
|
"changes_since_last_save:%lld\r\n"
|
||||||
"bgsave_in_progress:%d\r\n"
|
"bgsave_in_progress:%d\r\n"
|
||||||
"last_save_time:%ld\r\n"
|
"last_save_time:%ld\r\n"
|
||||||
|
"last_bgsave_status:%s\r\n"
|
||||||
"bgrewriteaof_in_progress:%d\r\n",
|
"bgrewriteaof_in_progress:%d\r\n",
|
||||||
server.loading,
|
server.loading,
|
||||||
server.aof_state != REDIS_AOF_OFF,
|
server.aof_state != REDIS_AOF_OFF,
|
||||||
server.dirty,
|
server.dirty,
|
||||||
server.rdb_child_pid != -1,
|
server.rdb_child_pid != -1,
|
||||||
server.lastsave,
|
server.lastsave,
|
||||||
|
server.lastbgsave_status == REDIS_OK ? "ok" : "err",
|
||||||
server.aof_child_pid != -1);
|
server.aof_child_pid != -1);
|
||||||
|
|
||||||
if (server.aof_state != REDIS_AOF_OFF) {
|
if (server.aof_state != REDIS_AOF_OFF) {
|
||||||
@ -1605,6 +1849,7 @@ sds genRedisInfoString(char *section) {
|
|||||||
"# Stats\r\n"
|
"# Stats\r\n"
|
||||||
"total_connections_received:%lld\r\n"
|
"total_connections_received:%lld\r\n"
|
||||||
"total_commands_processed:%lld\r\n"
|
"total_commands_processed:%lld\r\n"
|
||||||
|
"instantaneous_ops_per_sec:%lld\r\n"
|
||||||
"rejected_connections:%lld\r\n"
|
"rejected_connections:%lld\r\n"
|
||||||
"expired_keys:%lld\r\n"
|
"expired_keys:%lld\r\n"
|
||||||
"evicted_keys:%lld\r\n"
|
"evicted_keys:%lld\r\n"
|
||||||
@ -1615,6 +1860,7 @@ sds genRedisInfoString(char *section) {
|
|||||||
"latest_fork_usec:%lld\r\n",
|
"latest_fork_usec:%lld\r\n",
|
||||||
server.stat_numconnections,
|
server.stat_numconnections,
|
||||||
server.stat_numcommands,
|
server.stat_numcommands,
|
||||||
|
getOperationsPerSecond(),
|
||||||
server.stat_rejected_conn,
|
server.stat_rejected_conn,
|
||||||
server.stat_expiredkeys,
|
server.stat_expiredkeys,
|
||||||
server.stat_evictedkeys,
|
server.stat_evictedkeys,
|
||||||
@ -1990,8 +2236,8 @@ void daemonize(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void version() {
|
void version() {
|
||||||
printf("Redis server version %s (%s:%d)\n", REDIS_VERSION,
|
printf("Redis server v=%s sha=%s:%d malloc=%s\n", REDIS_VERSION,
|
||||||
redisGitSHA1(), atoi(redisGitDirty()) > 0);
|
redisGitSHA1(), atoi(redisGitDirty()) > 0, ZMALLOC_LIB);
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1999,7 +2245,8 @@ void usage() {
|
|||||||
fprintf(stderr,"Usage: ./redis-server [/path/to/redis.conf] [options]\n");
|
fprintf(stderr,"Usage: ./redis-server [/path/to/redis.conf] [options]\n");
|
||||||
fprintf(stderr," ./redis-server - (read config from stdin)\n");
|
fprintf(stderr," ./redis-server - (read config from stdin)\n");
|
||||||
fprintf(stderr," ./redis-server -v or --version\n");
|
fprintf(stderr," ./redis-server -v or --version\n");
|
||||||
fprintf(stderr," ./redis-server -h or --help\n\n");
|
fprintf(stderr," ./redis-server -h or --help\n");
|
||||||
|
fprintf(stderr," ./redis-server --test-memory <megabytes>\n\n");
|
||||||
fprintf(stderr,"Examples:\n");
|
fprintf(stderr,"Examples:\n");
|
||||||
fprintf(stderr," ./redis-server (run the server with default conf)\n");
|
fprintf(stderr," ./redis-server (run the server with default conf)\n");
|
||||||
fprintf(stderr," ./redis-server /etc/redis/6379.conf\n");
|
fprintf(stderr," ./redis-server /etc/redis/6379.conf\n");
|
||||||
@ -2055,6 +2302,8 @@ void setupSignalHandlers(void) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void memtest(size_t megabytes, int passes);
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
long long start;
|
long long start;
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
@ -2076,6 +2325,17 @@ int main(int argc, char **argv) {
|
|||||||
strcmp(argv[1], "--version") == 0) version();
|
strcmp(argv[1], "--version") == 0) version();
|
||||||
if (strcmp(argv[1], "--help") == 0 ||
|
if (strcmp(argv[1], "--help") == 0 ||
|
||||||
strcmp(argv[1], "-h") == 0) usage();
|
strcmp(argv[1], "-h") == 0) usage();
|
||||||
|
if (strcmp(argv[1], "--test-memory") == 0) {
|
||||||
|
if (argc == 3) {
|
||||||
|
memtest(atoi(argv[2]),50);
|
||||||
|
exit(0);
|
||||||
|
} else {
|
||||||
|
fprintf(stderr,"Please specify the amount of memory to test in megabytes.\n");
|
||||||
|
fprintf(stderr,"Example: ./redis-server --test-memory 4096\n\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* First argument is the config file name? */
|
/* First argument is the config file name? */
|
||||||
if (argv[j][0] != '-' || argv[j][1] != '-')
|
if (argv[j][0] != '-' || argv[j][1] != '-')
|
||||||
configfile = argv[j++];
|
configfile = argv[j++];
|
||||||
|
91
src/redis.h
91
src/redis.h
@ -28,7 +28,6 @@
|
|||||||
#include "adlist.h" /* Linked lists */
|
#include "adlist.h" /* Linked lists */
|
||||||
#include "zmalloc.h" /* total memory usage aware version of malloc/free */
|
#include "zmalloc.h" /* total memory usage aware version of malloc/free */
|
||||||
#include "anet.h" /* Networking the easy way */
|
#include "anet.h" /* Networking the easy way */
|
||||||
#include "zipmap.h" /* Compact string -> string data structure */
|
|
||||||
#include "ziplist.h" /* Compact list data structure */
|
#include "ziplist.h" /* Compact list data structure */
|
||||||
#include "intset.h" /* Compact integer set structure */
|
#include "intset.h" /* Compact integer set structure */
|
||||||
#include "version.h" /* Version macro */
|
#include "version.h" /* Version macro */
|
||||||
@ -58,6 +57,9 @@
|
|||||||
#define REDIS_REPL_TIMEOUT 60
|
#define REDIS_REPL_TIMEOUT 60
|
||||||
#define REDIS_REPL_PING_SLAVE_PERIOD 10
|
#define REDIS_REPL_PING_SLAVE_PERIOD 10
|
||||||
|
|
||||||
|
#define REDIS_RUN_ID_SIZE 40
|
||||||
|
#define REDIS_OPS_SEC_SAMPLES 16
|
||||||
|
|
||||||
/* Protocol and I/O related defines */
|
/* Protocol and I/O related defines */
|
||||||
#define REDIS_MAX_QUERYBUF_LEN (1024*1024*1024) /* 1GB max query buffer. */
|
#define REDIS_MAX_QUERYBUF_LEN (1024*1024*1024) /* 1GB max query buffer. */
|
||||||
#define REDIS_IOBUF_LEN (1024*16) /* Generic I/O buffer size */
|
#define REDIS_IOBUF_LEN (1024*16) /* Generic I/O buffer size */
|
||||||
@ -205,8 +207,8 @@
|
|||||||
#define AOF_FSYNC_EVERYSEC 2
|
#define AOF_FSYNC_EVERYSEC 2
|
||||||
|
|
||||||
/* Zip structure related defaults */
|
/* Zip structure related defaults */
|
||||||
#define REDIS_HASH_MAX_ZIPMAP_ENTRIES 512
|
#define REDIS_HASH_MAX_ZIPLIST_ENTRIES 512
|
||||||
#define REDIS_HASH_MAX_ZIPMAP_VALUE 64
|
#define REDIS_HASH_MAX_ZIPLIST_VALUE 64
|
||||||
#define REDIS_LIST_MAX_ZIPLIST_ENTRIES 512
|
#define REDIS_LIST_MAX_ZIPLIST_ENTRIES 512
|
||||||
#define REDIS_LIST_MAX_ZIPLIST_VALUE 64
|
#define REDIS_LIST_MAX_ZIPLIST_VALUE 64
|
||||||
#define REDIS_SET_MAX_INTSET_ENTRIES 512
|
#define REDIS_SET_MAX_INTSET_ENTRIES 512
|
||||||
@ -245,6 +247,11 @@
|
|||||||
#define REDIS_CALL_PROPAGATE 4
|
#define REDIS_CALL_PROPAGATE 4
|
||||||
#define REDIS_CALL_FULL (REDIS_CALL_SLOWLOG | REDIS_CALL_STATS | REDIS_CALL_PROPAGATE)
|
#define REDIS_CALL_FULL (REDIS_CALL_SLOWLOG | REDIS_CALL_STATS | REDIS_CALL_PROPAGATE)
|
||||||
|
|
||||||
|
/* Command propagation flags, see propagate() function */
|
||||||
|
#define REDIS_PROPAGATE_NONE 0
|
||||||
|
#define REDIS_PROPAGATE_AOF 1
|
||||||
|
#define REDIS_PROPAGATE_REPL 2
|
||||||
|
|
||||||
/* We can print the stacktrace, so our assert is defined this way: */
|
/* We can print the stacktrace, so our assert is defined this way: */
|
||||||
#define redisAssertWithInfo(_c,_o,_e) ((_e)?(void)0 : (_redisAssertWithInfo(_c,_o,#_e,__FILE__,__LINE__),_exit(1)))
|
#define redisAssertWithInfo(_c,_o,_e) ((_e)?(void)0 : (_redisAssertWithInfo(_c,_o,#_e,__FILE__,__LINE__),_exit(1)))
|
||||||
#define redisAssert(_e) ((_e)?(void)0 : (_redisAssert(#_e,__FILE__,__LINE__),_exit(1)))
|
#define redisAssert(_e) ((_e)?(void)0 : (_redisAssert(#_e,__FILE__,__LINE__),_exit(1)))
|
||||||
@ -316,6 +323,7 @@ typedef struct redisClient {
|
|||||||
redisDb *db;
|
redisDb *db;
|
||||||
int dictid;
|
int dictid;
|
||||||
sds querybuf;
|
sds querybuf;
|
||||||
|
size_t querybuf_peak; /* Recent (100ms or more) peak of querybuf size */
|
||||||
int argc;
|
int argc;
|
||||||
robj **argv;
|
robj **argv;
|
||||||
struct redisCommand *cmd, *lastcmd;
|
struct redisCommand *cmd, *lastcmd;
|
||||||
@ -325,6 +333,7 @@ typedef struct redisClient {
|
|||||||
list *reply;
|
list *reply;
|
||||||
unsigned long reply_bytes; /* Tot bytes of objects in reply list */
|
unsigned long reply_bytes; /* Tot bytes of objects in reply list */
|
||||||
int sentlen;
|
int sentlen;
|
||||||
|
time_t ctime; /* Client creation time */
|
||||||
time_t lastinteraction; /* time of the last interaction, used for timeout */
|
time_t lastinteraction; /* time of the last interaction, used for timeout */
|
||||||
time_t obuf_soft_limit_reached_time;
|
time_t obuf_soft_limit_reached_time;
|
||||||
int flags; /* REDIS_SLAVE | REDIS_MONITOR | REDIS_MULTI ... */
|
int flags; /* REDIS_SLAVE | REDIS_MONITOR | REDIS_MULTI ... */
|
||||||
@ -356,11 +365,11 @@ struct sharedObjectsStruct {
|
|||||||
robj *crlf, *ok, *err, *emptybulk, *czero, *cone, *cnegone, *pong, *space,
|
robj *crlf, *ok, *err, *emptybulk, *czero, *cone, *cnegone, *pong, *space,
|
||||||
*colon, *nullbulk, *nullmultibulk, *queued,
|
*colon, *nullbulk, *nullmultibulk, *queued,
|
||||||
*emptymultibulk, *wrongtypeerr, *nokeyerr, *syntaxerr, *sameobjecterr,
|
*emptymultibulk, *wrongtypeerr, *nokeyerr, *syntaxerr, *sameobjecterr,
|
||||||
*outofrangeerr, *noscripterr, *loadingerr, *slowscripterr, *plus,
|
*outofrangeerr, *noscripterr, *loadingerr, *slowscripterr, *bgsaveerr,
|
||||||
*select0, *select1, *select2, *select3, *select4,
|
*roslaveerr, *oomerr, *plus, *select0, *select1, *select2, *select3,
|
||||||
*select5, *select6, *select7, *select8, *select9,
|
*select4, *select5, *select6, *select7, *select8, *select9,
|
||||||
*messagebulk, *pmessagebulk, *subscribebulk, *unsubscribebulk,
|
*messagebulk, *pmessagebulk, *subscribebulk, *unsubscribebulk,
|
||||||
*psubscribebulk, *punsubscribebulk, *del,
|
*psubscribebulk, *punsubscribebulk, *del, *rpop, *lpop,
|
||||||
*integers[REDIS_SHARED_INTEGERS],
|
*integers[REDIS_SHARED_INTEGERS],
|
||||||
*mbulkhdr[REDIS_SHARED_BULKHDR_LEN], /* "*<value>\r\n" */
|
*mbulkhdr[REDIS_SHARED_BULKHDR_LEN], /* "*<value>\r\n" */
|
||||||
*bulkhdr[REDIS_SHARED_BULKHDR_LEN]; /* "$<value>\r\n" */
|
*bulkhdr[REDIS_SHARED_BULKHDR_LEN]; /* "$<value>\r\n" */
|
||||||
@ -394,6 +403,30 @@ typedef struct clientBufferLimitsConfig {
|
|||||||
time_t soft_limit_seconds;
|
time_t soft_limit_seconds;
|
||||||
} clientBufferLimitsConfig;
|
} clientBufferLimitsConfig;
|
||||||
|
|
||||||
|
/* The redisOp structure defines a Redis Operation, that is an instance of
|
||||||
|
* a command with an argument vector, database ID, propagation target
|
||||||
|
* (REDIS_PROPAGATE_*), and command pointer.
|
||||||
|
*
|
||||||
|
* Currently only used to additionally propagate more commands to AOF/Replication
|
||||||
|
* after the propagation of the executed command. */
|
||||||
|
typedef struct redisOp {
|
||||||
|
robj **argv;
|
||||||
|
int argc, dbid, target;
|
||||||
|
struct redisCommand *cmd;
|
||||||
|
} redisOp;
|
||||||
|
|
||||||
|
/* Defines an array of Redis operations. There is an API to add to this
|
||||||
|
* structure in a easy way.
|
||||||
|
*
|
||||||
|
* redisOpArrayInit();
|
||||||
|
* redisOpArrayAppend();
|
||||||
|
* redisOpArrayFree();
|
||||||
|
*/
|
||||||
|
typedef struct redisOpArray {
|
||||||
|
redisOp *ops;
|
||||||
|
int numops;
|
||||||
|
} redisOpArray;
|
||||||
|
|
||||||
/*-----------------------------------------------------------------------------
|
/*-----------------------------------------------------------------------------
|
||||||
* Redis cluster data structures
|
* Redis cluster data structures
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
@ -538,6 +571,8 @@ struct redisServer {
|
|||||||
char *requirepass; /* Pass for AUTH command, or NULL */
|
char *requirepass; /* Pass for AUTH command, or NULL */
|
||||||
char *pidfile; /* PID file path */
|
char *pidfile; /* PID file path */
|
||||||
int arch_bits; /* 32 or 64 depending on sizeof(long) */
|
int arch_bits; /* 32 or 64 depending on sizeof(long) */
|
||||||
|
int cronloops; /* Number of times the cron function run */
|
||||||
|
char runid[REDIS_RUN_ID_SIZE+1]; /* ID always different at every exec. */
|
||||||
/* Networking */
|
/* Networking */
|
||||||
int port; /* TCP listening port */
|
int port; /* TCP listening port */
|
||||||
char *bindaddr; /* Bind address or NULL */
|
char *bindaddr; /* Bind address or NULL */
|
||||||
@ -557,9 +592,7 @@ struct redisServer {
|
|||||||
off_t loading_loaded_bytes;
|
off_t loading_loaded_bytes;
|
||||||
time_t loading_start_time;
|
time_t loading_start_time;
|
||||||
/* Fast pointers to often looked up command */
|
/* Fast pointers to often looked up command */
|
||||||
struct redisCommand *delCommand, *multiCommand;
|
struct redisCommand *delCommand, *multiCommand, *lpushCommand;
|
||||||
int cronloops; /* Number of times the cron function run */
|
|
||||||
time_t lastsave; /* Unix time of last save succeeede */
|
|
||||||
/* Fields used only for stats */
|
/* Fields used only for stats */
|
||||||
time_t stat_starttime; /* Server start time */
|
time_t stat_starttime; /* Server start time */
|
||||||
long long stat_numcommands; /* Number of processed commands */
|
long long stat_numcommands; /* Number of processed commands */
|
||||||
@ -575,6 +608,12 @@ struct redisServer {
|
|||||||
long long slowlog_entry_id; /* SLOWLOG current entry ID */
|
long long slowlog_entry_id; /* SLOWLOG current entry ID */
|
||||||
long long slowlog_log_slower_than; /* SLOWLOG time limit (to get logged) */
|
long long slowlog_log_slower_than; /* SLOWLOG time limit (to get logged) */
|
||||||
unsigned long slowlog_max_len; /* SLOWLOG max number of items logged */
|
unsigned long slowlog_max_len; /* SLOWLOG max number of items logged */
|
||||||
|
/* The following two are used to track instantaneous "load" in terms
|
||||||
|
* of operations per second. */
|
||||||
|
long long ops_sec_last_sample_time; /* Timestamp of last sample (in ms) */
|
||||||
|
long long ops_sec_last_sample_ops; /* numcommands in last sample */
|
||||||
|
long long ops_sec_samples[REDIS_OPS_SEC_SAMPLES];
|
||||||
|
int ops_sec_idx;
|
||||||
/* Configuration */
|
/* Configuration */
|
||||||
int verbosity; /* Loglevel in redis.conf */
|
int verbosity; /* Loglevel in redis.conf */
|
||||||
int maxidletime; /* Client timeout in seconds */
|
int maxidletime; /* Client timeout in seconds */
|
||||||
@ -607,6 +646,11 @@ struct redisServer {
|
|||||||
int saveparamslen; /* Number of saving points */
|
int saveparamslen; /* Number of saving points */
|
||||||
char *rdb_filename; /* Name of RDB file */
|
char *rdb_filename; /* Name of RDB file */
|
||||||
int rdb_compression; /* Use compression in RDB? */
|
int rdb_compression; /* Use compression in RDB? */
|
||||||
|
time_t lastsave; /* Unix time of last save succeeede */
|
||||||
|
int lastbgsave_status; /* REDIS_OK or REDIS_ERR */
|
||||||
|
int stop_writes_on_bgsave_err; /* Don't allow writes if can't BGSAVE */
|
||||||
|
/* Propagation of commands in AOF / replication */
|
||||||
|
redisOpArray also_propagate; /* Additional command to propagate. */
|
||||||
/* Logging */
|
/* Logging */
|
||||||
char *logfile; /* Path of log file */
|
char *logfile; /* Path of log file */
|
||||||
int syslog_enabled; /* Is syslog enabled? */
|
int syslog_enabled; /* Is syslog enabled? */
|
||||||
@ -627,6 +671,7 @@ struct redisServer {
|
|||||||
char *repl_transfer_tmpfile; /* Slave-> master SYNC temp file name */
|
char *repl_transfer_tmpfile; /* Slave-> master SYNC temp file name */
|
||||||
time_t repl_transfer_lastio; /* Unix time of the latest read, for timeout */
|
time_t repl_transfer_lastio; /* Unix time of the latest read, for timeout */
|
||||||
int repl_serve_stale_data; /* Serve stale data when link is down? */
|
int repl_serve_stale_data; /* Serve stale data when link is down? */
|
||||||
|
int repl_slave_ro; /* Slave is read only? */
|
||||||
time_t repl_down_since; /* Unix time at which link with master went down */
|
time_t repl_down_since; /* Unix time at which link with master went down */
|
||||||
/* Limits */
|
/* Limits */
|
||||||
unsigned int maxclients; /* Max number of simultaneous clients */
|
unsigned int maxclients; /* Max number of simultaneous clients */
|
||||||
@ -643,8 +688,8 @@ struct redisServer {
|
|||||||
int sort_alpha;
|
int sort_alpha;
|
||||||
int sort_bypattern;
|
int sort_bypattern;
|
||||||
/* Zip structure config, see redis.conf for more information */
|
/* Zip structure config, see redis.conf for more information */
|
||||||
size_t hash_max_zipmap_entries;
|
size_t hash_max_ziplist_entries;
|
||||||
size_t hash_max_zipmap_value;
|
size_t hash_max_ziplist_value;
|
||||||
size_t list_max_ziplist_entries;
|
size_t list_max_ziplist_entries;
|
||||||
size_t list_max_ziplist_value;
|
size_t list_max_ziplist_value;
|
||||||
size_t set_max_intset_entries;
|
size_t set_max_intset_entries;
|
||||||
@ -748,10 +793,10 @@ typedef struct {
|
|||||||
* not both are required, store pointers in the iterator to avoid
|
* not both are required, store pointers in the iterator to avoid
|
||||||
* unnecessary memory allocation for fields/values. */
|
* unnecessary memory allocation for fields/values. */
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
robj *subject;
|
||||||
int encoding;
|
int encoding;
|
||||||
unsigned char *zi;
|
|
||||||
unsigned char *zk, *zv;
|
unsigned char *fptr, *vptr;
|
||||||
unsigned int zklen, zvlen;
|
|
||||||
|
|
||||||
dictIterator *di;
|
dictIterator *di;
|
||||||
dictEntry *de;
|
dictEntry *de;
|
||||||
@ -780,6 +825,7 @@ dictType hashDictType;
|
|||||||
/* Utils */
|
/* Utils */
|
||||||
long long ustime(void);
|
long long ustime(void);
|
||||||
long long mstime(void);
|
long long mstime(void);
|
||||||
|
void getRandomHexChars(char *p, unsigned int len);
|
||||||
|
|
||||||
/* networking.c -- Networking and Client related operations */
|
/* networking.c -- Networking and Client related operations */
|
||||||
redisClient *createClient(int fd);
|
redisClient *createClient(int fd);
|
||||||
@ -856,6 +902,7 @@ void freeClientMultiState(redisClient *c);
|
|||||||
void queueMultiCommand(redisClient *c);
|
void queueMultiCommand(redisClient *c);
|
||||||
void touchWatchedKey(redisDb *db, robj *key);
|
void touchWatchedKey(redisDb *db, robj *key);
|
||||||
void touchWatchedKeysOnFlush(int dbid);
|
void touchWatchedKeysOnFlush(int dbid);
|
||||||
|
void discardTransaction(redisClient *c);
|
||||||
|
|
||||||
/* Redis object implementation */
|
/* Redis object implementation */
|
||||||
void decrRefCount(void *o);
|
void decrRefCount(void *o);
|
||||||
@ -901,7 +948,7 @@ int syncReadLine(int fd, char *ptr, ssize_t size, int timeout);
|
|||||||
|
|
||||||
/* Replication */
|
/* Replication */
|
||||||
void replicationFeedSlaves(list *slaves, int dictid, robj **argv, int argc);
|
void replicationFeedSlaves(list *slaves, int dictid, robj **argv, int argc);
|
||||||
void replicationFeedMonitors(list *monitors, int dictid, robj **argv, int argc);
|
void replicationFeedMonitors(redisClient *c, list *monitors, int dictid, robj **argv, int argc);
|
||||||
void updateSlavesWaitingBgsave(int bgsaveerr);
|
void updateSlavesWaitingBgsave(int bgsaveerr);
|
||||||
void replicationCron(void);
|
void replicationCron(void);
|
||||||
|
|
||||||
@ -950,6 +997,8 @@ void setupSignalHandlers(void);
|
|||||||
struct redisCommand *lookupCommand(sds name);
|
struct redisCommand *lookupCommand(sds name);
|
||||||
struct redisCommand *lookupCommandByCString(char *s);
|
struct redisCommand *lookupCommandByCString(char *s);
|
||||||
void call(redisClient *c, int flags);
|
void call(redisClient *c, int flags);
|
||||||
|
void propagate(struct redisCommand *cmd, int dbid, robj **argv, int argc, int flags);
|
||||||
|
void alsoPropagate(struct redisCommand *cmd, int dbid, robj **argv, int argc, int target);
|
||||||
int prepareForShutdown();
|
int prepareForShutdown();
|
||||||
void redisLog(int level, const char *fmt, ...);
|
void redisLog(int level, const char *fmt, ...);
|
||||||
void redisLogRaw(int level, const char *msg);
|
void redisLogRaw(int level, const char *msg);
|
||||||
@ -974,10 +1023,9 @@ unsigned long setTypeSize(robj *subject);
|
|||||||
void setTypeConvert(robj *subject, int enc);
|
void setTypeConvert(robj *subject, int enc);
|
||||||
|
|
||||||
/* Hash data type */
|
/* Hash data type */
|
||||||
void convertToRealHash(robj *o);
|
void hashTypeConvert(robj *o, int enc);
|
||||||
void hashTypeTryConversion(robj *subject, robj **argv, int start, int end);
|
void hashTypeTryConversion(robj *subject, robj **argv, int start, int end);
|
||||||
void hashTypeTryObjectEncoding(robj *subject, robj **o1, robj **o2);
|
void hashTypeTryObjectEncoding(robj *subject, robj **o1, robj **o2);
|
||||||
int hashTypeGet(robj *o, robj *key, robj **objval, unsigned char **v, unsigned int *vlen);
|
|
||||||
robj *hashTypeGetObject(robj *o, robj *key);
|
robj *hashTypeGetObject(robj *o, robj *key);
|
||||||
int hashTypeExists(robj *o, robj *key);
|
int hashTypeExists(robj *o, robj *key);
|
||||||
int hashTypeSet(robj *o, robj *key, robj *value);
|
int hashTypeSet(robj *o, robj *key, robj *value);
|
||||||
@ -986,7 +1034,11 @@ unsigned long hashTypeLength(robj *o);
|
|||||||
hashTypeIterator *hashTypeInitIterator(robj *subject);
|
hashTypeIterator *hashTypeInitIterator(robj *subject);
|
||||||
void hashTypeReleaseIterator(hashTypeIterator *hi);
|
void hashTypeReleaseIterator(hashTypeIterator *hi);
|
||||||
int hashTypeNext(hashTypeIterator *hi);
|
int hashTypeNext(hashTypeIterator *hi);
|
||||||
int hashTypeCurrent(hashTypeIterator *hi, int what, robj **objval, unsigned char **v, unsigned int *vlen);
|
void hashTypeCurrentFromZiplist(hashTypeIterator *hi, int what,
|
||||||
|
unsigned char **vstr,
|
||||||
|
unsigned int *vlen,
|
||||||
|
long long *vll);
|
||||||
|
void hashTypeCurrentFromHashTable(hashTypeIterator *hi, int what, robj **dst);
|
||||||
robj *hashTypeCurrentObject(hashTypeIterator *hi, int what);
|
robj *hashTypeCurrentObject(hashTypeIterator *hi, int what);
|
||||||
robj *hashTypeLookupWriteOrCreate(redisClient *c, robj *key);
|
robj *hashTypeLookupWriteOrCreate(redisClient *c, robj *key);
|
||||||
|
|
||||||
@ -1185,6 +1237,7 @@ void clientCommand(redisClient *c);
|
|||||||
void evalCommand(redisClient *c);
|
void evalCommand(redisClient *c);
|
||||||
void evalShaCommand(redisClient *c);
|
void evalShaCommand(redisClient *c);
|
||||||
void scriptCommand(redisClient *c);
|
void scriptCommand(redisClient *c);
|
||||||
|
void timeCommand(redisClient *c);
|
||||||
|
|
||||||
#if defined(__GNUC__)
|
#if defined(__GNUC__)
|
||||||
void *calloc(size_t count, size_t size) __attribute__ ((deprecated));
|
void *calloc(size_t count, size_t size) __attribute__ ((deprecated));
|
||||||
|
@ -50,17 +50,23 @@ void replicationFeedSlaves(list *slaves, int dictid, robj **argv, int argc) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void replicationFeedMonitors(list *monitors, int dictid, robj **argv, int argc) {
|
void replicationFeedMonitors(redisClient *c, list *monitors, int dictid, robj **argv, int argc) {
|
||||||
listNode *ln;
|
listNode *ln;
|
||||||
listIter li;
|
listIter li;
|
||||||
int j;
|
int j, port;
|
||||||
sds cmdrepr = sdsnew("+");
|
sds cmdrepr = sdsnew("+");
|
||||||
robj *cmdobj;
|
robj *cmdobj;
|
||||||
|
char ip[32];
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
|
|
||||||
gettimeofday(&tv,NULL);
|
gettimeofday(&tv,NULL);
|
||||||
cmdrepr = sdscatprintf(cmdrepr,"%ld.%06ld ",(long)tv.tv_sec,(long)tv.tv_usec);
|
cmdrepr = sdscatprintf(cmdrepr,"%ld.%06ld ",(long)tv.tv_sec,(long)tv.tv_usec);
|
||||||
if (dictid != 0) cmdrepr = sdscatprintf(cmdrepr,"(db %d) ", dictid);
|
if (c->flags & REDIS_LUA_CLIENT) {
|
||||||
|
cmdrepr = sdscatprintf(cmdrepr,"[%d lua] ", dictid);
|
||||||
|
} else {
|
||||||
|
anetPeerToString(c->fd,ip,&port);
|
||||||
|
cmdrepr = sdscatprintf(cmdrepr,"[%d %s:%d] ", dictid,ip,port);
|
||||||
|
}
|
||||||
|
|
||||||
for (j = 0; j < argc; j++) {
|
for (j = 0; j < argc; j++) {
|
||||||
if (argv[j]->encoding == REDIS_ENCODING_INT) {
|
if (argv[j]->encoding == REDIS_ENCODING_INT) {
|
||||||
@ -596,7 +602,7 @@ void replicationCron(void) {
|
|||||||
if (slave->replstate == REDIS_REPL_SEND_BULK) continue;
|
if (slave->replstate == REDIS_REPL_SEND_BULK) continue;
|
||||||
if (slave->replstate == REDIS_REPL_ONLINE) {
|
if (slave->replstate == REDIS_REPL_ONLINE) {
|
||||||
/* If the slave is online send a normal ping */
|
/* If the slave is online send a normal ping */
|
||||||
addReplySds(slave,sdsnew("PING\r\n"));
|
addReplySds(slave,sdsnew("*1\r\n$4\r\nPING\r\n"));
|
||||||
} else {
|
} else {
|
||||||
/* Otherwise we are in the pre-synchronization stage.
|
/* Otherwise we are in the pre-synchronization stage.
|
||||||
* Just a newline will do the work of refreshing the
|
* Just a newline will do the work of refreshing the
|
||||||
|
@ -206,15 +206,45 @@ int luaRedisGenericCommand(lua_State *lua, int raise_error) {
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* There are commands that are not allowed inside scripts. */
|
||||||
if (cmd->flags & REDIS_CMD_NOSCRIPT) {
|
if (cmd->flags & REDIS_CMD_NOSCRIPT) {
|
||||||
luaPushError(lua, "This Redis command is not allowed from scripts");
|
luaPushError(lua, "This Redis command is not allowed from scripts");
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cmd->flags & REDIS_CMD_WRITE && server.lua_random_dirty) {
|
/* Write commands are forbidden against read-only slaves, or if a
|
||||||
luaPushError(lua,
|
* command marked as non-deterministic was already called in the context
|
||||||
"Write commands not allowed after non deterministic commands");
|
* of this script. */
|
||||||
goto cleanup;
|
if (cmd->flags & REDIS_CMD_WRITE) {
|
||||||
|
if (server.lua_random_dirty) {
|
||||||
|
luaPushError(lua,
|
||||||
|
"Write commands not allowed after non deterministic commands");
|
||||||
|
goto cleanup;
|
||||||
|
} else if (server.masterhost && server.repl_slave_ro &&
|
||||||
|
!(server.lua_caller->flags & REDIS_MASTER))
|
||||||
|
{
|
||||||
|
luaPushError(lua, shared.roslaveerr->ptr);
|
||||||
|
goto cleanup;
|
||||||
|
} else if (server.stop_writes_on_bgsave_err &&
|
||||||
|
server.saveparamslen > 0 &&
|
||||||
|
server.lastbgsave_status == REDIS_ERR)
|
||||||
|
{
|
||||||
|
luaPushError(lua, shared.bgsaveerr->ptr);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If we reached the memory limit configured via maxmemory, commands that
|
||||||
|
* could enlarge the memory usage are not allowed, but only if this is the
|
||||||
|
* first write in the context of this script, otherwise we can't stop
|
||||||
|
* in the middle. */
|
||||||
|
if (server.maxmemory && server.lua_write_dirty == 0 &&
|
||||||
|
(cmd->flags & REDIS_CMD_DENYOOM))
|
||||||
|
{
|
||||||
|
if (freeMemoryIfNeeded() == REDIS_ERR) {
|
||||||
|
luaPushError(lua, shared.oomerr->ptr);
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cmd->flags & REDIS_CMD_RANDOM) server.lua_random_dirty = 1;
|
if (cmd->flags & REDIS_CMD_RANDOM) server.lua_random_dirty = 1;
|
||||||
|
18
src/sds.c
18
src/sds.c
@ -111,6 +111,24 @@ sds sdsMakeRoomFor(sds s, size_t addlen) {
|
|||||||
return newsh->buf;
|
return newsh->buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Reallocate the sds string so that it has no free space at the end. The
|
||||||
|
* contained string remains not altered, but next concatenation operations
|
||||||
|
* will require a reallocation. */
|
||||||
|
sds sdsRemoveFreeSpace(sds s) {
|
||||||
|
struct sdshdr *sh;
|
||||||
|
|
||||||
|
sh = (void*) (s-(sizeof(struct sdshdr)));
|
||||||
|
sh = zrealloc(sh, sizeof(struct sdshdr)+sh->len+1);
|
||||||
|
sh->free = 0;
|
||||||
|
return sh->buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t sdsAllocSize(sds s) {
|
||||||
|
struct sdshdr *sh = (void*) (s-(sizeof(struct sdshdr)));
|
||||||
|
|
||||||
|
return sizeof(*sh)+sh->len+sh->free+1;
|
||||||
|
}
|
||||||
|
|
||||||
/* Increment the sds length and decrements the left free space at the
|
/* Increment the sds length and decrements the left free space at the
|
||||||
* end of the string accordingly to 'incr'. Also set the null term
|
* end of the string accordingly to 'incr'. Also set the null term
|
||||||
* in the new end of the string.
|
* in the new end of the string.
|
||||||
|
@ -94,5 +94,7 @@ sds sdsmapchars(sds s, char *from, char *to, size_t setlen);
|
|||||||
/* Low level functions exposed to the user API */
|
/* Low level functions exposed to the user API */
|
||||||
sds sdsMakeRoomFor(sds s, size_t addlen);
|
sds sdsMakeRoomFor(sds s, size_t addlen);
|
||||||
void sdsIncrLen(sds s, int incr);
|
void sdsIncrLen(sds s, int incr);
|
||||||
|
sds sdsRemoveFreeSpace(sds s);
|
||||||
|
size_t sdsAllocSize(sds s);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
600
src/t_hash.c
600
src/t_hash.c
@ -1,5 +1,4 @@
|
|||||||
#include "redis.h"
|
#include "redis.h"
|
||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
/*-----------------------------------------------------------------------------
|
/*-----------------------------------------------------------------------------
|
||||||
@ -7,18 +6,19 @@
|
|||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/* Check the length of a number of objects to see if we need to convert a
|
/* Check the length of a number of objects to see if we need to convert a
|
||||||
* zipmap to a real hash. Note that we only check string encoded objects
|
* ziplist to a real hash. Note that we only check string encoded objects
|
||||||
* as their string length can be queried in constant time. */
|
* as their string length can be queried in constant time. */
|
||||||
void hashTypeTryConversion(robj *subject, robj **argv, int start, int end) {
|
void hashTypeTryConversion(robj *o, robj **argv, int start, int end) {
|
||||||
int i;
|
int i;
|
||||||
if (subject->encoding != REDIS_ENCODING_ZIPMAP) return;
|
|
||||||
|
if (o->encoding != REDIS_ENCODING_ZIPLIST) return;
|
||||||
|
|
||||||
for (i = start; i <= end; i++) {
|
for (i = start; i <= end; i++) {
|
||||||
if (argv[i]->encoding == REDIS_ENCODING_RAW &&
|
if (argv[i]->encoding == REDIS_ENCODING_RAW &&
|
||||||
sdslen(argv[i]->ptr) > server.hash_max_zipmap_value)
|
sdslen(argv[i]->ptr) > server.hash_max_ziplist_value)
|
||||||
{
|
{
|
||||||
convertToRealHash(subject);
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
return;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -31,137 +31,233 @@ void hashTypeTryObjectEncoding(robj *subject, robj **o1, robj **o2) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the value from a hash identified by key.
|
/* Get the value from a ziplist encoded hash, identified by field.
|
||||||
*
|
* Returns -1 when the field cannot be found. */
|
||||||
* If the string is found either REDIS_ENCODING_HT or REDIS_ENCODING_ZIPMAP
|
int hashTypeGetFromZiplist(robj *o, robj *field,
|
||||||
* is returned, and either **objval or **v and *vlen are set accordingly,
|
unsigned char **vstr,
|
||||||
* so that objects in hash tables are returend as objects and pointers
|
unsigned int *vlen,
|
||||||
* inside a zipmap are returned as such.
|
long long *vll)
|
||||||
*
|
|
||||||
* If the object was not found -1 is returned.
|
|
||||||
*
|
|
||||||
* This function is copy on write friendly as there is no incr/decr
|
|
||||||
* of refcount needed if objects are accessed just for reading operations. */
|
|
||||||
int hashTypeGet(robj *o, robj *key, robj **objval, unsigned char **v,
|
|
||||||
unsigned int *vlen)
|
|
||||||
{
|
{
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
unsigned char *zl, *fptr = NULL, *vptr = NULL;
|
||||||
int found;
|
int ret;
|
||||||
|
|
||||||
key = getDecodedObject(key);
|
redisAssert(o->encoding == REDIS_ENCODING_ZIPLIST);
|
||||||
found = zipmapGet(o->ptr,key->ptr,sdslen(key->ptr),v,vlen);
|
|
||||||
decrRefCount(key);
|
field = getDecodedObject(field);
|
||||||
if (!found) return -1;
|
|
||||||
} else {
|
zl = o->ptr;
|
||||||
dictEntry *de = dictFind(o->ptr,key);
|
fptr = ziplistIndex(zl, ZIPLIST_HEAD);
|
||||||
if (de == NULL) return -1;
|
if (fptr != NULL) {
|
||||||
*objval = dictGetVal(de);
|
fptr = ziplistFind(fptr, field->ptr, sdslen(field->ptr), 1);
|
||||||
|
if (fptr != NULL) {
|
||||||
|
/* Grab pointer to the value (fptr points to the field) */
|
||||||
|
vptr = ziplistNext(zl, fptr);
|
||||||
|
redisAssert(vptr != NULL);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return o->encoding;
|
|
||||||
|
decrRefCount(field);
|
||||||
|
|
||||||
|
if (vptr != NULL) {
|
||||||
|
ret = ziplistGet(vptr, vstr, vlen, vll);
|
||||||
|
redisAssert(ret);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Higher level function of hashTypeGet() that always returns a Redis
|
/* Get the value from a hash table encoded hash, identified by field.
|
||||||
|
* Returns -1 when the field cannot be found. */
|
||||||
|
int hashTypeGetFromHashTable(robj *o, robj *field, robj **value) {
|
||||||
|
dictEntry *de;
|
||||||
|
|
||||||
|
redisAssert(o->encoding == REDIS_ENCODING_HT);
|
||||||
|
|
||||||
|
de = dictFind(o->ptr, field);
|
||||||
|
if (de == NULL) return -1;
|
||||||
|
*value = dictGetVal(de);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Higher level function of hashTypeGet*() that always returns a Redis
|
||||||
* object (either new or with refcount incremented), so that the caller
|
* object (either new or with refcount incremented), so that the caller
|
||||||
* can retain a reference or call decrRefCount after the usage.
|
* can retain a reference or call decrRefCount after the usage.
|
||||||
*
|
*
|
||||||
* The lower level function can prevent copy on write so it is
|
* The lower level function can prevent copy on write so it is
|
||||||
* the preferred way of doing read operations. */
|
* the preferred way of doing read operations. */
|
||||||
robj *hashTypeGetObject(robj *o, robj *key) {
|
robj *hashTypeGetObject(robj *o, robj *field) {
|
||||||
robj *objval;
|
robj *value = NULL;
|
||||||
unsigned char *v;
|
|
||||||
unsigned int vlen;
|
|
||||||
|
|
||||||
int encoding = hashTypeGet(o,key,&objval,&v,&vlen);
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
switch(encoding) {
|
unsigned char *vstr = NULL;
|
||||||
case REDIS_ENCODING_HT:
|
unsigned int vlen = UINT_MAX;
|
||||||
incrRefCount(objval);
|
long long vll = LLONG_MAX;
|
||||||
return objval;
|
|
||||||
case REDIS_ENCODING_ZIPMAP:
|
if (hashTypeGetFromZiplist(o, field, &vstr, &vlen, &vll) == 0) {
|
||||||
objval = createStringObject((char*)v,vlen);
|
if (vstr) {
|
||||||
return objval;
|
value = createStringObject((char*)vstr, vlen);
|
||||||
default: return NULL;
|
} else {
|
||||||
|
value = createStringObjectFromLongLong(vll);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
robj *aux;
|
||||||
|
|
||||||
|
if (hashTypeGetFromHashTable(o, field, &aux) == 0) {
|
||||||
|
incrRefCount(aux);
|
||||||
|
value = aux;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Test if the key exists in the given hash. Returns 1 if the key
|
/* Test if the specified field exists in the given hash. Returns 1 if the field
|
||||||
* exists and 0 when it doesn't. */
|
* exists, and 0 when it doesn't. */
|
||||||
int hashTypeExists(robj *o, robj *key) {
|
int hashTypeExists(robj *o, robj *field) {
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
key = getDecodedObject(key);
|
unsigned char *vstr = NULL;
|
||||||
if (zipmapExists(o->ptr,key->ptr,sdslen(key->ptr))) {
|
unsigned int vlen = UINT_MAX;
|
||||||
decrRefCount(key);
|
long long vll = LLONG_MAX;
|
||||||
return 1;
|
|
||||||
}
|
if (hashTypeGetFromZiplist(o, field, &vstr, &vlen, &vll) == 0) return 1;
|
||||||
decrRefCount(key);
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
robj *aux;
|
||||||
|
|
||||||
|
if (hashTypeGetFromHashTable(o, field, &aux) == 0) return 1;
|
||||||
} else {
|
} else {
|
||||||
if (dictFind(o->ptr,key) != NULL) {
|
redisPanic("Unknown hash encoding");
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Add an element, discard the old if the key already exists.
|
/* Add an element, discard the old if the key already exists.
|
||||||
* Return 0 on insert and 1 on update. */
|
* Return 0 on insert and 1 on update.
|
||||||
int hashTypeSet(robj *o, robj *key, robj *value) {
|
* This function will take care of incrementing the reference count of the
|
||||||
|
* retained fields and value objects. */
|
||||||
|
int hashTypeSet(robj *o, robj *field, robj *value) {
|
||||||
int update = 0;
|
int update = 0;
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
|
||||||
key = getDecodedObject(key);
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
unsigned char *zl, *fptr, *vptr;
|
||||||
|
|
||||||
|
field = getDecodedObject(field);
|
||||||
value = getDecodedObject(value);
|
value = getDecodedObject(value);
|
||||||
o->ptr = zipmapSet(o->ptr,
|
|
||||||
key->ptr,sdslen(key->ptr),
|
zl = o->ptr;
|
||||||
value->ptr,sdslen(value->ptr), &update);
|
fptr = ziplistIndex(zl, ZIPLIST_HEAD);
|
||||||
decrRefCount(key);
|
if (fptr != NULL) {
|
||||||
|
fptr = ziplistFind(fptr, field->ptr, sdslen(field->ptr), 1);
|
||||||
|
if (fptr != NULL) {
|
||||||
|
/* Grab pointer to the value (fptr points to the field) */
|
||||||
|
vptr = ziplistNext(zl, fptr);
|
||||||
|
redisAssert(vptr != NULL);
|
||||||
|
update = 1;
|
||||||
|
|
||||||
|
/* Delete value */
|
||||||
|
zl = ziplistDelete(zl, &vptr);
|
||||||
|
|
||||||
|
/* Insert new value */
|
||||||
|
zl = ziplistInsert(zl, vptr, value->ptr, sdslen(value->ptr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!update) {
|
||||||
|
/* Push new field/value pair onto the tail of the ziplist */
|
||||||
|
zl = ziplistPush(zl, field->ptr, sdslen(field->ptr), ZIPLIST_TAIL);
|
||||||
|
zl = ziplistPush(zl, value->ptr, sdslen(value->ptr), ZIPLIST_TAIL);
|
||||||
|
}
|
||||||
|
o->ptr = zl;
|
||||||
|
decrRefCount(field);
|
||||||
decrRefCount(value);
|
decrRefCount(value);
|
||||||
|
|
||||||
/* Check if the zipmap needs to be upgraded to a real hash table */
|
/* Check if the ziplist needs to be converted to a hash table */
|
||||||
if (zipmapLen(o->ptr) > server.hash_max_zipmap_entries)
|
if (hashTypeLength(o) > server.hash_max_ziplist_entries)
|
||||||
convertToRealHash(o);
|
hashTypeConvert(o, REDIS_ENCODING_HT);
|
||||||
} else {
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
if (dictReplace(o->ptr,key,value)) {
|
if (dictReplace(o->ptr, field, value)) { /* Insert */
|
||||||
/* Insert */
|
incrRefCount(field);
|
||||||
incrRefCount(key);
|
} else { /* Update */
|
||||||
} else {
|
|
||||||
/* Update */
|
|
||||||
update = 1;
|
update = 1;
|
||||||
}
|
}
|
||||||
incrRefCount(value);
|
incrRefCount(value);
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
return update;
|
return update;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Delete an element from a hash.
|
/* Delete an element from a hash.
|
||||||
* Return 1 on deleted and 0 on not found. */
|
* Return 1 on deleted and 0 on not found. */
|
||||||
int hashTypeDelete(robj *o, robj *key) {
|
int hashTypeDelete(robj *o, robj *field) {
|
||||||
int deleted = 0;
|
int deleted = 0;
|
||||||
if (o->encoding == REDIS_ENCODING_ZIPMAP) {
|
|
||||||
key = getDecodedObject(key);
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
o->ptr = zipmapDel(o->ptr,key->ptr,sdslen(key->ptr), &deleted);
|
unsigned char *zl, *fptr;
|
||||||
decrRefCount(key);
|
|
||||||
|
field = getDecodedObject(field);
|
||||||
|
|
||||||
|
zl = o->ptr;
|
||||||
|
fptr = ziplistIndex(zl, ZIPLIST_HEAD);
|
||||||
|
if (fptr != NULL) {
|
||||||
|
fptr = ziplistFind(fptr, field->ptr, sdslen(field->ptr), 1);
|
||||||
|
if (fptr != NULL) {
|
||||||
|
zl = ziplistDelete(zl,&fptr);
|
||||||
|
zl = ziplistDelete(zl,&fptr);
|
||||||
|
o->ptr = zl;
|
||||||
|
deleted = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
decrRefCount(field);
|
||||||
|
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
if (dictDelete((dict*)o->ptr, field) == REDIS_OK) {
|
||||||
|
deleted = 1;
|
||||||
|
|
||||||
|
/* Always check if the dictionary needs a resize after a delete. */
|
||||||
|
if (htNeedsResize(o->ptr)) dictResize(o->ptr);
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
deleted = dictDelete((dict*)o->ptr,key) == DICT_OK;
|
redisPanic("Unknown hash encoding");
|
||||||
/* Always check if the dictionary needs a resize after a delete. */
|
|
||||||
if (deleted && htNeedsResize(o->ptr)) dictResize(o->ptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return deleted;
|
return deleted;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the number of elements in a hash. */
|
/* Return the number of elements in a hash. */
|
||||||
unsigned long hashTypeLength(robj *o) {
|
unsigned long hashTypeLength(robj *o) {
|
||||||
return (o->encoding == REDIS_ENCODING_ZIPMAP) ?
|
unsigned long length = ULONG_MAX;
|
||||||
zipmapLen((unsigned char*)o->ptr) : dictSize((dict*)o->ptr);
|
|
||||||
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
length = ziplistLen(o->ptr) / 2;
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
length = dictSize((dict*)o->ptr);
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
}
|
||||||
|
|
||||||
|
return length;
|
||||||
}
|
}
|
||||||
|
|
||||||
hashTypeIterator *hashTypeInitIterator(robj *subject) {
|
hashTypeIterator *hashTypeInitIterator(robj *subject) {
|
||||||
hashTypeIterator *hi = zmalloc(sizeof(hashTypeIterator));
|
hashTypeIterator *hi = zmalloc(sizeof(hashTypeIterator));
|
||||||
|
hi->subject = subject;
|
||||||
hi->encoding = subject->encoding;
|
hi->encoding = subject->encoding;
|
||||||
if (hi->encoding == REDIS_ENCODING_ZIPMAP) {
|
|
||||||
hi->zi = zipmapRewind(subject->ptr);
|
if (hi->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
hi->fptr = NULL;
|
||||||
|
hi->vptr = NULL;
|
||||||
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
hi->di = dictGetIterator(subject->ptr);
|
hi->di = dictGetIterator(subject->ptr);
|
||||||
} else {
|
} else {
|
||||||
redisAssertWithInfo(NULL,subject,0);
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
|
|
||||||
return hi;
|
return hi;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -169,68 +265,108 @@ void hashTypeReleaseIterator(hashTypeIterator *hi) {
|
|||||||
if (hi->encoding == REDIS_ENCODING_HT) {
|
if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
dictReleaseIterator(hi->di);
|
dictReleaseIterator(hi->di);
|
||||||
}
|
}
|
||||||
|
|
||||||
zfree(hi);
|
zfree(hi);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the next entry in the hash. Return REDIS_OK when the next entry
|
/* Move to the next entry in the hash. Return REDIS_OK when the next entry
|
||||||
* could be found and REDIS_ERR when the iterator reaches the end. */
|
* could be found and REDIS_ERR when the iterator reaches the end. */
|
||||||
int hashTypeNext(hashTypeIterator *hi) {
|
int hashTypeNext(hashTypeIterator *hi) {
|
||||||
if (hi->encoding == REDIS_ENCODING_ZIPMAP) {
|
if (hi->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
if ((hi->zi = zipmapNext(hi->zi, &hi->zk, &hi->zklen,
|
unsigned char *zl;
|
||||||
&hi->zv, &hi->zvlen)) == NULL) return REDIS_ERR;
|
unsigned char *fptr, *vptr;
|
||||||
} else {
|
|
||||||
|
zl = hi->subject->ptr;
|
||||||
|
fptr = hi->fptr;
|
||||||
|
vptr = hi->vptr;
|
||||||
|
|
||||||
|
if (fptr == NULL) {
|
||||||
|
/* Initialize cursor */
|
||||||
|
redisAssert(vptr == NULL);
|
||||||
|
fptr = ziplistIndex(zl, 0);
|
||||||
|
} else {
|
||||||
|
/* Advance cursor */
|
||||||
|
redisAssert(vptr != NULL);
|
||||||
|
fptr = ziplistNext(zl, vptr);
|
||||||
|
}
|
||||||
|
if (fptr == NULL) return REDIS_ERR;
|
||||||
|
|
||||||
|
/* Grab pointer to the value (fptr points to the field) */
|
||||||
|
vptr = ziplistNext(zl, fptr);
|
||||||
|
redisAssert(vptr != NULL);
|
||||||
|
|
||||||
|
/* fptr, vptr now point to the first or next pair */
|
||||||
|
hi->fptr = fptr;
|
||||||
|
hi->vptr = vptr;
|
||||||
|
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
if ((hi->de = dictNext(hi->di)) == NULL) return REDIS_ERR;
|
if ((hi->de = dictNext(hi->di)) == NULL) return REDIS_ERR;
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
return REDIS_OK;
|
return REDIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get key or value object at current iteration position.
|
/* Get the field or value at iterator cursor, for an iterator on a hash value
|
||||||
* The returned item differs with the hash object encoding:
|
* encoded as a ziplist. Prototype is similar to `hashTypeGetFromZiplist`. */
|
||||||
* - When encoding is REDIS_ENCODING_HT, the objval pointer is populated
|
void hashTypeCurrentFromZiplist(hashTypeIterator *hi, int what,
|
||||||
* with the original object.
|
unsigned char **vstr,
|
||||||
* - When encoding is REDIS_ENCODING_ZIPMAP, a pointer to the string and
|
unsigned int *vlen,
|
||||||
* its length is retunred populating the v and vlen pointers.
|
long long *vll)
|
||||||
* This function is copy on write friendly as accessing objects in read only
|
{
|
||||||
* does not require writing to any memory page.
|
int ret;
|
||||||
*
|
|
||||||
* The function returns the encoding of the object, so that the caller
|
redisAssert(hi->encoding == REDIS_ENCODING_ZIPLIST);
|
||||||
* can underestand if the key or value was returned as object or C string. */
|
|
||||||
int hashTypeCurrent(hashTypeIterator *hi, int what, robj **objval, unsigned char **v, unsigned int *vlen) {
|
if (what & REDIS_HASH_KEY) {
|
||||||
if (hi->encoding == REDIS_ENCODING_ZIPMAP) {
|
ret = ziplistGet(hi->fptr, vstr, vlen, vll);
|
||||||
if (what & REDIS_HASH_KEY) {
|
redisAssert(ret);
|
||||||
*v = hi->zk;
|
|
||||||
*vlen = hi->zklen;
|
|
||||||
} else {
|
|
||||||
*v = hi->zv;
|
|
||||||
*vlen = hi->zvlen;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (what & REDIS_HASH_KEY)
|
ret = ziplistGet(hi->vptr, vstr, vlen, vll);
|
||||||
*objval = dictGetKey(hi->de);
|
redisAssert(ret);
|
||||||
else
|
|
||||||
*objval = dictGetVal(hi->de);
|
|
||||||
}
|
}
|
||||||
return hi->encoding;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* A non copy-on-write friendly but higher level version of hashTypeCurrent()
|
/* Get the field or value at iterator cursor, for an iterator on a hash value
|
||||||
* that always returns an object with refcount incremented by one (or a new
|
* encoded as a ziplist. Prototype is similar to `hashTypeGetFromHashTable`. */
|
||||||
* object), so it's up to the caller to decrRefCount() the object if no
|
void hashTypeCurrentFromHashTable(hashTypeIterator *hi, int what, robj **dst) {
|
||||||
* reference is retained. */
|
redisAssert(hi->encoding == REDIS_ENCODING_HT);
|
||||||
robj *hashTypeCurrentObject(hashTypeIterator *hi, int what) {
|
|
||||||
robj *obj;
|
|
||||||
unsigned char *v = NULL;
|
|
||||||
unsigned int vlen = 0;
|
|
||||||
int encoding = hashTypeCurrent(hi,what,&obj,&v,&vlen);
|
|
||||||
|
|
||||||
if (encoding == REDIS_ENCODING_HT) {
|
if (what & REDIS_HASH_KEY) {
|
||||||
incrRefCount(obj);
|
*dst = dictGetKey(hi->de);
|
||||||
return obj;
|
|
||||||
} else {
|
} else {
|
||||||
return createStringObject((char*)v,vlen);
|
*dst = dictGetVal(hi->de);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* A non copy-on-write friendly but higher level version of hashTypeCurrent*()
|
||||||
|
* that returns an object with incremented refcount (or a new object). It is up
|
||||||
|
* to the caller to decrRefCount() the object if no reference is retained. */
|
||||||
|
robj *hashTypeCurrentObject(hashTypeIterator *hi, int what) {
|
||||||
|
robj *dst;
|
||||||
|
|
||||||
|
if (hi->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
unsigned char *vstr = NULL;
|
||||||
|
unsigned int vlen = UINT_MAX;
|
||||||
|
long long vll = LLONG_MAX;
|
||||||
|
|
||||||
|
hashTypeCurrentFromZiplist(hi, what, &vstr, &vlen, &vll);
|
||||||
|
if (vstr) {
|
||||||
|
dst = createStringObject((char*)vstr, vlen);
|
||||||
|
} else {
|
||||||
|
dst = createStringObjectFromLongLong(vll);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
|
hashTypeCurrentFromHashTable(hi, what, &dst);
|
||||||
|
incrRefCount(dst);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
}
|
||||||
|
|
||||||
|
return dst;
|
||||||
|
}
|
||||||
|
|
||||||
robj *hashTypeLookupWriteOrCreate(redisClient *c, robj *key) {
|
robj *hashTypeLookupWriteOrCreate(redisClient *c, robj *key) {
|
||||||
robj *o = lookupKeyWrite(c->db,key);
|
robj *o = lookupKeyWrite(c->db,key);
|
||||||
if (o == NULL) {
|
if (o == NULL) {
|
||||||
@ -245,25 +381,50 @@ robj *hashTypeLookupWriteOrCreate(redisClient *c, robj *key) {
|
|||||||
return o;
|
return o;
|
||||||
}
|
}
|
||||||
|
|
||||||
void convertToRealHash(robj *o) {
|
void hashTypeConvertZiplist(robj *o, int enc) {
|
||||||
unsigned char *key, *val, *p, *zm = o->ptr;
|
redisAssert(o->encoding == REDIS_ENCODING_ZIPLIST);
|
||||||
unsigned int klen, vlen;
|
|
||||||
dict *dict = dictCreate(&hashDictType,NULL);
|
|
||||||
|
|
||||||
redisAssertWithInfo(NULL,o,o->type == REDIS_HASH && o->encoding != REDIS_ENCODING_HT);
|
if (enc == REDIS_ENCODING_ZIPLIST) {
|
||||||
p = zipmapRewind(zm);
|
/* Nothing to do... */
|
||||||
while((p = zipmapNext(p,&key,&klen,&val,&vlen)) != NULL) {
|
|
||||||
robj *keyobj, *valobj;
|
|
||||||
|
|
||||||
keyobj = createStringObject((char*)key,klen);
|
} else if (enc == REDIS_ENCODING_HT) {
|
||||||
valobj = createStringObject((char*)val,vlen);
|
hashTypeIterator *hi;
|
||||||
keyobj = tryObjectEncoding(keyobj);
|
dict *dict;
|
||||||
valobj = tryObjectEncoding(valobj);
|
int ret;
|
||||||
dictAdd(dict,keyobj,valobj);
|
|
||||||
|
hi = hashTypeInitIterator(o);
|
||||||
|
dict = dictCreate(&hashDictType, NULL);
|
||||||
|
|
||||||
|
while (hashTypeNext(hi) != REDIS_ERR) {
|
||||||
|
robj *field, *value;
|
||||||
|
|
||||||
|
field = hashTypeCurrentObject(hi, REDIS_HASH_KEY);
|
||||||
|
field = tryObjectEncoding(field);
|
||||||
|
value = hashTypeCurrentObject(hi, REDIS_HASH_VALUE);
|
||||||
|
value = tryObjectEncoding(value);
|
||||||
|
ret = dictAdd(dict, field, value);
|
||||||
|
redisAssert(ret == DICT_OK);
|
||||||
|
}
|
||||||
|
|
||||||
|
hashTypeReleaseIterator(hi);
|
||||||
|
zfree(o->ptr);
|
||||||
|
|
||||||
|
o->encoding = REDIS_ENCODING_HT;
|
||||||
|
o->ptr = dict;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void hashTypeConvert(robj *o, int enc) {
|
||||||
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
hashTypeConvertZiplist(o, enc);
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
redisPanic("Not implemented");
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
o->encoding = REDIS_ENCODING_HT;
|
|
||||||
o->ptr = dict;
|
|
||||||
zfree(zm);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------------------
|
/*-----------------------------------------------------------------------------
|
||||||
@ -354,7 +515,7 @@ void hincrbyCommand(redisClient *c) {
|
|||||||
|
|
||||||
void hincrbyfloatCommand(redisClient *c) {
|
void hincrbyfloatCommand(redisClient *c) {
|
||||||
double long value, incr;
|
double long value, incr;
|
||||||
robj *o, *current, *new;
|
robj *o, *current, *new, *aux;
|
||||||
|
|
||||||
if (getLongDoubleFromObjectOrReply(c,c->argv[3],&incr,NULL) != REDIS_OK) return;
|
if (getLongDoubleFromObjectOrReply(c,c->argv[3],&incr,NULL) != REDIS_OK) return;
|
||||||
if ((o = hashTypeLookupWriteOrCreate(c,c->argv[1])) == NULL) return;
|
if ((o = hashTypeLookupWriteOrCreate(c,c->argv[1])) == NULL) return;
|
||||||
@ -374,56 +535,82 @@ void hincrbyfloatCommand(redisClient *c) {
|
|||||||
hashTypeTryObjectEncoding(o,&c->argv[2],NULL);
|
hashTypeTryObjectEncoding(o,&c->argv[2],NULL);
|
||||||
hashTypeSet(o,c->argv[2],new);
|
hashTypeSet(o,c->argv[2],new);
|
||||||
addReplyBulk(c,new);
|
addReplyBulk(c,new);
|
||||||
decrRefCount(new);
|
|
||||||
signalModifiedKey(c->db,c->argv[1]);
|
signalModifiedKey(c->db,c->argv[1]);
|
||||||
server.dirty++;
|
server.dirty++;
|
||||||
|
|
||||||
|
/* Always replicate HINCRBYFLOAT as an HSET command with the final value
|
||||||
|
* in order to make sure that differences in float pricision or formatting
|
||||||
|
* will not create differences in replicas or after an AOF restart. */
|
||||||
|
aux = createStringObject("HSET",4);
|
||||||
|
rewriteClientCommandArgument(c,0,aux);
|
||||||
|
decrRefCount(aux);
|
||||||
|
rewriteClientCommandArgument(c,3,new);
|
||||||
|
decrRefCount(new);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void addHashFieldToReply(redisClient *c, robj *o, robj *field) {
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (o == NULL) {
|
||||||
|
addReply(c, shared.nullbulk);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (o->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
unsigned char *vstr = NULL;
|
||||||
|
unsigned int vlen = UINT_MAX;
|
||||||
|
long long vll = LLONG_MAX;
|
||||||
|
|
||||||
|
ret = hashTypeGetFromZiplist(o, field, &vstr, &vlen, &vll);
|
||||||
|
if (ret < 0) {
|
||||||
|
addReply(c, shared.nullbulk);
|
||||||
|
} else {
|
||||||
|
if (vstr) {
|
||||||
|
addReplyBulkCBuffer(c, vstr, vlen);
|
||||||
|
} else {
|
||||||
|
addReplyBulkLongLong(c, vll);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (o->encoding == REDIS_ENCODING_HT) {
|
||||||
|
robj *value;
|
||||||
|
|
||||||
|
ret = hashTypeGetFromHashTable(o, field, &value);
|
||||||
|
if (ret < 0) {
|
||||||
|
addReply(c, shared.nullbulk);
|
||||||
|
} else {
|
||||||
|
addReplyBulk(c, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void hgetCommand(redisClient *c) {
|
void hgetCommand(redisClient *c) {
|
||||||
robj *o, *value;
|
robj *o;
|
||||||
unsigned char *v;
|
|
||||||
unsigned int vlen;
|
|
||||||
int encoding;
|
|
||||||
|
|
||||||
if ((o = lookupKeyReadOrReply(c,c->argv[1],shared.nullbulk)) == NULL ||
|
if ((o = lookupKeyReadOrReply(c,c->argv[1],shared.nullbulk)) == NULL ||
|
||||||
checkType(c,o,REDIS_HASH)) return;
|
checkType(c,o,REDIS_HASH)) return;
|
||||||
|
|
||||||
if ((encoding = hashTypeGet(o,c->argv[2],&value,&v,&vlen)) != -1) {
|
addHashFieldToReply(c, o, c->argv[2]);
|
||||||
if (encoding == REDIS_ENCODING_HT)
|
|
||||||
addReplyBulk(c,value);
|
|
||||||
else
|
|
||||||
addReplyBulkCBuffer(c,v,vlen);
|
|
||||||
} else {
|
|
||||||
addReply(c,shared.nullbulk);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void hmgetCommand(redisClient *c) {
|
void hmgetCommand(redisClient *c) {
|
||||||
int i, encoding;
|
robj *o;
|
||||||
robj *o, *value;
|
int i;
|
||||||
unsigned char *v;
|
|
||||||
unsigned int vlen;
|
|
||||||
|
|
||||||
o = lookupKeyRead(c->db,c->argv[1]);
|
/* Don't abort when the key cannot be found. Non-existing keys are empty
|
||||||
|
* hashes, where HMGET should respond with a series of null bulks. */
|
||||||
|
o = lookupKeyRead(c->db, c->argv[1]);
|
||||||
if (o != NULL && o->type != REDIS_HASH) {
|
if (o != NULL && o->type != REDIS_HASH) {
|
||||||
addReply(c,shared.wrongtypeerr);
|
addReply(c, shared.wrongtypeerr);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Note the check for o != NULL happens inside the loop. This is
|
addReplyMultiBulkLen(c, c->argc-2);
|
||||||
* done because objects that cannot be found are considered to be
|
|
||||||
* an empty hash. The reply should then be a series of NULLs. */
|
|
||||||
addReplyMultiBulkLen(c,c->argc-2);
|
|
||||||
for (i = 2; i < c->argc; i++) {
|
for (i = 2; i < c->argc; i++) {
|
||||||
if (o != NULL &&
|
addHashFieldToReply(c, o, c->argv[i]);
|
||||||
(encoding = hashTypeGet(o,c->argv[i],&value,&v,&vlen)) != -1) {
|
|
||||||
if (encoding == REDIS_ENCODING_HT)
|
|
||||||
addReplyBulk(c,value);
|
|
||||||
else
|
|
||||||
addReplyBulkCBuffer(c,v,vlen);
|
|
||||||
} else {
|
|
||||||
addReply(c,shared.nullbulk);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -458,42 +645,59 @@ void hlenCommand(redisClient *c) {
|
|||||||
addReplyLongLong(c,hashTypeLength(o));
|
addReplyLongLong(c,hashTypeLength(o));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void addHashIteratorCursorToReply(redisClient *c, hashTypeIterator *hi, int what) {
|
||||||
|
if (hi->encoding == REDIS_ENCODING_ZIPLIST) {
|
||||||
|
unsigned char *vstr = NULL;
|
||||||
|
unsigned int vlen = UINT_MAX;
|
||||||
|
long long vll = LLONG_MAX;
|
||||||
|
|
||||||
|
hashTypeCurrentFromZiplist(hi, what, &vstr, &vlen, &vll);
|
||||||
|
if (vstr) {
|
||||||
|
addReplyBulkCBuffer(c, vstr, vlen);
|
||||||
|
} else {
|
||||||
|
addReplyBulkLongLong(c, vll);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (hi->encoding == REDIS_ENCODING_HT) {
|
||||||
|
robj *value;
|
||||||
|
|
||||||
|
hashTypeCurrentFromHashTable(hi, what, &value);
|
||||||
|
addReplyBulk(c, value);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
redisPanic("Unknown hash encoding");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void genericHgetallCommand(redisClient *c, int flags) {
|
void genericHgetallCommand(redisClient *c, int flags) {
|
||||||
robj *o;
|
robj *o;
|
||||||
unsigned long count = 0;
|
|
||||||
hashTypeIterator *hi;
|
hashTypeIterator *hi;
|
||||||
void *replylen = NULL;
|
int multiplier = 0;
|
||||||
|
int length, count = 0;
|
||||||
|
|
||||||
if ((o = lookupKeyReadOrReply(c,c->argv[1],shared.emptymultibulk)) == NULL
|
if ((o = lookupKeyReadOrReply(c,c->argv[1],shared.emptymultibulk)) == NULL
|
||||||
|| checkType(c,o,REDIS_HASH)) return;
|
|| checkType(c,o,REDIS_HASH)) return;
|
||||||
|
|
||||||
replylen = addDeferredMultiBulkLength(c);
|
if (flags & REDIS_HASH_KEY) multiplier++;
|
||||||
|
if (flags & REDIS_HASH_VALUE) multiplier++;
|
||||||
|
|
||||||
|
length = hashTypeLength(o) * multiplier;
|
||||||
|
addReplyMultiBulkLen(c, length);
|
||||||
|
|
||||||
hi = hashTypeInitIterator(o);
|
hi = hashTypeInitIterator(o);
|
||||||
while (hashTypeNext(hi) != REDIS_ERR) {
|
while (hashTypeNext(hi) != REDIS_ERR) {
|
||||||
robj *obj;
|
|
||||||
unsigned char *v = NULL;
|
|
||||||
unsigned int vlen = 0;
|
|
||||||
int encoding;
|
|
||||||
|
|
||||||
if (flags & REDIS_HASH_KEY) {
|
if (flags & REDIS_HASH_KEY) {
|
||||||
encoding = hashTypeCurrent(hi,REDIS_HASH_KEY,&obj,&v,&vlen);
|
addHashIteratorCursorToReply(c, hi, REDIS_HASH_KEY);
|
||||||
if (encoding == REDIS_ENCODING_HT)
|
|
||||||
addReplyBulk(c,obj);
|
|
||||||
else
|
|
||||||
addReplyBulkCBuffer(c,v,vlen);
|
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
if (flags & REDIS_HASH_VALUE) {
|
if (flags & REDIS_HASH_VALUE) {
|
||||||
encoding = hashTypeCurrent(hi,REDIS_HASH_VALUE,&obj,&v,&vlen);
|
addHashIteratorCursorToReply(c, hi, REDIS_HASH_VALUE);
|
||||||
if (encoding == REDIS_ENCODING_HT)
|
|
||||||
addReplyBulk(c,obj);
|
|
||||||
else
|
|
||||||
addReplyBulkCBuffer(c,v,vlen);
|
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
hashTypeReleaseIterator(hi);
|
hashTypeReleaseIterator(hi);
|
||||||
setDeferredMultiBulkLength(c,replylen,count);
|
redisAssert(count == length);
|
||||||
}
|
}
|
||||||
|
|
||||||
void hkeysCommand(redisClient *c) {
|
void hkeysCommand(redisClient *c) {
|
||||||
|
97
src/t_list.c
97
src/t_list.c
@ -259,7 +259,7 @@ void listTypeConvert(robj *subject, int enc) {
|
|||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
void pushGenericCommand(redisClient *c, int where) {
|
void pushGenericCommand(redisClient *c, int where) {
|
||||||
int j, addlen = 0, pushed = 0;
|
int j, waiting = 0, pushed = 0;
|
||||||
robj *lobj = lookupKeyWrite(c->db,c->argv[1]);
|
robj *lobj = lookupKeyWrite(c->db,c->argv[1]);
|
||||||
int may_have_waiting_clients = (lobj == NULL);
|
int may_have_waiting_clients = (lobj == NULL);
|
||||||
|
|
||||||
@ -272,7 +272,7 @@ void pushGenericCommand(redisClient *c, int where) {
|
|||||||
c->argv[j] = tryObjectEncoding(c->argv[j]);
|
c->argv[j] = tryObjectEncoding(c->argv[j]);
|
||||||
if (may_have_waiting_clients) {
|
if (may_have_waiting_clients) {
|
||||||
if (handleClientsWaitingListPush(c,c->argv[1],c->argv[j])) {
|
if (handleClientsWaitingListPush(c,c->argv[1],c->argv[j])) {
|
||||||
addlen++;
|
waiting++;
|
||||||
continue;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
may_have_waiting_clients = 0;
|
may_have_waiting_clients = 0;
|
||||||
@ -285,9 +285,21 @@ void pushGenericCommand(redisClient *c, int where) {
|
|||||||
listTypePush(lobj,c->argv[j],where);
|
listTypePush(lobj,c->argv[j],where);
|
||||||
pushed++;
|
pushed++;
|
||||||
}
|
}
|
||||||
addReplyLongLong(c,addlen + (lobj ? listTypeLength(lobj) : 0));
|
addReplyLongLong(c, waiting + (lobj ? listTypeLength(lobj) : 0));
|
||||||
if (pushed) signalModifiedKey(c->db,c->argv[1]);
|
if (pushed) signalModifiedKey(c->db,c->argv[1]);
|
||||||
server.dirty += pushed;
|
server.dirty += pushed;
|
||||||
|
|
||||||
|
/* Alter the replication of the command accordingly to the number of
|
||||||
|
* list elements delivered to clients waiting into a blocking operation.
|
||||||
|
* We do that only if there were waiting clients, and only if still some
|
||||||
|
* element was pushed into the list (othewise dirty is 0 and nothign will
|
||||||
|
* be propagated). */
|
||||||
|
if (waiting && pushed) {
|
||||||
|
/* CMD KEY a b C D E */
|
||||||
|
for (j = 0; j < waiting; j++) decrRefCount(c->argv[j+2]);
|
||||||
|
memmove(c->argv+2,c->argv+2+waiting,sizeof(robj*)*pushed);
|
||||||
|
c->argc -= waiting;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void lpushCommand(redisClient *c) {
|
void lpushCommand(redisClient *c) {
|
||||||
@ -655,8 +667,6 @@ void lremCommand(redisClient *c) {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
void rpoplpushHandlePush(redisClient *origclient, redisClient *c, robj *dstkey, robj *dstobj, robj *value) {
|
void rpoplpushHandlePush(redisClient *origclient, redisClient *c, robj *dstkey, robj *dstobj, robj *value) {
|
||||||
robj *aux;
|
|
||||||
|
|
||||||
if (!handleClientsWaitingListPush(origclient,dstkey,value)) {
|
if (!handleClientsWaitingListPush(origclient,dstkey,value)) {
|
||||||
/* Create the list if the key does not exist */
|
/* Create the list if the key does not exist */
|
||||||
if (!dstobj) {
|
if (!dstobj) {
|
||||||
@ -666,27 +676,19 @@ void rpoplpushHandlePush(redisClient *origclient, redisClient *c, robj *dstkey,
|
|||||||
signalModifiedKey(c->db,dstkey);
|
signalModifiedKey(c->db,dstkey);
|
||||||
}
|
}
|
||||||
listTypePush(dstobj,value,REDIS_HEAD);
|
listTypePush(dstobj,value,REDIS_HEAD);
|
||||||
/* If we are pushing as a result of LPUSH against a key
|
/* Additionally propagate this PUSH operation together with
|
||||||
* watched by BRPOPLPUSH, we need to rewrite the command vector
|
* the operation performed by the command. */
|
||||||
* as an LPUSH.
|
{
|
||||||
*
|
robj **argv = zmalloc(sizeof(robj*)*3);
|
||||||
* If this is called directly by RPOPLPUSH (either directly
|
argv[0] = createStringObject("LPUSH",5);
|
||||||
* or via a BRPOPLPUSH where the popped list exists)
|
argv[1] = dstkey;
|
||||||
* we should replicate the RPOPLPUSH command itself. */
|
argv[2] = value;
|
||||||
if (c != origclient) {
|
incrRefCount(argv[1]);
|
||||||
aux = createStringObject("LPUSH",5);
|
incrRefCount(argv[2]);
|
||||||
rewriteClientCommandVector(origclient,3,aux,dstkey,value);
|
alsoPropagate(server.lpushCommand,c->db->id,argv,3,
|
||||||
decrRefCount(aux);
|
REDIS_PROPAGATE_AOF|REDIS_PROPAGATE_REPL);
|
||||||
} else {
|
|
||||||
/* Make sure to always use RPOPLPUSH in the replication / AOF,
|
|
||||||
* even if the original command was BRPOPLPUSH. */
|
|
||||||
aux = createStringObject("RPOPLPUSH",9);
|
|
||||||
rewriteClientCommandVector(origclient,3,aux,c->argv[1],c->argv[2]);
|
|
||||||
decrRefCount(aux);
|
|
||||||
}
|
}
|
||||||
server.dirty++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Always send the pushed value to the client. */
|
/* Always send the pushed value to the client. */
|
||||||
addReplyBulk(c,value);
|
addReplyBulk(c,value);
|
||||||
}
|
}
|
||||||
@ -717,6 +719,13 @@ void rpoplpushCommand(redisClient *c) {
|
|||||||
signalModifiedKey(c->db,touchedkey);
|
signalModifiedKey(c->db,touchedkey);
|
||||||
decrRefCount(touchedkey);
|
decrRefCount(touchedkey);
|
||||||
server.dirty++;
|
server.dirty++;
|
||||||
|
|
||||||
|
/* Replicate this as a simple RPOP since the LPUSH side is replicated
|
||||||
|
* by rpoplpushHandlePush() call if needed (it may not be needed
|
||||||
|
* if a client is blocking wait a push against the list). */
|
||||||
|
rewriteClientCommandVector(c,2,
|
||||||
|
resetRefCount(createStringObject("RPOP",4)),
|
||||||
|
c->argv[1]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -924,36 +933,22 @@ void blockingPopGenericCommand(redisClient *c, int where) {
|
|||||||
return;
|
return;
|
||||||
} else {
|
} else {
|
||||||
if (listTypeLength(o) != 0) {
|
if (listTypeLength(o) != 0) {
|
||||||
/* If the list contains elements fall back to the usual
|
/* Non empty list, this is like a non normal [LR]POP. */
|
||||||
* non-blocking POP operation */
|
robj *value = listTypePop(o,where);
|
||||||
struct redisCommand *orig_cmd;
|
redisAssert(value != NULL);
|
||||||
robj *argv[2], **orig_argv;
|
|
||||||
int orig_argc;
|
|
||||||
|
|
||||||
/* We need to alter the command arguments before to call
|
|
||||||
* popGenericCommand() as the command takes a single key. */
|
|
||||||
orig_argv = c->argv;
|
|
||||||
orig_argc = c->argc;
|
|
||||||
orig_cmd = c->cmd;
|
|
||||||
argv[1] = c->argv[j];
|
|
||||||
c->argv = argv;
|
|
||||||
c->argc = 2;
|
|
||||||
|
|
||||||
/* Also the return value is different, we need to output
|
|
||||||
* the multi bulk reply header and the key name. The
|
|
||||||
* "real" command will add the last element (the value)
|
|
||||||
* for us. If this souds like an hack to you it's just
|
|
||||||
* because it is... */
|
|
||||||
addReplyMultiBulkLen(c,2);
|
addReplyMultiBulkLen(c,2);
|
||||||
addReplyBulk(c,argv[1]);
|
addReplyBulk(c,c->argv[j]);
|
||||||
|
addReplyBulk(c,value);
|
||||||
popGenericCommand(c,where);
|
decrRefCount(value);
|
||||||
|
if (listTypeLength(o) == 0) dbDelete(c->db,c->argv[j]);
|
||||||
/* Fix the client structure with the original stuff */
|
signalModifiedKey(c->db,c->argv[j]);
|
||||||
c->argv = orig_argv;
|
server.dirty++;
|
||||||
c->argc = orig_argc;
|
|
||||||
c->cmd = orig_cmd;
|
|
||||||
|
|
||||||
|
/* Replicate it as an [LR]POP instead of B[LR]POP. */
|
||||||
|
rewriteClientCommandVector(c,2,
|
||||||
|
(where == REDIS_HEAD) ? shared.lpop : shared.rpop,
|
||||||
|
c->argv[j]);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
54
src/util.c
54
src/util.c
@ -5,6 +5,8 @@
|
|||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
|
||||||
@ -209,8 +211,8 @@ int ll2string(char *s, size_t len, long long value) {
|
|||||||
/* Convert a string into a long long. Returns 1 if the string could be parsed
|
/* Convert a string into a long long. Returns 1 if the string could be parsed
|
||||||
* into a (non-overflowing) long long, 0 otherwise. The value will be set to
|
* into a (non-overflowing) long long, 0 otherwise. The value will be set to
|
||||||
* the parsed value when appropriate. */
|
* the parsed value when appropriate. */
|
||||||
int string2ll(char *s, size_t slen, long long *value) {
|
int string2ll(const char *s, size_t slen, long long *value) {
|
||||||
char *p = s;
|
const char *p = s;
|
||||||
size_t plen = 0;
|
size_t plen = 0;
|
||||||
int negative = 0;
|
int negative = 0;
|
||||||
unsigned long long v;
|
unsigned long long v;
|
||||||
@ -275,7 +277,7 @@ int string2ll(char *s, size_t slen, long long *value) {
|
|||||||
/* Convert a string into a long. Returns 1 if the string could be parsed into a
|
/* Convert a string into a long. Returns 1 if the string could be parsed into a
|
||||||
* (non-overflowing) long, 0 otherwise. The value will be set to the parsed
|
* (non-overflowing) long, 0 otherwise. The value will be set to the parsed
|
||||||
* value when appropriate. */
|
* value when appropriate. */
|
||||||
int string2l(char *s, size_t slen, long *lval) {
|
int string2l(const char *s, size_t slen, long *lval) {
|
||||||
long long llval;
|
long long llval;
|
||||||
|
|
||||||
if (!string2ll(s,slen,&llval))
|
if (!string2ll(s,slen,&llval))
|
||||||
@ -327,6 +329,52 @@ int d2string(char *buf, size_t len, double value) {
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Generate the Redis "Run ID", a SHA1-sized random number that identifies a
|
||||||
|
* given execution of Redis, so that if you are talking with an instance
|
||||||
|
* having run_id == A, and you reconnect and it has run_id == B, you can be
|
||||||
|
* sure that it is either a different instance or it was restarted. */
|
||||||
|
void getRandomHexChars(char *p, unsigned int len) {
|
||||||
|
FILE *fp = fopen("/dev/urandom","r");
|
||||||
|
char *charset = "0123456789abcdef";
|
||||||
|
unsigned int j;
|
||||||
|
|
||||||
|
if (fp == NULL || fread(p,len,1,fp) == 0) {
|
||||||
|
/* If we can't read from /dev/urandom, do some reasonable effort
|
||||||
|
* in order to create some entropy, since this function is used to
|
||||||
|
* generate run_id and cluster instance IDs */
|
||||||
|
char *x = p;
|
||||||
|
unsigned int l = len;
|
||||||
|
struct timeval tv;
|
||||||
|
pid_t pid = getpid();
|
||||||
|
|
||||||
|
/* Use time and PID to fill the initial array. */
|
||||||
|
gettimeofday(&tv,NULL);
|
||||||
|
if (l >= sizeof(tv.tv_usec)) {
|
||||||
|
memcpy(x,&tv.tv_usec,sizeof(tv.tv_usec));
|
||||||
|
l -= sizeof(tv.tv_usec);
|
||||||
|
x += sizeof(tv.tv_usec);
|
||||||
|
}
|
||||||
|
if (l >= sizeof(tv.tv_sec)) {
|
||||||
|
memcpy(x,&tv.tv_sec,sizeof(tv.tv_sec));
|
||||||
|
l -= sizeof(tv.tv_sec);
|
||||||
|
x += sizeof(tv.tv_sec);
|
||||||
|
}
|
||||||
|
if (l >= sizeof(pid)) {
|
||||||
|
memcpy(x,&pid,sizeof(pid));
|
||||||
|
l -= sizeof(pid);
|
||||||
|
x += sizeof(pid);
|
||||||
|
}
|
||||||
|
/* Finally xor it with rand() output, that was already seeded with
|
||||||
|
* time() at startup. */
|
||||||
|
for (j = 0; j < len; j++)
|
||||||
|
p[j] ^= rand();
|
||||||
|
}
|
||||||
|
/* Turn it into hex digits taking just 4 bits out of 8 for every byte. */
|
||||||
|
for (j = 0; j < len; j++)
|
||||||
|
p[j] = charset[p[j] & 0x0F];
|
||||||
|
fclose(fp);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef UTIL_TEST_MAIN
|
#ifdef UTIL_TEST_MAIN
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
|
@ -5,8 +5,8 @@ int stringmatchlen(const char *p, int plen, const char *s, int slen, int nocase)
|
|||||||
int stringmatch(const char *p, const char *s, int nocase);
|
int stringmatch(const char *p, const char *s, int nocase);
|
||||||
long long memtoll(const char *p, int *err);
|
long long memtoll(const char *p, int *err);
|
||||||
int ll2string(char *s, size_t len, long long value);
|
int ll2string(char *s, size_t len, long long value);
|
||||||
int string2ll(char *s, size_t slen, long long *value);
|
int string2ll(const char *s, size_t slen, long long *value);
|
||||||
int string2l(char *s, size_t slen, long *value);
|
int string2l(const char *s, size_t slen, long *value);
|
||||||
int d2string(char *buf, size_t len, double value);
|
int d2string(char *buf, size_t len, double value);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -1 +1 @@
|
|||||||
#define REDIS_VERSION "2.9.4"
|
#define REDIS_VERSION "2.9.5"
|
||||||
|
215
src/ziplist.c
215
src/ziplist.c
@ -75,6 +75,8 @@
|
|||||||
#define ZIP_BIGLEN 254
|
#define ZIP_BIGLEN 254
|
||||||
|
|
||||||
/* Different encoding/length possibilities */
|
/* Different encoding/length possibilities */
|
||||||
|
#define ZIP_STR_MASK (0xc0)
|
||||||
|
#define ZIP_INT_MASK (0x30)
|
||||||
#define ZIP_STR_06B (0 << 6)
|
#define ZIP_STR_06B (0 << 6)
|
||||||
#define ZIP_STR_14B (1 << 6)
|
#define ZIP_STR_14B (1 << 6)
|
||||||
#define ZIP_STR_32B (2 << 6)
|
#define ZIP_STR_32B (2 << 6)
|
||||||
@ -82,9 +84,8 @@
|
|||||||
#define ZIP_INT_32B (0xc0 | 1<<4)
|
#define ZIP_INT_32B (0xc0 | 1<<4)
|
||||||
#define ZIP_INT_64B (0xc0 | 2<<4)
|
#define ZIP_INT_64B (0xc0 | 2<<4)
|
||||||
|
|
||||||
/* Macro's to determine type */
|
/* Macro to determine type */
|
||||||
#define ZIP_IS_STR(enc) (((enc) & 0xc0) < 0xc0)
|
#define ZIP_IS_STR(enc) (((enc) & ZIP_STR_MASK) < ZIP_STR_MASK)
|
||||||
#define ZIP_IS_INT(enc) (!ZIP_IS_STR(enc) && ((enc) & 0x30) < 0x30)
|
|
||||||
|
|
||||||
/* Utility macros */
|
/* Utility macros */
|
||||||
#define ZIPLIST_BYTES(zl) (*((uint32_t*)(zl)))
|
#define ZIPLIST_BYTES(zl) (*((uint32_t*)(zl)))
|
||||||
@ -110,19 +111,13 @@ typedef struct zlentry {
|
|||||||
unsigned char *p;
|
unsigned char *p;
|
||||||
} zlentry;
|
} zlentry;
|
||||||
|
|
||||||
/* Return the encoding pointer to by 'p'. */
|
#define ZIP_ENTRY_ENCODING(ptr, encoding) do { \
|
||||||
static unsigned int zipEntryEncoding(unsigned char *p) {
|
(encoding) = (ptr[0]) & (ZIP_STR_MASK | ZIP_INT_MASK); \
|
||||||
/* String encoding: 2 MSBs */
|
if (((encoding) & ZIP_STR_MASK) < ZIP_STR_MASK) { \
|
||||||
unsigned char b = p[0] & 0xc0;
|
/* String encoding: 2 MSBs */ \
|
||||||
if (b < 0xc0) {
|
(encoding) &= ZIP_STR_MASK; \
|
||||||
return b;
|
} \
|
||||||
} else {
|
} while(0)
|
||||||
/* Integer encoding: 4 MSBs */
|
|
||||||
return p[0] & 0xf0;
|
|
||||||
}
|
|
||||||
assert(NULL);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Return bytes needed to store integer encoded by 'encoding' */
|
/* Return bytes needed to store integer encoded by 'encoding' */
|
||||||
static unsigned int zipIntSize(unsigned char encoding) {
|
static unsigned int zipIntSize(unsigned char encoding) {
|
||||||
@ -135,36 +130,6 @@ static unsigned int zipIntSize(unsigned char encoding) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Decode the encoded length pointed by 'p'. If a pointer to 'lensize' is
|
|
||||||
* provided, it is set to the number of bytes required to encode the length. */
|
|
||||||
static unsigned int zipDecodeLength(unsigned char *p, unsigned int *lensize) {
|
|
||||||
unsigned char encoding = zipEntryEncoding(p);
|
|
||||||
unsigned int len = 0;
|
|
||||||
|
|
||||||
if (ZIP_IS_STR(encoding)) {
|
|
||||||
switch(encoding) {
|
|
||||||
case ZIP_STR_06B:
|
|
||||||
len = p[0] & 0x3f;
|
|
||||||
if (lensize) *lensize = 1;
|
|
||||||
break;
|
|
||||||
case ZIP_STR_14B:
|
|
||||||
len = ((p[0] & 0x3f) << 8) | p[1];
|
|
||||||
if (lensize) *lensize = 2;
|
|
||||||
break;
|
|
||||||
case ZIP_STR_32B:
|
|
||||||
len = (p[1] << 24) | (p[2] << 16) | (p[3] << 8) | p[4];
|
|
||||||
if (lensize) *lensize = 5;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
assert(NULL);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
len = zipIntSize(encoding);
|
|
||||||
if (lensize) *lensize = 1;
|
|
||||||
}
|
|
||||||
return len;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Encode the length 'l' writing it in 'p'. If p is NULL it just returns
|
/* Encode the length 'l' writing it in 'p'. If p is NULL it just returns
|
||||||
* the amount of bytes required to encode such a length. */
|
* the amount of bytes required to encode such a length. */
|
||||||
static unsigned int zipEncodeLength(unsigned char *p, unsigned char encoding, unsigned int rawlen) {
|
static unsigned int zipEncodeLength(unsigned char *p, unsigned char encoding, unsigned int rawlen) {
|
||||||
@ -201,18 +166,33 @@ static unsigned int zipEncodeLength(unsigned char *p, unsigned char encoding, un
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Decode the length of the previous element stored at "p". */
|
/* Decode the length encoded in 'ptr'. The 'encoding' variable will hold the
|
||||||
static unsigned int zipPrevDecodeLength(unsigned char *p, unsigned int *lensize) {
|
* entries encoding, the 'lensize' variable will hold the number of bytes
|
||||||
unsigned int len = *p;
|
* required to encode the entries length, and the 'len' variable will hold the
|
||||||
if (len < ZIP_BIGLEN) {
|
* entries length. */
|
||||||
if (lensize) *lensize = 1;
|
#define ZIP_DECODE_LENGTH(ptr, encoding, lensize, len) do { \
|
||||||
} else {
|
ZIP_ENTRY_ENCODING((ptr), (encoding)); \
|
||||||
if (lensize) *lensize = 1+sizeof(len);
|
if ((encoding) < ZIP_STR_MASK) { \
|
||||||
memcpy(&len,p+1,sizeof(len));
|
if ((encoding) == ZIP_STR_06B) { \
|
||||||
memrev32ifbe(&len);
|
(lensize) = 1; \
|
||||||
}
|
(len) = (ptr)[0] & 0x3f; \
|
||||||
return len;
|
} else if ((encoding) == ZIP_STR_14B) { \
|
||||||
}
|
(lensize) = 2; \
|
||||||
|
(len) = (((ptr)[0] & 0x3f) << 8) | (ptr)[1]; \
|
||||||
|
} else if (encoding == ZIP_STR_32B) { \
|
||||||
|
(lensize) = 5; \
|
||||||
|
(len) = ((ptr)[1] << 24) | \
|
||||||
|
((ptr)[2] << 16) | \
|
||||||
|
((ptr)[3] << 8) | \
|
||||||
|
((ptr)[4]); \
|
||||||
|
} else { \
|
||||||
|
assert(NULL); \
|
||||||
|
} \
|
||||||
|
} else { \
|
||||||
|
(lensize) = 1; \
|
||||||
|
(len) = zipIntSize(encoding); \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
/* Encode the length of the previous entry and write it to "p". Return the
|
/* Encode the length of the previous entry and write it to "p". Return the
|
||||||
* number of bytes needed to encode this length if "p" is NULL. */
|
* number of bytes needed to encode this length if "p" is NULL. */
|
||||||
@ -241,12 +221,43 @@ static void zipPrevEncodeLengthForceLarge(unsigned char *p, unsigned int len) {
|
|||||||
memrev32ifbe(p+1);
|
memrev32ifbe(p+1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the difference in number of bytes needed to store the new length
|
/* Decode the number of bytes required to store the length of the previous
|
||||||
* "len" on the entry pointed to by "p". */
|
* element, from the perspective of the entry pointed to by 'ptr'. */
|
||||||
|
#define ZIP_DECODE_PREVLENSIZE(ptr, prevlensize) do { \
|
||||||
|
if ((ptr)[0] < ZIP_BIGLEN) { \
|
||||||
|
(prevlensize) = 1; \
|
||||||
|
} else { \
|
||||||
|
(prevlensize) = 5; \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
|
/* Decode the length of the previous element, from the perspective of the entry
|
||||||
|
* pointed to by 'ptr'. */
|
||||||
|
#define ZIP_DECODE_PREVLEN(ptr, prevlensize, prevlen) do { \
|
||||||
|
ZIP_DECODE_PREVLENSIZE(ptr, prevlensize); \
|
||||||
|
if ((prevlensize) == 1) { \
|
||||||
|
(prevlen) = (ptr)[0]; \
|
||||||
|
} else if ((prevlensize) == 5) { \
|
||||||
|
assert(sizeof((prevlensize)) == 4); \
|
||||||
|
memcpy(&(prevlen), ((char*)(ptr)) + 1, 4); \
|
||||||
|
memrev32ifbe(&prevlen); \
|
||||||
|
} \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
|
/* Return the difference in number of bytes needed to store the length of the
|
||||||
|
* previous element 'len', in the entry pointed to by 'p'. */
|
||||||
static int zipPrevLenByteDiff(unsigned char *p, unsigned int len) {
|
static int zipPrevLenByteDiff(unsigned char *p, unsigned int len) {
|
||||||
unsigned int prevlensize;
|
unsigned int prevlensize;
|
||||||
zipPrevDecodeLength(p,&prevlensize);
|
ZIP_DECODE_PREVLENSIZE(p, prevlensize);
|
||||||
return zipPrevEncodeLength(NULL,len)-prevlensize;
|
return zipPrevEncodeLength(NULL, len) - prevlensize;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the total number of bytes used by the entry pointed to by 'p'. */
|
||||||
|
static unsigned int zipRawEntryLength(unsigned char *p) {
|
||||||
|
unsigned int prevlensize, encoding, lensize, len;
|
||||||
|
ZIP_DECODE_PREVLENSIZE(p, prevlensize);
|
||||||
|
ZIP_DECODE_LENGTH(p + prevlensize, encoding, lensize, len);
|
||||||
|
return prevlensize + lensize + len;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if string pointed to by 'entry' can be encoded as an integer.
|
/* Check if string pointed to by 'entry' can be encoded as an integer.
|
||||||
@ -319,20 +330,14 @@ static int64_t zipLoadInteger(unsigned char *p, unsigned char encoding) {
|
|||||||
/* Return a struct with all information about an entry. */
|
/* Return a struct with all information about an entry. */
|
||||||
static zlentry zipEntry(unsigned char *p) {
|
static zlentry zipEntry(unsigned char *p) {
|
||||||
zlentry e;
|
zlentry e;
|
||||||
e.prevrawlen = zipPrevDecodeLength(p,&e.prevrawlensize);
|
|
||||||
e.len = zipDecodeLength(p+e.prevrawlensize,&e.lensize);
|
ZIP_DECODE_PREVLEN(p, e.prevrawlensize, e.prevrawlen);
|
||||||
e.headersize = e.prevrawlensize+e.lensize;
|
ZIP_DECODE_LENGTH(p + e.prevrawlensize, e.encoding, e.lensize, e.len);
|
||||||
e.encoding = zipEntryEncoding(p+e.prevrawlensize);
|
e.headersize = e.prevrawlensize + e.lensize;
|
||||||
e.p = p;
|
e.p = p;
|
||||||
return e;
|
return e;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the total number of bytes used by the entry at "p". */
|
|
||||||
static unsigned int zipRawEntryLength(unsigned char *p) {
|
|
||||||
zlentry e = zipEntry(p);
|
|
||||||
return e.headersize + e.len;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Create a new empty ziplist. */
|
/* Create a new empty ziplist. */
|
||||||
unsigned char *ziplistNew(void) {
|
unsigned char *ziplistNew(void) {
|
||||||
unsigned int bytes = ZIPLIST_HEADER_SIZE+1;
|
unsigned int bytes = ZIPLIST_HEADER_SIZE+1;
|
||||||
@ -628,10 +633,14 @@ unsigned char *ziplistNext(unsigned char *zl, unsigned char *p) {
|
|||||||
* when the *next* element is ZIP_END (there is no next entry). */
|
* when the *next* element is ZIP_END (there is no next entry). */
|
||||||
if (p[0] == ZIP_END) {
|
if (p[0] == ZIP_END) {
|
||||||
return NULL;
|
return NULL;
|
||||||
} else {
|
|
||||||
p = p+zipRawEntryLength(p);
|
|
||||||
return (p[0] == ZIP_END) ? NULL : p;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
p += zipRawEntryLength(p);
|
||||||
|
if (p[0] == ZIP_END) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return pointer to previous entry in ziplist. */
|
/* Return pointer to previous entry in ziplist. */
|
||||||
@ -729,6 +738,62 @@ unsigned int ziplistCompare(unsigned char *p, unsigned char *sstr, unsigned int
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Find pointer to the entry equal to the specified entry. Skip 'skip' entries
|
||||||
|
* between every comparison. Returns NULL when the field could not be found. */
|
||||||
|
unsigned char *ziplistFind(unsigned char *p, unsigned char *vstr, unsigned int vlen, unsigned int skip) {
|
||||||
|
int skipcnt = 0;
|
||||||
|
unsigned char vencoding = 0;
|
||||||
|
long long vll = 0;
|
||||||
|
|
||||||
|
while (p[0] != ZIP_END) {
|
||||||
|
unsigned int prevlensize, encoding, lensize, len;
|
||||||
|
unsigned char *q;
|
||||||
|
|
||||||
|
ZIP_DECODE_PREVLENSIZE(p, prevlensize);
|
||||||
|
ZIP_DECODE_LENGTH(p + prevlensize, encoding, lensize, len);
|
||||||
|
q = p + prevlensize + lensize;
|
||||||
|
|
||||||
|
if (skipcnt == 0) {
|
||||||
|
/* Compare current entry with specified entry */
|
||||||
|
if (ZIP_IS_STR(encoding)) {
|
||||||
|
if (len == vlen && memcmp(q, vstr, vlen) == 0) {
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* Find out if the specified entry can be encoded */
|
||||||
|
if (vencoding == 0) {
|
||||||
|
/* UINT_MAX when the entry CANNOT be encoded */
|
||||||
|
if (!zipTryEncoding(vstr, vlen, &vll, &vencoding)) {
|
||||||
|
vencoding = UCHAR_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Must be non-zero by now */
|
||||||
|
assert(vencoding);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Compare current entry with specified entry */
|
||||||
|
if (encoding == vencoding) {
|
||||||
|
long long ll = zipLoadInteger(q, encoding);
|
||||||
|
if (ll == vll) {
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Reset skip count */
|
||||||
|
skipcnt = skip;
|
||||||
|
} else {
|
||||||
|
/* Skip entry */
|
||||||
|
skipcnt--;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Move to next entry */
|
||||||
|
p = q + len;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/* Return length of ziplist. */
|
/* Return length of ziplist. */
|
||||||
unsigned int ziplistLen(unsigned char *zl) {
|
unsigned int ziplistLen(unsigned char *zl) {
|
||||||
unsigned int len = 0;
|
unsigned int len = 0;
|
||||||
|
@ -11,5 +11,6 @@ unsigned char *ziplistInsert(unsigned char *zl, unsigned char *p, unsigned char
|
|||||||
unsigned char *ziplistDelete(unsigned char *zl, unsigned char **p);
|
unsigned char *ziplistDelete(unsigned char *zl, unsigned char **p);
|
||||||
unsigned char *ziplistDeleteRange(unsigned char *zl, unsigned int index, unsigned int num);
|
unsigned char *ziplistDeleteRange(unsigned char *zl, unsigned int index, unsigned int num);
|
||||||
unsigned int ziplistCompare(unsigned char *p, unsigned char *s, unsigned int slen);
|
unsigned int ziplistCompare(unsigned char *p, unsigned char *s, unsigned int slen);
|
||||||
|
unsigned char *ziplistFind(unsigned char *p, unsigned char *vstr, unsigned int vlen, unsigned int skip);
|
||||||
unsigned int ziplistLen(unsigned char *zl);
|
unsigned int ziplistLen(unsigned char *zl);
|
||||||
size_t ziplistBlobLen(unsigned char *zl);
|
size_t ziplistBlobLen(unsigned char *zl);
|
||||||
|
@ -297,8 +297,8 @@ no-appendfsync-on-rewrite no
|
|||||||
# have at max a given numer of elements, and the biggest element does not
|
# have at max a given numer of elements, and the biggest element does not
|
||||||
# exceed a given threshold. You can configure this limits with the following
|
# exceed a given threshold. You can configure this limits with the following
|
||||||
# configuration directives.
|
# configuration directives.
|
||||||
hash-max-zipmap-entries 64
|
hash-max-ziplist-entries 64
|
||||||
hash-max-zipmap-value 512
|
hash-max-ziplist-value 512
|
||||||
|
|
||||||
# Similarly to hashes, small lists are also encoded in a special way in order
|
# Similarly to hashes, small lists are also encoded in a special way in order
|
||||||
# to save a lot of space. The special representation is only used when
|
# to save a lot of space. The special representation is only used when
|
||||||
|
BIN
tests/assets/encodings.rdb
Normal file
BIN
tests/assets/encodings.rdb
Normal file
Binary file not shown.
BIN
tests/assets/hash-zipmap.rdb
Normal file
BIN
tests/assets/hash-zipmap.rdb
Normal file
Binary file not shown.
34
tests/integration/convert-zipmap-hash-on-load.tcl
Normal file
34
tests/integration/convert-zipmap-hash-on-load.tcl
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
set server_path [tmpdir "server.convert-zipmap-hash-on-load"]
|
||||||
|
|
||||||
|
# Copy RDB with zipmap encoded hash to server path
|
||||||
|
exec cp tests/assets/hash-zipmap.rdb $server_path
|
||||||
|
|
||||||
|
start_server [list overrides [list "dir" $server_path "dbfilename" "hash-zipmap.rdb"]] {
|
||||||
|
test "RDB load zipmap hash: converts to ziplist" {
|
||||||
|
r select 0
|
||||||
|
|
||||||
|
assert_match "*ziplist*" [r debug object hash]
|
||||||
|
assert_equal 2 [r hlen hash]
|
||||||
|
assert_match {v1 v2} [r hmget hash f1 f2]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
start_server [list overrides [list "dir" $server_path "dbfilename" "hash-zipmap.rdb" "hash-max-ziplist-entries" 1]] {
|
||||||
|
test "RDB load zipmap hash: converts to hash table when hash-max-ziplist-entries is exceeded" {
|
||||||
|
r select 0
|
||||||
|
|
||||||
|
assert_match "*hashtable*" [r debug object hash]
|
||||||
|
assert_equal 2 [r hlen hash]
|
||||||
|
assert_match {v1 v2} [r hmget hash f1 f2]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
start_server [list overrides [list "dir" $server_path "dbfilename" "hash-zipmap.rdb" "hash-max-ziplist-value" 1]] {
|
||||||
|
test "RDB load zipmap hash: converts to hash table when hash-max-ziplist-value is exceeded" {
|
||||||
|
r select 0
|
||||||
|
|
||||||
|
assert_match "*hashtable*" [r debug object hash]
|
||||||
|
assert_equal 2 [r hlen hash]
|
||||||
|
assert_match {v1 v2} [r hmget hash f1 f2]
|
||||||
|
}
|
||||||
|
}
|
25
tests/integration/rdb.tcl
Normal file
25
tests/integration/rdb.tcl
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
set server_path [tmpdir "server.rdb-encoding-test"]
|
||||||
|
|
||||||
|
# Copy RDB with different encodings in server path
|
||||||
|
exec cp tests/assets/encodings.rdb $server_path
|
||||||
|
|
||||||
|
start_server [list overrides [list "dir" $server_path "dbfilename" "encodings.rdb"]] {
|
||||||
|
test "RDB encoding loading test" {
|
||||||
|
r select 0
|
||||||
|
csvdump r
|
||||||
|
} {"compressible","string","aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
||||||
|
"hash","hash","a","1","aa","10","aaa","100","b","2","bb","20","bbb","200","c","3","cc","30","ccc","300","ddd","400","eee","5000000000",
|
||||||
|
"hash_zipped","hash","a","1","b","2","c","3",
|
||||||
|
"list","list","1","2","3","a","b","c","100000","6000000000","1","2","3","a","b","c","100000","6000000000","1","2","3","a","b","c","100000","6000000000",
|
||||||
|
"list_zipped","list","1","2","3","a","b","c","100000","6000000000",
|
||||||
|
"number","string","10"
|
||||||
|
"set","set","1","100000","2","3","6000000000","a","b","c",
|
||||||
|
"set_zipped_1","set","1","2","3","4",
|
||||||
|
"set_zipped_2","set","100000","200000","300000","400000",
|
||||||
|
"set_zipped_3","set","1000000000","2000000000","3000000000","4000000000","5000000000","6000000000",
|
||||||
|
"string","string","Hello World"
|
||||||
|
"zset","zset","a","1","b","2","c","3","aa","10","bb","20","cc","30","aaa","100","bbb","200","ccc","300","aaaa","1000","cccc","123456789","bbbb","5000000000",
|
||||||
|
"zset_zipped","zset","a","1","b","2","c","3",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -160,7 +160,7 @@ proc ::redis::redis_read_reply fd {
|
|||||||
- {return -code error [redis_read_line $fd]}
|
- {return -code error [redis_read_line $fd]}
|
||||||
$ {redis_bulk_read $fd}
|
$ {redis_bulk_read $fd}
|
||||||
* {redis_multi_bulk_read $fd}
|
* {redis_multi_bulk_read $fd}
|
||||||
default {return -code error "Bad protocol, $type as reply type byte"}
|
default {return -code error "Bad protocol, '$type' as reply type byte"}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,6 +30,8 @@ set ::all_tests {
|
|||||||
integration/replication-2
|
integration/replication-2
|
||||||
integration/replication-3
|
integration/replication-3
|
||||||
integration/aof
|
integration/aof
|
||||||
|
integration/rdb
|
||||||
|
integration/convert-zipmap-hash-on-load
|
||||||
unit/pubsub
|
unit/pubsub
|
||||||
unit/slowlog
|
unit/slowlog
|
||||||
unit/scripting
|
unit/scripting
|
||||||
|
@ -54,10 +54,10 @@ start_server {tags {"aofrw"}} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
foreach d {string int} {
|
foreach d {string int} {
|
||||||
foreach e {zipmap hashtable} {
|
foreach e {ziplist hashtable} {
|
||||||
test "AOF rewrite of hash with $e encoding, $d data" {
|
test "AOF rewrite of hash with $e encoding, $d data" {
|
||||||
r flushall
|
r flushall
|
||||||
if {$e eq {zipmap}} {set len 10} else {set len 1000}
|
if {$e eq {ziplist}} {set len 10} else {set len 1000}
|
||||||
for {set j 0} {$j < $len} {incr j} {
|
for {set j 0} {$j < $len} {incr j} {
|
||||||
if {$d eq {string}} {
|
if {$d eq {string}} {
|
||||||
set data [randstring 0 16 alpha]
|
set data [randstring 0 16 alpha]
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
start_server {tags {"introspection"}} {
|
start_server {tags {"introspection"}} {
|
||||||
test {CLIENT LIST} {
|
test {CLIENT LIST} {
|
||||||
r client list
|
r client list
|
||||||
} {*addr=*:* fd=* idle=* flags=N db=9 sub=0 psub=0 qbuf=0 obl=0 oll=0 omem=0 events=r cmd=client*}
|
} {*addr=*:* fd=* age=* idle=* flags=N db=9 sub=0 psub=0 qbuf=0 qbuf-free=* obl=0 oll=0 omem=0 events=r cmd=client*}
|
||||||
}
|
}
|
||||||
|
@ -14,8 +14,8 @@ start_server {tags {"hash"}} {
|
|||||||
list [r hlen smallhash]
|
list [r hlen smallhash]
|
||||||
} {8}
|
} {8}
|
||||||
|
|
||||||
test {Is the small hash encoded with a zipmap?} {
|
test {Is the small hash encoded with a ziplist?} {
|
||||||
assert_encoding zipmap smallhash
|
assert_encoding ziplist smallhash
|
||||||
}
|
}
|
||||||
|
|
||||||
test {HSET/HLEN - Big hash creation} {
|
test {HSET/HLEN - Big hash creation} {
|
||||||
@ -33,7 +33,7 @@ start_server {tags {"hash"}} {
|
|||||||
list [r hlen bighash]
|
list [r hlen bighash]
|
||||||
} {1024}
|
} {1024}
|
||||||
|
|
||||||
test {Is the big hash encoded with a zipmap?} {
|
test {Is the big hash encoded with a ziplist?} {
|
||||||
assert_encoding hashtable bighash
|
assert_encoding hashtable bighash
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -252,7 +252,7 @@ start_server {tags {"hash"}} {
|
|||||||
lappend rv [r hexists bighash nokey]
|
lappend rv [r hexists bighash nokey]
|
||||||
} {1 0 1 0}
|
} {1 0 1 0}
|
||||||
|
|
||||||
test {Is a zipmap encoded Hash promoted on big payload?} {
|
test {Is a ziplist encoded Hash promoted on big payload?} {
|
||||||
r hset smallhash foo [string repeat a 1024]
|
r hset smallhash foo [string repeat a 1024]
|
||||||
r debug object smallhash
|
r debug object smallhash
|
||||||
} {*hashtable*}
|
} {*hashtable*}
|
||||||
@ -390,7 +390,7 @@ start_server {tags {"hash"}} {
|
|||||||
lappend rv [string match "ERR*not*float*" $bigerr]
|
lappend rv [string match "ERR*not*float*" $bigerr]
|
||||||
} {1 1}
|
} {1 1}
|
||||||
|
|
||||||
test {Hash zipmap regression test for large keys} {
|
test {Hash ziplist regression test for large keys} {
|
||||||
r hset hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk a
|
r hset hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk a
|
||||||
r hset hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk b
|
r hset hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk b
|
||||||
r hget hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk
|
r hget hash kkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkkk
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Redis configuration file example
|
# Redis configuration file example
|
||||||
|
|
||||||
# Note on units: when memory size is needed, it is possible to specifiy
|
# Note on units: when memory size is needed, it is possible to specify
|
||||||
# it in the usual form of 1k 5GB 4M and so forth:
|
# it in the usual form of 1k 5GB 4M and so forth:
|
||||||
#
|
#
|
||||||
# 1k => 1000 bytes
|
# 1k => 1000 bytes
|
||||||
@ -34,9 +34,10 @@ port $REDIS_PORT
|
|||||||
# on a unix socket when not specified.
|
# on a unix socket when not specified.
|
||||||
#
|
#
|
||||||
# unixsocket /tmp/redis.sock
|
# unixsocket /tmp/redis.sock
|
||||||
|
# unixsocketperm 755
|
||||||
|
|
||||||
# Close the connection after a client is idle for N seconds (0 to disable)
|
# Close the connection after a client is idle for N seconds (0 to disable)
|
||||||
timeout 300
|
timeout 0
|
||||||
|
|
||||||
# Set server verbosity to 'debug'
|
# Set server verbosity to 'debug'
|
||||||
# it can be one of:
|
# it can be one of:
|
||||||
@ -44,7 +45,7 @@ timeout 300
|
|||||||
# verbose (many rarely useful info, but not a mess like the debug level)
|
# verbose (many rarely useful info, but not a mess like the debug level)
|
||||||
# notice (moderately verbose, what you want in production probably)
|
# notice (moderately verbose, what you want in production probably)
|
||||||
# warning (only very important / critical messages are logged)
|
# warning (only very important / critical messages are logged)
|
||||||
loglevel verbose
|
loglevel notice
|
||||||
|
|
||||||
# Specify the log file name. Also 'stdout' can be used to force
|
# Specify the log file name. Also 'stdout' can be used to force
|
||||||
# Redis to log on the standard output. Note that if you use standard
|
# Redis to log on the standard output. Note that if you use standard
|
||||||
@ -81,11 +82,32 @@ databases 16
|
|||||||
# after 60 sec if at least 10000 keys changed
|
# after 60 sec if at least 10000 keys changed
|
||||||
#
|
#
|
||||||
# Note: you can disable saving at all commenting all the "save" lines.
|
# Note: you can disable saving at all commenting all the "save" lines.
|
||||||
|
#
|
||||||
|
# It is also possible to remove all the previously configured save
|
||||||
|
# points by adding a save directive with a single empty string argument
|
||||||
|
# like in the following example:
|
||||||
|
#
|
||||||
|
# save ""
|
||||||
|
|
||||||
save 900 1
|
save 900 1
|
||||||
save 300 10
|
save 300 10
|
||||||
save 60 10000
|
save 60 10000
|
||||||
|
|
||||||
|
# By default Redis will stop accepting writes if RDB snapshots are enabled
|
||||||
|
# (at least one save point) and the latest background save failed.
|
||||||
|
# This will make the user aware (in an hard way) that data is not persisting
|
||||||
|
# on disk properly, otherwise chances are that no one will notice and some
|
||||||
|
# distater will happen.
|
||||||
|
#
|
||||||
|
# If the background saving process will start working again Redis will
|
||||||
|
# automatically allow writes again.
|
||||||
|
#
|
||||||
|
# However if you have setup your proper monitoring of the Redis server
|
||||||
|
# and persistence, you may want to disable this feature so that Redis will
|
||||||
|
# continue to work as usually even if there are problems with disk,
|
||||||
|
# permissions, and so forth.
|
||||||
|
stop-writes-on-bgsave-error yes
|
||||||
|
|
||||||
# Compress string objects using LZF when dump .rdb databases?
|
# Compress string objects using LZF when dump .rdb databases?
|
||||||
# For default that's set to 'yes' as it's almost always a win.
|
# For default that's set to 'yes' as it's almost always a win.
|
||||||
# If you want to save some CPU in the saving child set it to 'no' but
|
# If you want to save some CPU in the saving child set it to 'no' but
|
||||||
@ -125,7 +147,7 @@ dir $REDIS_DATA_DIR
|
|||||||
# is still in progress, the slave can act in two different ways:
|
# is still in progress, the slave can act in two different ways:
|
||||||
#
|
#
|
||||||
# 1) if slave-serve-stale-data is set to 'yes' (the default) the slave will
|
# 1) if slave-serve-stale-data is set to 'yes' (the default) the slave will
|
||||||
# still reply to client requests, possibly with out of data data, or the
|
# still reply to client requests, possibly with out of date data, or the
|
||||||
# data set may just be empty if this is the first synchronization.
|
# data set may just be empty if this is the first synchronization.
|
||||||
#
|
#
|
||||||
# 2) if slave-serve-stale data is set to 'no' the slave will reply with
|
# 2) if slave-serve-stale data is set to 'no' the slave will reply with
|
||||||
@ -134,6 +156,21 @@ dir $REDIS_DATA_DIR
|
|||||||
#
|
#
|
||||||
slave-serve-stale-data yes
|
slave-serve-stale-data yes
|
||||||
|
|
||||||
|
# Slaves send PINGs to server in a predefined interval. It's possible to change
|
||||||
|
# this interval with the repl_ping_slave_period option. The default value is 10
|
||||||
|
# seconds.
|
||||||
|
#
|
||||||
|
# repl-ping-slave-period 10
|
||||||
|
|
||||||
|
# The following option sets a timeout for both Bulk transfer I/O timeout and
|
||||||
|
# master data or ping response timeout. The default value is 60 seconds.
|
||||||
|
#
|
||||||
|
# It is important to make sure that this value is greater than the value
|
||||||
|
# specified for repl-ping-slave-period otherwise a timeout will be detected
|
||||||
|
# every time there is low traffic between the master and the slave.
|
||||||
|
#
|
||||||
|
# repl-timeout 60
|
||||||
|
|
||||||
################################## SECURITY ###################################
|
################################## SECURITY ###################################
|
||||||
|
|
||||||
# Require clients to issue AUTH <PASSWORD> before processing any other
|
# Require clients to issue AUTH <PASSWORD> before processing any other
|
||||||
@ -151,7 +188,7 @@ slave-serve-stale-data yes
|
|||||||
|
|
||||||
# Command renaming.
|
# Command renaming.
|
||||||
#
|
#
|
||||||
# It is possilbe to change the name of dangerous commands in a shared
|
# It is possible to change the name of dangerous commands in a shared
|
||||||
# environment. For instance the CONFIG command may be renamed into something
|
# environment. For instance the CONFIG command may be renamed into something
|
||||||
# of hard to guess so that it will be still available for internal-use
|
# of hard to guess so that it will be still available for internal-use
|
||||||
# tools but not available for general clients.
|
# tools but not available for general clients.
|
||||||
@ -160,37 +197,46 @@ slave-serve-stale-data yes
|
|||||||
#
|
#
|
||||||
# rename-command CONFIG b840fc02d524045429941cc15f59e41cb7be6c52
|
# rename-command CONFIG b840fc02d524045429941cc15f59e41cb7be6c52
|
||||||
#
|
#
|
||||||
# It is also possilbe to completely kill a command renaming it into
|
# It is also possible to completely kill a command renaming it into
|
||||||
# an empty string:
|
# an empty string:
|
||||||
#
|
#
|
||||||
# rename-command CONFIG ""
|
# rename-command CONFIG ""
|
||||||
|
|
||||||
################################### LIMITS ####################################
|
################################### LIMITS ####################################
|
||||||
|
|
||||||
# Set the max number of connected clients at the same time. By default there
|
# Set the max number of connected clients at the same time. By default
|
||||||
# is no limit, and it's up to the number of file descriptors the Redis process
|
# this limit is set to 10000 clients, however if the Redis server is not
|
||||||
# is able to open. The special value '0' means no limits.
|
# able ot configure the process file limit to allow for the specified limit
|
||||||
|
# the max number of allowed clients is set to the current file limit
|
||||||
|
# minus 32 (as Redis reserves a few file descriptors for internal uses).
|
||||||
|
#
|
||||||
# Once the limit is reached Redis will close all the new connections sending
|
# Once the limit is reached Redis will close all the new connections sending
|
||||||
# an error 'max number of clients reached'.
|
# an error 'max number of clients reached'.
|
||||||
#
|
#
|
||||||
# maxclients 128
|
# maxclients 10000
|
||||||
|
|
||||||
# Don't use more memory than the specified amount of bytes.
|
# Don't use more memory than the specified amount of bytes.
|
||||||
# When the memory limit is reached Redis will try to remove keys with an
|
# When the memory limit is reached Redis will try to remove keys
|
||||||
# EXPIRE set. It will try to start freeing keys that are going to expire
|
# accordingly to the eviction policy selected (see maxmemmory-policy).
|
||||||
# in little time and preserve keys with a longer time to live.
|
|
||||||
# Redis will also try to remove objects from free lists if possible.
|
|
||||||
#
|
#
|
||||||
# If all this fails, Redis will start to reply with errors to commands
|
# If Redis can't remove keys according to the policy, or if the policy is
|
||||||
# that will use more memory, like SET, LPUSH, and so on, and will continue
|
# set to 'noeviction', Redis will start to reply with errors to commands
|
||||||
# to reply to most read-only commands like GET.
|
# that would use more memory, like SET, LPUSH, and so on, and will continue
|
||||||
|
# to reply to read-only commands like GET.
|
||||||
#
|
#
|
||||||
# WARNING: maxmemory can be a good idea mainly if you want to use Redis as a
|
# This option is usually useful when using Redis as an LRU cache, or to set
|
||||||
# 'state' server or cache, not as a real DB. When Redis is used as a real
|
# an hard memory limit for an instance (using the 'noeviction' policy).
|
||||||
# database the memory usage will grow over the weeks, it will be obvious if
|
#
|
||||||
# it is going to use too much memory in the long run, and you'll have the time
|
# WARNING: If you have slaves attached to an instance with maxmemory on,
|
||||||
# to upgrade. With maxmemory after the limit is reached you'll start to get
|
# the size of the output buffers needed to feed the slaves are subtracted
|
||||||
# errors for write operations, and this may even lead to DB inconsistency.
|
# from the used memory count, so that network problems / resyncs will
|
||||||
|
# not trigger a loop where keys are evicted, and in turn the output
|
||||||
|
# buffer of slaves is full with DELs of keys evicted triggering the deletion
|
||||||
|
# of more keys, and so forth until the database is completely emptied.
|
||||||
|
#
|
||||||
|
# In short... if you have slaves attached it is suggested that you set a lower
|
||||||
|
# limit for maxmemory so that there is some free RAM on the system for slave
|
||||||
|
# output buffers (but this is not needed if the policy is 'noeviction').
|
||||||
#
|
#
|
||||||
# maxmemory <bytes>
|
# maxmemory <bytes>
|
||||||
|
|
||||||
@ -200,7 +246,7 @@ slave-serve-stale-data yes
|
|||||||
# volatile-lru -> remove the key with an expire set using an LRU algorithm
|
# volatile-lru -> remove the key with an expire set using an LRU algorithm
|
||||||
# allkeys-lru -> remove any key accordingly to the LRU algorithm
|
# allkeys-lru -> remove any key accordingly to the LRU algorithm
|
||||||
# volatile-random -> remove a random key with an expire set
|
# volatile-random -> remove a random key with an expire set
|
||||||
# allkeys->random -> remove a random key, any key
|
# allkeys-random -> remove a random key, any key
|
||||||
# volatile-ttl -> remove the key with the nearest expire time (minor TTL)
|
# volatile-ttl -> remove the key with the nearest expire time (minor TTL)
|
||||||
# noeviction -> don't expire at all, just return an error on write operations
|
# noeviction -> don't expire at all, just return an error on write operations
|
||||||
#
|
#
|
||||||
@ -260,7 +306,7 @@ appendonly no
|
|||||||
#
|
#
|
||||||
# The default is "everysec" that's usually the right compromise between
|
# The default is "everysec" that's usually the right compromise between
|
||||||
# speed and data safety. It's up to you to understand if you can relax this to
|
# speed and data safety. It's up to you to understand if you can relax this to
|
||||||
# "no" that will will let the operating system flush the output buffer when
|
# "no" that will let the operating system flush the output buffer when
|
||||||
# it wants, for better performances (but if you can live with the idea of
|
# it wants, for better performances (but if you can live with the idea of
|
||||||
# some data loss consider the default persistence mode that's snapshotting),
|
# some data loss consider the default persistence mode that's snapshotting),
|
||||||
# or on the contrary, use "always" that's very slow but a bit safer than
|
# or on the contrary, use "always" that's very slow but a bit safer than
|
||||||
@ -284,7 +330,7 @@ appendfsync everysec
|
|||||||
# BGSAVE or BGREWRITEAOF is in progress.
|
# BGSAVE or BGREWRITEAOF is in progress.
|
||||||
#
|
#
|
||||||
# This means that while another child is saving the durability of Redis is
|
# This means that while another child is saving the durability of Redis is
|
||||||
# the same as "appendfsync none", that in pratical terms means that it is
|
# the same as "appendfsync none", that in practical terms means that it is
|
||||||
# possible to lost up to 30 seconds of log in the worst scenario (with the
|
# possible to lost up to 30 seconds of log in the worst scenario (with the
|
||||||
# default Linux settings).
|
# default Linux settings).
|
||||||
#
|
#
|
||||||
@ -306,7 +352,7 @@ no-appendfsync-on-rewrite no
|
|||||||
# is useful to avoid rewriting the AOF file even if the percentage increase
|
# is useful to avoid rewriting the AOF file even if the percentage increase
|
||||||
# is reached but it is still pretty small.
|
# is reached but it is still pretty small.
|
||||||
#
|
#
|
||||||
# Specify a precentage of zero in order to disable the automatic AOF
|
# Specify a percentage of zero in order to disable the automatic AOF
|
||||||
# rewrite feature.
|
# rewrite feature.
|
||||||
|
|
||||||
auto-aof-rewrite-percentage 100
|
auto-aof-rewrite-percentage 100
|
||||||
@ -315,9 +361,39 @@ auto-aof-rewrite-min-size 64mb
|
|||||||
################################ LUA SCRIPTING ###############################
|
################################ LUA SCRIPTING ###############################
|
||||||
|
|
||||||
# Max execution time of a Lua script in milliseconds.
|
# Max execution time of a Lua script in milliseconds.
|
||||||
# This prevents that a programming error generating an infinite loop will block
|
#
|
||||||
# your server forever. Set it to 0 or a negative value for unlimited execution.
|
# If the maximum execution time is reached Redis will log that a script is
|
||||||
#lua-time-limit 60000
|
# still in execution after the maximum allowed time and will start to
|
||||||
|
# reply to queries with an error.
|
||||||
|
#
|
||||||
|
# When a long running script exceed the maximum execution time only the
|
||||||
|
# SCRIPT KILL and SHUTDOWN NOSAVE commands are available. The first can be
|
||||||
|
# used to stop a script that did not yet called write commands. The second
|
||||||
|
# is the only way to shut down the server in the case a write commands was
|
||||||
|
# already issue by the script but the user don't want to wait for the natural
|
||||||
|
# termination of the script.
|
||||||
|
#
|
||||||
|
# Set it to 0 or a negative value for unlimited execution without warnings.
|
||||||
|
lua-time-limit 5000
|
||||||
|
|
||||||
|
################################ REDIS CLUSTER ###############################
|
||||||
|
#
|
||||||
|
# Normal Redis instances can't be part of a Redis Cluster, only nodes that are
|
||||||
|
# started as cluster nodes can. In order to start a Redis instance as a
|
||||||
|
# cluster node enable the cluster support uncommenting the following:
|
||||||
|
#
|
||||||
|
# cluster-enabled yes
|
||||||
|
|
||||||
|
# Every cluster node has a cluster configuration file. This file is not
|
||||||
|
# intended to be edited by hand. It is created and updated by Redis nodes.
|
||||||
|
# Every Redis Cluster node requires a different cluster configuration file.
|
||||||
|
# Make sure that instances running in the same system does not have
|
||||||
|
# overlapping cluster configuration file names.
|
||||||
|
#
|
||||||
|
# cluster-config-file nodes-6379.conf
|
||||||
|
|
||||||
|
# In order to setup your cluster make sure to read the documentation
|
||||||
|
# available at http://redis.io web site.
|
||||||
|
|
||||||
################################## SLOW LOG ###################################
|
################################## SLOW LOG ###################################
|
||||||
|
|
||||||
@ -345,12 +421,11 @@ slowlog-max-len 1024
|
|||||||
|
|
||||||
############################### ADVANCED CONFIG ###############################
|
############################### ADVANCED CONFIG ###############################
|
||||||
|
|
||||||
# Hashes are encoded in a special way (much more memory efficient) when they
|
# Hashes are encoded using a memory efficient data structure when they have a
|
||||||
# have at max a given numer of elements, and the biggest element does not
|
# small number of entries, and the biggest entry does not exceed a given
|
||||||
# exceed a given threshold. You can configure this limits with the following
|
# threshold. These thresholds can be configured using the following directives.
|
||||||
# configuration directives.
|
hash-max-ziplist-entries 512
|
||||||
hash-max-zipmap-entries 512
|
hash-max-ziplist-value 64
|
||||||
hash-max-zipmap-value 64
|
|
||||||
|
|
||||||
# Similarly to hashes, small lists are also encoded in a special way in order
|
# Similarly to hashes, small lists are also encoded in a special way in order
|
||||||
# to save a lot of space. The special representation is only used when
|
# to save a lot of space. The special representation is only used when
|
||||||
@ -373,9 +448,9 @@ zset-max-ziplist-value 64
|
|||||||
|
|
||||||
# Active rehashing uses 1 millisecond every 100 milliseconds of CPU time in
|
# Active rehashing uses 1 millisecond every 100 milliseconds of CPU time in
|
||||||
# order to help rehashing the main Redis hash table (the one mapping top-level
|
# order to help rehashing the main Redis hash table (the one mapping top-level
|
||||||
# keys to values). The hash table implementation redis uses (see dict.c)
|
# keys to values). The hash table implementation Redis uses (see dict.c)
|
||||||
# performs a lazy rehashing: the more operation you run into an hash table
|
# performs a lazy rehashing: the more operation you run into an hash table
|
||||||
# that is rhashing, the more rehashing "steps" are performed, so if the
|
# that is rehashing, the more rehashing "steps" are performed, so if the
|
||||||
# server is idle the rehashing is never complete and some more memory is used
|
# server is idle the rehashing is never complete and some more memory is used
|
||||||
# by the hash table.
|
# by the hash table.
|
||||||
#
|
#
|
||||||
@ -391,10 +466,47 @@ zset-max-ziplist-value 64
|
|||||||
# want to free memory asap when possible.
|
# want to free memory asap when possible.
|
||||||
activerehashing yes
|
activerehashing yes
|
||||||
|
|
||||||
|
# The client output buffer limits can be used to force disconnection of clients
|
||||||
|
# that are not reading data from the server fast enough for some reason (a
|
||||||
|
# common reason is that a Pub/Sub client can't consume messages as fast as the
|
||||||
|
# publisher can produce them).
|
||||||
|
#
|
||||||
|
# The limit can be set differently for the three different classes of clients:
|
||||||
|
#
|
||||||
|
# normal -> normal clients
|
||||||
|
# slave -> slave clients and MONITOR clients
|
||||||
|
# pubsub -> clients subcribed to at least one pubsub channel or pattern
|
||||||
|
#
|
||||||
|
# The syntax of every client-output-buffer-limit directive is the following:
|
||||||
|
#
|
||||||
|
# client-output-buffer-limit <class> <hard limit> <soft limit> <soft seconds>
|
||||||
|
#
|
||||||
|
# A client is immediately disconnected once the hard limit is reached, or if
|
||||||
|
# the soft limit is reached and remains reached for the specified number of
|
||||||
|
# seconds (continuously).
|
||||||
|
# So for instance if the hard limit is 32 megabytes and the soft limit is
|
||||||
|
# 16 megabytes / 10 seconds, the client will get disconnected immediately
|
||||||
|
# if the size of the output buffers reach 32 megabytes, but will also get
|
||||||
|
# disconnected if the client reaches 16 megabytes and continuously overcomes
|
||||||
|
# the limit for 10 seconds.
|
||||||
|
#
|
||||||
|
# By default normal clients are not limited because they don't receive data
|
||||||
|
# without asking (in a push way), but just after a request, so only
|
||||||
|
# asynchronous clients may create a scenario where data is requested faster
|
||||||
|
# than it can read.
|
||||||
|
#
|
||||||
|
# Instead there is a default limit for pubsub and slave clients, since
|
||||||
|
# subscribers and slaves receive data in a push fashion.
|
||||||
|
#
|
||||||
|
# Both the hard or the soft limit can be disabled just setting it to zero.
|
||||||
|
client-output-buffer-limit normal 0 0 0
|
||||||
|
client-output-buffer-limit slave 256mb 64mb 60
|
||||||
|
client-output-buffer-limit pubsub 32mb 8mb 60
|
||||||
|
|
||||||
################################## INCLUDES ###################################
|
################################## INCLUDES ###################################
|
||||||
|
|
||||||
# Include one or more other config files here. This is useful if you
|
# Include one or more other config files here. This is useful if you
|
||||||
# have a standard template that goes to all redis server but also need
|
# have a standard template that goes to all Redis server but also need
|
||||||
# to customize a few per-server settings. Include files can include
|
# to customize a few per-server settings. Include files can include
|
||||||
# other files, so use this wisely.
|
# other files, so use this wisely.
|
||||||
#
|
#
|
||||||
|
Loading…
Reference in New Issue
Block a user