mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 08:08:53 -05:00
Fixed some typos, add a spell check ci and others minor fix (#8890)
This PR adds a spell checker CI action that will fail future PRs if they introduce typos and spelling mistakes. This spell checker is based on blacklist of common spelling mistakes, so it will not catch everything, but at least it is also unlikely to cause false positives. Besides that, the PR also fixes many spelling mistakes and types, not all are a result of the spell checker we use. Here's a summary of other changes: 1. Scanned the entire source code and fixes all sorts of typos and spelling mistakes (including missing or extra spaces). 2. Outdated function / variable / argument names in comments 3. Fix outdated keyspace masks error log when we check `config.notify-keyspace-events` in loadServerConfigFromString. 4. Trim the white space at the end of line in `module.c`. Check: https://github.com/redis/redis/pull/7751 5. Some outdated https link URLs. 6. Fix some outdated comment. Such as: - In README: about the rdb, we used to said create a `thread`, change to `process` - dbRandomKey function coment (about the dictGetRandomKey, change to dictGetFairRandomKey) - notifyKeyspaceEvent fucntion comment (add type arg) - Some others minor fix in comment (Most of them are incorrectly quoted by variable names) 7. Modified the error log so that users can easily distinguish between TCP and TLS in `changeBindAddr`
This commit is contained in:
parent
8a86bca5ed
commit
0bfccc55e2
5
.github/.codespellrc
vendored
Normal file
5
.github/.codespellrc
vendored
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
[codespell]
|
||||||
|
quiet-level = 2
|
||||||
|
count =
|
||||||
|
skip = ./deps,./src/crc16_slottable.h
|
||||||
|
ignore-words = ./.github/wordlist.txt
|
2
.github/ISSUE_TEMPLATE/crash_report.md
vendored
2
.github/ISSUE_TEMPLATE/crash_report.md
vendored
@ -14,7 +14,7 @@ Paste the complete crash log between the quotes below. Please include a few line
|
|||||||
```
|
```
|
||||||
```
|
```
|
||||||
|
|
||||||
**Aditional information**
|
**Additional information**
|
||||||
|
|
||||||
1. OS distribution and version
|
1. OS distribution and version
|
||||||
2. Steps to reproduce (if any)
|
2. Steps to reproduce (if any)
|
||||||
|
15
.github/wordlist.txt
vendored
Normal file
15
.github/wordlist.txt
vendored
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
ake
|
||||||
|
bale
|
||||||
|
fle
|
||||||
|
fo
|
||||||
|
gameboy
|
||||||
|
mutli
|
||||||
|
nd
|
||||||
|
nees
|
||||||
|
oll
|
||||||
|
optin
|
||||||
|
ot
|
||||||
|
smove
|
||||||
|
te
|
||||||
|
tre
|
||||||
|
cancelability
|
25
.github/workflows/spell-check.yml
vendored
Normal file
25
.github/workflows/spell-check.yml
vendored
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
# A CI action that using codespell to check spell.
|
||||||
|
# .github/.codespellrc is a config file.
|
||||||
|
# .github/wordlist.txt is a list of words that will ignore word checks.
|
||||||
|
# More details please check the following link:
|
||||||
|
# https://github.com/codespell-project/codespell
|
||||||
|
name: Spellcheck
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
pull_request:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Spellcheck
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Install prerequisites
|
||||||
|
run: sudo pip install codespell==2.0.0
|
||||||
|
|
||||||
|
- name: Spell check
|
||||||
|
run: codespell --config=./.github/.codespellrc
|
@ -18,7 +18,7 @@ all the support in the mailing list.
|
|||||||
|
|
||||||
There is also an active community of Redis users at Stack Overflow:
|
There is also an active community of Redis users at Stack Overflow:
|
||||||
|
|
||||||
http://stackoverflow.com/questions/tagged/redis
|
https://stackoverflow.com/questions/tagged/redis
|
||||||
|
|
||||||
Issues and pull requests for documentation belong on the redis-doc repo:
|
Issues and pull requests for documentation belong on the redis-doc repo:
|
||||||
|
|
||||||
@ -38,10 +38,10 @@ Here you'll see if there is consensus about your idea.
|
|||||||
2. If in step 1 you get an acknowledgment from the project leaders, use the
|
2. If in step 1 you get an acknowledgment from the project leaders, use the
|
||||||
following procedure to submit a patch:
|
following procedure to submit a patch:
|
||||||
|
|
||||||
a. Fork Redis on github ( http://help.github.com/fork-a-repo/ )
|
a. Fork Redis on github ( https://docs.github.com/en/github/getting-started-with-github/fork-a-repo )
|
||||||
b. Create a topic branch (git checkout -b my_branch)
|
b. Create a topic branch (git checkout -b my_branch)
|
||||||
c. Push to your branch (git push origin my_branch)
|
c. Push to your branch (git push origin my_branch)
|
||||||
d. Initiate a pull request on github ( https://help.github.com/articles/creating-a-pull-request/ )
|
d. Initiate a pull request on github ( https://docs.github.com/en/github/collaborating-with-issues-and-pull-requests/creating-a-pull-request )
|
||||||
e. Done :)
|
e. Done :)
|
||||||
|
|
||||||
3. Keep in mind that we are very overloaded, so issues and PRs sometimes wait
|
3. Keep in mind that we are very overloaded, so issues and PRs sometimes wait
|
||||||
|
@ -380,8 +380,8 @@ aof.c and rdb.c
|
|||||||
|
|
||||||
As you can guess from the names, these files implement the RDB and AOF
|
As you can guess from the names, these files implement the RDB and AOF
|
||||||
persistence for Redis. Redis uses a persistence model based on the `fork()`
|
persistence for Redis. Redis uses a persistence model based on the `fork()`
|
||||||
system call in order to create a thread with the same (shared) memory
|
system call in order to create a process with the same (shared) memory
|
||||||
content of the main Redis thread. This secondary thread dumps the content
|
content of the main Redis process. This secondary process dumps the content
|
||||||
of the memory on disk. This is used by `rdb.c` to create the snapshots
|
of the memory on disk. This is used by `rdb.c` to create the snapshots
|
||||||
on disk and by `aof.c` in order to perform the AOF rewrite when the
|
on disk and by `aof.c` in order to perform the AOF rewrite when the
|
||||||
append only file gets too big.
|
append only file gets too big.
|
||||||
|
@ -178,7 +178,7 @@ acllog-max-len 128
|
|||||||
# incoming connections (via ACL), and for outgoing connections (via
|
# incoming connections (via ACL), and for outgoing connections (via
|
||||||
# sentinel-user and sentinel-pass)
|
# sentinel-user and sentinel-pass)
|
||||||
#
|
#
|
||||||
# The requirepass is not compatable with aclfile option and the ACL LOAD
|
# The requirepass is not compatible with aclfile option and the ACL LOAD
|
||||||
# command, these will cause requirepass to be ignored.
|
# command, these will cause requirepass to be ignored.
|
||||||
|
|
||||||
# sentinel sentinel-user <username>
|
# sentinel sentinel-user <username>
|
||||||
|
@ -1363,7 +1363,7 @@ int ACLCheckPubsubPerm(client *c, int idx, int count, int literal, int *idxptr)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether the command is ready to be exceuted by ACLCheckCommandPerm.
|
/* Check whether the command is ready to be executed by ACLCheckCommandPerm.
|
||||||
* If check passes, then check whether pub/sub channels of the command is
|
* If check passes, then check whether pub/sub channels of the command is
|
||||||
* ready to be executed by ACLCheckPubsubPerm */
|
* ready to be executed by ACLCheckPubsubPerm */
|
||||||
int ACLCheckAllPerm(client *c, int *idxptr) {
|
int ACLCheckAllPerm(client *c, int *idxptr) {
|
||||||
@ -2254,8 +2254,8 @@ void authCommand(client *c) {
|
|||||||
* will just use "default" as username. */
|
* will just use "default" as username. */
|
||||||
robj *username, *password;
|
robj *username, *password;
|
||||||
if (c->argc == 2) {
|
if (c->argc == 2) {
|
||||||
/* Mimic the old behavior of giving an error for the two commands
|
/* Mimic the old behavior of giving an error for the two argument
|
||||||
* from if no password is configured. */
|
* form if no password is configured. */
|
||||||
if (DefaultUser->flags & USER_FLAG_NOPASS) {
|
if (DefaultUser->flags & USER_FLAG_NOPASS) {
|
||||||
addReplyError(c,"AUTH <password> called without any password "
|
addReplyError(c,"AUTH <password> called without any password "
|
||||||
"configured for the default user. Are you sure "
|
"configured for the default user. Are you sure "
|
||||||
|
@ -419,7 +419,7 @@ void flushAppendOnlyFile(int force) {
|
|||||||
* than two seconds this is still ok. Postpone again. */
|
* than two seconds this is still ok. Postpone again. */
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
/* Otherwise fall trough, and go write since we can't wait
|
/* Otherwise fall through, and go write since we can't wait
|
||||||
* over two seconds. */
|
* over two seconds. */
|
||||||
server.aof_delayed_fsync++;
|
server.aof_delayed_fsync++;
|
||||||
serverLog(LL_NOTICE,"Asynchronous AOF fsync is taking too long (disk is busy?). Writing the AOF buffer without waiting for fsync to complete, this may slow down Redis.");
|
serverLog(LL_NOTICE,"Asynchronous AOF fsync is taking too long (disk is busy?). Writing the AOF buffer without waiting for fsync to complete, this may slow down Redis.");
|
||||||
@ -1745,7 +1745,7 @@ int rewriteAppendOnlyFileBackground(void) {
|
|||||||
server.aof_rewrite_scheduled = 0;
|
server.aof_rewrite_scheduled = 0;
|
||||||
server.aof_rewrite_time_start = time(NULL);
|
server.aof_rewrite_time_start = time(NULL);
|
||||||
|
|
||||||
/* We set appendseldb to -1 in order to force the next call to the
|
/* We set aof_selected_db to -1 in order to force the next call to the
|
||||||
* feedAppendOnlyFile() to issue a SELECT command, so the differences
|
* feedAppendOnlyFile() to issue a SELECT command, so the differences
|
||||||
* accumulated by the parent into server.aof_rewrite_buf will start
|
* accumulated by the parent into server.aof_rewrite_buf will start
|
||||||
* with a SELECT statement and it will be safe to merge. */
|
* with a SELECT statement and it will be safe to merge. */
|
||||||
@ -1885,7 +1885,7 @@ void backgroundRewriteDoneHandler(int exitcode, int bysignal) {
|
|||||||
oldfd = open(server.aof_filename,O_RDONLY|O_NONBLOCK);
|
oldfd = open(server.aof_filename,O_RDONLY|O_NONBLOCK);
|
||||||
} else {
|
} else {
|
||||||
/* AOF enabled */
|
/* AOF enabled */
|
||||||
oldfd = -1; /* We'll set this to the current AOF filedes later. */
|
oldfd = -1; /* We'll set this to the current AOF file descriptor later. */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Rename the temporary file. This will not unlink the target file if
|
/* Rename the temporary file. This will not unlink the target file if
|
||||||
|
@ -472,7 +472,7 @@ int getBitfieldTypeFromArgument(client *c, robj *o, int *sign, int *bits) {
|
|||||||
return C_OK;
|
return C_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is an helper function for commands implementations that need to write
|
/* This is a helper function for commands implementations that need to write
|
||||||
* bits to a string object. The command creates or pad with zeroes the string
|
* bits to a string object. The command creates or pad with zeroes the string
|
||||||
* so that the 'maxbit' bit can be addressed. The object is finally
|
* so that the 'maxbit' bit can be addressed. The object is finally
|
||||||
* returned. Otherwise if the key holds a wrong type NULL is returned and
|
* returned. Otherwise if the key holds a wrong type NULL is returned and
|
||||||
@ -1025,7 +1025,7 @@ void bitfieldGeneric(client *c, int flags) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Lookup by making room up to the farest bit reached by
|
/* Lookup by making room up to the farthest bit reached by
|
||||||
* this operation. */
|
* this operation. */
|
||||||
if ((o = lookupStringForBitCommand(c,
|
if ((o = lookupStringForBitCommand(c,
|
||||||
highest_write_offset)) == NULL) {
|
highest_write_offset)) == NULL) {
|
||||||
|
@ -166,7 +166,7 @@ ssize_t cliWriteConn(redisContext *c, const char *buf, size_t buf_len)
|
|||||||
*
|
*
|
||||||
* Do we still have data that was there prior to our buf? If so,
|
* Do we still have data that was there prior to our buf? If so,
|
||||||
* restore buffer to it's original state and report no new data was
|
* restore buffer to it's original state and report no new data was
|
||||||
* writen.
|
* written.
|
||||||
*/
|
*/
|
||||||
if (sdslen(c->obuf) > buf_len) {
|
if (sdslen(c->obuf) > buf_len) {
|
||||||
sdsrange(c->obuf, 0, -(buf_len+1));
|
sdsrange(c->obuf, 0, -(buf_len+1));
|
||||||
|
@ -16,9 +16,9 @@ typedef struct cliSSLconfig {
|
|||||||
char *cert;
|
char *cert;
|
||||||
/* Private key file to authenticate with, or NULL */
|
/* Private key file to authenticate with, or NULL */
|
||||||
char *key;
|
char *key;
|
||||||
/* Prefered cipher list, or NULL (applies only to <= TLSv1.2) */
|
/* Preferred cipher list, or NULL (applies only to <= TLSv1.2) */
|
||||||
char* ciphers;
|
char* ciphers;
|
||||||
/* Prefered ciphersuites list, or NULL (applies only to TLSv1.3) */
|
/* Preferred ciphersuites list, or NULL (applies only to TLSv1.3) */
|
||||||
char* ciphersuites;
|
char* ciphersuites;
|
||||||
} cliSSLconfig;
|
} cliSSLconfig;
|
||||||
|
|
||||||
|
@ -1185,7 +1185,7 @@ void clusterHandleConfigEpochCollision(clusterNode *sender) {
|
|||||||
* CLUSTER nodes blacklist
|
* CLUSTER nodes blacklist
|
||||||
*
|
*
|
||||||
* The nodes blacklist is just a way to ensure that a given node with a given
|
* The nodes blacklist is just a way to ensure that a given node with a given
|
||||||
* Node ID is not readded before some time elapsed (this time is specified
|
* Node ID is not re-added before some time elapsed (this time is specified
|
||||||
* in seconds in CLUSTER_BLACKLIST_TTL).
|
* in seconds in CLUSTER_BLACKLIST_TTL).
|
||||||
*
|
*
|
||||||
* This is useful when we want to remove a node from the cluster completely:
|
* This is useful when we want to remove a node from the cluster completely:
|
||||||
@ -4203,7 +4203,7 @@ sds clusterGenNodeDescription(clusterNode *node, int use_pport) {
|
|||||||
"connected" : "disconnected");
|
"connected" : "disconnected");
|
||||||
|
|
||||||
/* Slots served by this instance. If we already have slots info,
|
/* Slots served by this instance. If we already have slots info,
|
||||||
* append it diretly, otherwise, generate slots only if it has. */
|
* append it directly, otherwise, generate slots only if it has. */
|
||||||
if (node->slots_info) {
|
if (node->slots_info) {
|
||||||
ci = sdscatsds(ci, node->slots_info);
|
ci = sdscatsds(ci, node->slots_info);
|
||||||
} else if (node->numslots > 0) {
|
} else if (node->numslots > 0) {
|
||||||
|
@ -581,7 +581,7 @@ void loadServerConfigFromString(char *config) {
|
|||||||
int flags = keyspaceEventsStringToFlags(argv[1]);
|
int flags = keyspaceEventsStringToFlags(argv[1]);
|
||||||
|
|
||||||
if (flags == -1) {
|
if (flags == -1) {
|
||||||
err = "Invalid event class character. Use 'g$lshzxeA'.";
|
err = "Invalid event class character. Use 'Ag$lshzxeKEtmd'.";
|
||||||
goto loaderr;
|
goto loaderr;
|
||||||
}
|
}
|
||||||
server.notify_keyspace_events = flags;
|
server.notify_keyspace_events = flags;
|
||||||
|
@ -41,7 +41,7 @@
|
|||||||
* of connections from within a handler.
|
* of connections from within a handler.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* Incremenet connection references.
|
/* Increment connection references.
|
||||||
*
|
*
|
||||||
* Inside a connection handler, we guarantee refs >= 1 so it is always
|
* Inside a connection handler, we guarantee refs >= 1 so it is always
|
||||||
* safe to connClose().
|
* safe to connClose().
|
||||||
|
@ -248,7 +248,7 @@ uint16_t crcspeed16big(uint16_t big_table[8][256], uint16_t crc_in, void *buf,
|
|||||||
|
|
||||||
/* Return the CRC of buf[0..len-1] with initial crc, processing eight bytes
|
/* Return the CRC of buf[0..len-1] with initial crc, processing eight bytes
|
||||||
at a time using passed-in lookup table.
|
at a time using passed-in lookup table.
|
||||||
This selects one of two routines depending on the endianess of
|
This selects one of two routines depending on the endianness of
|
||||||
the architecture. */
|
the architecture. */
|
||||||
uint64_t crcspeed64native(uint64_t table[8][256], uint64_t crc, void *buf,
|
uint64_t crcspeed64native(uint64_t table[8][256], uint64_t crc, void *buf,
|
||||||
size_t len) {
|
size_t len) {
|
||||||
|
2
src/db.c
2
src/db.c
@ -295,7 +295,7 @@ robj *dbRandomKey(redisDb *db) {
|
|||||||
* it could happen that all the keys are already logically
|
* it could happen that all the keys are already logically
|
||||||
* expired in the slave, so the function cannot stop because
|
* expired in the slave, so the function cannot stop because
|
||||||
* expireIfNeeded() is false, nor it can stop because
|
* expireIfNeeded() is false, nor it can stop because
|
||||||
* dictGetRandomKey() returns NULL (there are keys to return).
|
* dictGetFairRandomKey() returns NULL (there are keys to return).
|
||||||
* To prevent the infinite loop we do some tries, but if there
|
* To prevent the infinite loop we do some tries, but if there
|
||||||
* are the conditions for an infinite loop, eventually we
|
* are the conditions for an infinite loop, eventually we
|
||||||
* return a key name that may be already expired. */
|
* return a key name that may be already expired. */
|
||||||
|
@ -1980,7 +1980,7 @@ void disableWatchdog(void) {
|
|||||||
* of microseconds, i.e. -10 means 100 nanoseconds. */
|
* of microseconds, i.e. -10 means 100 nanoseconds. */
|
||||||
void debugDelay(int usec) {
|
void debugDelay(int usec) {
|
||||||
/* Since even the shortest sleep results in context switch and system call,
|
/* Since even the shortest sleep results in context switch and system call,
|
||||||
* the way we achive short sleeps is by statistically sleeping less often. */
|
* the way we achieve short sleeps is by statistically sleeping less often. */
|
||||||
if (usec < 0) usec = (rand() % -usec) == 0 ? 1: 0;
|
if (usec < 0) usec = (rand() % -usec) == 0 ? 1: 0;
|
||||||
if (usec) usleep(usec);
|
if (usec) usleep(usec);
|
||||||
}
|
}
|
||||||
|
@ -133,7 +133,7 @@ void evictionPoolAlloc(void) {
|
|||||||
EvictionPoolLRU = ep;
|
EvictionPoolLRU = ep;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is an helper function for performEvictions(), it is used in order
|
/* This is a helper function for performEvictions(), it is used in order
|
||||||
* to populate the evictionPool with a few entries every time we want to
|
* to populate the evictionPool with a few entries every time we want to
|
||||||
* expire a key. Keys with idle time bigger than one of the current
|
* expire a key. Keys with idle time bigger than one of the current
|
||||||
* keys are added. Keys are always added if there are free entries.
|
* keys are added. Keys are always added if there are free entries.
|
||||||
|
@ -213,7 +213,7 @@ void addReplyDoubleDistance(client *c, double d) {
|
|||||||
* representing a point, and a GeoShape, appends this entry as a geoPoint
|
* representing a point, and a GeoShape, appends this entry as a geoPoint
|
||||||
* into the specified geoArray only if the point is within the search area.
|
* into the specified geoArray only if the point is within the search area.
|
||||||
*
|
*
|
||||||
* returns C_OK if the point is included, or REIDS_ERR if it is outside. */
|
* returns C_OK if the point is included, or C_ERR if it is outside. */
|
||||||
int geoAppendIfWithinShape(geoArray *ga, GeoShape *shape, double score, sds member) {
|
int geoAppendIfWithinShape(geoArray *ga, GeoShape *shape, double score, sds member) {
|
||||||
double distance = 0, xy[2];
|
double distance = 0, xy[2];
|
||||||
|
|
||||||
@ -241,10 +241,10 @@ int geoAppendIfWithinShape(geoArray *ga, GeoShape *shape, double score, sds memb
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Query a Redis sorted set to extract all the elements between 'min' and
|
/* Query a Redis sorted set to extract all the elements between 'min' and
|
||||||
* 'max', appending them into the array of geoPoint structures 'gparray'.
|
* 'max', appending them into the array of geoPoint structures 'geoArray'.
|
||||||
* The command returns the number of elements added to the array.
|
* The command returns the number of elements added to the array.
|
||||||
*
|
*
|
||||||
* Elements which are farest than 'radius' from the specified 'x' and 'y'
|
* Elements which are farther than 'radius' from the specified 'x' and 'y'
|
||||||
* coordinates are not included.
|
* coordinates are not included.
|
||||||
*
|
*
|
||||||
* The ability of this function to append to an existing set of points is
|
* The ability of this function to append to an existing set of points is
|
||||||
@ -330,7 +330,7 @@ void scoresOfGeoHashBox(GeoHashBits hash, GeoHashFix52Bits *min, GeoHashFix52Bit
|
|||||||
*
|
*
|
||||||
* To get the min score we just use the initial hash value left
|
* To get the min score we just use the initial hash value left
|
||||||
* shifted enough to get the 52 bit value. Later we increment the
|
* shifted enough to get the 52 bit value. Later we increment the
|
||||||
* 6 bit prefis (see the hash.bits++ statement), and get the new
|
* 6 bit prefix (see the hash.bits++ statement), and get the new
|
||||||
* prefix: 101011, which we align again to 52 bits to get the maximum
|
* prefix: 101011, which we align again to 52 bits to get the maximum
|
||||||
* value (which is excluded from the search). So we get everything
|
* value (which is excluded from the search). So we get everything
|
||||||
* between the two following scores (represented in binary):
|
* between the two following scores (represented in binary):
|
||||||
|
@ -216,7 +216,7 @@ GeoHashFix52Bits geohashAlign52Bits(const GeoHashBits hash) {
|
|||||||
return bits;
|
return bits;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Calculate distance using haversin great circle distance formula. */
|
/* Calculate distance using haversine great circle distance formula. */
|
||||||
double geohashGetDistance(double lon1d, double lat1d, double lon2d, double lat2d) {
|
double geohashGetDistance(double lon1d, double lat1d, double lon2d, double lat2d) {
|
||||||
double lat1r, lon1r, lat2r, lon2r, u, v;
|
double lat1r, lon1r, lat2r, lon2r, u, v;
|
||||||
lat1r = deg_rad(lat1d);
|
lat1r = deg_rad(lat1d);
|
||||||
|
@ -899,7 +899,7 @@ promote: /* Promote to dense representation. */
|
|||||||
* the element belongs to is incremented if needed.
|
* the element belongs to is incremented if needed.
|
||||||
*
|
*
|
||||||
* This function is actually a wrapper for hllSparseSet(), it only performs
|
* This function is actually a wrapper for hllSparseSet(), it only performs
|
||||||
* the hashshing of the element to obtain the index and zeros run length. */
|
* the hashing of the element to obtain the index and zeros run length. */
|
||||||
int hllSparseAdd(robj *o, unsigned char *ele, size_t elesize) {
|
int hllSparseAdd(robj *o, unsigned char *ele, size_t elesize) {
|
||||||
long index;
|
long index;
|
||||||
uint8_t count = hllPatLen(ele,elesize,&index);
|
uint8_t count = hllPatLen(ele,elesize,&index);
|
||||||
|
@ -306,7 +306,7 @@ int intsetValidateIntegrity(const unsigned char *p, size_t size, int deep) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* check that the size matchies (all records are inside the buffer). */
|
/* check that the size matches (all records are inside the buffer). */
|
||||||
uint32_t count = intrev32ifbe(is->length);
|
uint32_t count = intrev32ifbe(is->length);
|
||||||
if (sizeof(*is) + count*record_size != size)
|
if (sizeof(*is) + count*record_size != size)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -116,7 +116,7 @@
|
|||||||
(p)[5] = ((v)>>8)&0xff; \
|
(p)[5] = ((v)>>8)&0xff; \
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
/* Validates that 'p' is not ouside the listpack.
|
/* Validates that 'p' is not outside the listpack.
|
||||||
* All function that return a pointer to an element in the listpack will assert
|
* All function that return a pointer to an element in the listpack will assert
|
||||||
* that this element is valid, so it can be freely used.
|
* that this element is valid, so it can be freely used.
|
||||||
* Generally functions such lpNext and lpDelete assume the input pointer is
|
* Generally functions such lpNext and lpDelete assume the input pointer is
|
||||||
@ -125,7 +125,7 @@
|
|||||||
assert((p) >= (lp)+LP_HDR_SIZE && (p) < (lp)+lpGetTotalBytes((lp))); \
|
assert((p) >= (lp)+LP_HDR_SIZE && (p) < (lp)+lpGetTotalBytes((lp))); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
/* Similar to the above, but validates the entire element lenth rather than just
|
/* Similar to the above, but validates the entire element length rather than just
|
||||||
* it's pointer. */
|
* it's pointer. */
|
||||||
#define ASSERT_INTEGRITY_LEN(lp, p, len) do { \
|
#define ASSERT_INTEGRITY_LEN(lp, p, len) do { \
|
||||||
assert((p) >= (lp)+LP_HDR_SIZE && (p)+(len) < (lp)+lpGetTotalBytes((lp))); \
|
assert((p) >= (lp)+LP_HDR_SIZE && (p)+(len) < (lp)+lpGetTotalBytes((lp))); \
|
||||||
@ -218,7 +218,7 @@ int lpStringToInt64(const char *s, unsigned long slen, int64_t *value) {
|
|||||||
/* Create a new, empty listpack.
|
/* Create a new, empty listpack.
|
||||||
* On success the new listpack is returned, otherwise an error is returned.
|
* On success the new listpack is returned, otherwise an error is returned.
|
||||||
* Pre-allocate at least `capacity` bytes of memory,
|
* Pre-allocate at least `capacity` bytes of memory,
|
||||||
* over-allocated memory can be shrinked by `lpShrinkToFit`.
|
* over-allocated memory can be shrunk by `lpShrinkToFit`.
|
||||||
* */
|
* */
|
||||||
unsigned char *lpNew(size_t capacity) {
|
unsigned char *lpNew(size_t capacity) {
|
||||||
unsigned char *lp = lp_malloc(capacity > LP_HDR_SIZE+1 ? capacity : LP_HDR_SIZE+1);
|
unsigned char *lp = lp_malloc(capacity > LP_HDR_SIZE+1 ? capacity : LP_HDR_SIZE+1);
|
||||||
@ -416,7 +416,7 @@ uint32_t lpCurrentEncodedSizeUnsafe(unsigned char *p) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Return bytes needed to encode the length of the listpack element pointed by 'p'.
|
/* Return bytes needed to encode the length of the listpack element pointed by 'p'.
|
||||||
* This includes just the encodign byte, and the bytes needed to encode the length
|
* This includes just the encoding byte, and the bytes needed to encode the length
|
||||||
* of the element (excluding the element data itself)
|
* of the element (excluding the element data itself)
|
||||||
* If the element encoding is wrong then 0 is returned. */
|
* If the element encoding is wrong then 0 is returned. */
|
||||||
uint32_t lpCurrentEncodedSizeBytes(unsigned char *p) {
|
uint32_t lpCurrentEncodedSizeBytes(unsigned char *p) {
|
||||||
@ -641,7 +641,7 @@ unsigned char *lpGet(unsigned char *p, int64_t *count, unsigned char *intbuf) {
|
|||||||
*
|
*
|
||||||
* If 'newp' is not NULL, at the end of a successful call '*newp' will be set
|
* If 'newp' is not NULL, at the end of a successful call '*newp' will be set
|
||||||
* to the address of the element just added, so that it will be possible to
|
* to the address of the element just added, so that it will be possible to
|
||||||
* continue an interation with lpNext() and lpPrev().
|
* continue an interaction with lpNext() and lpPrev().
|
||||||
*
|
*
|
||||||
* For deletion operations ('ele' set to NULL) 'newp' is set to the next
|
* For deletion operations ('ele' set to NULL) 'newp' is set to the next
|
||||||
* element, on the right of the deleted one, or to NULL if the deleted element
|
* element, on the right of the deleted one, or to NULL if the deleted element
|
||||||
@ -879,7 +879,7 @@ int lpValidateNext(unsigned char *lp, unsigned char **pp, size_t lpbytes) {
|
|||||||
if (!lenbytes)
|
if (!lenbytes)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* make sure the encoded entry length doesn't rech outside the edge of the listpack */
|
/* make sure the encoded entry length doesn't reach outside the edge of the listpack */
|
||||||
if (OUT_OF_RANGE(p + lenbytes))
|
if (OUT_OF_RANGE(p + lenbytes))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -888,7 +888,7 @@ int lpValidateNext(unsigned char *lp, unsigned char **pp, size_t lpbytes) {
|
|||||||
unsigned long encodedBacklen = lpEncodeBacklen(NULL,entrylen);
|
unsigned long encodedBacklen = lpEncodeBacklen(NULL,entrylen);
|
||||||
entrylen += encodedBacklen;
|
entrylen += encodedBacklen;
|
||||||
|
|
||||||
/* make sure the entry doesn't rech outside the edge of the listpack */
|
/* make sure the entry doesn't reach outside the edge of the listpack */
|
||||||
if (OUT_OF_RANGE(p + entrylen))
|
if (OUT_OF_RANGE(p + entrylen))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -925,7 +925,7 @@ int lpValidateIntegrity(unsigned char *lp, size_t size, int deep){
|
|||||||
if (!deep)
|
if (!deep)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
/* Validate the invividual entries. */
|
/* Validate the individual entries. */
|
||||||
uint32_t count = 0;
|
uint32_t count = 0;
|
||||||
unsigned char *p = lpFirst(lp);
|
unsigned char *p = lpFirst(lp);
|
||||||
while(p) {
|
while(p) {
|
||||||
|
@ -84,7 +84,7 @@ void lolwutCommand(client *c) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ========================== LOLWUT Canvase ===============================
|
/* ========================== LOLWUT Canvas ===============================
|
||||||
* Many LOLWUT versions will likely print some computer art to the screen.
|
* Many LOLWUT versions will likely print some computer art to the screen.
|
||||||
* This is the case with LOLWUT 5 and LOLWUT 6, so here there is a generic
|
* This is the case with LOLWUT 5 and LOLWUT 6, so here there is a generic
|
||||||
* canvas implementation that can be reused. */
|
* canvas implementation that can be reused. */
|
||||||
|
@ -102,7 +102,7 @@ lwCanvas *lwDrawSchotter(int console_cols, int squares_per_row, int squares_per_
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Converts the canvas to an SDS string representing the UTF8 characters to
|
/* Converts the canvas to an SDS string representing the UTF8 characters to
|
||||||
* print to the terminal in order to obtain a graphical representaiton of the
|
* print to the terminal in order to obtain a graphical representation of the
|
||||||
* logical canvas. The actual returned string will require a terminal that is
|
* logical canvas. The actual returned string will require a terminal that is
|
||||||
* width/2 large and height/4 tall in order to hold the whole image without
|
* width/2 large and height/4 tall in order to hold the whole image without
|
||||||
* overflowing or scrolling, since each Barille character is 2x4. */
|
* overflowing or scrolling, since each Barille character is 2x4. */
|
||||||
|
@ -32,7 +32,7 @@
|
|||||||
* fun and interesting, and should be replaced by a new implementation at
|
* fun and interesting, and should be replaced by a new implementation at
|
||||||
* each new version of Redis.
|
* each new version of Redis.
|
||||||
*
|
*
|
||||||
* Thanks to Michele Hiki Falcone for the original image that ispired
|
* Thanks to Michele Hiki Falcone for the original image that inspired
|
||||||
* the image, part of his game, Plaguemon.
|
* the image, part of his game, Plaguemon.
|
||||||
*
|
*
|
||||||
* Thanks to the Shhh computer art collective for the help in tuning the
|
* Thanks to the Shhh computer art collective for the help in tuning the
|
||||||
@ -180,7 +180,7 @@ void lolwut6Command(client *c) {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
/* Limits. We want LOLWUT to be always reasonably fast and cheap to execute
|
/* Limits. We want LOLWUT to be always reasonably fast and cheap to execute
|
||||||
* so we have maximum number of columns, rows, and output resulution. */
|
* so we have maximum number of columns, rows, and output resolution. */
|
||||||
if (cols < 1) cols = 1;
|
if (cols < 1) cols = 1;
|
||||||
if (cols > 1000) cols = 1000;
|
if (cols > 1000) cols = 1000;
|
||||||
if (rows < 1) rows = 1;
|
if (rows < 1) rows = 1;
|
||||||
|
52
src/module.c
52
src/module.c
@ -962,7 +962,7 @@ long long RM_Milliseconds(void) {
|
|||||||
* the elapsed execution time when RM_BlockedClientMeasureTimeEnd() is called.
|
* the elapsed execution time when RM_BlockedClientMeasureTimeEnd() is called.
|
||||||
* Within the same command, you can call multiple times
|
* Within the same command, you can call multiple times
|
||||||
* RM_BlockedClientMeasureTimeStart() and RM_BlockedClientMeasureTimeEnd()
|
* RM_BlockedClientMeasureTimeStart() and RM_BlockedClientMeasureTimeEnd()
|
||||||
* to accummulate indepedent time intervals to the background duration.
|
* to accumulate independent time intervals to the background duration.
|
||||||
* This method always return REDISMODULE_OK. */
|
* This method always return REDISMODULE_OK. */
|
||||||
int RM_BlockedClientMeasureTimeStart(RedisModuleBlockedClient *bc) {
|
int RM_BlockedClientMeasureTimeStart(RedisModuleBlockedClient *bc) {
|
||||||
elapsedStart(&(bc->background_timer));
|
elapsedStart(&(bc->background_timer));
|
||||||
@ -1902,7 +1902,7 @@ RedisModuleString *RM_GetClientUserNameById(RedisModuleCtx *ctx, uint64_t id) {
|
|||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is an helper for RM_GetClientInfoById() and other functions: given
|
/* This is a helper for RM_GetClientInfoById() and other functions: given
|
||||||
* a client, it populates the client info structure with the appropriate
|
* a client, it populates the client info structure with the appropriate
|
||||||
* fields depending on the version provided. If the version is not valid
|
* fields depending on the version provided. If the version is not valid
|
||||||
* then REDISMODULE_ERR is returned. Otherwise the function returns
|
* then REDISMODULE_ERR is returned. Otherwise the function returns
|
||||||
@ -1935,7 +1935,7 @@ int modulePopulateClientInfoStructure(void *ci, client *client, int structver) {
|
|||||||
return REDISMODULE_OK;
|
return REDISMODULE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is an helper for moduleFireServerEvent() and other functions:
|
/* This is a helper for moduleFireServerEvent() and other functions:
|
||||||
* It populates the replication info structure with the appropriate
|
* It populates the replication info structure with the appropriate
|
||||||
* fields depending on the version provided. If the version is not valid
|
* fields depending on the version provided. If the version is not valid
|
||||||
* then REDISMODULE_ERR is returned. Otherwise the function returns
|
* then REDISMODULE_ERR is returned. Otherwise the function returns
|
||||||
@ -3093,7 +3093,7 @@ int RM_ZsetRangePrev(RedisModuleKey *key) {
|
|||||||
*
|
*
|
||||||
* The number of fields existing in the hash prior to the call, which have been
|
* The number of fields existing in the hash prior to the call, which have been
|
||||||
* updated (its old value has been replaced by a new value) or deleted. If the
|
* updated (its old value has been replaced by a new value) or deleted. If the
|
||||||
* flag REDISMODULE_HASH_COUNT_ALL is set, insterted fields not previously
|
* flag REDISMODULE_HASH_COUNT_ALL is set, inserted fields not previously
|
||||||
* existing in the hash are also counted.
|
* existing in the hash are also counted.
|
||||||
*
|
*
|
||||||
* If the return value is zero, `errno` is set (since Redis 6.2) as follows:
|
* If the return value is zero, `errno` is set (since Redis 6.2) as follows:
|
||||||
@ -3333,7 +3333,7 @@ int RM_StreamAdd(RedisModuleKey *key, int flags, RedisModuleStreamID *id, RedisM
|
|||||||
return REDISMODULE_ERR;
|
return REDISMODULE_ERR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Create key if necessery */
|
/* Create key if necessary */
|
||||||
int created = 0;
|
int created = 0;
|
||||||
if (key->value == NULL) {
|
if (key->value == NULL) {
|
||||||
moduleCreateEmptyKey(key, REDISMODULE_KEYTYPE_STREAM);
|
moduleCreateEmptyKey(key, REDISMODULE_KEYTYPE_STREAM);
|
||||||
@ -4470,7 +4470,7 @@ robj *moduleTypeDupOrReply(client *c, robj *fromkey, robj *tokey, robj *value) {
|
|||||||
* to have significant internal complexity. To determine this, the defrag mechanism
|
* to have significant internal complexity. To determine this, the defrag mechanism
|
||||||
* uses the free_effort callback and the 'active-defrag-max-scan-fields' config directive.
|
* uses the free_effort callback and the 'active-defrag-max-scan-fields' config directive.
|
||||||
* NOTE: The value is passed as a `void**` and the function is expected to update the
|
* NOTE: The value is passed as a `void**` and the function is expected to update the
|
||||||
* pointer if the top-level value pointer is defragmented and consequentially changes.
|
* pointer if the top-level value pointer is defragmented and consequently changes.
|
||||||
*
|
*
|
||||||
* Note: the module name "AAAAAAAAA" is reserved and produces an error, it
|
* Note: the module name "AAAAAAAAA" is reserved and produces an error, it
|
||||||
* happens to be pretty lame as well.
|
* happens to be pretty lame as well.
|
||||||
@ -4925,7 +4925,7 @@ ssize_t rdbSaveModulesAux(rio *rdb, int when) {
|
|||||||
* foreach key,value {
|
* foreach key,value {
|
||||||
* AddElement(key);
|
* AddElement(key);
|
||||||
* AddElement(value);
|
* AddElement(value);
|
||||||
* EndSquence();
|
* EndSequence();
|
||||||
* }
|
* }
|
||||||
*
|
*
|
||||||
* Because the key and value will be always in the above order, while instead
|
* Because the key and value will be always in the above order, while instead
|
||||||
@ -6058,13 +6058,13 @@ int RM_SendClusterMessage(RedisModuleCtx *ctx, char *target_id, uint8_t type, un
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Return an array of string pointers, each string pointer points to a cluster
|
/* Return an array of string pointers, each string pointer points to a cluster
|
||||||
* node ID of exactly REDISMODULE_NODE_ID_SIZE bytes (without any null term).
|
* node ID of exactly REDISMODULE_NODE_ID_LEN bytes (without any null term).
|
||||||
* The number of returned node IDs is stored into `*numnodes`.
|
* The number of returned node IDs is stored into `*numnodes`.
|
||||||
* However if this function is called by a module not running an a Redis
|
* However if this function is called by a module not running an a Redis
|
||||||
* instance with Redis Cluster enabled, NULL is returned instead.
|
* instance with Redis Cluster enabled, NULL is returned instead.
|
||||||
*
|
*
|
||||||
* The IDs returned can be used with RedisModule_GetClusterNodeInfo() in order
|
* The IDs returned can be used with RedisModule_GetClusterNodeInfo() in order
|
||||||
* to get more information about single nodes.
|
* to get more information about single node.
|
||||||
*
|
*
|
||||||
* The array returned by this function must be freed using the function
|
* The array returned by this function must be freed using the function
|
||||||
* RedisModule_FreeClusterNodesList().
|
* RedisModule_FreeClusterNodesList().
|
||||||
@ -6074,7 +6074,7 @@ int RM_SendClusterMessage(RedisModuleCtx *ctx, char *target_id, uint8_t type, un
|
|||||||
* size_t count, j;
|
* size_t count, j;
|
||||||
* char **ids = RedisModule_GetClusterNodesList(ctx,&count);
|
* char **ids = RedisModule_GetClusterNodesList(ctx,&count);
|
||||||
* for (j = 0; j < count; j++) {
|
* for (j = 0; j < count; j++) {
|
||||||
* RedisModule_Log("notice","Node %.*s",
|
* RedisModule_Log(ctx,"notice","Node %.*s",
|
||||||
* REDISMODULE_NODE_ID_LEN,ids[j]);
|
* REDISMODULE_NODE_ID_LEN,ids[j]);
|
||||||
* }
|
* }
|
||||||
* RedisModule_FreeClusterNodesList(ids);
|
* RedisModule_FreeClusterNodesList(ids);
|
||||||
@ -6568,7 +6568,7 @@ int RM_AuthenticateClientWithACLUser(RedisModuleCtx *ctx, const char *name, size
|
|||||||
|
|
||||||
/* Deauthenticate and close the client. The client resources will not be
|
/* Deauthenticate and close the client. The client resources will not be
|
||||||
* be immediately freed, but will be cleaned up in a background job. This is
|
* be immediately freed, but will be cleaned up in a background job. This is
|
||||||
* the recommended way to deauthenicate a client since most clients can't
|
* the recommended way to deauthenticate a client since most clients can't
|
||||||
* handle users becoming deauthenticated. Returns REDISMODULE_ERR when the
|
* handle users becoming deauthenticated. Returns REDISMODULE_ERR when the
|
||||||
* client doesn't exist and REDISMODULE_OK when the operation was successful.
|
* client doesn't exist and REDISMODULE_OK when the operation was successful.
|
||||||
*
|
*
|
||||||
@ -6598,7 +6598,7 @@ int RM_DeauthenticateAndCloseClient(RedisModuleCtx *ctx, uint64_t client_id) {
|
|||||||
*
|
*
|
||||||
* - Connection ID does not exist
|
* - Connection ID does not exist
|
||||||
* - Connection is not a TLS connection
|
* - Connection is not a TLS connection
|
||||||
* - Connection is a TLS connection but no client ceritifcate was used
|
* - Connection is a TLS connection but no client certificate was used
|
||||||
*/
|
*/
|
||||||
RedisModuleString *RM_GetClientCertificate(RedisModuleCtx *ctx, uint64_t client_id) {
|
RedisModuleString *RM_GetClientCertificate(RedisModuleCtx *ctx, uint64_t client_id) {
|
||||||
client *c = lookupClientByID(client_id);
|
client *c = lookupClientByID(client_id);
|
||||||
@ -6698,7 +6698,7 @@ void *RM_DictGet(RedisModuleDict *d, RedisModuleString *key, int *nokey) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Remove the specified key from the dictionary, returning REDISMODULE_OK if
|
/* Remove the specified key from the dictionary, returning REDISMODULE_OK if
|
||||||
* the key was found and delted, or REDISMODULE_ERR if instead there was
|
* the key was found and deleted, or REDISMODULE_ERR if instead there was
|
||||||
* no such key in the dictionary. When the operation is successful, if
|
* no such key in the dictionary. When the operation is successful, if
|
||||||
* 'oldval' is not NULL, then '*oldval' is set to the value stored at the
|
* 'oldval' is not NULL, then '*oldval' is set to the value stored at the
|
||||||
* key before it was deleted. Using this feature it is possible to get
|
* key before it was deleted. Using this feature it is possible to get
|
||||||
@ -6720,7 +6720,7 @@ int RM_DictDel(RedisModuleDict *d, RedisModuleString *key, void *oldval) {
|
|||||||
* operators available are:
|
* operators available are:
|
||||||
*
|
*
|
||||||
* * `^` -- Seek the first (lexicographically smaller) key.
|
* * `^` -- Seek the first (lexicographically smaller) key.
|
||||||
* * `$` -- Seek the last (lexicographically biffer) key.
|
* * `$` -- Seek the last (lexicographically bigger) key.
|
||||||
* * `>` -- Seek the first element greater than the specified key.
|
* * `>` -- Seek the first element greater than the specified key.
|
||||||
* * `>=` -- Seek the first element greater or equal than the specified key.
|
* * `>=` -- Seek the first element greater or equal than the specified key.
|
||||||
* * `<` -- Seek the first element smaller than the specified key.
|
* * `<` -- Seek the first element smaller than the specified key.
|
||||||
@ -6806,7 +6806,7 @@ void *RM_DictNextC(RedisModuleDictIter *di, size_t *keylen, void **dataptr) {
|
|||||||
|
|
||||||
/* This function is exactly like RedisModule_DictNext() but after returning
|
/* This function is exactly like RedisModule_DictNext() but after returning
|
||||||
* the currently selected element in the iterator, it selects the previous
|
* the currently selected element in the iterator, it selects the previous
|
||||||
* element (laxicographically smaller) instead of the next one. */
|
* element (lexicographically smaller) instead of the next one. */
|
||||||
void *RM_DictPrevC(RedisModuleDictIter *di, size_t *keylen, void **dataptr) {
|
void *RM_DictPrevC(RedisModuleDictIter *di, size_t *keylen, void **dataptr) {
|
||||||
if (!raxPrev(&di->ri)) return NULL;
|
if (!raxPrev(&di->ri)) return NULL;
|
||||||
if (keylen) *keylen = di->ri.key_len;
|
if (keylen) *keylen = di->ri.key_len;
|
||||||
@ -6829,7 +6829,7 @@ RedisModuleString *RM_DictNext(RedisModuleCtx *ctx, RedisModuleDictIter *di, voi
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Like RedisModule_DictNext() but after returning the currently selected
|
/* Like RedisModule_DictNext() but after returning the currently selected
|
||||||
* element in the iterator, it selects the previous element (laxicographically
|
* element in the iterator, it selects the previous element (lexicographically
|
||||||
* smaller) instead of the next one. */
|
* smaller) instead of the next one. */
|
||||||
RedisModuleString *RM_DictPrev(RedisModuleCtx *ctx, RedisModuleDictIter *di, void **dataptr) {
|
RedisModuleString *RM_DictPrev(RedisModuleCtx *ctx, RedisModuleDictIter *di, void **dataptr) {
|
||||||
size_t keylen;
|
size_t keylen;
|
||||||
@ -6841,7 +6841,7 @@ RedisModuleString *RM_DictPrev(RedisModuleCtx *ctx, RedisModuleDictIter *di, voi
|
|||||||
/* Compare the element currently pointed by the iterator to the specified
|
/* Compare the element currently pointed by the iterator to the specified
|
||||||
* element given by key/keylen, according to the operator 'op' (the set of
|
* element given by key/keylen, according to the operator 'op' (the set of
|
||||||
* valid operators are the same valid for RedisModule_DictIteratorStart).
|
* valid operators are the same valid for RedisModule_DictIteratorStart).
|
||||||
* If the comparision is successful the command returns REDISMODULE_OK
|
* If the comparison is successful the command returns REDISMODULE_OK
|
||||||
* otherwise REDISMODULE_ERR is returned.
|
* otherwise REDISMODULE_ERR is returned.
|
||||||
*
|
*
|
||||||
* This is useful when we want to just emit a lexicographical range, so
|
* This is useful when we want to just emit a lexicographical range, so
|
||||||
@ -7614,7 +7614,7 @@ void RM_ScanCursorDestroy(RedisModuleScanCursor *cursor) {
|
|||||||
* RedisModule_ScanCursorDestroy(c);
|
* RedisModule_ScanCursorDestroy(c);
|
||||||
*
|
*
|
||||||
* It is also possible to use this API from another thread while the lock
|
* It is also possible to use this API from another thread while the lock
|
||||||
* is acquired during the actuall call to RM_Scan:
|
* is acquired during the actual call to RM_Scan:
|
||||||
*
|
*
|
||||||
* RedisModuleCursor *c = RedisModule_ScanCursorCreate();
|
* RedisModuleCursor *c = RedisModule_ScanCursorCreate();
|
||||||
* RedisModule_ThreadSafeContextLock(ctx);
|
* RedisModule_ThreadSafeContextLock(ctx);
|
||||||
@ -7711,7 +7711,7 @@ static void moduleScanKeyCallback(void *privdata, const dictEntry *de) {
|
|||||||
* RedisModule_ScanCursorDestroy(c);
|
* RedisModule_ScanCursorDestroy(c);
|
||||||
*
|
*
|
||||||
* It is also possible to use this API from another thread while the lock is acquired during
|
* It is also possible to use this API from another thread while the lock is acquired during
|
||||||
* the actuall call to RM_ScanKey, and re-opening the key each time:
|
* the actual call to RM_ScanKey, and re-opening the key each time:
|
||||||
*
|
*
|
||||||
* RedisModuleCursor *c = RedisModule_ScanCursorCreate();
|
* RedisModuleCursor *c = RedisModule_ScanCursorCreate();
|
||||||
* RedisModule_ThreadSafeContextLock(ctx);
|
* RedisModule_ThreadSafeContextLock(ctx);
|
||||||
@ -7813,7 +7813,7 @@ int RM_ScanKey(RedisModuleKey *key, RedisModuleScanCursor *cursor, RedisModuleSc
|
|||||||
* ## Module fork API
|
* ## Module fork API
|
||||||
* -------------------------------------------------------------------------- */
|
* -------------------------------------------------------------------------- */
|
||||||
|
|
||||||
/* Create a background child process with the current frozen snaphost of the
|
/* Create a background child process with the current frozen snapshot of the
|
||||||
* main process where you can do some processing in the background without
|
* main process where you can do some processing in the background without
|
||||||
* affecting / freezing the traffic and no need for threads and GIL locking.
|
* affecting / freezing the traffic and no need for threads and GIL locking.
|
||||||
* Note that Redis allows for only one concurrent fork.
|
* Note that Redis allows for only one concurrent fork.
|
||||||
@ -7973,7 +7973,7 @@ void ModuleForkDoneHandler(int exitcode, int bysignal) {
|
|||||||
* The above events are triggered not just when the user calls the
|
* The above events are triggered not just when the user calls the
|
||||||
* relevant commands like BGSAVE, but also when a saving operation
|
* relevant commands like BGSAVE, but also when a saving operation
|
||||||
* or AOF rewriting occurs because of internal server triggers.
|
* or AOF rewriting occurs because of internal server triggers.
|
||||||
* The SYNC_RDB_START sub events are happening in the forground due to
|
* The SYNC_RDB_START sub events are happening in the foreground due to
|
||||||
* SAVE command, FLUSHALL, or server shutdown, and the other RDB and
|
* SAVE command, FLUSHALL, or server shutdown, and the other RDB and
|
||||||
* AOF sub events are executed in a background fork child, so any
|
* AOF sub events are executed in a background fork child, so any
|
||||||
* action the module takes can only affect the generated AOF or RDB,
|
* action the module takes can only affect the generated AOF or RDB,
|
||||||
@ -8001,7 +8001,7 @@ void ModuleForkDoneHandler(int exitcode, int bysignal) {
|
|||||||
* int32_t dbnum; // Flushed database number, -1 for all the DBs
|
* int32_t dbnum; // Flushed database number, -1 for all the DBs
|
||||||
* // in the case of the FLUSHALL operation.
|
* // in the case of the FLUSHALL operation.
|
||||||
*
|
*
|
||||||
* The start event is called *before* the operation is initated, thus
|
* The start event is called *before* the operation is initiated, thus
|
||||||
* allowing the callback to call DBSIZE or other operation on the
|
* allowing the callback to call DBSIZE or other operation on the
|
||||||
* yet-to-free keyspace.
|
* yet-to-free keyspace.
|
||||||
*
|
*
|
||||||
@ -8093,7 +8093,7 @@ void ModuleForkDoneHandler(int exitcode, int bysignal) {
|
|||||||
*
|
*
|
||||||
* This event is called repeatedly called while an RDB or AOF file
|
* This event is called repeatedly called while an RDB or AOF file
|
||||||
* is being loaded.
|
* is being loaded.
|
||||||
* The following sub events are availble:
|
* The following sub events are available:
|
||||||
*
|
*
|
||||||
* * `REDISMODULE_SUBEVENT_LOADING_PROGRESS_RDB`
|
* * `REDISMODULE_SUBEVENT_LOADING_PROGRESS_RDB`
|
||||||
* * `REDISMODULE_SUBEVENT_LOADING_PROGRESS_AOF`
|
* * `REDISMODULE_SUBEVENT_LOADING_PROGRESS_AOF`
|
||||||
@ -8219,7 +8219,7 @@ int RM_IsSubEventSupported(RedisModuleEvent event, int64_t subevent) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* This is called by the Redis internals every time we want to fire an
|
/* This is called by the Redis internals every time we want to fire an
|
||||||
* event that can be interceppted by some module. The pointer 'data' is useful
|
* event that can be intercepted by some module. The pointer 'data' is useful
|
||||||
* in order to populate the event-specific structure when needed, in order
|
* in order to populate the event-specific structure when needed, in order
|
||||||
* to return the structure with more information to the callback.
|
* to return the structure with more information to the callback.
|
||||||
*
|
*
|
||||||
@ -8963,11 +8963,11 @@ int RM_ModuleTypeReplaceValue(RedisModuleKey *key, moduleType *mt, void *new_val
|
|||||||
|
|
||||||
/* For a specified command, parse its arguments and return an array that
|
/* For a specified command, parse its arguments and return an array that
|
||||||
* contains the indexes of all key name arguments. This function is
|
* contains the indexes of all key name arguments. This function is
|
||||||
* essnetially a more efficient way to do COMMAND GETKEYS.
|
* essentially a more efficient way to do COMMAND GETKEYS.
|
||||||
*
|
*
|
||||||
* A NULL return value indicates the specified command has no keys, or
|
* A NULL return value indicates the specified command has no keys, or
|
||||||
* an error condition. Error conditions are indicated by setting errno
|
* an error condition. Error conditions are indicated by setting errno
|
||||||
* as folllows:
|
* as follows:
|
||||||
*
|
*
|
||||||
* * ENOENT: Specified command does not exist.
|
* * ENOENT: Specified command does not exist.
|
||||||
* * EINVAL: Invalid command arity specified.
|
* * EINVAL: Invalid command arity specified.
|
||||||
|
@ -88,7 +88,7 @@ int cmd_KEYRANGE(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
|
|||||||
/* Reply with the matching items. */
|
/* Reply with the matching items. */
|
||||||
char *key;
|
char *key;
|
||||||
size_t keylen;
|
size_t keylen;
|
||||||
long long replylen = 0; /* Keep track of the amitted array len. */
|
long long replylen = 0; /* Keep track of the emitted array len. */
|
||||||
RedisModule_ReplyWithArray(ctx,REDISMODULE_POSTPONED_ARRAY_LEN);
|
RedisModule_ReplyWithArray(ctx,REDISMODULE_POSTPONED_ARRAY_LEN);
|
||||||
while((key = RedisModule_DictNextC(iter,&keylen,NULL)) != NULL) {
|
while((key = RedisModule_DictNextC(iter,&keylen,NULL)) != NULL) {
|
||||||
if (replylen >= count) break;
|
if (replylen >= count) break;
|
||||||
|
@ -229,7 +229,7 @@ void HelloBlock_FreeData(RedisModuleCtx *ctx, void *privdata) {
|
|||||||
RedisModule_Free(privdata);
|
RedisModule_Free(privdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* HELLOTYPE.BRANGE key first count timeout -- This is a blocking verison of
|
/* HELLOTYPE.BRANGE key first count timeout -- This is a blocking version of
|
||||||
* the RANGE operation, in order to show how to use the API
|
* the RANGE operation, in order to show how to use the API
|
||||||
* RedisModule_BlockClientOnKeys(). */
|
* RedisModule_BlockClientOnKeys(). */
|
||||||
int HelloTypeBRange_RedisCommand(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
|
int HelloTypeBRange_RedisCommand(RedisModuleCtx *ctx, RedisModuleString **argv, int argc) {
|
||||||
|
@ -302,7 +302,7 @@ void _addReplyProtoToList(client *c, const char *s, size_t len) {
|
|||||||
|
|
||||||
/* Note that 'tail' may be NULL even if we have a tail node, because when
|
/* Note that 'tail' may be NULL even if we have a tail node, because when
|
||||||
* addReplyDeferredLen() is used, it sets a dummy node to NULL just
|
* addReplyDeferredLen() is used, it sets a dummy node to NULL just
|
||||||
* fo fill it later, when the size of the bulk length is set. */
|
* to fill it later, when the size of the bulk length is set. */
|
||||||
|
|
||||||
/* Append to tail string when possible. */
|
/* Append to tail string when possible. */
|
||||||
if (tail) {
|
if (tail) {
|
||||||
@ -1752,7 +1752,7 @@ int processInlineBuffer(client *c) {
|
|||||||
/* Masters should never send us inline protocol to run actual
|
/* Masters should never send us inline protocol to run actual
|
||||||
* commands. If this happens, it is likely due to a bug in Redis where
|
* commands. If this happens, it is likely due to a bug in Redis where
|
||||||
* we got some desynchronization in the protocol, for example
|
* we got some desynchronization in the protocol, for example
|
||||||
* beause of a PSYNC gone bad.
|
* because of a PSYNC gone bad.
|
||||||
*
|
*
|
||||||
* However the is an exception: masters may send us just a newline
|
* However the is an exception: masters may send us just a newline
|
||||||
* to keep the connection active. */
|
* to keep the connection active. */
|
||||||
@ -1783,7 +1783,7 @@ int processInlineBuffer(client *c) {
|
|||||||
return C_OK;
|
return C_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Helper function. Record protocol erro details in server log,
|
/* Helper function. Record protocol error details in server log,
|
||||||
* and set the client as CLIENT_CLOSE_AFTER_REPLY and
|
* and set the client as CLIENT_CLOSE_AFTER_REPLY and
|
||||||
* CLIENT_PROTOCOL_ERROR. */
|
* CLIENT_PROTOCOL_ERROR. */
|
||||||
#define PROTO_DUMP_LEN 128
|
#define PROTO_DUMP_LEN 128
|
||||||
@ -2367,7 +2367,7 @@ sds getAllClientsInfoString(int type) {
|
|||||||
|
|
||||||
/* This function implements CLIENT SETNAME, including replying to the
|
/* This function implements CLIENT SETNAME, including replying to the
|
||||||
* user with an error if the charset is wrong (in that case C_ERR is
|
* user with an error if the charset is wrong (in that case C_ERR is
|
||||||
* returned). If the function succeeeded C_OK is returned, and it's up
|
* returned). If the function succeeded C_OK is returned, and it's up
|
||||||
* to the caller to send a reply if needed.
|
* to the caller to send a reply if needed.
|
||||||
*
|
*
|
||||||
* Setting an empty string as name has the effect of unsetting the
|
* Setting an empty string as name has the effect of unsetting the
|
||||||
@ -2484,7 +2484,7 @@ void clientCommand(client *c) {
|
|||||||
"UNPAUSE",
|
"UNPAUSE",
|
||||||
" Stop the current client pause, resuming traffic.",
|
" Stop the current client pause, resuming traffic.",
|
||||||
"PAUSE <timeout> [WRITE|ALL]",
|
"PAUSE <timeout> [WRITE|ALL]",
|
||||||
" Suspend all, or just write, clients for <timout> milliseconds.",
|
" Suspend all, or just write, clients for <timeout> milliseconds.",
|
||||||
"REPLY (ON|OFF|SKIP)",
|
"REPLY (ON|OFF|SKIP)",
|
||||||
" Control the replies sent to the current connection.",
|
" Control the replies sent to the current connection.",
|
||||||
"SETNAME <name>",
|
"SETNAME <name>",
|
||||||
@ -3312,7 +3312,7 @@ void flushSlavesOutputBuffers(void) {
|
|||||||
*
|
*
|
||||||
* A main use case of this function is to allow pausing replication traffic
|
* A main use case of this function is to allow pausing replication traffic
|
||||||
* so that a failover without data loss to occur. Replicas will continue to receive
|
* so that a failover without data loss to occur. Replicas will continue to receive
|
||||||
* traffic to faciliate this functionality.
|
* traffic to facilitate this functionality.
|
||||||
*
|
*
|
||||||
* This function is also internally used by Redis Cluster for the manual
|
* This function is also internally used by Redis Cluster for the manual
|
||||||
* failover procedure implemented by CLUSTER FAILOVER.
|
* failover procedure implemented by CLUSTER FAILOVER.
|
||||||
@ -3401,7 +3401,7 @@ void processEventsWhileBlocked(void) {
|
|||||||
AE_FILE_EVENTS|AE_DONT_WAIT|
|
AE_FILE_EVENTS|AE_DONT_WAIT|
|
||||||
AE_CALL_BEFORE_SLEEP|AE_CALL_AFTER_SLEEP);
|
AE_CALL_BEFORE_SLEEP|AE_CALL_AFTER_SLEEP);
|
||||||
/* Note that server.events_processed_while_blocked will also get
|
/* Note that server.events_processed_while_blocked will also get
|
||||||
* incremeted by callbacks called by the event loop handlers. */
|
* incremented by callbacks called by the event loop handlers. */
|
||||||
server.events_processed_while_blocked += ae_events;
|
server.events_processed_while_blocked += ae_events;
|
||||||
long long events = server.events_processed_while_blocked - startval;
|
long long events = server.events_processed_while_blocked - startval;
|
||||||
if (!events) break;
|
if (!events) break;
|
||||||
|
@ -93,8 +93,9 @@ sds keyspaceEventsFlagsToString(int flags) {
|
|||||||
|
|
||||||
/* The API provided to the rest of the Redis core is a simple function:
|
/* The API provided to the rest of the Redis core is a simple function:
|
||||||
*
|
*
|
||||||
* notifyKeyspaceEvent(char *event, robj *key, int dbid);
|
* notifyKeyspaceEvent(int type, char *event, robj *key, int dbid);
|
||||||
*
|
*
|
||||||
|
* 'type' is the notification class we define in `server.h`.
|
||||||
* 'event' is a C string representing the event name.
|
* 'event' is a C string representing the event name.
|
||||||
* 'key' is a Redis object representing the key name.
|
* 'key' is a Redis object representing the key name.
|
||||||
* 'dbid' is the database ID where the key lives. */
|
* 'dbid' is the database ID where the key lives. */
|
||||||
|
@ -762,7 +762,7 @@ char *strEncoding(int encoding) {
|
|||||||
/* =========================== Memory introspection ========================= */
|
/* =========================== Memory introspection ========================= */
|
||||||
|
|
||||||
|
|
||||||
/* This is an helper function with the goal of estimating the memory
|
/* This is a helper function with the goal of estimating the memory
|
||||||
* size of a radix tree that is used to store Stream IDs.
|
* size of a radix tree that is used to store Stream IDs.
|
||||||
*
|
*
|
||||||
* Note: to guess the size of the radix tree is not trivial, so we
|
* Note: to guess the size of the radix tree is not trivial, so we
|
||||||
@ -1208,9 +1208,9 @@ int objectSetLRUOrLFU(robj *val, long long lfu_freq, long long lru_idle,
|
|||||||
* below statement will expand to lru_idle*1000/1000. */
|
* below statement will expand to lru_idle*1000/1000. */
|
||||||
lru_idle = lru_idle*lru_multiplier/LRU_CLOCK_RESOLUTION;
|
lru_idle = lru_idle*lru_multiplier/LRU_CLOCK_RESOLUTION;
|
||||||
long lru_abs = lru_clock - lru_idle; /* Absolute access time. */
|
long lru_abs = lru_clock - lru_idle; /* Absolute access time. */
|
||||||
/* If the LRU field underflows (since LRU it is a wrapping
|
/* If the LRU field underflow (since LRU it is a wrapping
|
||||||
* clock), the best we can do is to provide a large enough LRU
|
* clock), the best we can do is to provide a large enough LRU
|
||||||
* that is half-way in the circlular LRU clock we use: this way
|
* that is half-way in the circular LRU clock we use: this way
|
||||||
* the computed idle time for this object will stay high for quite
|
* the computed idle time for this object will stay high for quite
|
||||||
* some time. */
|
* some time. */
|
||||||
if (lru_abs < 0)
|
if (lru_abs < 0)
|
||||||
|
@ -344,7 +344,7 @@ void subscribeCommand(client *c) {
|
|||||||
* expect a reply per command and so can not execute subscribe.
|
* expect a reply per command and so can not execute subscribe.
|
||||||
*
|
*
|
||||||
* Notice that we have a special treatment for multi because of
|
* Notice that we have a special treatment for multi because of
|
||||||
* backword compatibility
|
* backward compatibility
|
||||||
*/
|
*/
|
||||||
addReplyError(c, "SUBSCRIBE isn't allowed for a DENY BLOCKING client");
|
addReplyError(c, "SUBSCRIBE isn't allowed for a DENY BLOCKING client");
|
||||||
return;
|
return;
|
||||||
@ -377,7 +377,7 @@ void psubscribeCommand(client *c) {
|
|||||||
* expect a reply per command and so can not execute subscribe.
|
* expect a reply per command and so can not execute subscribe.
|
||||||
*
|
*
|
||||||
* Notice that we have a special treatment for multi because of
|
* Notice that we have a special treatment for multi because of
|
||||||
* backword compatibility
|
* backward compatibility
|
||||||
*/
|
*/
|
||||||
addReplyError(c, "PSUBSCRIBE isn't allowed for a DENY BLOCKING client");
|
addReplyError(c, "PSUBSCRIBE isn't allowed for a DENY BLOCKING client");
|
||||||
return;
|
return;
|
||||||
|
20
src/rax.c
20
src/rax.c
@ -182,7 +182,7 @@ static inline void raxStackFree(raxStack *ts) {
|
|||||||
)
|
)
|
||||||
|
|
||||||
/* Allocate a new non compressed node with the specified number of children.
|
/* Allocate a new non compressed node with the specified number of children.
|
||||||
* If datafiled is true, the allocation is made large enough to hold the
|
* If datafield is true, the allocation is made large enough to hold the
|
||||||
* associated data pointer.
|
* associated data pointer.
|
||||||
* Returns the new node pointer. On out of memory NULL is returned. */
|
* Returns the new node pointer. On out of memory NULL is returned. */
|
||||||
raxNode *raxNewNode(size_t children, int datafield) {
|
raxNode *raxNewNode(size_t children, int datafield) {
|
||||||
@ -259,7 +259,7 @@ raxNode *raxAddChild(raxNode *n, unsigned char c, raxNode **childptr, raxNode **
|
|||||||
size_t curlen = raxNodeCurrentLength(n);
|
size_t curlen = raxNodeCurrentLength(n);
|
||||||
n->size++;
|
n->size++;
|
||||||
size_t newlen = raxNodeCurrentLength(n);
|
size_t newlen = raxNodeCurrentLength(n);
|
||||||
n->size--; /* For now restore the orignal size. We'll update it only on
|
n->size--; /* For now restore the original size. We'll update it only on
|
||||||
success at the end. */
|
success at the end. */
|
||||||
|
|
||||||
/* Alloc the new child we will link to 'n'. */
|
/* Alloc the new child we will link to 'n'. */
|
||||||
@ -352,8 +352,8 @@ raxNode *raxAddChild(raxNode *n, unsigned char c, raxNode **childptr, raxNode **
|
|||||||
* we don't need to do anything if there was already some padding to use. In
|
* we don't need to do anything if there was already some padding to use. In
|
||||||
* that case the final destination of the pointers will be the same, however
|
* that case the final destination of the pointers will be the same, however
|
||||||
* in our example there was no pre-existing padding, so we added one byte
|
* in our example there was no pre-existing padding, so we added one byte
|
||||||
* plus thre bytes of padding. After the next memmove() things will look
|
* plus there bytes of padding. After the next memmove() things will look
|
||||||
* like thata:
|
* like that:
|
||||||
*
|
*
|
||||||
* [HDR*][abde][....][Aptr][Bptr][....][Dptr][Eptr]|AUXP|
|
* [HDR*][abde][....][Aptr][Bptr][....][Dptr][Eptr]|AUXP|
|
||||||
*/
|
*/
|
||||||
@ -653,7 +653,7 @@ int raxGenericInsert(rax *rax, unsigned char *s, size_t len, void *data, void **
|
|||||||
* Let $SPLITPOS be the zero-based index at which, in the
|
* Let $SPLITPOS be the zero-based index at which, in the
|
||||||
* compressed node array of characters, we stopped iterating because
|
* compressed node array of characters, we stopped iterating because
|
||||||
* there were no more keys character to match. So in the example of
|
* there were no more keys character to match. So in the example of
|
||||||
* the node "ANNIBALE", addig the string "ANNI", the $SPLITPOS is 4.
|
* the node "ANNIBALE", adding the string "ANNI", the $SPLITPOS is 4.
|
||||||
*
|
*
|
||||||
* 1. Save the current compressed node $NEXT pointer (the pointer to the
|
* 1. Save the current compressed node $NEXT pointer (the pointer to the
|
||||||
* child element, that is always present in compressed nodes).
|
* child element, that is always present in compressed nodes).
|
||||||
@ -666,7 +666,7 @@ int raxGenericInsert(rax *rax, unsigned char *s, size_t len, void *data, void **
|
|||||||
*
|
*
|
||||||
* 3. Trim the current node to contain the first $SPLITPOS characters.
|
* 3. Trim the current node to contain the first $SPLITPOS characters.
|
||||||
* As usually if the new node length is just 1, set iscompr to 0.
|
* As usually if the new node length is just 1, set iscompr to 0.
|
||||||
* Take the iskey / associated value as it was in the orignal node.
|
* Take the iskey / associated value as it was in the original node.
|
||||||
* Fix the parent's reference.
|
* Fix the parent's reference.
|
||||||
*
|
*
|
||||||
* 4. Set the postfix node as the only child pointer of the trimmed
|
* 4. Set the postfix node as the only child pointer of the trimmed
|
||||||
@ -1102,9 +1102,9 @@ int raxRemove(rax *rax, unsigned char *s, size_t len, void **old) {
|
|||||||
* We try to navigate upward till there are other nodes that can be
|
* We try to navigate upward till there are other nodes that can be
|
||||||
* compressed, when we reach the upper node which is not a key and has
|
* compressed, when we reach the upper node which is not a key and has
|
||||||
* a single child, we scan the chain of children to collect the
|
* a single child, we scan the chain of children to collect the
|
||||||
* compressable part of the tree, and replace the current node with the
|
* compressible part of the tree, and replace the current node with the
|
||||||
* new one, fixing the child pointer to reference the first non
|
* new one, fixing the child pointer to reference the first non
|
||||||
* compressable node.
|
* compressible node.
|
||||||
*
|
*
|
||||||
* Example of case "1". A tree stores the keys "FOO" = 1 and
|
* Example of case "1". A tree stores the keys "FOO" = 1 and
|
||||||
* "FOOBAR" = 2:
|
* "FOOBAR" = 2:
|
||||||
@ -1341,7 +1341,7 @@ int raxIteratorNextStep(raxIterator *it, int noup) {
|
|||||||
if (it->node_cb && it->node_cb(&it->node))
|
if (it->node_cb && it->node_cb(&it->node))
|
||||||
memcpy(cp,&it->node,sizeof(it->node));
|
memcpy(cp,&it->node,sizeof(it->node));
|
||||||
/* For "next" step, stop every time we find a key along the
|
/* For "next" step, stop every time we find a key along the
|
||||||
* way, since the key is lexicograhically smaller compared to
|
* way, since the key is lexicographically smaller compared to
|
||||||
* what follows in the sub-children. */
|
* what follows in the sub-children. */
|
||||||
if (it->node->iskey) {
|
if (it->node->iskey) {
|
||||||
it->data = raxGetData(it->node);
|
it->data = raxGetData(it->node);
|
||||||
@ -1409,7 +1409,7 @@ int raxIteratorNextStep(raxIterator *it, int noup) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Seek the greatest key in the subtree at the current node. Return 0 on
|
/* Seek the greatest key in the subtree at the current node. Return 0 on
|
||||||
* out of memory, otherwise 1. This is an helper function for different
|
* out of memory, otherwise 1. This is a helper function for different
|
||||||
* iteration functions below. */
|
* iteration functions below. */
|
||||||
int raxSeekGreatest(raxIterator *it) {
|
int raxSeekGreatest(raxIterator *it) {
|
||||||
while(it->node->size) {
|
while(it->node->size) {
|
||||||
|
10
src/rdb.c
10
src/rdb.c
@ -699,7 +699,7 @@ int rdbLoadObjectType(rio *rdb) {
|
|||||||
|
|
||||||
/* This helper function serializes a consumer group Pending Entries List (PEL)
|
/* This helper function serializes a consumer group Pending Entries List (PEL)
|
||||||
* into the RDB file. The 'nacks' argument tells the function if also persist
|
* into the RDB file. The 'nacks' argument tells the function if also persist
|
||||||
* the informations about the not acknowledged message, or if to persist
|
* the information about the not acknowledged message, or if to persist
|
||||||
* just the IDs: this is useful because for the global consumer group PEL
|
* just the IDs: this is useful because for the global consumer group PEL
|
||||||
* we serialized the NACKs as well, but when serializing the local consumer
|
* we serialized the NACKs as well, but when serializing the local consumer
|
||||||
* PELs we just add the ID, that will be resolved inside the global PEL to
|
* PELs we just add the ID, that will be resolved inside the global PEL to
|
||||||
@ -1452,7 +1452,7 @@ void rdbRemoveTempFile(pid_t childpid, int from_signal) {
|
|||||||
char tmpfile[256];
|
char tmpfile[256];
|
||||||
char pid[32];
|
char pid[32];
|
||||||
|
|
||||||
/* Generate temp rdb file name using aync-signal safe functions. */
|
/* Generate temp rdb file name using async-signal safe functions. */
|
||||||
int pid_len = ll2string(pid, sizeof(pid), childpid);
|
int pid_len = ll2string(pid, sizeof(pid), childpid);
|
||||||
strcpy(tmpfile, "temp-");
|
strcpy(tmpfile, "temp-");
|
||||||
strncpy(tmpfile+5, pid, pid_len);
|
strncpy(tmpfile+5, pid, pid_len);
|
||||||
@ -1614,7 +1614,7 @@ robj *rdbLoadObject(int rdbtype, rio *rdb, sds key) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (rdbtype == RDB_TYPE_ZSET_2 || rdbtype == RDB_TYPE_ZSET) {
|
} else if (rdbtype == RDB_TYPE_ZSET_2 || rdbtype == RDB_TYPE_ZSET) {
|
||||||
/* Read list/set value. */
|
/* Read sorted set value. */
|
||||||
uint64_t zsetlen;
|
uint64_t zsetlen;
|
||||||
size_t maxelelen = 0;
|
size_t maxelelen = 0;
|
||||||
zset *zs;
|
zset *zs;
|
||||||
@ -2625,7 +2625,7 @@ eoferr:
|
|||||||
* to do the actual loading. Moreover the ETA displayed in the INFO
|
* to do the actual loading. Moreover the ETA displayed in the INFO
|
||||||
* output is initialized and finalized.
|
* output is initialized and finalized.
|
||||||
*
|
*
|
||||||
* If you pass an 'rsi' structure initialied with RDB_SAVE_OPTION_INIT, the
|
* If you pass an 'rsi' structure initialized with RDB_SAVE_OPTION_INIT, the
|
||||||
* loading code will fill the information fields in the structure. */
|
* loading code will fill the information fields in the structure. */
|
||||||
int rdbLoad(char *filename, rdbSaveInfo *rsi, int rdbflags) {
|
int rdbLoad(char *filename, rdbSaveInfo *rsi, int rdbflags) {
|
||||||
FILE *fp;
|
FILE *fp;
|
||||||
@ -2901,7 +2901,7 @@ void bgsaveCommand(client *c) {
|
|||||||
* information inside the RDB file. Currently the structure explicitly
|
* information inside the RDB file. Currently the structure explicitly
|
||||||
* contains just the currently selected DB from the master stream, however
|
* contains just the currently selected DB from the master stream, however
|
||||||
* if the rdbSave*() family functions receive a NULL rsi structure also
|
* if the rdbSave*() family functions receive a NULL rsi structure also
|
||||||
* the Replication ID/offset is not saved. The function popultes 'rsi'
|
* the Replication ID/offset is not saved. The function populates 'rsi'
|
||||||
* that is normally stack-allocated in the caller, returns the populated
|
* that is normally stack-allocated in the caller, returns the populated
|
||||||
* pointer if the instance has a valid master client, otherwise NULL
|
* pointer if the instance has a valid master client, otherwise NULL
|
||||||
* is returned, and the RDB saving will not persist any replication related
|
* is returned, and the RDB saving will not persist any replication related
|
||||||
|
@ -1466,7 +1466,7 @@ int parseOptions(int argc, const char **argv) {
|
|||||||
config.idlemode = 1;
|
config.idlemode = 1;
|
||||||
} else if (!strcmp(argv[i],"-e")) {
|
} else if (!strcmp(argv[i],"-e")) {
|
||||||
printf("WARNING: -e option has been deprecated. "
|
printf("WARNING: -e option has been deprecated. "
|
||||||
"We now immediatly exit on error to avoid false results.\n");
|
"We now immediately exit on error to avoid false results.\n");
|
||||||
} else if (!strcmp(argv[i],"-t")) {
|
} else if (!strcmp(argv[i],"-t")) {
|
||||||
if (lastarg) goto invalid;
|
if (lastarg) goto invalid;
|
||||||
/* We get the list of tests to run as a string in the form
|
/* We get the list of tests to run as a string in the form
|
||||||
@ -1586,11 +1586,11 @@ usage:
|
|||||||
" --insecure Allow insecure TLS connection by skipping cert validation.\n"
|
" --insecure Allow insecure TLS connection by skipping cert validation.\n"
|
||||||
" --cert <file> Client certificate to authenticate with.\n"
|
" --cert <file> Client certificate to authenticate with.\n"
|
||||||
" --key <file> Private key file to authenticate with.\n"
|
" --key <file> Private key file to authenticate with.\n"
|
||||||
" --tls-ciphers <list> Sets the list of prefered ciphers (TLSv1.2 and below)\n"
|
" --tls-ciphers <list> Sets the list of preferred ciphers (TLSv1.2 and below)\n"
|
||||||
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
||||||
" See the ciphers(1ssl) manpage for more information about the syntax of this string.\n"
|
" See the ciphers(1ssl) manpage for more information about the syntax of this string.\n"
|
||||||
#ifdef TLS1_3_VERSION
|
#ifdef TLS1_3_VERSION
|
||||||
" --tls-ciphersuites <list> Sets the list of prefered ciphersuites (TLSv1.3)\n"
|
" --tls-ciphersuites <list> Sets the list of preferred ciphersuites (TLSv1.3)\n"
|
||||||
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
||||||
" See the ciphers(1ssl) manpage for more information about the syntax of this string,\n"
|
" See the ciphers(1ssl) manpage for more information about the syntax of this string,\n"
|
||||||
" and specifically for TLSv1.3 ciphersuites.\n"
|
" and specifically for TLSv1.3 ciphersuites.\n"
|
||||||
|
@ -128,7 +128,7 @@ void rdbCheckError(const char *fmt, ...) {
|
|||||||
rdbShowGenericInfo();
|
rdbShowGenericInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Print informations during RDB checking. */
|
/* Print information during RDB checking. */
|
||||||
void rdbCheckInfo(const char *fmt, ...) {
|
void rdbCheckInfo(const char *fmt, ...) {
|
||||||
char msg[1024];
|
char msg[1024];
|
||||||
va_list ap;
|
va_list ap;
|
||||||
@ -265,7 +265,7 @@ int redis_check_rdb(char *rdbfilename, FILE *fp) {
|
|||||||
} else if (type == RDB_OPCODE_AUX) {
|
} else if (type == RDB_OPCODE_AUX) {
|
||||||
/* AUX: generic string-string fields. Use to add state to RDB
|
/* AUX: generic string-string fields. Use to add state to RDB
|
||||||
* which is backward compatible. Implementations of RDB loading
|
* which is backward compatible. Implementations of RDB loading
|
||||||
* are requierd to skip AUX fields they don't understand.
|
* are required to skip AUX fields they don't understand.
|
||||||
*
|
*
|
||||||
* An AUX field is composed of two strings: key and value. */
|
* An AUX field is composed of two strings: key and value. */
|
||||||
robj *auxkey, *auxval;
|
robj *auxkey, *auxval;
|
||||||
|
@ -1893,11 +1893,11 @@ static void usage(void) {
|
|||||||
" --insecure Allow insecure TLS connection by skipping cert validation.\n"
|
" --insecure Allow insecure TLS connection by skipping cert validation.\n"
|
||||||
" --cert <file> Client certificate to authenticate with.\n"
|
" --cert <file> Client certificate to authenticate with.\n"
|
||||||
" --key <file> Private key file to authenticate with.\n"
|
" --key <file> Private key file to authenticate with.\n"
|
||||||
" --tls-ciphers <list> Sets the list of prefered ciphers (TLSv1.2 and below)\n"
|
" --tls-ciphers <list> Sets the list of preferred ciphers (TLSv1.2 and below)\n"
|
||||||
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
||||||
" See the ciphers(1ssl) manpage for more information about the syntax of this string.\n"
|
" See the ciphers(1ssl) manpage for more information about the syntax of this string.\n"
|
||||||
#ifdef TLS1_3_VERSION
|
#ifdef TLS1_3_VERSION
|
||||||
" --tls-ciphersuites <list> Sets the list of prefered ciphersuites (TLSv1.3)\n"
|
" --tls-ciphersuites <list> Sets the list of preferred ciphersuites (TLSv1.3)\n"
|
||||||
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
" in order of preference from highest to lowest separated by colon (\":\").\n"
|
||||||
" See the ciphers(1ssl) manpage for more information about the syntax of this string,\n"
|
" See the ciphers(1ssl) manpage for more information about the syntax of this string,\n"
|
||||||
" and specifically for TLSv1.3 ciphersuites.\n"
|
" and specifically for TLSv1.3 ciphersuites.\n"
|
||||||
@ -1909,7 +1909,7 @@ static void usage(void) {
|
|||||||
" --quoted-input Force input to be handled as quoted strings.\n"
|
" --quoted-input Force input to be handled as quoted strings.\n"
|
||||||
" --csv Output in CSV format.\n"
|
" --csv Output in CSV format.\n"
|
||||||
" --show-pushes <yn> Whether to print RESP3 PUSH messages. Enabled by default when\n"
|
" --show-pushes <yn> Whether to print RESP3 PUSH messages. Enabled by default when\n"
|
||||||
" STDOUT is a tty but can be overriden with --show-pushes no.\n"
|
" STDOUT is a tty but can be overridden with --show-pushes no.\n"
|
||||||
" --stat Print rolling stats about server: mem, clients, ...\n"
|
" --stat Print rolling stats about server: mem, clients, ...\n"
|
||||||
" --latency Enter a special mode continuously sampling latency.\n"
|
" --latency Enter a special mode continuously sampling latency.\n"
|
||||||
" If you use this mode in an interactive session it runs\n"
|
" If you use this mode in an interactive session it runs\n"
|
||||||
@ -2992,7 +2992,7 @@ result:
|
|||||||
* So a greater score means a worse anti-affinity level, while zero
|
* So a greater score means a worse anti-affinity level, while zero
|
||||||
* means perfect anti-affinity.
|
* means perfect anti-affinity.
|
||||||
*
|
*
|
||||||
* The anti affinity optimizator will try to get a score as low as
|
* The anti affinity optimization will try to get a score as low as
|
||||||
* possible. Since we do not want to sacrifice the fact that slaves should
|
* possible. Since we do not want to sacrifice the fact that slaves should
|
||||||
* not be in the same host as the master, we assign 10000 times the score
|
* not be in the same host as the master, we assign 10000 times the score
|
||||||
* to this violation, so that we'll optimize for the second factor only
|
* to this violation, so that we'll optimize for the second factor only
|
||||||
@ -8183,7 +8183,7 @@ static void LRUTestMode(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Intrisic latency mode.
|
* Intrinsic latency mode.
|
||||||
*
|
*
|
||||||
* Measure max latency of a running process that does not result from
|
* Measure max latency of a running process that does not result from
|
||||||
* syscalls. Basically this software should provide a hint about how much
|
* syscalls. Basically this software should provide a hint about how much
|
||||||
|
@ -789,7 +789,7 @@ void syncCommand(client *c) {
|
|||||||
|
|
||||||
/* Increment stats for failed PSYNCs, but only if the
|
/* Increment stats for failed PSYNCs, but only if the
|
||||||
* replid is not "?", as this is used by slaves to force a full
|
* replid is not "?", as this is used by slaves to force a full
|
||||||
* resync on purpose when they are not albe to partially
|
* resync on purpose when they are not able to partially
|
||||||
* resync. */
|
* resync. */
|
||||||
if (master_replid[0] != '?') server.stat_sync_partial_err++;
|
if (master_replid[0] != '?') server.stat_sync_partial_err++;
|
||||||
}
|
}
|
||||||
@ -870,7 +870,7 @@ void syncCommand(client *c) {
|
|||||||
* in order to synchronize. */
|
* in order to synchronize. */
|
||||||
serverLog(LL_NOTICE,"Current BGSAVE has socket target. Waiting for next BGSAVE for SYNC");
|
serverLog(LL_NOTICE,"Current BGSAVE has socket target. Waiting for next BGSAVE for SYNC");
|
||||||
|
|
||||||
/* CASE 3: There is no BGSAVE is progress. */
|
/* CASE 3: There is no BGSAVE is in progress. */
|
||||||
} else {
|
} else {
|
||||||
if (server.repl_diskless_sync && (c->slave_capa & SLAVE_CAPA_EOF) &&
|
if (server.repl_diskless_sync && (c->slave_capa & SLAVE_CAPA_EOF) &&
|
||||||
server.repl_diskless_sync_delay)
|
server.repl_diskless_sync_delay)
|
||||||
@ -1234,7 +1234,7 @@ void rdbPipeReadHandler(struct aeEventLoop *eventLoop, int fd, void *clientData,
|
|||||||
}
|
}
|
||||||
serverLog(LL_WARNING,"Diskless rdb transfer, done reading from pipe, %d replicas still up.", stillUp);
|
serverLog(LL_WARNING,"Diskless rdb transfer, done reading from pipe, %d replicas still up.", stillUp);
|
||||||
/* Now that the replicas have finished reading, notify the child that it's safe to exit.
|
/* Now that the replicas have finished reading, notify the child that it's safe to exit.
|
||||||
* When the server detectes the child has exited, it can mark the replica as online, and
|
* When the server detects the child has exited, it can mark the replica as online, and
|
||||||
* start streaming the replication buffers. */
|
* start streaming the replication buffers. */
|
||||||
close(server.rdb_child_exit_pipe);
|
close(server.rdb_child_exit_pipe);
|
||||||
server.rdb_child_exit_pipe = -1;
|
server.rdb_child_exit_pipe = -1;
|
||||||
@ -1336,7 +1336,7 @@ void updateSlavesWaitingBgsave(int bgsaveerr, int type) {
|
|||||||
*
|
*
|
||||||
* So things work like that:
|
* So things work like that:
|
||||||
*
|
*
|
||||||
* 1. We end trasnferring the RDB file via socket.
|
* 1. We end transferring the RDB file via socket.
|
||||||
* 2. The replica is put ONLINE but the write handler
|
* 2. The replica is put ONLINE but the write handler
|
||||||
* is not installed.
|
* is not installed.
|
||||||
* 3. The replica however goes really online, and pings us
|
* 3. The replica however goes really online, and pings us
|
||||||
@ -1351,7 +1351,7 @@ void updateSlavesWaitingBgsave(int bgsaveerr, int type) {
|
|||||||
* in advance). Detecting such final EOF string is much
|
* in advance). Detecting such final EOF string is much
|
||||||
* simpler and less CPU intensive if no more data is sent
|
* simpler and less CPU intensive if no more data is sent
|
||||||
* after such final EOF. So we don't want to glue the end of
|
* after such final EOF. So we don't want to glue the end of
|
||||||
* the RDB trasfer with the start of the other replication
|
* the RDB transfer with the start of the other replication
|
||||||
* data. */
|
* data. */
|
||||||
slave->replstate = SLAVE_STATE_ONLINE;
|
slave->replstate = SLAVE_STATE_ONLINE;
|
||||||
slave->repl_put_online_on_ack = 1;
|
slave->repl_put_online_on_ack = 1;
|
||||||
@ -1717,7 +1717,7 @@ void readSyncBulkPayload(connection *conn) {
|
|||||||
* such case we want just to read the RDB file in memory. */
|
* such case we want just to read the RDB file in memory. */
|
||||||
serverLog(LL_NOTICE, "MASTER <-> REPLICA sync: Flushing old data");
|
serverLog(LL_NOTICE, "MASTER <-> REPLICA sync: Flushing old data");
|
||||||
|
|
||||||
/* We need to stop any AOF rewriting child before flusing and parsing
|
/* We need to stop any AOF rewriting child before flushing and parsing
|
||||||
* the RDB, otherwise we'll create a copy-on-write disaster. */
|
* the RDB, otherwise we'll create a copy-on-write disaster. */
|
||||||
if (server.aof_state != AOF_OFF) stopAppendOnly();
|
if (server.aof_state != AOF_OFF) stopAppendOnly();
|
||||||
|
|
||||||
@ -2408,7 +2408,7 @@ void syncWithMaster(connection *conn) {
|
|||||||
server.repl_state = REPL_STATE_SEND_PSYNC;
|
server.repl_state = REPL_STATE_SEND_PSYNC;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Try a partial resynchonization. If we don't have a cached master
|
/* Try a partial resynchronization. If we don't have a cached master
|
||||||
* slaveTryPartialResynchronization() will at least try to use PSYNC
|
* slaveTryPartialResynchronization() will at least try to use PSYNC
|
||||||
* to start a full resynchronization so that we get the master replid
|
* to start a full resynchronization so that we get the master replid
|
||||||
* and the global offset, to try a partial resync at the next
|
* and the global offset, to try a partial resync at the next
|
||||||
@ -2901,7 +2901,7 @@ void replicationCacheMaster(client *c) {
|
|||||||
unlinkClient(c);
|
unlinkClient(c);
|
||||||
|
|
||||||
/* Reset the master client so that's ready to accept new commands:
|
/* Reset the master client so that's ready to accept new commands:
|
||||||
* we want to discard te non processed query buffers and non processed
|
* we want to discard the non processed query buffers and non processed
|
||||||
* offsets, including pending transactions, already populated arguments,
|
* offsets, including pending transactions, already populated arguments,
|
||||||
* pending outputs to the master. */
|
* pending outputs to the master. */
|
||||||
sdsclear(server.master->querybuf);
|
sdsclear(server.master->querybuf);
|
||||||
@ -2935,13 +2935,13 @@ void replicationCacheMaster(client *c) {
|
|||||||
replicationHandleMasterDisconnection();
|
replicationHandleMasterDisconnection();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This function is called when a master is turend into a slave, in order to
|
/* This function is called when a master is turned into a slave, in order to
|
||||||
* create from scratch a cached master for the new client, that will allow
|
* create from scratch a cached master for the new client, that will allow
|
||||||
* to PSYNC with the slave that was promoted as the new master after a
|
* to PSYNC with the slave that was promoted as the new master after a
|
||||||
* failover.
|
* failover.
|
||||||
*
|
*
|
||||||
* Assuming this instance was previously the master instance of the new master,
|
* Assuming this instance was previously the master instance of the new master,
|
||||||
* the new master will accept its replication ID, and potentiall also the
|
* the new master will accept its replication ID, and potential also the
|
||||||
* current offset if no data was lost during the failover. So we use our
|
* current offset if no data was lost during the failover. So we use our
|
||||||
* current replication ID and offset in order to synthesize a cached master. */
|
* current replication ID and offset in order to synthesize a cached master. */
|
||||||
void replicationCacheMasterUsingMyself(void) {
|
void replicationCacheMasterUsingMyself(void) {
|
||||||
|
@ -310,7 +310,7 @@ static size_t rioFdWrite(rio *r, const void *buf, size_t len) {
|
|||||||
if (!doflush)
|
if (!doflush)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
/* Flusing the buffered data. set 'p' and 'len' accordintly. */
|
/* Flushing the buffered data. set 'p' and 'len' accordingly. */
|
||||||
p = (unsigned char*) r->io.fd.buf;
|
p = (unsigned char*) r->io.fd.buf;
|
||||||
len = sdslen(r->io.fd.buf);
|
len = sdslen(r->io.fd.buf);
|
||||||
}
|
}
|
||||||
|
@ -2275,7 +2275,7 @@ sds ldbCatStackValue(sds s, lua_State *lua, int idx) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Produce a debugger log entry representing the value of the Lua object
|
/* Produce a debugger log entry representing the value of the Lua object
|
||||||
* currently on the top of the stack. The element is ot popped nor modified.
|
* currently on the top of the stack. The element is not popped nor modified.
|
||||||
* Check ldbCatStackValue() for the actual implementation. */
|
* Check ldbCatStackValue() for the actual implementation. */
|
||||||
void ldbLogStackValue(lua_State *lua, char *prefix) {
|
void ldbLogStackValue(lua_State *lua, char *prefix) {
|
||||||
sds s = sdsnew(prefix);
|
sds s = sdsnew(prefix);
|
||||||
|
@ -92,7 +92,7 @@ static inline size_t sdsTypeMaxSize(char type) {
|
|||||||
* If NULL is used for 'init' the string is initialized with zero bytes.
|
* If NULL is used for 'init' the string is initialized with zero bytes.
|
||||||
* If SDS_NOINIT is used, the buffer is left uninitialized;
|
* If SDS_NOINIT is used, the buffer is left uninitialized;
|
||||||
*
|
*
|
||||||
* The string is always null-termined (all the sds strings are, always) so
|
* The string is always null-terminated (all the sds strings are, always) so
|
||||||
* even if you create an sds string with:
|
* even if you create an sds string with:
|
||||||
*
|
*
|
||||||
* mystring = sdsnewlen("abc",3);
|
* mystring = sdsnewlen("abc",3);
|
||||||
@ -469,7 +469,7 @@ sds sdscpylen(sds s, const char *t, size_t len) {
|
|||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Like sdscpylen() but 't' must be a null-termined string so that the length
|
/* Like sdscpylen() but 't' must be a null-terminated string so that the length
|
||||||
* of the string is obtained with strlen(). */
|
* of the string is obtained with strlen(). */
|
||||||
sds sdscpy(sds s, const char *t) {
|
sds sdscpy(sds s, const char *t) {
|
||||||
return sdscpylen(s, t, strlen(t));
|
return sdscpylen(s, t, strlen(t));
|
||||||
@ -731,7 +731,7 @@ sds sdscatfmt(sds s, char const *fmt, ...) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Remove the part of the string from left and from right composed just of
|
/* Remove the part of the string from left and from right composed just of
|
||||||
* contiguous characters found in 'cset', that is a null terminted C string.
|
* contiguous characters found in 'cset', that is a null terminated C string.
|
||||||
*
|
*
|
||||||
* After the call, the modified sds string is no longer valid and all the
|
* After the call, the modified sds string is no longer valid and all the
|
||||||
* references must be substituted with the new pointer returned by the call.
|
* references must be substituted with the new pointer returned by the call.
|
||||||
@ -1179,7 +1179,7 @@ sds sdstemplate(const char *template, sdstemplate_callback_t cb_func, void *cb_a
|
|||||||
res = sdscat(res, p);
|
res = sdscat(res, p);
|
||||||
break;
|
break;
|
||||||
} else if (sv > p) {
|
} else if (sv > p) {
|
||||||
/* Found: copy anything up to the begining of the variable */
|
/* Found: copy anything up to the beginning of the variable */
|
||||||
res = sdscatlen(res, p, sv - p);
|
res = sdscatlen(res, p, sv - p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1422,7 +1422,7 @@ sentinelRedisInstance *sentinelRedisInstanceLookupSlave(
|
|||||||
|
|
||||||
/* We need to handle a slave_addr that is potentially a hostname.
|
/* We need to handle a slave_addr that is potentially a hostname.
|
||||||
* If that is the case, depending on configuration we either resolve
|
* If that is the case, depending on configuration we either resolve
|
||||||
* it and use the IP addres or fail.
|
* it and use the IP address or fail.
|
||||||
*/
|
*/
|
||||||
addr = createSentinelAddr(slave_addr, port);
|
addr = createSentinelAddr(slave_addr, port);
|
||||||
if (!addr) return NULL;
|
if (!addr) return NULL;
|
||||||
@ -3550,7 +3550,7 @@ void sentinelCommand(client *c) {
|
|||||||
"SENTINELS <master-name>",
|
"SENTINELS <master-name>",
|
||||||
" Show a list of Sentinel instances for this master and their state.",
|
" Show a list of Sentinel instances for this master and their state.",
|
||||||
"SET <master-name> <option> <value>",
|
"SET <master-name> <option> <value>",
|
||||||
" Set configuration paramters for certain masters.",
|
" Set configuration parameters for certain masters.",
|
||||||
"SIMULATE-FAILURE (CRASH-AFTER-ELECTION|CRASH-AFTER-PROMOTION|HELP)",
|
"SIMULATE-FAILURE (CRASH-AFTER-ELECTION|CRASH-AFTER-PROMOTION|HELP)",
|
||||||
" Simulate a Sentinel crash.",
|
" Simulate a Sentinel crash.",
|
||||||
NULL
|
NULL
|
||||||
@ -3990,7 +3990,7 @@ void sentinelSetCommand(client *c) {
|
|||||||
int old_j = j; /* Used to know what to log as an event. */
|
int old_j = j; /* Used to know what to log as an event. */
|
||||||
|
|
||||||
if (!strcasecmp(option,"down-after-milliseconds") && moreargs > 0) {
|
if (!strcasecmp(option,"down-after-milliseconds") && moreargs > 0) {
|
||||||
/* down-after-millisecodns <milliseconds> */
|
/* down-after-milliseconds <milliseconds> */
|
||||||
robj *o = c->argv[++j];
|
robj *o = c->argv[++j];
|
||||||
if (getLongLongFromObject(o,&ll) == C_ERR || ll <= 0) {
|
if (getLongLongFromObject(o,&ll) == C_ERR || ll <= 0) {
|
||||||
badarg = j;
|
badarg = j;
|
||||||
|
24
src/server.c
24
src/server.c
@ -1759,7 +1759,7 @@ int clientsCronTrackExpansiveClients(client *c, int time_idx) {
|
|||||||
/* Iterating all the clients in getMemoryOverheadData() is too slow and
|
/* Iterating all the clients in getMemoryOverheadData() is too slow and
|
||||||
* in turn would make the INFO command too slow. So we perform this
|
* in turn would make the INFO command too slow. So we perform this
|
||||||
* computation incrementally and track the (not instantaneous but updated
|
* computation incrementally and track the (not instantaneous but updated
|
||||||
* to the second) total memory used by clients using clinetsCron() in
|
* to the second) total memory used by clients using clientsCron() in
|
||||||
* a more incremental way (depending on server.hz). */
|
* a more incremental way (depending on server.hz). */
|
||||||
int clientsCronTrackClientsMemUsage(client *c) {
|
int clientsCronTrackClientsMemUsage(client *c) {
|
||||||
size_t mem = 0;
|
size_t mem = 0;
|
||||||
@ -2203,7 +2203,7 @@ int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Just for the sake of defensive programming, to avoid forgeting to
|
/* Just for the sake of defensive programming, to avoid forgetting to
|
||||||
* call this function when need. */
|
* call this function when need. */
|
||||||
updateDictResizePolicy();
|
updateDictResizePolicy();
|
||||||
|
|
||||||
@ -2482,7 +2482,7 @@ void afterSleep(struct aeEventLoop *eventLoop) {
|
|||||||
|
|
||||||
/* Do NOT add anything above moduleAcquireGIL !!! */
|
/* Do NOT add anything above moduleAcquireGIL !!! */
|
||||||
|
|
||||||
/* Aquire the modules GIL so that their threads won't touch anything. */
|
/* Acquire the modules GIL so that their threads won't touch anything. */
|
||||||
if (!ProcessingEventsWhileBlocked) {
|
if (!ProcessingEventsWhileBlocked) {
|
||||||
if (moduleCount()) moduleAcquireGIL();
|
if (moduleCount()) moduleAcquireGIL();
|
||||||
}
|
}
|
||||||
@ -2637,7 +2637,7 @@ void createSharedObjects(void) {
|
|||||||
shared.bulkhdr[j] = createObject(OBJ_STRING,
|
shared.bulkhdr[j] = createObject(OBJ_STRING,
|
||||||
sdscatprintf(sdsempty(),"$%d\r\n",j));
|
sdscatprintf(sdsempty(),"$%d\r\n",j));
|
||||||
}
|
}
|
||||||
/* The following two shared objects, minstring and maxstrings, are not
|
/* The following two shared objects, minstring and maxstring, are not
|
||||||
* actually used for their value but as a special object meaning
|
* actually used for their value but as a special object meaning
|
||||||
* respectively the minimum possible string and the maximum possible
|
* respectively the minimum possible string and the maximum possible
|
||||||
* string in string comparisons for the ZRANGEBYLEX command. */
|
* string in string comparisons for the ZRANGEBYLEX command. */
|
||||||
@ -2829,7 +2829,7 @@ int restartServer(int flags, mstime_t delay) {
|
|||||||
return C_ERR;
|
return C_ERR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Close all file descriptors, with the exception of stdin, stdout, strerr
|
/* Close all file descriptors, with the exception of stdin, stdout, stderr
|
||||||
* which are useful if we restart a Redis server which is not daemonized. */
|
* which are useful if we restart a Redis server which is not daemonized. */
|
||||||
for (j = 3; j < (int)server.maxclients + 1024; j++) {
|
for (j = 3; j < (int)server.maxclients + 1024; j++) {
|
||||||
/* Test the descriptor validity before closing it, otherwise
|
/* Test the descriptor validity before closing it, otherwise
|
||||||
@ -3596,7 +3596,7 @@ void propagate(struct redisCommand *cmd, int dbid, robj **argv, int argc,
|
|||||||
execCommandPropagateMulti(dbid);
|
execCommandPropagateMulti(dbid);
|
||||||
|
|
||||||
/* This needs to be unreachable since the dataset should be fixed during
|
/* This needs to be unreachable since the dataset should be fixed during
|
||||||
* client pause, otherwise data may be lossed during a failover. */
|
* client pause, otherwise data may be lost during a failover. */
|
||||||
serverAssert(!(areClientsPaused() && !server.client_pause_in_transaction));
|
serverAssert(!(areClientsPaused() && !server.client_pause_in_transaction));
|
||||||
|
|
||||||
if (server.aof_state != AOF_OFF && flags & PROPAGATE_AOF)
|
if (server.aof_state != AOF_OFF && flags & PROPAGATE_AOF)
|
||||||
@ -3912,7 +3912,7 @@ void call(client *c, int flags) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Used when a command that is ready for execution needs to be rejected, due to
|
/* Used when a command that is ready for execution needs to be rejected, due to
|
||||||
* varios pre-execution checks. it returns the appropriate error to the client.
|
* various pre-execution checks. it returns the appropriate error to the client.
|
||||||
* If there's a transaction is flags it as dirty, and if the command is EXEC,
|
* If there's a transaction is flags it as dirty, and if the command is EXEC,
|
||||||
* it aborts the transaction.
|
* it aborts the transaction.
|
||||||
* Note: 'reply' is expected to end with \r\n */
|
* Note: 'reply' is expected to end with \r\n */
|
||||||
@ -4230,7 +4230,7 @@ int processCommand(client *c) {
|
|||||||
* The main objective here is to prevent abuse of client pause check
|
* The main objective here is to prevent abuse of client pause check
|
||||||
* from which replicas are exempt. */
|
* from which replicas are exempt. */
|
||||||
if ((c->flags & CLIENT_SLAVE) && (is_may_replicate_command || is_write_command || is_read_command)) {
|
if ((c->flags & CLIENT_SLAVE) && (is_may_replicate_command || is_write_command || is_read_command)) {
|
||||||
rejectCommandFormat(c, "Replica can't interract with the keyspace");
|
rejectCommandFormat(c, "Replica can't interact with the keyspace");
|
||||||
return C_OK;
|
return C_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4322,7 +4322,7 @@ int prepareForShutdown(int flags) {
|
|||||||
/* Note that, in killRDBChild normally has backgroundSaveDoneHandler
|
/* Note that, in killRDBChild normally has backgroundSaveDoneHandler
|
||||||
* doing it's cleanup, but in this case this code will not be reached,
|
* doing it's cleanup, but in this case this code will not be reached,
|
||||||
* so we need to call rdbRemoveTempFile which will close fd(in order
|
* so we need to call rdbRemoveTempFile which will close fd(in order
|
||||||
* to unlink file actully) in background thread.
|
* to unlink file actually) in background thread.
|
||||||
* The temp rdb file fd may won't be closed when redis exits quickly,
|
* The temp rdb file fd may won't be closed when redis exits quickly,
|
||||||
* but OS will close this fd when process exits. */
|
* but OS will close this fd when process exits. */
|
||||||
rdbRemoveTempFile(server.child_pid, 0);
|
rdbRemoveTempFile(server.child_pid, 0);
|
||||||
@ -5691,12 +5691,12 @@ int changeBindAddr(sds *addrlist, int addrlist_len) {
|
|||||||
/* Re-Listen TCP and TLS */
|
/* Re-Listen TCP and TLS */
|
||||||
server.ipfd.count = 0;
|
server.ipfd.count = 0;
|
||||||
if (server.port != 0 && listenToPort(server.port, &server.ipfd) != C_OK) {
|
if (server.port != 0 && listenToPort(server.port, &server.ipfd) != C_OK) {
|
||||||
serverPanic("Failed to restore old listening sockets.");
|
serverPanic("Failed to restore old listening TCP socket.");
|
||||||
}
|
}
|
||||||
|
|
||||||
server.tlsfd.count = 0;
|
server.tlsfd.count = 0;
|
||||||
if (server.tls_port != 0 && listenToPort(server.tls_port, &server.tlsfd) != C_OK) {
|
if (server.tls_port != 0 && listenToPort(server.tls_port, &server.tlsfd) != C_OK) {
|
||||||
serverPanic("Failed to restore old listening sockets.");
|
serverPanic("Failed to restore old listening TLS socket.");
|
||||||
}
|
}
|
||||||
|
|
||||||
result = C_ERR;
|
result = C_ERR;
|
||||||
@ -5959,7 +5959,7 @@ void loadDataFromDisk(void) {
|
|||||||
memcpy(server.replid,rsi.repl_id,sizeof(server.replid));
|
memcpy(server.replid,rsi.repl_id,sizeof(server.replid));
|
||||||
server.master_repl_offset = rsi.repl_offset;
|
server.master_repl_offset = rsi.repl_offset;
|
||||||
/* If we are a slave, create a cached master from this
|
/* If we are a slave, create a cached master from this
|
||||||
* information, in order to allow partial resynchronizations
|
* information, in order to allow partial resynchronization
|
||||||
* with masters. */
|
* with masters. */
|
||||||
replicationCacheMasterUsingMyself();
|
replicationCacheMasterUsingMyself();
|
||||||
selectDb(server.cached_master,rsi.repl_stream_db);
|
selectDb(server.cached_master,rsi.repl_stream_db);
|
||||||
|
14
src/server.h
14
src/server.h
@ -314,7 +314,7 @@ typedef enum {
|
|||||||
REPL_STATE_CONNECTING, /* Connecting to master */
|
REPL_STATE_CONNECTING, /* Connecting to master */
|
||||||
/* --- Handshake states, must be ordered --- */
|
/* --- Handshake states, must be ordered --- */
|
||||||
REPL_STATE_RECEIVE_PING_REPLY, /* Wait for PING reply */
|
REPL_STATE_RECEIVE_PING_REPLY, /* Wait for PING reply */
|
||||||
REPL_STATE_SEND_HANDSHAKE, /* Send handshake sequance to master */
|
REPL_STATE_SEND_HANDSHAKE, /* Send handshake sequence to master */
|
||||||
REPL_STATE_RECEIVE_AUTH_REPLY, /* Wait for AUTH reply */
|
REPL_STATE_RECEIVE_AUTH_REPLY, /* Wait for AUTH reply */
|
||||||
REPL_STATE_RECEIVE_PORT_REPLY, /* Wait for REPLCONF reply */
|
REPL_STATE_RECEIVE_PORT_REPLY, /* Wait for REPLCONF reply */
|
||||||
REPL_STATE_RECEIVE_IP_REPLY, /* Wait for REPLCONF reply */
|
REPL_STATE_RECEIVE_IP_REPLY, /* Wait for REPLCONF reply */
|
||||||
@ -963,7 +963,7 @@ typedef struct client {
|
|||||||
/* In clientsCronTrackClientsMemUsage() we track the memory usage of
|
/* In clientsCronTrackClientsMemUsage() we track the memory usage of
|
||||||
* each client and add it to the sum of all the clients of a given type,
|
* each client and add it to the sum of all the clients of a given type,
|
||||||
* however we need to remember what was the old contribution of each
|
* however we need to remember what was the old contribution of each
|
||||||
* client, and in which categoty the client was, in order to remove it
|
* client, and in which category the client was, in order to remove it
|
||||||
* before adding it the new value. */
|
* before adding it the new value. */
|
||||||
uint64_t client_cron_last_memory_usage;
|
uint64_t client_cron_last_memory_usage;
|
||||||
int client_cron_last_memory_type;
|
int client_cron_last_memory_type;
|
||||||
@ -1291,7 +1291,7 @@ struct redisServer {
|
|||||||
long long stat_numconnections; /* Number of connections received */
|
long long stat_numconnections; /* Number of connections received */
|
||||||
long long stat_expiredkeys; /* Number of expired keys */
|
long long stat_expiredkeys; /* Number of expired keys */
|
||||||
double stat_expired_stale_perc; /* Percentage of keys probably expired */
|
double stat_expired_stale_perc; /* Percentage of keys probably expired */
|
||||||
long long stat_expired_time_cap_reached_count; /* Early expire cylce stops.*/
|
long long stat_expired_time_cap_reached_count; /* Early expire cycle stops.*/
|
||||||
long long stat_expire_cycle_time_used; /* Cumulative microseconds used. */
|
long long stat_expire_cycle_time_used; /* Cumulative microseconds used. */
|
||||||
long long stat_evictedkeys; /* Number of evicted keys (maxmemory) */
|
long long stat_evictedkeys; /* Number of evicted keys (maxmemory) */
|
||||||
long long stat_keyspace_hits; /* Number of successful lookups of keys */
|
long long stat_keyspace_hits; /* Number of successful lookups of keys */
|
||||||
@ -1348,7 +1348,7 @@ struct redisServer {
|
|||||||
int active_expire_effort; /* From 1 (default) to 10, active effort. */
|
int active_expire_effort; /* From 1 (default) to 10, active effort. */
|
||||||
int active_defrag_enabled;
|
int active_defrag_enabled;
|
||||||
int sanitize_dump_payload; /* Enables deep sanitization for ziplist and listpack in RDB and RESTORE. */
|
int sanitize_dump_payload; /* Enables deep sanitization for ziplist and listpack in RDB and RESTORE. */
|
||||||
int skip_checksum_validation; /* Disables checksum validateion for RDB and RESTORE payload. */
|
int skip_checksum_validation; /* Disable checksum validation for RDB and RESTORE payload. */
|
||||||
int jemalloc_bg_thread; /* Enable jemalloc background thread */
|
int jemalloc_bg_thread; /* Enable jemalloc background thread */
|
||||||
size_t active_defrag_ignore_bytes; /* minimum amount of fragmentation waste to start active defrag */
|
size_t active_defrag_ignore_bytes; /* minimum amount of fragmentation waste to start active defrag */
|
||||||
int active_defrag_threshold_lower; /* minimum percentage of fragmentation to start active defrag */
|
int active_defrag_threshold_lower; /* minimum percentage of fragmentation to start active defrag */
|
||||||
@ -1433,10 +1433,10 @@ struct redisServer {
|
|||||||
int rdb_pipe_bufflen; /* that was read from the the rdb pipe. */
|
int rdb_pipe_bufflen; /* that was read from the the rdb pipe. */
|
||||||
int rdb_key_save_delay; /* Delay in microseconds between keys while
|
int rdb_key_save_delay; /* Delay in microseconds between keys while
|
||||||
* writing the RDB. (for testings). negative
|
* writing the RDB. (for testings). negative
|
||||||
* value means fractions of microsecons (on average). */
|
* value means fractions of microseconds (on average). */
|
||||||
int key_load_delay; /* Delay in microseconds between keys while
|
int key_load_delay; /* Delay in microseconds between keys while
|
||||||
* loading aof or rdb. (for testings). negative
|
* loading aof or rdb. (for testings). negative
|
||||||
* value means fractions of microsecons (on average). */
|
* value means fractions of microseconds (on average). */
|
||||||
/* Pipe and data structures for child -> parent info sharing. */
|
/* Pipe and data structures for child -> parent info sharing. */
|
||||||
int child_info_pipe[2]; /* Pipe used to write the child_info_data. */
|
int child_info_pipe[2]; /* Pipe used to write the child_info_data. */
|
||||||
int child_info_nread; /* Num of bytes of the last read from pipe */
|
int child_info_nread; /* Num of bytes of the last read from pipe */
|
||||||
@ -1649,7 +1649,7 @@ struct redisServer {
|
|||||||
struct sentinelConfig *sentinel_config; /* sentinel config to load at startup time. */
|
struct sentinelConfig *sentinel_config; /* sentinel config to load at startup time. */
|
||||||
/* Coordinate failover info */
|
/* Coordinate failover info */
|
||||||
mstime_t failover_end_time; /* Deadline for failover command. */
|
mstime_t failover_end_time; /* Deadline for failover command. */
|
||||||
int force_failover; /* If true then failover will be foreced at the
|
int force_failover; /* If true then failover will be forced at the
|
||||||
* deadline, otherwise failover is aborted. */
|
* deadline, otherwise failover is aborted. */
|
||||||
char *target_replica_host; /* Failover target host. If null during a
|
char *target_replica_host; /* Failover target host. If null during a
|
||||||
* failover then any replica can be used. */
|
* failover then any replica can be used. */
|
||||||
|
@ -312,7 +312,7 @@ void sortCommand(client *c) {
|
|||||||
if (sortval->type == OBJ_ZSET)
|
if (sortval->type == OBJ_ZSET)
|
||||||
zsetConvert(sortval, OBJ_ENCODING_SKIPLIST);
|
zsetConvert(sortval, OBJ_ENCODING_SKIPLIST);
|
||||||
|
|
||||||
/* Objtain the length of the object to sort. */
|
/* Obtain the length of the object to sort. */
|
||||||
switch(sortval->type) {
|
switch(sortval->type) {
|
||||||
case OBJ_LIST: vectorlen = listTypeLength(sortval); break;
|
case OBJ_LIST: vectorlen = listTypeLength(sortval); break;
|
||||||
case OBJ_SET: vectorlen = setTypeSize(sortval); break;
|
case OBJ_SET: vectorlen = setTypeSize(sortval); break;
|
||||||
|
@ -86,7 +86,7 @@ typedef struct streamNACK {
|
|||||||
in the last delivery. */
|
in the last delivery. */
|
||||||
} streamNACK;
|
} streamNACK;
|
||||||
|
|
||||||
/* Stream propagation informations, passed to functions in order to propagate
|
/* Stream propagation information, passed to functions in order to propagate
|
||||||
* XCLAIM commands to AOF and slaves. */
|
* XCLAIM commands to AOF and slaves. */
|
||||||
typedef struct streamPropInfo {
|
typedef struct streamPropInfo {
|
||||||
robj *keyname;
|
robj *keyname;
|
||||||
|
@ -626,7 +626,7 @@ int streamAppendItem(stream *s, robj **argv, int64_t numfields, streamID *added_
|
|||||||
lp_count += 3; /* Add the 3 fixed fields flags + ms-diff + seq-diff. */
|
lp_count += 3; /* Add the 3 fixed fields flags + ms-diff + seq-diff. */
|
||||||
if (!(flags & STREAM_ITEM_FLAG_SAMEFIELDS)) {
|
if (!(flags & STREAM_ITEM_FLAG_SAMEFIELDS)) {
|
||||||
/* If the item is not compressed, it also has the fields other than
|
/* If the item is not compressed, it also has the fields other than
|
||||||
* the values, and an additional num-fileds field. */
|
* the values, and an additional num-fields field. */
|
||||||
lp_count += numfields+1;
|
lp_count += numfields+1;
|
||||||
}
|
}
|
||||||
lp = lpAppendInteger(lp,lp_count);
|
lp = lpAppendInteger(lp,lp_count);
|
||||||
@ -968,7 +968,7 @@ static int streamParseAddOrTrimArgsOrReply(client *c, streamAddTrimArgs *args, i
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (c == server.master || c->id == CLIENT_ID_AOF) {
|
if (c == server.master || c->id == CLIENT_ID_AOF) {
|
||||||
/* If command cam from master or from AOF we must not enforce maxnodes
|
/* If command came from master or from AOF we must not enforce maxnodes
|
||||||
* (The maxlen/minid argument was re-written to make sure there's no
|
* (The maxlen/minid argument was re-written to make sure there's no
|
||||||
* inconsistency). */
|
* inconsistency). */
|
||||||
args->limit = 0;
|
args->limit = 0;
|
||||||
@ -1365,7 +1365,7 @@ void streamPropagateXCLAIM(client *c, robj *key, streamCG *group, robj *groupnam
|
|||||||
argv[12] = shared.lastid;
|
argv[12] = shared.lastid;
|
||||||
argv[13] = createObjectFromStreamID(&group->last_id);
|
argv[13] = createObjectFromStreamID(&group->last_id);
|
||||||
|
|
||||||
/* We use progagate() because this code path is not always called from
|
/* We use propagate() because this code path is not always called from
|
||||||
* the command execution context. Moreover this will just alter the
|
* the command execution context. Moreover this will just alter the
|
||||||
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
||||||
* there is no message state cross-message atomicity required. */
|
* there is no message state cross-message atomicity required. */
|
||||||
@ -1390,7 +1390,7 @@ void streamPropagateGroupID(client *c, robj *key, streamCG *group, robj *groupna
|
|||||||
argv[3] = groupname;
|
argv[3] = groupname;
|
||||||
argv[4] = createObjectFromStreamID(&group->last_id);
|
argv[4] = createObjectFromStreamID(&group->last_id);
|
||||||
|
|
||||||
/* We use progagate() because this code path is not always called from
|
/* We use propagate() because this code path is not always called from
|
||||||
* the command execution context. Moreover this will just alter the
|
* the command execution context. Moreover this will just alter the
|
||||||
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
||||||
* there is no message state cross-message atomicity required. */
|
* there is no message state cross-message atomicity required. */
|
||||||
@ -1412,7 +1412,7 @@ void streamPropagateConsumerCreation(client *c, robj *key, robj *groupname, sds
|
|||||||
argv[3] = groupname;
|
argv[3] = groupname;
|
||||||
argv[4] = createObject(OBJ_STRING,sdsdup(consumername));
|
argv[4] = createObject(OBJ_STRING,sdsdup(consumername));
|
||||||
|
|
||||||
/* We use progagate() because this code path is not always called from
|
/* We use propagate() because this code path is not always called from
|
||||||
* the command execution context. Moreover this will just alter the
|
* the command execution context. Moreover this will just alter the
|
||||||
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
* consumer group state, and we don't need MULTI/EXEC wrapping because
|
||||||
* there is no message state cross-message atomicity required. */
|
* there is no message state cross-message atomicity required. */
|
||||||
@ -1576,7 +1576,7 @@ size_t streamReplyWithRange(client *c, stream *s, streamID *start, streamID *end
|
|||||||
return arraylen;
|
return arraylen;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is an helper function for streamReplyWithRange() when called with
|
/* This is a helper function for streamReplyWithRange() when called with
|
||||||
* group and consumer arguments, but with a range that is referring to already
|
* group and consumer arguments, but with a range that is referring to already
|
||||||
* delivered messages. In this case we just emit messages that are already
|
* delivered messages. In this case we just emit messages that are already
|
||||||
* in the history of the consumer, fetching the IDs from its PEL.
|
* in the history of the consumer, fetching the IDs from its PEL.
|
||||||
@ -1944,7 +1944,7 @@ void xreadCommand(client *c) {
|
|||||||
if (c->flags & CLIENT_LUA) {
|
if (c->flags & CLIENT_LUA) {
|
||||||
/*
|
/*
|
||||||
* Although the CLIENT_DENY_BLOCKING flag should protect from blocking the client
|
* Although the CLIENT_DENY_BLOCKING flag should protect from blocking the client
|
||||||
* on Lua/MULTI/RM_Call we want special treatment for Lua to keep backword compatibility.
|
* on Lua/MULTI/RM_Call we want special treatment for Lua to keep backward compatibility.
|
||||||
* There is no sense to use BLOCK option within Lua. */
|
* There is no sense to use BLOCK option within Lua. */
|
||||||
addReplyErrorFormat(c, "%s command is not allowed with BLOCK option from scripts", (char *)c->argv[0]->ptr);
|
addReplyErrorFormat(c, "%s command is not allowed with BLOCK option from scripts", (char *)c->argv[0]->ptr);
|
||||||
return;
|
return;
|
||||||
@ -2506,7 +2506,7 @@ void xsetidCommand(client *c) {
|
|||||||
/* XACK <key> <group> <id> <id> ... <id>
|
/* XACK <key> <group> <id> <id> ... <id>
|
||||||
*
|
*
|
||||||
* Acknowledge a message as processed. In practical terms we just check the
|
* Acknowledge a message as processed. In practical terms we just check the
|
||||||
* pendine entries list (PEL) of the group, and delete the PEL entry both from
|
* pending entries list (PEL) of the group, and delete the PEL entry both from
|
||||||
* the group and the consumer (pending messages are referenced in both places).
|
* the group and the consumer (pending messages are referenced in both places).
|
||||||
*
|
*
|
||||||
* Return value of the command is the number of messages successfully
|
* Return value of the command is the number of messages successfully
|
||||||
@ -2572,7 +2572,7 @@ cleanup:
|
|||||||
* delivery time and so forth. */
|
* delivery time and so forth. */
|
||||||
void xpendingCommand(client *c) {
|
void xpendingCommand(client *c) {
|
||||||
int justinfo = c->argc == 3; /* Without the range just outputs general
|
int justinfo = c->argc == 3; /* Without the range just outputs general
|
||||||
informations about the PEL. */
|
information about the PEL. */
|
||||||
robj *key = c->argv[1];
|
robj *key = c->argv[1];
|
||||||
robj *groupname = c->argv[2];
|
robj *groupname = c->argv[2];
|
||||||
robj *consumername = NULL;
|
robj *consumername = NULL;
|
||||||
@ -2928,7 +2928,7 @@ void xclaimCommand(client *c) {
|
|||||||
streamNACK *nack = raxFind(group->pel,buf,sizeof(buf));
|
streamNACK *nack = raxFind(group->pel,buf,sizeof(buf));
|
||||||
|
|
||||||
/* If FORCE is passed, let's check if at least the entry
|
/* If FORCE is passed, let's check if at least the entry
|
||||||
* exists in the Stream. In such case, we'll crate a new
|
* exists in the Stream. In such case, we'll create a new
|
||||||
* entry in the PEL from scratch, so that XCLAIM can also
|
* entry in the PEL from scratch, so that XCLAIM can also
|
||||||
* be used to create entries in the PEL. Useful for AOF
|
* be used to create entries in the PEL. Useful for AOF
|
||||||
* and replication of consumer groups. */
|
* and replication of consumer groups. */
|
||||||
@ -3548,7 +3548,7 @@ NULL
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Validate the integrity stream listpack entries structure. Both in term of a
|
/* Validate the integrity stream listpack entries structure. Both in term of a
|
||||||
* valid listpack, but also that the structure of the entires matches a valid
|
* valid listpack, but also that the structure of the entries matches a valid
|
||||||
* stream. return 1 if valid 0 if not valid. */
|
* stream. return 1 if valid 0 if not valid. */
|
||||||
int streamValidateListpackIntegrity(unsigned char *lp, size_t size, int deep) {
|
int streamValidateListpackIntegrity(unsigned char *lp, size_t size, int deep) {
|
||||||
int valid_record;
|
int valid_record;
|
||||||
|
10
src/t_zset.c
10
src/t_zset.c
@ -594,7 +594,7 @@ int zslParseLexRangeItem(robj *item, sds *dest, int *ex) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free a lex range structure, must be called only after zelParseLexRange()
|
/* Free a lex range structure, must be called only after zslParseLexRange()
|
||||||
* populated the structure with success (C_OK returned). */
|
* populated the structure with success (C_OK returned). */
|
||||||
void zslFreeLexRange(zlexrangespec *spec) {
|
void zslFreeLexRange(zlexrangespec *spec) {
|
||||||
if (spec->min != shared.minstring &&
|
if (spec->min != shared.minstring &&
|
||||||
@ -806,7 +806,7 @@ void zzlNext(unsigned char *zl, unsigned char **eptr, unsigned char **sptr) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Move to the previous entry based on the values in eptr and sptr. Both are
|
/* Move to the previous entry based on the values in eptr and sptr. Both are
|
||||||
* set to NULL when there is no next entry. */
|
* set to NULL when there is no prev entry. */
|
||||||
void zzlPrev(unsigned char *zl, unsigned char **eptr, unsigned char **sptr) {
|
void zzlPrev(unsigned char *zl, unsigned char **eptr, unsigned char **sptr) {
|
||||||
unsigned char *_eptr, *_sptr;
|
unsigned char *_eptr, *_sptr;
|
||||||
serverAssert(*eptr != NULL && *sptr != NULL);
|
serverAssert(*eptr != NULL && *sptr != NULL);
|
||||||
@ -1610,7 +1610,7 @@ robj *zsetDup(robj *o) {
|
|||||||
return zobj;
|
return zobj;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* callback for to check the ziplist doesn't have duplicate recoreds */
|
/* callback for to check the ziplist doesn't have duplicate records */
|
||||||
static int _zsetZiplistValidateIntegrity(unsigned char *p, void *userdata) {
|
static int _zsetZiplistValidateIntegrity(unsigned char *p, void *userdata) {
|
||||||
struct {
|
struct {
|
||||||
long count;
|
long count;
|
||||||
@ -2481,7 +2481,7 @@ static void zdiffAlgorithm2(zsetopsrc *src, long setnum, zset *dstzset, size_t *
|
|||||||
if (cardinality == 0) break;
|
if (cardinality == 0) break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Redize dict if needed after removing multiple elements */
|
/* Resize dict if needed after removing multiple elements */
|
||||||
if (htNeedsResize(dstzset->dict)) dictResize(dstzset->dict);
|
if (htNeedsResize(dstzset->dict)) dictResize(dstzset->dict);
|
||||||
|
|
||||||
/* Using this algorithm, we can't calculate the max element as we go,
|
/* Using this algorithm, we can't calculate the max element as we go,
|
||||||
@ -3600,7 +3600,7 @@ void zrangeGenericCommand(zrange_result_handler *handler, int argc_start, int st
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use defaults if not overriden by arguments. */
|
/* Use defaults if not overridden by arguments. */
|
||||||
if (direction == ZRANGE_DIRECTION_AUTO)
|
if (direction == ZRANGE_DIRECTION_AUTO)
|
||||||
direction = ZRANGE_DIRECTION_FORWARD;
|
direction = ZRANGE_DIRECTION_FORWARD;
|
||||||
if (rangetype == ZRANGE_AUTO)
|
if (rangetype == ZRANGE_AUTO)
|
||||||
|
@ -475,7 +475,7 @@ static void tlsEventHandler(struct aeEventLoop *el, int fd, void *clientData, in
|
|||||||
|
|
||||||
/* Process the return code received from OpenSSL>
|
/* Process the return code received from OpenSSL>
|
||||||
* Update the want parameter with expected I/O.
|
* Update the want parameter with expected I/O.
|
||||||
* Update the connection's error state if a real error has occured.
|
* Update the connection's error state if a real error has occurred.
|
||||||
* Returns an SSL error code, or 0 if no further handling is required.
|
* Returns an SSL error code, or 0 if no further handling is required.
|
||||||
*/
|
*/
|
||||||
static int handleSSLReturnCode(tls_connection *conn, int ret_value, WantIOType *want) {
|
static int handleSSLReturnCode(tls_connection *conn, int ret_value, WantIOType *want) {
|
||||||
|
@ -249,7 +249,7 @@ void trackingRememberKeys(client *c) {
|
|||||||
|
|
||||||
/* Given a key name, this function sends an invalidation message in the
|
/* Given a key name, this function sends an invalidation message in the
|
||||||
* proper channel (depending on RESP version: PubSub or Push message) and
|
* proper channel (depending on RESP version: PubSub or Push message) and
|
||||||
* to the proper client (in case fo redirection), in the context of the
|
* to the proper client (in case of redirection), in the context of the
|
||||||
* client 'c' with tracking enabled.
|
* client 'c' with tracking enabled.
|
||||||
*
|
*
|
||||||
* In case the 'proto' argument is non zero, the function will assume that
|
* In case the 'proto' argument is non zero, the function will assume that
|
||||||
@ -448,7 +448,7 @@ void trackingInvalidateKeysOnFlush(int async) {
|
|||||||
*
|
*
|
||||||
* So Redis allows the user to configure a maximum number of keys for the
|
* So Redis allows the user to configure a maximum number of keys for the
|
||||||
* invalidation table. This function makes sure that we don't go over the
|
* invalidation table. This function makes sure that we don't go over the
|
||||||
* specified fill rate: if we are over, we can just evict informations about
|
* specified fill rate: if we are over, we can just evict information about
|
||||||
* a random key, and send invalidation messages to clients like if the key was
|
* a random key, and send invalidation messages to clients like if the key was
|
||||||
* modified. */
|
* modified. */
|
||||||
void trackingLimitUsedSlots(void) {
|
void trackingLimitUsedSlots(void) {
|
||||||
@ -493,7 +493,7 @@ void trackingLimitUsedSlots(void) {
|
|||||||
* include keys that were modified the last time by this client, in order
|
* include keys that were modified the last time by this client, in order
|
||||||
* to implement the NOLOOP option.
|
* to implement the NOLOOP option.
|
||||||
*
|
*
|
||||||
* If the resultin array would be empty, NULL is returned instead. */
|
* If the resulting array would be empty, NULL is returned instead. */
|
||||||
sds trackingBuildBroadcastReply(client *c, rax *keys) {
|
sds trackingBuildBroadcastReply(client *c, rax *keys) {
|
||||||
raxIterator ri;
|
raxIterator ri;
|
||||||
uint64_t count;
|
uint64_t count;
|
||||||
|
@ -38,7 +38,7 @@
|
|||||||
* This should be the size of the buffer given to ld2string */
|
* This should be the size of the buffer given to ld2string */
|
||||||
#define MAX_LONG_DOUBLE_CHARS 5*1024
|
#define MAX_LONG_DOUBLE_CHARS 5*1024
|
||||||
|
|
||||||
/* long double to string convertion options */
|
/* long double to string conversion options */
|
||||||
typedef enum {
|
typedef enum {
|
||||||
LD_STR_AUTO, /* %.17Lg */
|
LD_STR_AUTO, /* %.17Lg */
|
||||||
LD_STR_HUMAN, /* %.17Lf + Trimming of trailing zeros */
|
LD_STR_HUMAN, /* %.17Lf + Trimming of trailing zeros */
|
||||||
|
@ -54,7 +54,7 @@
|
|||||||
*
|
*
|
||||||
* The length of the previous entry, <prevlen>, is encoded in the following way:
|
* The length of the previous entry, <prevlen>, is encoded in the following way:
|
||||||
* If this length is smaller than 254 bytes, it will only consume a single
|
* If this length is smaller than 254 bytes, it will only consume a single
|
||||||
* byte representing the length as an unsinged 8 bit integer. When the length
|
* byte representing the length as an unsigned 8 bit integer. When the length
|
||||||
* is greater than or equal to 254, it will consume 5 bytes. The first byte is
|
* is greater than or equal to 254, it will consume 5 bytes. The first byte is
|
||||||
* set to 254 (FE) to indicate a larger value is following. The remaining 4
|
* set to 254 (FE) to indicate a larger value is following. The remaining 4
|
||||||
* bytes take the length of the previous entry as value.
|
* bytes take the length of the previous entry as value.
|
||||||
@ -620,7 +620,7 @@ static inline int zipEntrySafe(unsigned char* zl, size_t zlbytes, unsigned char
|
|||||||
unsigned char *zllast = zl + zlbytes - ZIPLIST_END_SIZE;
|
unsigned char *zllast = zl + zlbytes - ZIPLIST_END_SIZE;
|
||||||
#define OUT_OF_RANGE(p) (unlikely((p) < zlfirst || (p) > zllast))
|
#define OUT_OF_RANGE(p) (unlikely((p) < zlfirst || (p) > zllast))
|
||||||
|
|
||||||
/* If threre's no possibility for the header to reach outside the ziplist,
|
/* If there's no possibility for the header to reach outside the ziplist,
|
||||||
* take the fast path. (max lensize and prevrawlensize are both 5 bytes) */
|
* take the fast path. (max lensize and prevrawlensize are both 5 bytes) */
|
||||||
if (p >= zlfirst && p + 10 < zllast) {
|
if (p >= zlfirst && p + 10 < zllast) {
|
||||||
ZIP_DECODE_PREVLEN(p, e->prevrawlensize, e->prevrawlen);
|
ZIP_DECODE_PREVLEN(p, e->prevrawlensize, e->prevrawlen);
|
||||||
@ -631,16 +631,16 @@ static inline int zipEntrySafe(unsigned char* zl, size_t zlbytes, unsigned char
|
|||||||
/* We didn't call ZIP_ASSERT_ENCODING, so we check lensize was set to 0. */
|
/* We didn't call ZIP_ASSERT_ENCODING, so we check lensize was set to 0. */
|
||||||
if (unlikely(e->lensize == 0))
|
if (unlikely(e->lensize == 0))
|
||||||
return 0;
|
return 0;
|
||||||
/* Make sure the entry doesn't rech outside the edge of the ziplist */
|
/* Make sure the entry doesn't reach outside the edge of the ziplist */
|
||||||
if (OUT_OF_RANGE(p + e->headersize + e->len))
|
if (OUT_OF_RANGE(p + e->headersize + e->len))
|
||||||
return 0;
|
return 0;
|
||||||
/* Make sure prevlen doesn't rech outside the edge of the ziplist */
|
/* Make sure prevlen doesn't reach outside the edge of the ziplist */
|
||||||
if (validate_prevlen && OUT_OF_RANGE(p - e->prevrawlen))
|
if (validate_prevlen && OUT_OF_RANGE(p - e->prevrawlen))
|
||||||
return 0;
|
return 0;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Make sure the pointer doesn't rech outside the edge of the ziplist */
|
/* Make sure the pointer doesn't reach outside the edge of the ziplist */
|
||||||
if (OUT_OF_RANGE(p))
|
if (OUT_OF_RANGE(p))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -664,11 +664,11 @@ static inline int zipEntrySafe(unsigned char* zl, size_t zlbytes, unsigned char
|
|||||||
ZIP_DECODE_LENGTH(p + e->prevrawlensize, e->encoding, e->lensize, e->len);
|
ZIP_DECODE_LENGTH(p + e->prevrawlensize, e->encoding, e->lensize, e->len);
|
||||||
e->headersize = e->prevrawlensize + e->lensize;
|
e->headersize = e->prevrawlensize + e->lensize;
|
||||||
|
|
||||||
/* Make sure the entry doesn't rech outside the edge of the ziplist */
|
/* Make sure the entry doesn't reach outside the edge of the ziplist */
|
||||||
if (OUT_OF_RANGE(p + e->headersize + e->len))
|
if (OUT_OF_RANGE(p + e->headersize + e->len))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Make sure prevlen doesn't rech outside the edge of the ziplist */
|
/* Make sure prevlen doesn't reach outside the edge of the ziplist */
|
||||||
if (validate_prevlen && OUT_OF_RANGE(p - e->prevrawlen))
|
if (validate_prevlen && OUT_OF_RANGE(p - e->prevrawlen))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -827,7 +827,7 @@ unsigned char *__ziplistCascadeUpdate(unsigned char *zl, unsigned char *p) {
|
|||||||
/* An entry's prevlen can only increment 4 bytes. */
|
/* An entry's prevlen can only increment 4 bytes. */
|
||||||
zipStorePrevEntryLength(p, cur.prevrawlen+delta);
|
zipStorePrevEntryLength(p, cur.prevrawlen+delta);
|
||||||
}
|
}
|
||||||
/* Foward to previous entry. */
|
/* Forward to previous entry. */
|
||||||
prevoffset -= cur.prevrawlen;
|
prevoffset -= cur.prevrawlen;
|
||||||
cnt--;
|
cnt--;
|
||||||
}
|
}
|
||||||
|
@ -399,7 +399,7 @@ int zipmapValidateIntegrity(unsigned char *zm, size_t size, int deep) {
|
|||||||
while(*p != ZIPMAP_END) {
|
while(*p != ZIPMAP_END) {
|
||||||
/* read the field name length encoding type */
|
/* read the field name length encoding type */
|
||||||
s = zipmapGetEncodedLengthSize(p);
|
s = zipmapGetEncodedLengthSize(p);
|
||||||
/* make sure the entry length doesn't rech outside the edge of the zipmap */
|
/* make sure the entry length doesn't reach outside the edge of the zipmap */
|
||||||
if (OUT_OF_RANGE(p+s))
|
if (OUT_OF_RANGE(p+s))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -408,13 +408,13 @@ int zipmapValidateIntegrity(unsigned char *zm, size_t size, int deep) {
|
|||||||
p += s; /* skip the encoded field size */
|
p += s; /* skip the encoded field size */
|
||||||
p += l; /* skip the field */
|
p += l; /* skip the field */
|
||||||
|
|
||||||
/* make sure the entry doesn't rech outside the edge of the zipmap */
|
/* make sure the entry doesn't reach outside the edge of the zipmap */
|
||||||
if (OUT_OF_RANGE(p))
|
if (OUT_OF_RANGE(p))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* read the value length encoding type */
|
/* read the value length encoding type */
|
||||||
s = zipmapGetEncodedLengthSize(p);
|
s = zipmapGetEncodedLengthSize(p);
|
||||||
/* make sure the entry length doesn't rech outside the edge of the zipmap */
|
/* make sure the entry length doesn't reach outside the edge of the zipmap */
|
||||||
if (OUT_OF_RANGE(p+s))
|
if (OUT_OF_RANGE(p+s))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -425,7 +425,7 @@ int zipmapValidateIntegrity(unsigned char *zm, size_t size, int deep) {
|
|||||||
p += l+e; /* skip the value and free space */
|
p += l+e; /* skip the value and free space */
|
||||||
count++;
|
count++;
|
||||||
|
|
||||||
/* make sure the entry doesn't rech outside the edge of the zipmap */
|
/* make sure the entry doesn't reach outside the edge of the zipmap */
|
||||||
if (OUT_OF_RANGE(p))
|
if (OUT_OF_RANGE(p))
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# Failover stress test.
|
# Failover stress test.
|
||||||
# In this test a different node is killed in a loop for N
|
# In this test a different node is killed in a loop for N
|
||||||
# iterations. The test checks that certain properties
|
# iterations. The test checks that certain properties
|
||||||
# are preseved across iterations.
|
# are preserved across iterations.
|
||||||
|
|
||||||
source "../tests/includes/init-tests.tcl"
|
source "../tests/includes/init-tests.tcl"
|
||||||
|
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
# Test UPDATE messages sent by other nodes when the currently authorirative
|
# Test UPDATE messages sent by other nodes when the currently authorirative
|
||||||
# master is unavaialble. The test is performed in the following steps:
|
# master is unavailable. The test is performed in the following steps:
|
||||||
#
|
#
|
||||||
# 1) Master goes down.
|
# 1) Master goes down.
|
||||||
# 2) Slave failover and becomes new master.
|
# 2) Slave failover and becomes new master.
|
||||||
# 3) New master is partitoned away.
|
# 3) New master is partitioned away.
|
||||||
# 4) Old master returns.
|
# 4) Old master returns.
|
||||||
# 5) At this point we expect the old master to turn into a slave ASAP because
|
# 5) At this point we expect the old master to turn into a slave ASAP because
|
||||||
# of the UPDATE messages it will receive from the other nodes when its
|
# of the UPDATE messages it will receive from the other nodes when its
|
||||||
|
@ -53,7 +53,7 @@ test "Wait cluster to be stable" {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test "Master #0 stil should have its replicas" {
|
test "Master #0 still should have its replicas" {
|
||||||
assert { [llength [lindex [R 0 role] 2]] >= 2 }
|
assert { [llength [lindex [R 0 role] 2]] >= 2 }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ test "Cluster should start ok" {
|
|||||||
set primary [Rn 0]
|
set primary [Rn 0]
|
||||||
set replica [Rn 1]
|
set replica [Rn 1]
|
||||||
|
|
||||||
test "Cant read from replica without READONLY" {
|
test "Can't read from replica without READONLY" {
|
||||||
$primary SET a 1
|
$primary SET a 1
|
||||||
wait_for_ofs_sync $primary $replica
|
wait_for_ofs_sync $primary $replica
|
||||||
catch {$replica GET a} err
|
catch {$replica GET a} err
|
||||||
@ -25,7 +25,7 @@ test "Can read from replica after READONLY" {
|
|||||||
assert {[$replica GET a] eq {1}}
|
assert {[$replica GET a] eq {1}}
|
||||||
}
|
}
|
||||||
|
|
||||||
test "Can preform HSET primary and HGET from replica" {
|
test "Can perform HSET primary and HGET from replica" {
|
||||||
$primary HSET h a 1
|
$primary HSET h a 1
|
||||||
$primary HSET h b 2
|
$primary HSET h b 2
|
||||||
$primary HSET h c 3
|
$primary HSET h c 3
|
||||||
|
@ -13,12 +13,12 @@ test "Cluster should start ok" {
|
|||||||
set primary1 [Rn 0]
|
set primary1 [Rn 0]
|
||||||
set primary2 [Rn 1]
|
set primary2 [Rn 1]
|
||||||
|
|
||||||
proc cmdstat {instace cmd} {
|
proc cmdstat {instance cmd} {
|
||||||
return [cmdrstat $cmd $instace]
|
return [cmdrstat $cmd $instance]
|
||||||
}
|
}
|
||||||
|
|
||||||
proc errorstat {instace cmd} {
|
proc errorstat {instance cmd} {
|
||||||
return [errorrstat $cmd $instace]
|
return [errorrstat $cmd $instance]
|
||||||
}
|
}
|
||||||
|
|
||||||
test "errorstats: rejected call due to MOVED Redirection" {
|
test "errorstats: rejected call due to MOVED Redirection" {
|
||||||
|
@ -52,7 +52,7 @@ proc generate_types {} {
|
|||||||
generate_collections big 10
|
generate_collections big 10
|
||||||
|
|
||||||
# make sure our big stream also has a listpack record that has different
|
# make sure our big stream also has a listpack record that has different
|
||||||
# field names than the master recored
|
# field names than the master recorded
|
||||||
r xadd streambig * item 1 value 1
|
r xadd streambig * item 1 value 1
|
||||||
r xadd streambig * item 1 unique value
|
r xadd streambig * item 1 unique value
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# tests of corrupt ziplist payload with valid CRC
|
# tests of corrupt ziplist payload with valid CRC
|
||||||
# * setting crash-memcheck-enabled to no to avoid issues with valgrind
|
# * setting crash-memcheck-enabled to no to avoid issues with valgrind
|
||||||
# * setting use-exit-on-panic to yes so that valgrind can search for leaks
|
# * setting use-exit-on-panic to yes so that valgrind can search for leaks
|
||||||
# * settng debug set-skip-checksum-validation to 1 on some tests for which we
|
# * setting debug set-skip-checksum-validation to 1 on some tests for which we
|
||||||
# didn't bother to fake a valid checksum
|
# didn't bother to fake a valid checksum
|
||||||
# * some tests set sanitize-dump-payload to no and some to yet, depending on
|
# * some tests set sanitize-dump-payload to no and some to yet, depending on
|
||||||
# what we want to test
|
# what we want to test
|
||||||
@ -214,7 +214,7 @@ test {corrupt payload: hash ziplist uneven record count} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test {corrupt payload: hash dupliacte records} {
|
test {corrupt payload: hash duplicate records} {
|
||||||
# when we do perform full sanitization, we expect duplicate records to fail the restore
|
# when we do perform full sanitization, we expect duplicate records to fail the restore
|
||||||
start_server [list overrides [list loglevel verbose use-exit-on-panic yes crash-memcheck-enabled no] ] {
|
start_server [list overrides [list loglevel verbose use-exit-on-panic yes crash-memcheck-enabled no] ] {
|
||||||
r config set sanitize-dump-payload yes
|
r config set sanitize-dump-payload yes
|
||||||
|
@ -117,7 +117,7 @@ start_server {} {
|
|||||||
set used [list $master_id]
|
set used [list $master_id]
|
||||||
test "PSYNC2: \[NEW LAYOUT\] Set #$master_id as master" {
|
test "PSYNC2: \[NEW LAYOUT\] Set #$master_id as master" {
|
||||||
$R($master_id) slaveof no one
|
$R($master_id) slaveof no one
|
||||||
$R($master_id) config set repl-ping-replica-period 1 ;# increse the chance that random ping will cause issues
|
$R($master_id) config set repl-ping-replica-period 1 ;# increase the chance that random ping will cause issues
|
||||||
if {$counter_value == 0} {
|
if {$counter_value == 0} {
|
||||||
$R($master_id) set x $counter_value
|
$R($master_id) set x $counter_value
|
||||||
}
|
}
|
||||||
|
@ -130,7 +130,7 @@ start_server_and_kill_it [list "dir" $server_path] {
|
|||||||
|
|
||||||
start_server {} {
|
start_server {} {
|
||||||
test {Test FLUSHALL aborts bgsave} {
|
test {Test FLUSHALL aborts bgsave} {
|
||||||
# 1000 keys with 1ms sleep per key shuld take 1 second
|
# 1000 keys with 1ms sleep per key should take 1 second
|
||||||
r config set rdb-key-save-delay 1000
|
r config set rdb-key-save-delay 1000
|
||||||
r debug populate 1000
|
r debug populate 1000
|
||||||
r bgsave
|
r bgsave
|
||||||
|
@ -419,7 +419,7 @@ test {slave fails full sync and diskless load swapdb recovers it} {
|
|||||||
fail "Replica didn't get into loading mode"
|
fail "Replica didn't get into loading mode"
|
||||||
}
|
}
|
||||||
|
|
||||||
# make sure that next sync will not start immediately so that we can catch the slave in betweeen syncs
|
# make sure that next sync will not start immediately so that we can catch the slave in between syncs
|
||||||
$master config set repl-diskless-sync-delay 5
|
$master config set repl-diskless-sync-delay 5
|
||||||
# for faster server shutdown, make rdb saving fast again (the fork is already uses the slow one)
|
# for faster server shutdown, make rdb saving fast again (the fork is already uses the slow one)
|
||||||
$master config set rdb-key-save-delay 0
|
$master config set rdb-key-save-delay 0
|
||||||
|
@ -82,7 +82,7 @@ void CommandFilter_CommandFilter(RedisModuleCommandFilterCtx *filter)
|
|||||||
* - Remove @delme
|
* - Remove @delme
|
||||||
* - Replace @replaceme
|
* - Replace @replaceme
|
||||||
* - Append @insertbefore or @insertafter
|
* - Append @insertbefore or @insertafter
|
||||||
* - Prefix with Log command if @log encounterd
|
* - Prefix with Log command if @log encountered
|
||||||
*/
|
*/
|
||||||
int log = 0;
|
int log = 0;
|
||||||
int pos = 0;
|
int pos = 0;
|
||||||
|
@ -36,7 +36,7 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
/** strores all the keys on which we got 'loaded' keyspace notification **/
|
/** stores all the keys on which we got 'loaded' keyspace notification **/
|
||||||
RedisModuleDict *loaded_event_log = NULL;
|
RedisModuleDict *loaded_event_log = NULL;
|
||||||
/** stores all the keys on which we got 'module' keyspace notification **/
|
/** stores all the keys on which we got 'module' keyspace notification **/
|
||||||
RedisModuleDict *module_event_log = NULL;
|
RedisModuleDict *module_event_log = NULL;
|
||||||
|
@ -75,7 +75,7 @@ void timerNestedHandler(RedisModuleCtx *ctx, void *data) {
|
|||||||
int repl = (long long)data;
|
int repl = (long long)data;
|
||||||
|
|
||||||
/* The goal is the trigger a module command that calls RM_Replicate
|
/* The goal is the trigger a module command that calls RM_Replicate
|
||||||
* in order to test MULTI/EXEC structre */
|
* in order to test MULTI/EXEC structure */
|
||||||
RedisModule_Replicate(ctx,"INCRBY","cc","timer-nested-start","1");
|
RedisModule_Replicate(ctx,"INCRBY","cc","timer-nested-start","1");
|
||||||
RedisModuleCallReply *reply = RedisModule_Call(ctx,"propagate-test.nested", repl? "!" : "");
|
RedisModuleCallReply *reply = RedisModule_Call(ctx,"propagate-test.nested", repl? "!" : "");
|
||||||
RedisModule_FreeCallReply(reply);
|
RedisModule_FreeCallReply(reply);
|
||||||
|
@ -52,7 +52,7 @@ test "SDOWN is triggered by masters advertising as slaves" {
|
|||||||
ensure_master_up
|
ensure_master_up
|
||||||
}
|
}
|
||||||
|
|
||||||
test "SDOWN is triggered by misconfigured instance repling with errors" {
|
test "SDOWN is triggered by misconfigured instance replying with errors" {
|
||||||
ensure_master_up
|
ensure_master_up
|
||||||
set orig_dir [lindex [R 0 config get dir] 1]
|
set orig_dir [lindex [R 0 config get dir] 1]
|
||||||
set orig_save [lindex [R 0 config get save] 1]
|
set orig_save [lindex [R 0 config get save] 1]
|
||||||
|
@ -202,7 +202,7 @@ proc tags_acceptable {tags err_return} {
|
|||||||
|
|
||||||
# doesn't really belong here, but highly coupled to code in start_server
|
# doesn't really belong here, but highly coupled to code in start_server
|
||||||
proc tags {tags code} {
|
proc tags {tags code} {
|
||||||
# If we 'tags' contain multiple tags, quoted and seperated by spaces,
|
# If we 'tags' contain multiple tags, quoted and separated by spaces,
|
||||||
# we want to get rid of the quotes in order to have a proper list
|
# we want to get rid of the quotes in order to have a proper list
|
||||||
set tags [string map { \" "" } $tags]
|
set tags [string map { \" "" } $tags]
|
||||||
set ::tags [concat $::tags $tags]
|
set ::tags [concat $::tags $tags]
|
||||||
@ -366,7 +366,7 @@ proc start_server {options {code undefined}} {
|
|||||||
set omit $value
|
set omit $value
|
||||||
}
|
}
|
||||||
"tags" {
|
"tags" {
|
||||||
# If we 'tags' contain multiple tags, quoted and seperated by spaces,
|
# If we 'tags' contain multiple tags, quoted and separated by spaces,
|
||||||
# we want to get rid of the quotes in order to have a proper list
|
# we want to get rid of the quotes in order to have a proper list
|
||||||
set tags [string map { \" "" } $value]
|
set tags [string map { \" "" } $value]
|
||||||
set ::tags [concat $::tags $tags]
|
set ::tags [concat $::tags $tags]
|
||||||
|
@ -498,7 +498,7 @@ proc find_valgrind_errors {stderr on_termination} {
|
|||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
# Look for the absense of a leak free summary (happens when redis isn't terminated properly).
|
# Look for the absence of a leak free summary (happens when redis isn't terminated properly).
|
||||||
if {(![regexp -- {definitely lost: 0 bytes} $buf] &&
|
if {(![regexp -- {definitely lost: 0 bytes} $buf] &&
|
||||||
![regexp -- {no leaks are possible} $buf])} {
|
![regexp -- {no leaks are possible} $buf])} {
|
||||||
return $buf
|
return $buf
|
||||||
|
@ -251,6 +251,6 @@ start_server {tags {"introspection"}} {
|
|||||||
}
|
}
|
||||||
} {} {external:skip}
|
} {} {external:skip}
|
||||||
|
|
||||||
# Config file at this point is at a wierd state, and includes all
|
# Config file at this point is at a weird state, and includes all
|
||||||
# known keywords. Might be a good idea to avoid adding tests here.
|
# known keywords. Might be a good idea to avoid adding tests here.
|
||||||
}
|
}
|
||||||
|
@ -393,7 +393,7 @@ start_server {tags {"defrag external:skip"} overrides {appendonly yes auto-aof-r
|
|||||||
# there was an edge case in defrag where all the slabs of a certain bin are exact the same
|
# there was an edge case in defrag where all the slabs of a certain bin are exact the same
|
||||||
# % utilization, with the exception of the current slab from which new allocations are made
|
# % utilization, with the exception of the current slab from which new allocations are made
|
||||||
# if the current slab is lower in utilization the defragger would have ended up in stagnation,
|
# if the current slab is lower in utilization the defragger would have ended up in stagnation,
|
||||||
# keept running and not move any allocation.
|
# kept running and not move any allocation.
|
||||||
# this test is more consistent on a fresh server with no history
|
# this test is more consistent on a fresh server with no history
|
||||||
start_server {tags {"defrag"} overrides {save ""}} {
|
start_server {tags {"defrag"} overrides {save ""}} {
|
||||||
r flushdb
|
r flushdb
|
||||||
|
@ -30,7 +30,7 @@ start_server {tags {"modules"}} {
|
|||||||
assert_equal [r auth.changecount] 1
|
assert_equal [r auth.changecount] 1
|
||||||
}
|
}
|
||||||
|
|
||||||
test {Modules cant authenticate with ACLs users that dont exist} {
|
test {Modules can't authenticate with ACLs users that dont exist} {
|
||||||
catch { [r auth.authrealuser auth-module-test-fake] } e
|
catch { [r auth.authrealuser auth-module-test-fake] } e
|
||||||
assert_match {*Invalid user*} $e
|
assert_match {*Invalid user*} $e
|
||||||
}
|
}
|
||||||
|
@ -40,7 +40,7 @@ start_server {tags {"modules"}} {
|
|||||||
assert_equal [r test.dbsize] 0
|
assert_equal [r test.dbsize] 0
|
||||||
}
|
}
|
||||||
|
|
||||||
test {test modle lru api} {
|
test {test module lru api} {
|
||||||
r config set maxmemory-policy allkeys-lru
|
r config set maxmemory-policy allkeys-lru
|
||||||
r set x foo
|
r set x foo
|
||||||
set lru [r test.getlru x]
|
set lru [r test.getlru x]
|
||||||
@ -59,7 +59,7 @@ start_server {tags {"modules"}} {
|
|||||||
}
|
}
|
||||||
r config set maxmemory-policy allkeys-lru
|
r config set maxmemory-policy allkeys-lru
|
||||||
|
|
||||||
test {test modle lfu api} {
|
test {test module lfu api} {
|
||||||
r config set maxmemory-policy allkeys-lfu
|
r config set maxmemory-policy allkeys-lfu
|
||||||
r set x foo
|
r set x foo
|
||||||
set lfu [r test.getlfu x]
|
set lfu [r test.getlfu x]
|
||||||
|
@ -414,7 +414,7 @@ start_server {tags {"multi"}} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
test {MULTI-EXEC body and script timeout} {
|
test {MULTI-EXEC body and script timeout} {
|
||||||
# check that we don't run an imcomplete transaction due to some commands
|
# check that we don't run an incomplete transaction due to some commands
|
||||||
# arriving during busy script
|
# arriving during busy script
|
||||||
set rd1 [redis_deferring_client]
|
set rd1 [redis_deferring_client]
|
||||||
set r2 [redis_client]
|
set r2 [redis_client]
|
||||||
|
@ -131,7 +131,7 @@ start_server {tags {"obuf-limits external:skip"}} {
|
|||||||
after 100
|
after 100
|
||||||
|
|
||||||
# Create a pipeline of commands that will be processed in one socket read.
|
# Create a pipeline of commands that will be processed in one socket read.
|
||||||
# It is important to use one write, in TLS mode independant writes seem
|
# It is important to use one write, in TLS mode independent writes seem
|
||||||
# to wait for response from the server.
|
# to wait for response from the server.
|
||||||
# Total size should be less than OS socket buffer, redis can
|
# Total size should be less than OS socket buffer, redis can
|
||||||
# execute all commands in this pipeline when it wakes up.
|
# execute all commands in this pipeline when it wakes up.
|
||||||
|
@ -272,7 +272,7 @@ start_server {tags {"scripting"}} {
|
|||||||
local encoded = cmsgpack.pack(a)
|
local encoded = cmsgpack.pack(a)
|
||||||
local h = ""
|
local h = ""
|
||||||
-- cmsgpack encodes to a depth of 16, but can't encode
|
-- cmsgpack encodes to a depth of 16, but can't encode
|
||||||
-- references, so the encoded object has a deep copy recusive
|
-- references, so the encoded object has a deep copy recursive
|
||||||
-- depth of 16.
|
-- depth of 16.
|
||||||
for i = 1, #encoded do
|
for i = 1, #encoded do
|
||||||
h = h .. string.format("%02x",string.byte(encoded,i))
|
h = h .. string.format("%02x",string.byte(encoded,i))
|
||||||
|
@ -640,7 +640,7 @@ start_server {
|
|||||||
set item [$slave xreadgroup group mygroup myconsumer \
|
set item [$slave xreadgroup group mygroup myconsumer \
|
||||||
COUNT 1 STREAMS stream >]
|
COUNT 1 STREAMS stream >]
|
||||||
|
|
||||||
# The consumed enty should be the third
|
# The consumed entry should be the third
|
||||||
set myentry [lindex $item 0 1 0 1]
|
set myentry [lindex $item 0 1 0 1]
|
||||||
assert {$myentry eq {a 3}}
|
assert {$myentry eq {a 3}}
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,7 @@ uint16_t minutes_diff(uint16_t now, uint16_t prev) {
|
|||||||
return 65535-prev+now;
|
return 65535-prev+now;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Increment a couter logaritmically: the greatest is its value, the
|
/* Increment a counter logarithmically: the greatest is its value, the
|
||||||
* less likely is that the counter is really incremented.
|
* less likely is that the counter is really incremented.
|
||||||
* The maximum value of the counter is saturated at 255. */
|
* The maximum value of the counter is saturated at 255. */
|
||||||
uint8_t log_incr(uint8_t counter) {
|
uint8_t log_incr(uint8_t counter) {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# redis-sha1.rb - Copyright (C) 2009 Salvatore Sanfilippo
|
# redis-sha1.rb - Copyright (C) 2009 Salvatore Sanfilippo
|
||||||
# BSD license, See the COPYING file for more information.
|
# BSD license, See the COPYING file for more information.
|
||||||
#
|
#
|
||||||
# Performs the SHA1 sum of the whole datset.
|
# Performs the SHA1 sum of the whole dataset.
|
||||||
# This is useful to spot bugs in persistence related code and to make sure
|
# This is useful to spot bugs in persistence related code and to make sure
|
||||||
# Slaves and Masters are in SYNC.
|
# Slaves and Masters are in SYNC.
|
||||||
#
|
#
|
||||||
|
@ -101,7 +101,7 @@ if {![file exists speed-regression.tcl]} {
|
|||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
# Make sure there is not already a server runnign on port 12123
|
# Make sure there is not already a server running on port 12123
|
||||||
set is_not_running [catch {set r [redis 127.0.0.1 $::port]}]
|
set is_not_running [catch {set r [redis 127.0.0.1 $::port]}]
|
||||||
if {!$is_not_running} {
|
if {!$is_not_running} {
|
||||||
puts "Sorry, you have a running server on port $::port"
|
puts "Sorry, you have a running server on port $::port"
|
||||||
|
@ -17,7 +17,7 @@ freq = {}
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# Print the frequency each element was yeld to process it with gnuplot
|
# Print the frequency each element was yield to process it with gnuplot
|
||||||
freq.each{|item,count|
|
freq.each{|item,count|
|
||||||
puts "#{item} #{count}"
|
puts "#{item} #{count}"
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* This is a small program used in order to understand the collison rate
|
/* This is a small program used in order to understand the collision rate
|
||||||
* of CRC64 (ISO version) VS other stronger hashing functions in the context
|
* of CRC64 (ISO version) VS other stronger hashing functions in the context
|
||||||
* of hashing keys for the Redis "tracking" feature (client side caching
|
* of hashing keys for the Redis "tracking" feature (client side caching
|
||||||
* assisted by the server).
|
* assisted by the server).
|
||||||
@ -7,7 +7,7 @@
|
|||||||
*
|
*
|
||||||
* prefix:<counter>
|
* prefix:<counter>
|
||||||
*
|
*
|
||||||
* And counts the resulting collisons generated in the 24 bits of output
|
* And counts the resulting collisions generated in the 24 bits of output
|
||||||
* needed for the tracking feature invalidation table (16 millions + entries)
|
* needed for the tracking feature invalidation table (16 millions + entries)
|
||||||
*
|
*
|
||||||
* Compile with:
|
* Compile with:
|
||||||
|
Loading…
Reference in New Issue
Block a user