DEBUG DIGEST refactoring: extract function to digest a value.

This commit is contained in:
antirez 2018-12-07 16:30:33 +01:00
parent 0006c989d2
commit e9400e8efd

View File

@ -74,7 +74,7 @@ void xorDigest(unsigned char *digest, void *ptr, size_t len) {
digest[j] ^= hash[j]; digest[j] ^= hash[j];
} }
void xorObjectDigest(unsigned char *digest, robj *o) { void xorStringObjectDigest(unsigned char *digest, robj *o) {
o = getDecodedObject(o); o = getDecodedObject(o);
xorDigest(digest,o->ptr,sdslen(o->ptr)); xorDigest(digest,o->ptr,sdslen(o->ptr));
decrRefCount(o); decrRefCount(o);
@ -104,66 +104,35 @@ void mixDigest(unsigned char *digest, void *ptr, size_t len) {
SHA1Final(digest,&ctx); SHA1Final(digest,&ctx);
} }
void mixObjectDigest(unsigned char *digest, robj *o) { void mixStringObjectDigest(unsigned char *digest, robj *o) {
o = getDecodedObject(o); o = getDecodedObject(o);
mixDigest(digest,o->ptr,sdslen(o->ptr)); mixDigest(digest,o->ptr,sdslen(o->ptr));
decrRefCount(o); decrRefCount(o);
} }
/* Compute the dataset digest. Since keys, sets elements, hashes elements /* This function computes the digest of a data structure stored in the
* are not ordered, we use a trick: every aggregate digest is the xor * object 'o'. It is the core of the DEBUG DIGEST command: when taking the
* of the digests of their elements. This way the order will not change * digest of a whole dataset, we take the digest of the key and the value
* the result. For list instead we use a feedback entering the output digest * pair, and xor all those together.
* as input in order to ensure that a different ordered list will result in *
* a different digest. */ * Note that this function does not reset the initial 'digest' passed, it
void computeDatasetDigest(unsigned char *final) { * will continue mixing this object digest to anything that was already
unsigned char digest[20]; * present. */
char buf[128]; void xorObjectDigest(redisDb *db, robj *keyobj, unsigned char *digest, robj *o) {
dictIterator *di = NULL; uint32_t aux = htonl(o->type);
dictEntry *de;
int j;
uint32_t aux;
memset(final,0,20); /* Start with a clean result */
for (j = 0; j < server.dbnum; j++) {
redisDb *db = server.db+j;
if (dictSize(db->dict) == 0) continue;
di = dictGetSafeIterator(db->dict);
/* hash the DB id, so the same dataset moved in a different
* DB will lead to a different digest */
aux = htonl(j);
mixDigest(final,&aux,sizeof(aux));
/* Iterate this DB writing every entry */
while((de = dictNext(di)) != NULL) {
sds key;
robj *keyobj, *o;
long long expiretime;
memset(digest,0,20); /* This key-val digest */
key = dictGetKey(de);
keyobj = createStringObject(key,sdslen(key));
mixDigest(digest,key,sdslen(key));
o = dictGetVal(de);
aux = htonl(o->type);
mixDigest(digest,&aux,sizeof(aux)); mixDigest(digest,&aux,sizeof(aux));
expiretime = getExpire(db,keyobj); long long expiretime = getExpire(db,keyobj);
char buf[128];
/* Save the key and associated value */ /* Save the key and associated value */
if (o->type == OBJ_STRING) { if (o->type == OBJ_STRING) {
mixObjectDigest(digest,o); mixStringObjectDigest(digest,o);
} else if (o->type == OBJ_LIST) { } else if (o->type == OBJ_LIST) {
listTypeIterator *li = listTypeInitIterator(o,0,LIST_TAIL); listTypeIterator *li = listTypeInitIterator(o,0,LIST_TAIL);
listTypeEntry entry; listTypeEntry entry;
while(listTypeNext(li,&entry)) { while(listTypeNext(li,&entry)) {
robj *eleobj = listTypeGet(&entry); robj *eleobj = listTypeGet(&entry);
mixObjectDigest(digest,eleobj); mixStringObjectDigest(digest,eleobj);
decrRefCount(eleobj); decrRefCount(eleobj);
} }
listTypeReleaseIterator(li); listTypeReleaseIterator(li);
@ -278,6 +247,48 @@ void computeDatasetDigest(unsigned char *final) {
} }
/* If the key has an expire, add it to the mix */ /* If the key has an expire, add it to the mix */
if (expiretime != -1) xorDigest(digest,"!!expire!!",10); if (expiretime != -1) xorDigest(digest,"!!expire!!",10);
}
/* Compute the dataset digest. Since keys, sets elements, hashes elements
* are not ordered, we use a trick: every aggregate digest is the xor
* of the digests of their elements. This way the order will not change
* the result. For list instead we use a feedback entering the output digest
* as input in order to ensure that a different ordered list will result in
* a different digest. */
void computeDatasetDigest(unsigned char *final) {
unsigned char digest[20];
dictIterator *di = NULL;
dictEntry *de;
int j;
uint32_t aux;
memset(final,0,20); /* Start with a clean result */
for (j = 0; j < server.dbnum; j++) {
redisDb *db = server.db+j;
if (dictSize(db->dict) == 0) continue;
di = dictGetSafeIterator(db->dict);
/* hash the DB id, so the same dataset moved in a different
* DB will lead to a different digest */
aux = htonl(j);
mixDigest(final,&aux,sizeof(aux));
/* Iterate this DB writing every entry */
while((de = dictNext(di)) != NULL) {
sds key;
robj *keyobj, *o;
memset(digest,0,20); /* This key-val digest */
key = dictGetKey(de);
keyobj = createStringObject(key,sdslen(key));
mixDigest(digest,key,sdslen(key));
o = dictGetVal(de);
xorObjectDigest(db,keyobj,digest,o);
/* We can finally xor the key-val digest to the final digest */ /* We can finally xor the key-val digest to the final digest */
xorDigest(final,digest,20); xorDigest(final,digest,20);
decrRefCount(keyobj); decrRefCount(keyobj);