2019-07-03 13:16:20 -04:00
|
|
|
/* tracking.c - Client side caching: keys tracking and invalidation
|
|
|
|
*
|
|
|
|
* Copyright (c) 2019, Salvatore Sanfilippo <antirez at gmail dot com>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* * Neither the name of Redis nor the names of its contributors may be used
|
|
|
|
* to endorse or promote products derived from this software without
|
|
|
|
* specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "server.h"
|
|
|
|
|
2020-02-07 08:03:43 -05:00
|
|
|
/* The tracking table is constituted by a radix tree of keys, each pointing
|
|
|
|
* to a radix tree of client IDs, used to track the clients that may have
|
|
|
|
* certain keys in their local, client side, cache.
|
2019-07-03 13:16:20 -04:00
|
|
|
*
|
|
|
|
* When a client enables tracking with "CLIENT TRACKING on", each key served to
|
2020-02-07 08:03:43 -05:00
|
|
|
* the client is remembered in the table mapping the keys to the client IDs.
|
|
|
|
* Later, when a key is modified, all the clients that may have local copy
|
|
|
|
* of such key will receive an invalidation message.
|
2019-07-03 13:16:20 -04:00
|
|
|
*
|
|
|
|
* Clients will normally take frequently requested objects in memory, removing
|
2020-02-07 08:03:43 -05:00
|
|
|
* them when invalidation messages are received. */
|
|
|
|
rax *TrackingTable = NULL;
|
2020-02-10 11:18:11 -05:00
|
|
|
rax *PrefixTable = NULL;
|
2020-02-07 08:03:43 -05:00
|
|
|
uint64_t TrackingTableTotalItems = 0; /* Total number of IDs stored across
|
2020-02-16 08:16:51 -05:00
|
|
|
the whole tracking table. This gives
|
2020-02-07 08:03:43 -05:00
|
|
|
an hint about the total memory we
|
|
|
|
are using server side for CSC. */
|
2019-07-05 06:24:28 -04:00
|
|
|
robj *TrackingChannelName;
|
2019-07-03 13:16:20 -04:00
|
|
|
|
2020-02-11 11:26:27 -05:00
|
|
|
/* This is the structure that we have as value of the PrefixTable, and
|
|
|
|
* represents the list of keys modified, and the list of clients that need
|
|
|
|
* to be notified, for a given prefix. */
|
|
|
|
typedef struct bcastState {
|
|
|
|
rax *keys; /* Keys modified in the current event loop cycle. */
|
|
|
|
rax *clients; /* Clients subscribed to the notification events for this
|
|
|
|
prefix. */
|
|
|
|
} bcastState;
|
|
|
|
|
2019-07-03 13:16:20 -04:00
|
|
|
/* Remove the tracking state from the client 'c'. Note that there is not much
|
|
|
|
* to do for us here, if not to decrement the counter of the clients in
|
|
|
|
* tracking mode, because we just store the ID of the client in the tracking
|
|
|
|
* table, so we'll remove the ID reference in a lazy way. Otherwise when a
|
|
|
|
* client with many entries in the table is removed, it would cost a lot of
|
|
|
|
* time to do the cleanup. */
|
|
|
|
void disableTracking(client *c) {
|
2020-02-11 11:26:27 -05:00
|
|
|
/* If this client is in broadcasting mode, we need to unsubscribe it
|
|
|
|
* from all the prefixes it is registered to. */
|
|
|
|
if (c->flags & CLIENT_TRACKING_BCAST) {
|
|
|
|
raxIterator ri;
|
|
|
|
raxStart(&ri,c->client_tracking_prefixes);
|
|
|
|
raxSeek(&ri,"^",NULL,0);
|
|
|
|
while(raxNext(&ri)) {
|
|
|
|
bcastState *bs = raxFind(PrefixTable,ri.key,ri.key_len);
|
|
|
|
serverAssert(bs != raxNotFound);
|
|
|
|
raxRemove(bs->clients,(unsigned char*)&c,sizeof(c),NULL);
|
|
|
|
/* Was it the last client? Remove the prefix from the
|
|
|
|
* table. */
|
|
|
|
if (raxSize(bs->clients) == 0) {
|
|
|
|
raxFree(bs->clients);
|
|
|
|
raxFree(bs->keys);
|
|
|
|
zfree(bs);
|
|
|
|
raxRemove(PrefixTable,ri.key,ri.key_len,NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
raxStop(&ri);
|
2020-02-12 13:22:04 -05:00
|
|
|
raxFree(c->client_tracking_prefixes);
|
|
|
|
c->client_tracking_prefixes = NULL;
|
2020-02-11 11:26:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear flags and adjust the count. */
|
2019-07-03 13:16:20 -04:00
|
|
|
if (c->flags & CLIENT_TRACKING) {
|
|
|
|
server.tracking_clients--;
|
2020-02-11 11:26:27 -05:00
|
|
|
c->flags &= ~(CLIENT_TRACKING|CLIENT_TRACKING_BROKEN_REDIR|
|
2020-02-21 10:39:42 -05:00
|
|
|
CLIENT_TRACKING_BCAST|CLIENT_TRACKING_OPTIN|
|
2020-03-23 01:04:49 -04:00
|
|
|
CLIENT_TRACKING_OPTOUT|CLIENT_TRACKING_CACHING);
|
2020-02-11 11:26:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the client 'c' to track the prefix 'prefix'. If the client 'c' is
|
|
|
|
* already registered for the specified prefix, no operation is performed. */
|
|
|
|
void enableBcastTrackingForPrefix(client *c, char *prefix, size_t plen) {
|
|
|
|
bcastState *bs = raxFind(PrefixTable,(unsigned char*)prefix,sdslen(prefix));
|
|
|
|
/* If this is the first client subscribing to such prefix, create
|
|
|
|
* the prefix in the table. */
|
|
|
|
if (bs == raxNotFound) {
|
|
|
|
bs = zmalloc(sizeof(*bs));
|
|
|
|
bs->keys = raxNew();
|
|
|
|
bs->clients = raxNew();
|
|
|
|
raxInsert(PrefixTable,(unsigned char*)prefix,plen,bs,NULL);
|
|
|
|
}
|
|
|
|
if (raxTryInsert(bs->clients,(unsigned char*)&c,sizeof(c),NULL,NULL)) {
|
2020-02-12 13:22:04 -05:00
|
|
|
if (c->client_tracking_prefixes == NULL)
|
|
|
|
c->client_tracking_prefixes = raxNew();
|
2020-02-11 11:26:27 -05:00
|
|
|
raxInsert(c->client_tracking_prefixes,
|
|
|
|
(unsigned char*)prefix,plen,NULL,NULL);
|
2019-07-03 13:16:20 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable the tracking state for the client 'c', and as a side effect allocates
|
|
|
|
* the tracking table if needed. If the 'redirect_to' argument is non zero, the
|
|
|
|
* invalidation messages for this client will be sent to the client ID
|
|
|
|
* specified by the 'redirect_to' argument. Note that if such client will
|
|
|
|
* eventually get freed, we'll send a message to the original client to
|
|
|
|
* inform it of the condition. Multiple clients can redirect the invalidation
|
|
|
|
* messages to the same client ID. */
|
2020-02-21 10:39:42 -05:00
|
|
|
void enableTracking(client *c, uint64_t redirect_to, uint64_t options, robj **prefix, size_t numprefix) {
|
2020-02-12 13:22:04 -05:00
|
|
|
if (!(c->flags & CLIENT_TRACKING)) server.tracking_clients++;
|
2019-07-03 13:16:20 -04:00
|
|
|
c->flags |= CLIENT_TRACKING;
|
2020-02-21 10:39:42 -05:00
|
|
|
c->flags &= ~(CLIENT_TRACKING_BROKEN_REDIR|CLIENT_TRACKING_BCAST|
|
|
|
|
CLIENT_TRACKING_OPTIN|CLIENT_TRACKING_OPTOUT);
|
2019-07-03 13:16:20 -04:00
|
|
|
c->client_tracking_redirection = redirect_to;
|
2019-07-05 06:24:28 -04:00
|
|
|
if (TrackingTable == NULL) {
|
2020-02-07 08:03:43 -05:00
|
|
|
TrackingTable = raxNew();
|
2020-02-10 11:18:11 -05:00
|
|
|
PrefixTable = raxNew();
|
2019-07-05 06:24:28 -04:00
|
|
|
TrackingChannelName = createStringObject("__redis__:invalidate",20);
|
|
|
|
}
|
2020-02-10 11:18:11 -05:00
|
|
|
|
2020-02-21 10:39:42 -05:00
|
|
|
if (options & CLIENT_TRACKING_BCAST) {
|
2020-02-10 11:18:11 -05:00
|
|
|
c->flags |= CLIENT_TRACKING_BCAST;
|
|
|
|
if (numprefix == 0) enableBcastTrackingForPrefix(c,"",0);
|
2020-02-11 11:26:27 -05:00
|
|
|
for (size_t j = 0; j < numprefix; j++) {
|
2020-02-10 11:18:11 -05:00
|
|
|
sds sdsprefix = prefix[j]->ptr;
|
2020-02-11 11:26:27 -05:00
|
|
|
enableBcastTrackingForPrefix(c,sdsprefix,sdslen(sdsprefix));
|
2020-02-10 11:18:11 -05:00
|
|
|
}
|
|
|
|
}
|
2020-02-21 10:39:42 -05:00
|
|
|
c->flags |= options & (CLIENT_TRACKING_OPTIN|CLIENT_TRACKING_OPTOUT);
|
2019-07-03 13:16:20 -04:00
|
|
|
}
|
|
|
|
|
2020-02-16 08:16:51 -05:00
|
|
|
/* This function is called after the execution of a readonly command in the
|
2020-02-21 10:39:42 -05:00
|
|
|
* case the client 'c' has keys tracking enabled and the tracking is not
|
|
|
|
* in BCAST mode. It will populate the tracking invalidation table according
|
|
|
|
* to the keys the user fetched, so that Redis will know what are the clients
|
|
|
|
* that should receive an invalidation message with certain groups of keys
|
|
|
|
* are modified. */
|
2019-07-03 13:16:20 -04:00
|
|
|
void trackingRememberKeys(client *c) {
|
2020-02-21 10:39:42 -05:00
|
|
|
/* Return if we are in optin/out mode and the right CACHING command
|
|
|
|
* was/wasn't given in order to modify the default behavior. */
|
|
|
|
uint64_t optin = c->flags & CLIENT_TRACKING_OPTIN;
|
|
|
|
uint64_t optout = c->flags & CLIENT_TRACKING_OPTOUT;
|
|
|
|
uint64_t caching_given = c->flags & CLIENT_TRACKING_CACHING;
|
|
|
|
if ((optin && !caching_given) || (optout && caching_given)) return;
|
|
|
|
|
2019-07-03 13:16:20 -04:00
|
|
|
int numkeys;
|
|
|
|
int *keys = getKeysFromCommand(c->cmd,c->argv,c->argc,&numkeys);
|
|
|
|
if (keys == NULL) return;
|
|
|
|
|
|
|
|
for(int j = 0; j < numkeys; j++) {
|
|
|
|
int idx = keys[j];
|
|
|
|
sds sdskey = c->argv[idx]->ptr;
|
2020-02-07 08:03:43 -05:00
|
|
|
rax *ids = raxFind(TrackingTable,(unsigned char*)sdskey,sdslen(sdskey));
|
|
|
|
if (ids == raxNotFound) {
|
|
|
|
ids = raxNew();
|
|
|
|
int inserted = raxTryInsert(TrackingTable,(unsigned char*)sdskey,
|
|
|
|
sdslen(sdskey),ids, NULL);
|
|
|
|
serverAssert(inserted == 1);
|
2019-07-22 05:47:44 -04:00
|
|
|
}
|
2020-02-07 08:03:43 -05:00
|
|
|
if (raxTryInsert(ids,(unsigned char*)&c->id,sizeof(c->id),NULL,NULL))
|
|
|
|
TrackingTableTotalItems++;
|
2019-07-03 13:16:20 -04:00
|
|
|
}
|
|
|
|
getKeysFreeResult(keys);
|
|
|
|
}
|
|
|
|
|
2020-02-11 12:11:59 -05:00
|
|
|
/* Given a key name, this function sends an invalidation message in the
|
|
|
|
* proper channel (depending on RESP version: PubSub or Push message) and
|
|
|
|
* to the proper client (in case fo redirection), in the context of the
|
|
|
|
* client 'c' with tracking enabled.
|
|
|
|
*
|
|
|
|
* In case the 'proto' argument is non zero, the function will assume that
|
|
|
|
* 'keyname' points to a buffer of 'keylen' bytes already expressed in the
|
|
|
|
* form of Redis RESP protocol, representing an array of keys to send
|
|
|
|
* to the client as value of the invalidation. This is used in BCAST mode
|
|
|
|
* in order to optimized the implementation to use less CPU time. */
|
|
|
|
void sendTrackingMessage(client *c, char *keyname, size_t keylen, int proto) {
|
2019-07-17 08:33:52 -04:00
|
|
|
int using_redirection = 0;
|
|
|
|
if (c->client_tracking_redirection) {
|
|
|
|
client *redir = lookupClientByID(c->client_tracking_redirection);
|
|
|
|
if (!redir) {
|
|
|
|
/* We need to signal to the original connection that we
|
|
|
|
* are unable to send invalidation messages to the redirected
|
|
|
|
* connection, because the client no longer exist. */
|
|
|
|
if (c->resp > 2) {
|
|
|
|
addReplyPushLen(c,3);
|
|
|
|
addReplyBulkCBuffer(c,"tracking-redir-broken",21);
|
|
|
|
addReplyLongLong(c,c->client_tracking_redirection);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
c = redir;
|
|
|
|
using_redirection = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only send such info for clients in RESP version 3 or more. However
|
|
|
|
* if redirection is active, and the connection we redirect to is
|
|
|
|
* in Pub/Sub mode, we can support the feature with RESP 2 as well,
|
|
|
|
* by sending Pub/Sub messages in the __redis__:invalidate channel. */
|
|
|
|
if (c->resp > 2) {
|
|
|
|
addReplyPushLen(c,2);
|
|
|
|
addReplyBulkCBuffer(c,"invalidate",10);
|
|
|
|
} else if (using_redirection && c->flags & CLIENT_PUBSUB) {
|
2020-02-07 08:03:43 -05:00
|
|
|
/* We use a static object to speedup things, however we assume
|
|
|
|
* that addReplyPubsubMessage() will not take a reference. */
|
2020-02-10 07:42:18 -05:00
|
|
|
addReplyPubsubMessage(c,TrackingChannelName,NULL);
|
2020-02-13 10:58:07 -05:00
|
|
|
} else {
|
|
|
|
/* If are here, the client is not using RESP3, nor is
|
|
|
|
* redirecting to another client. We can't send anything to
|
|
|
|
* it since RESP2 does not support push messages in the same
|
|
|
|
* connection. */
|
|
|
|
return;
|
2020-02-11 12:11:59 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Send the "value" part, which is the array of keys. */
|
|
|
|
if (proto) {
|
|
|
|
addReplyProto(c,keyname,keylen);
|
|
|
|
} else {
|
2020-02-10 07:42:18 -05:00
|
|
|
addReplyArrayLen(c,1);
|
2020-02-11 12:11:59 -05:00
|
|
|
addReplyBulkCBuffer(c,keyname,keylen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This function is called when a key is modified in Redis and in the case
|
|
|
|
* we have at least one client with the BCAST mode enabled.
|
|
|
|
* Its goal is to set the key in the right broadcast state if the key
|
|
|
|
* matches one or more prefixes in the prefix table. Later when we
|
|
|
|
* return to the event loop, we'll send invalidation messages to the
|
|
|
|
* clients subscribed to each prefix. */
|
|
|
|
void trackingRememberKeyToBroadcast(char *keyname, size_t keylen) {
|
|
|
|
raxIterator ri;
|
|
|
|
raxStart(&ri,PrefixTable);
|
|
|
|
raxSeek(&ri,"^",NULL,0);
|
|
|
|
while(raxNext(&ri)) {
|
2020-02-12 13:22:04 -05:00
|
|
|
if (ri.key_len > keylen) continue;
|
|
|
|
if (ri.key_len != 0 && memcmp(ri.key,keyname,ri.key_len) != 0)
|
|
|
|
continue;
|
|
|
|
bcastState *bs = ri.data;
|
|
|
|
raxTryInsert(bs->keys,(unsigned char*)keyname,keylen,NULL,NULL);
|
2019-07-17 08:33:52 -04:00
|
|
|
}
|
2020-02-11 12:11:59 -05:00
|
|
|
raxStop(&ri);
|
2019-07-17 08:33:52 -04:00
|
|
|
}
|
|
|
|
|
2020-02-07 08:03:43 -05:00
|
|
|
/* This function is called from signalModifiedKey() or other places in Redis
|
|
|
|
* when a key changes value. In the context of keys tracking, our task here is
|
|
|
|
* to send a notification to every client that may have keys about such caching
|
|
|
|
* slot. */
|
|
|
|
void trackingInvalidateKey(robj *keyobj) {
|
|
|
|
if (TrackingTable == NULL) return;
|
|
|
|
sds sdskey = keyobj->ptr;
|
2020-02-11 12:11:59 -05:00
|
|
|
|
|
|
|
if (raxSize(PrefixTable) > 0)
|
|
|
|
trackingRememberKeyToBroadcast(sdskey,sdslen(sdskey));
|
|
|
|
|
2020-02-07 08:03:43 -05:00
|
|
|
rax *ids = raxFind(TrackingTable,(unsigned char*)sdskey,sdslen(sdskey));
|
2020-03-23 01:07:46 -04:00
|
|
|
if (ids == raxNotFound) return;
|
2019-07-03 13:16:20 -04:00
|
|
|
|
|
|
|
raxIterator ri;
|
2020-02-07 08:03:43 -05:00
|
|
|
raxStart(&ri,ids);
|
2019-07-03 13:16:20 -04:00
|
|
|
raxSeek(&ri,"^",NULL,0);
|
|
|
|
while(raxNext(&ri)) {
|
|
|
|
uint64_t id;
|
2019-12-05 08:37:11 -05:00
|
|
|
memcpy(&id,ri.key,sizeof(id));
|
2019-07-03 13:16:20 -04:00
|
|
|
client *c = lookupClientByID(id);
|
2020-02-14 08:17:10 -05:00
|
|
|
/* Note that if the client is in BCAST mode, we don't want to
|
|
|
|
* send invalidation messages that were pending in the case
|
|
|
|
* previously the client was not in BCAST mode. This can happen if
|
|
|
|
* TRACKING is enabled normally, and then the client switches to
|
|
|
|
* BCAST mode. */
|
|
|
|
if (c == NULL ||
|
|
|
|
!(c->flags & CLIENT_TRACKING)||
|
|
|
|
c->flags & CLIENT_TRACKING_BCAST)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2020-02-11 12:11:59 -05:00
|
|
|
sendTrackingMessage(c,sdskey,sdslen(sdskey),0);
|
2019-07-03 13:16:20 -04:00
|
|
|
}
|
|
|
|
raxStop(&ri);
|
|
|
|
|
|
|
|
/* Free the tracking table: we'll create the radix tree and populate it
|
2019-07-22 12:59:53 -04:00
|
|
|
* again if more keys will be modified in this caching slot. */
|
2020-02-07 08:03:43 -05:00
|
|
|
TrackingTableTotalItems -= raxSize(ids);
|
|
|
|
raxFree(ids);
|
|
|
|
raxRemove(TrackingTable,(unsigned char*)sdskey,sdslen(sdskey),NULL);
|
2019-07-22 12:59:53 -04:00
|
|
|
}
|
|
|
|
|
2019-07-22 06:31:46 -04:00
|
|
|
/* This function is called when one or all the Redis databases are flushed
|
2020-02-16 08:16:51 -05:00
|
|
|
* (dbid == -1 in case of FLUSHALL). Caching keys are not specific for
|
|
|
|
* each DB but are global: currently what we do is send a special
|
2019-07-22 06:31:46 -04:00
|
|
|
* notification to clients with tracking enabled, invalidating the caching
|
2020-02-16 08:16:51 -05:00
|
|
|
* key "", which means, "all the keys", in order to avoid flooding clients
|
2019-07-22 06:31:46 -04:00
|
|
|
* with many invalidation messages for all the keys they may hold.
|
2020-02-16 08:16:51 -05:00
|
|
|
*/
|
2020-02-07 08:03:43 -05:00
|
|
|
void freeTrackingRadixTree(void *rt) {
|
|
|
|
raxFree(rt);
|
|
|
|
}
|
|
|
|
|
2019-07-17 08:33:52 -04:00
|
|
|
void trackingInvalidateKeysOnFlush(int dbid) {
|
2019-07-22 06:31:46 -04:00
|
|
|
if (server.tracking_clients) {
|
|
|
|
listNode *ln;
|
|
|
|
listIter li;
|
|
|
|
listRewind(server.clients,&li);
|
|
|
|
while ((ln = listNext(&li)) != NULL) {
|
|
|
|
client *c = listNodeValue(ln);
|
|
|
|
if (c->flags & CLIENT_TRACKING) {
|
2020-02-11 12:11:59 -05:00
|
|
|
sendTrackingMessage(c,"",1,0);
|
2019-07-22 06:31:46 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* In case of FLUSHALL, reclaim all the memory used by tracking. */
|
|
|
|
if (dbid == -1 && TrackingTable) {
|
2020-02-07 08:03:43 -05:00
|
|
|
raxFreeWithCallback(TrackingTable,freeTrackingRadixTree);
|
2020-02-16 08:16:51 -05:00
|
|
|
TrackingTable = raxNew();
|
2020-02-07 08:03:43 -05:00
|
|
|
TrackingTableTotalItems = 0;
|
2019-07-17 08:33:52 -04:00
|
|
|
}
|
|
|
|
}
|
2019-07-22 12:59:53 -04:00
|
|
|
|
|
|
|
/* Tracking forces Redis to remember information about which client may have
|
2020-02-07 08:03:43 -05:00
|
|
|
* certain keys. In workloads where there are a lot of reads, but keys are
|
|
|
|
* hardly modified, the amount of information we have to remember server side
|
|
|
|
* could be a lot, with the number of keys being totally not bound.
|
2019-07-22 12:59:53 -04:00
|
|
|
*
|
2020-02-07 08:03:43 -05:00
|
|
|
* So Redis allows the user to configure a maximum number of keys for the
|
2019-07-22 12:59:53 -04:00
|
|
|
* 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
|
2020-02-07 08:03:43 -05:00
|
|
|
* a random key, and send invalidation messages to clients like if the key was
|
|
|
|
* modified. */
|
2019-07-22 12:59:53 -04:00
|
|
|
void trackingLimitUsedSlots(void) {
|
2019-07-23 04:57:22 -04:00
|
|
|
static unsigned int timeout_counter = 0;
|
2020-02-07 08:03:43 -05:00
|
|
|
if (TrackingTable == NULL) return;
|
2020-02-07 12:12:45 -05:00
|
|
|
if (server.tracking_table_max_keys == 0) return; /* No limits set. */
|
|
|
|
size_t max_keys = server.tracking_table_max_keys;
|
2020-02-07 08:03:43 -05:00
|
|
|
if (raxSize(TrackingTable) <= max_keys) {
|
2019-07-23 04:57:22 -04:00
|
|
|
timeout_counter = 0;
|
|
|
|
return; /* Limit not reached. */
|
|
|
|
}
|
|
|
|
|
2020-02-07 08:03:43 -05:00
|
|
|
/* We have to invalidate a few keys to reach the limit again. The effort
|
2019-07-23 04:57:22 -04:00
|
|
|
* we do here is proportional to the number of times we entered this
|
|
|
|
* function and found that we are still over the limit. */
|
|
|
|
int effort = 100 * (timeout_counter+1);
|
|
|
|
|
2020-02-07 08:03:43 -05:00
|
|
|
/* We just remove one key after another by using a random walk. */
|
|
|
|
raxIterator ri;
|
|
|
|
raxStart(&ri,TrackingTable);
|
2019-07-23 04:57:22 -04:00
|
|
|
while(effort > 0) {
|
2020-02-07 08:03:43 -05:00
|
|
|
effort--;
|
|
|
|
raxSeek(&ri,"^",NULL,0);
|
|
|
|
raxRandomWalk(&ri,0);
|
|
|
|
rax *ids = ri.data;
|
|
|
|
TrackingTableTotalItems -= raxSize(ids);
|
|
|
|
raxFree(ids);
|
|
|
|
raxRemove(TrackingTable,ri.key,ri.key_len,NULL);
|
|
|
|
if (raxSize(TrackingTable) <= max_keys) {
|
|
|
|
timeout_counter = 0;
|
|
|
|
raxStop(&ri);
|
|
|
|
return; /* Return ASAP: we are again under the limit. */
|
|
|
|
}
|
2019-07-23 04:57:22 -04:00
|
|
|
}
|
2020-02-07 08:03:43 -05:00
|
|
|
|
|
|
|
/* If we reach this point, we were not able to go under the configured
|
|
|
|
* limit using the maximum effort we had for this run. */
|
|
|
|
raxStop(&ri);
|
2019-07-23 04:57:22 -04:00
|
|
|
timeout_counter++;
|
2019-07-22 12:59:53 -04:00
|
|
|
}
|
2019-07-23 05:02:14 -04:00
|
|
|
|
2020-02-11 12:11:59 -05:00
|
|
|
/* This function will run the prefixes of clients in BCAST mode and
|
|
|
|
* keys that were modified about each prefix, and will send the
|
|
|
|
* notifications to each client in each prefix. */
|
|
|
|
void trackingBroadcastInvalidationMessages(void) {
|
|
|
|
raxIterator ri, ri2;
|
2020-02-12 13:22:04 -05:00
|
|
|
|
|
|
|
/* Return ASAP if there is nothing to do here. */
|
|
|
|
if (TrackingTable == NULL || !server.tracking_clients) return;
|
|
|
|
|
2020-02-11 12:11:59 -05:00
|
|
|
raxStart(&ri,PrefixTable);
|
|
|
|
raxSeek(&ri,"^",NULL,0);
|
|
|
|
while(raxNext(&ri)) {
|
|
|
|
bcastState *bs = ri.data;
|
2020-02-12 13:22:04 -05:00
|
|
|
if (raxSize(bs->keys)) {
|
|
|
|
/* Create the array reply with the list of keys once, then send
|
|
|
|
* it to all the clients subscribed to this prefix. */
|
|
|
|
char buf[32];
|
|
|
|
size_t len = ll2string(buf,sizeof(buf),raxSize(bs->keys));
|
|
|
|
sds proto = sdsempty();
|
|
|
|
proto = sdsMakeRoomFor(proto,raxSize(bs->keys)*15);
|
|
|
|
proto = sdscatlen(proto,"*",1);
|
|
|
|
proto = sdscatlen(proto,buf,len);
|
2020-02-11 12:11:59 -05:00
|
|
|
proto = sdscatlen(proto,"\r\n",2);
|
2020-02-12 13:22:04 -05:00
|
|
|
raxStart(&ri2,bs->keys);
|
|
|
|
raxSeek(&ri2,"^",NULL,0);
|
|
|
|
while(raxNext(&ri2)) {
|
|
|
|
len = ll2string(buf,sizeof(buf),ri2.key_len);
|
|
|
|
proto = sdscatlen(proto,"$",1);
|
|
|
|
proto = sdscatlen(proto,buf,len);
|
|
|
|
proto = sdscatlen(proto,"\r\n",2);
|
|
|
|
proto = sdscatlen(proto,ri2.key,ri2.key_len);
|
|
|
|
proto = sdscatlen(proto,"\r\n",2);
|
|
|
|
}
|
|
|
|
raxStop(&ri2);
|
|
|
|
|
|
|
|
/* Send this array of keys to every client in the list. */
|
|
|
|
raxStart(&ri2,bs->clients);
|
|
|
|
raxSeek(&ri2,"^",NULL,0);
|
|
|
|
while(raxNext(&ri2)) {
|
|
|
|
client *c;
|
|
|
|
memcpy(&c,ri2.key,sizeof(c));
|
|
|
|
sendTrackingMessage(c,proto,sdslen(proto),1);
|
|
|
|
}
|
|
|
|
raxStop(&ri2);
|
2020-02-11 12:11:59 -05:00
|
|
|
|
2020-02-12 13:22:04 -05:00
|
|
|
/* Clean up: we can remove everything from this state, because we
|
|
|
|
* want to only track the new keys that will be accumulated starting
|
|
|
|
* from now. */
|
|
|
|
sdsfree(proto);
|
|
|
|
}
|
2020-02-11 12:11:59 -05:00
|
|
|
raxFree(bs->keys);
|
|
|
|
bs->keys = raxNew();
|
|
|
|
}
|
|
|
|
raxStop(&ri);
|
|
|
|
}
|
|
|
|
|
2019-07-23 05:02:14 -04:00
|
|
|
/* This is just used in order to access the amount of used slots in the
|
|
|
|
* tracking table. */
|
2020-02-07 08:03:43 -05:00
|
|
|
uint64_t trackingGetTotalItems(void) {
|
|
|
|
return TrackingTableTotalItems;
|
2019-07-23 05:02:14 -04:00
|
|
|
}
|
2020-02-07 12:12:45 -05:00
|
|
|
|
|
|
|
uint64_t trackingGetTotalKeys(void) {
|
2020-02-14 10:13:58 -05:00
|
|
|
if (TrackingTable == NULL) return 0;
|
2020-02-07 12:12:45 -05:00
|
|
|
return raxSize(TrackingTable);
|
|
|
|
}
|