mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 08:08:53 -05:00
objectComputeSize(): estimate collections sampling N elements.
For most tasks, we need the memory estimation to be O(1) by default. This commit also implements an initial MEMORY command. Note that objectComputeSize() takes the number of samples to check as argument, so MEMORY should be able to get the sample size as option to make precision VS CPU tradeoff tunable. Related to: PR #3223.
This commit is contained in:
parent
8c24325f8f
commit
89dec6921d
78
src/object.c
78
src/object.c
@ -694,16 +694,17 @@ char *strEncoding(int encoding) {
|
|||||||
|
|
||||||
/* ========================== Objects introspection ========================= */
|
/* ========================== Objects introspection ========================= */
|
||||||
|
|
||||||
/* Returns the size in bytes consumed by the key's value in RAM */
|
/* Returns the size in bytes consumed by the key's value in RAM.
|
||||||
size_t objectComputeSize(robj *o) {
|
* Note that the returned value is just an approximation, especially in the
|
||||||
|
* case of aggregated data types where only "sample_size" elements
|
||||||
|
* are checked and averaged to estimate the total size. */
|
||||||
|
#define OBJ_COMPUTE_SIZE_DEF_SAMPLES 5 /* Default sample size. */
|
||||||
|
size_t objectComputeSize(robj *o, size_t sample_size) {
|
||||||
robj *ele;
|
robj *ele;
|
||||||
list *l;
|
|
||||||
listNode *ln;
|
|
||||||
dict *d;
|
dict *d;
|
||||||
dictIterator *di;
|
dictIterator *di;
|
||||||
listIter li;
|
|
||||||
struct dictEntry *de;
|
struct dictEntry *de;
|
||||||
size_t asize = 0, elesize;
|
size_t asize = 0, elesize = 0, samples = 0;
|
||||||
|
|
||||||
if (o->type == OBJ_STRING) {
|
if (o->type == OBJ_STRING) {
|
||||||
if(o->encoding == OBJ_ENCODING_INT) {
|
if(o->encoding == OBJ_ENCODING_INT) {
|
||||||
@ -722,20 +723,12 @@ size_t objectComputeSize(robj *o) {
|
|||||||
quicklistNode *node = ql->head;
|
quicklistNode *node = ql->head;
|
||||||
asize = sizeof(*o)+sizeof(quicklist);
|
asize = sizeof(*o)+sizeof(quicklist);
|
||||||
do {
|
do {
|
||||||
asize += sizeof(quicklistNode)+ziplistBlobLen(node->zl);
|
elesize += sizeof(quicklistNode)+ziplistBlobLen(node->zl);
|
||||||
} while ((node = node->next));
|
samples++;
|
||||||
|
} while ((node = node->next) && samples < sample_size);
|
||||||
|
asize += (double)elesize/samples*listTypeLength(o);
|
||||||
} else if (o->encoding == OBJ_ENCODING_ZIPLIST) {
|
} else if (o->encoding == OBJ_ENCODING_ZIPLIST) {
|
||||||
asize = sizeof(*o)+ziplistBlobLen(o->ptr);
|
asize = sizeof(*o)+ziplistBlobLen(o->ptr);
|
||||||
} else if (o->encoding == OBJ_ENCODING_LINKEDLIST) {
|
|
||||||
l = o->ptr;
|
|
||||||
asize = sizeof(*o)+sizeof(list);
|
|
||||||
listRewind(l,&li);
|
|
||||||
while((ln = listNext(&li))) {
|
|
||||||
ele = ln->value;
|
|
||||||
elesize = (ele->encoding == OBJ_ENCODING_RAW) ?
|
|
||||||
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
|
||||||
asize += (sizeof(listNode)+elesize);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
serverPanic("Unknown list encoding");
|
serverPanic("Unknown list encoding");
|
||||||
}
|
}
|
||||||
@ -744,13 +737,15 @@ size_t objectComputeSize(robj *o) {
|
|||||||
d = o->ptr;
|
d = o->ptr;
|
||||||
di = dictGetIterator(d);
|
di = dictGetIterator(d);
|
||||||
asize = sizeof(*o)+sizeof(dict)+(sizeof(struct dictEntry*)*dictSlots(d));
|
asize = sizeof(*o)+sizeof(dict)+(sizeof(struct dictEntry*)*dictSlots(d));
|
||||||
while((de = dictNext(di)) != NULL) {
|
while((de = dictNext(di)) != NULL && samples < sample_size) {
|
||||||
ele = dictGetKey(de);
|
ele = dictGetKey(de);
|
||||||
elesize = (ele->encoding == OBJ_ENCODING_RAW) ?
|
elesize += (ele->encoding == OBJ_ENCODING_RAW) ?
|
||||||
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
||||||
asize += (sizeof(struct dictEntry)+elesize);
|
elesize += sizeof(struct dictEntry);
|
||||||
|
samples++;
|
||||||
}
|
}
|
||||||
dictReleaseIterator(di);
|
dictReleaseIterator(di);
|
||||||
|
if (samples) asize += (double)elesize/samples*dictSize(d);
|
||||||
} else if (o->encoding == OBJ_ENCODING_INTSET) {
|
} else if (o->encoding == OBJ_ENCODING_INTSET) {
|
||||||
intset *is = o->ptr;
|
intset *is = o->ptr;
|
||||||
asize = sizeof(*o)+sizeof(*is)+is->encoding*is->length;
|
asize = sizeof(*o)+sizeof(*is)+is->encoding*is->length;
|
||||||
@ -764,14 +759,16 @@ size_t objectComputeSize(robj *o) {
|
|||||||
d = ((zset*)o->ptr)->dict;
|
d = ((zset*)o->ptr)->dict;
|
||||||
di = dictGetIterator(d);
|
di = dictGetIterator(d);
|
||||||
asize = sizeof(*o)+sizeof(zset)+(sizeof(struct dictEntry*)*dictSlots(d));
|
asize = sizeof(*o)+sizeof(zset)+(sizeof(struct dictEntry*)*dictSlots(d));
|
||||||
while((de = dictNext(di)) != NULL) {
|
while((de = dictNext(di)) != NULL && samples < sample_size) {
|
||||||
ele = dictGetKey(de);
|
ele = dictGetKey(de);
|
||||||
elesize = (ele->encoding == OBJ_ENCODING_RAW) ?
|
elesize += (ele->encoding == OBJ_ENCODING_RAW) ?
|
||||||
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
||||||
asize += (sizeof(struct dictEntry)+elesize);
|
elesize += sizeof(struct dictEntry);
|
||||||
asize += sizeof(zskiplistNode)*dictSize(d);
|
elesize += sizeof(zskiplistNode)*dictSize(d);
|
||||||
|
samples++;
|
||||||
}
|
}
|
||||||
dictReleaseIterator(di);
|
dictReleaseIterator(di);
|
||||||
|
if (samples) asize += (double)elesize/samples*dictSize(d);
|
||||||
} else {
|
} else {
|
||||||
serverPanic("Unknown sorted set encoding");
|
serverPanic("Unknown sorted set encoding");
|
||||||
}
|
}
|
||||||
@ -782,16 +779,19 @@ size_t objectComputeSize(robj *o) {
|
|||||||
d = o->ptr;
|
d = o->ptr;
|
||||||
di = dictGetIterator(d);
|
di = dictGetIterator(d);
|
||||||
asize = sizeof(*o)+sizeof(dict)+(sizeof(struct dictEntry*)*dictSlots(d));
|
asize = sizeof(*o)+sizeof(dict)+(sizeof(struct dictEntry*)*dictSlots(d));
|
||||||
while((de = dictNext(di)) != NULL) {
|
while((de = dictNext(di)) != NULL && samples < sample_size) {
|
||||||
ele = dictGetKey(de);
|
ele = dictGetKey(de);
|
||||||
elesize = (ele->encoding == OBJ_ENCODING_RAW) ?
|
elesize += (ele->encoding == OBJ_ENCODING_RAW) ?
|
||||||
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
||||||
ele = dictGetVal(de);
|
ele = dictGetVal(de);
|
||||||
elesize = (ele->encoding == OBJ_ENCODING_RAW) ?
|
elesize += (ele->encoding == OBJ_ENCODING_RAW) ?
|
||||||
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
(sizeof(*o)+sdsAllocSize(ele->ptr)) : sizeof(*o);
|
||||||
asize += (sizeof(struct dictEntry)+elesize);
|
elesize += sizeof(struct dictEntry);
|
||||||
|
samples++;
|
||||||
|
printf("%zu samples: %zu usage\n", samples, elesize);
|
||||||
}
|
}
|
||||||
dictReleaseIterator(di);
|
dictReleaseIterator(di);
|
||||||
|
if (samples) asize += (double)elesize/samples*dictSize(d);
|
||||||
} else {
|
} else {
|
||||||
serverPanic("Unknown hash encoding");
|
serverPanic("Unknown hash encoding");
|
||||||
}
|
}
|
||||||
@ -801,7 +801,7 @@ size_t objectComputeSize(robj *o) {
|
|||||||
return asize;
|
return asize;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ============================ The OBJECT command ========================== */
|
/* ======================= The OBJECT and MEMORY commands =================== */
|
||||||
|
|
||||||
/* This is a helper function for the OBJECT command. We need to lookup keys
|
/* This is a helper function for the OBJECT command. We need to lookup keys
|
||||||
* without any modification of LRU or other parameters. */
|
* without any modification of LRU or other parameters. */
|
||||||
@ -853,3 +853,21 @@ void objectCommand(client *c) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* The memory command will eventually be a complete interface for the
|
||||||
|
* memory introspection capabilities of Redis.
|
||||||
|
*
|
||||||
|
* Usage: MEMORY usage <key> */
|
||||||
|
void memoryCommand(client *c) {
|
||||||
|
robj *o;
|
||||||
|
|
||||||
|
if (!strcasecmp(c->argv[1]->ptr,"usage") && c->argc == 3) {
|
||||||
|
if ((o = objectCommandLookupOrReply(c,c->argv[2],shared.nullbulk))
|
||||||
|
== NULL) return;
|
||||||
|
size_t usage = objectComputeSize(o,OBJ_COMPUTE_SIZE_DEF_SAMPLES);
|
||||||
|
usage += sdsAllocSize(c->argv[1]->ptr);
|
||||||
|
usage += sizeof(dictEntry);
|
||||||
|
addReplyLongLong(c,usage);
|
||||||
|
} else {
|
||||||
|
addReplyError(c,"Syntax error. Try MEMORY usage <key>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -274,6 +274,7 @@ struct redisCommand redisCommandTable[] = {
|
|||||||
{"readwrite",readwriteCommand,1,"F",0,NULL,0,0,0,0,0},
|
{"readwrite",readwriteCommand,1,"F",0,NULL,0,0,0,0,0},
|
||||||
{"dump",dumpCommand,2,"r",0,NULL,1,1,1,0,0},
|
{"dump",dumpCommand,2,"r",0,NULL,1,1,1,0,0},
|
||||||
{"object",objectCommand,3,"r",0,NULL,2,2,2,0,0},
|
{"object",objectCommand,3,"r",0,NULL,2,2,2,0,0},
|
||||||
|
{"memory",memoryCommand,3,"r",0,NULL,0,0,0,0,0},
|
||||||
{"client",clientCommand,-2,"as",0,NULL,0,0,0,0,0},
|
{"client",clientCommand,-2,"as",0,NULL,0,0,0,0,0},
|
||||||
{"eval",evalCommand,-3,"s",0,evalGetKeys,0,0,0,0,0},
|
{"eval",evalCommand,-3,"s",0,evalGetKeys,0,0,0,0,0},
|
||||||
{"evalsha",evalShaCommand,-3,"s",0,evalGetKeys,0,0,0,0,0},
|
{"evalsha",evalShaCommand,-3,"s",0,evalGetKeys,0,0,0,0,0},
|
||||||
|
@ -532,7 +532,7 @@ typedef struct RedisModuleIO {
|
|||||||
#define OBJ_ENCODING_INT 1 /* Encoded as integer */
|
#define OBJ_ENCODING_INT 1 /* Encoded as integer */
|
||||||
#define OBJ_ENCODING_HT 2 /* Encoded as hash table */
|
#define OBJ_ENCODING_HT 2 /* Encoded as hash table */
|
||||||
#define OBJ_ENCODING_ZIPMAP 3 /* Encoded as zipmap */
|
#define OBJ_ENCODING_ZIPMAP 3 /* Encoded as zipmap */
|
||||||
#define OBJ_ENCODING_LINKEDLIST 4 /* Encoded as regular linked list */
|
#define OBJ_ENCODING_LINKEDLIST 4 /* No longer used: old list encoding. */
|
||||||
#define OBJ_ENCODING_ZIPLIST 5 /* Encoded as ziplist */
|
#define OBJ_ENCODING_ZIPLIST 5 /* Encoded as ziplist */
|
||||||
#define OBJ_ENCODING_INTSET 6 /* Encoded as intset */
|
#define OBJ_ENCODING_INTSET 6 /* Encoded as intset */
|
||||||
#define OBJ_ENCODING_SKIPLIST 7 /* Encoded as skiplist */
|
#define OBJ_ENCODING_SKIPLIST 7 /* Encoded as skiplist */
|
||||||
@ -1792,6 +1792,7 @@ void readonlyCommand(client *c);
|
|||||||
void readwriteCommand(client *c);
|
void readwriteCommand(client *c);
|
||||||
void dumpCommand(client *c);
|
void dumpCommand(client *c);
|
||||||
void objectCommand(client *c);
|
void objectCommand(client *c);
|
||||||
|
void memoryCommand(client *c);
|
||||||
void clientCommand(client *c);
|
void clientCommand(client *c);
|
||||||
void evalCommand(client *c);
|
void evalCommand(client *c);
|
||||||
void evalShaCommand(client *c);
|
void evalShaCommand(client *c);
|
||||||
|
Loading…
Reference in New Issue
Block a user