Skip to content

Commit ed10f73

Browse files
sundbtezcmoticless
authored
Add new hexpired notification for HFE (#13329)
When the hash field expired, we will send a new `hexpired` notification. It mainly includes the following three cases: 1. When field expired by active expiration. 2. When field expired by lazy expiration. 3. When the user uses the `h(p)expire(at)` command, the user will also get a `hexpired` notification if the field expires during the command. ## Improvement 1. Now if more than one field expires in the hmget command, we will only send a `hexpired` notification. 2. When a field with TTL is deleted by commands like hdel without updating the global DS, active expire will not send a notification. --------- Co-authored-by: Ozan Tezcan <[email protected]> Co-authored-by: Moti Cohen <[email protected]>
1 parent f01fdc3 commit ed10f73

File tree

3 files changed

+81
-31
lines changed

3 files changed

+81
-31
lines changed

src/server.h

+2
Original file line numberDiff line numberDiff line change
@@ -3195,6 +3195,8 @@ typedef struct dictExpireMetadata {
31953195
#define HFE_LAZY_EXPIRE (0) /* Delete expired field, and if last field also the hash */
31963196
#define HFE_LAZY_AVOID_FIELD_DEL (1<<0) /* Avoid deleting expired field */
31973197
#define HFE_LAZY_AVOID_HASH_DEL (1<<1) /* Avoid deleting hash if the field is the last one */
3198+
#define HFE_LAZY_NO_NOTIFICATION (1<<2) /* Do not send notification, used when multiple fields
3199+
* may expire and only one notification is desired. */
31983200

31993201
void hashTypeConvert(robj *o, int enc, ebuckets *hexpires);
32003202
void hashTypeTryConversion(redisDb *db, robj *subject, robj **argv, int start, int end);

src/t_hash.c

+45-28
Original file line numberDiff line numberDiff line change
@@ -750,16 +750,19 @@ GetFieldRes hashTypeGetValue(redisDb *db, robj *o, sds field, unsigned char **vs
750750
propagateHashFieldDeletion(db, key, field, sdslen(field));
751751

752752
/* If the field is the last one in the hash, then the hash will be deleted */
753+
res = GETF_EXPIRED;
754+
robj *keyObj = createStringObject(key, sdslen(key));
755+
if (!(hfeFlags & HFE_LAZY_NO_NOTIFICATION))
756+
notifyKeyspaceEvent(NOTIFY_HASH, "hexpired", keyObj, db->id);
753757
if ((hashTypeLength(o, 0) == 0) && (!(hfeFlags & HFE_LAZY_AVOID_HASH_DEL))) {
754-
robj *keyObj = createStringObject(key, sdslen(key));
755-
notifyKeyspaceEvent(NOTIFY_GENERIC, "del", keyObj, db->id);
758+
if (!(hfeFlags & HFE_LAZY_NO_NOTIFICATION))
759+
notifyKeyspaceEvent(NOTIFY_GENERIC, "del", keyObj, db->id);
756760
dbDelete(db,keyObj);
757-
signalModifiedKey(NULL, db, keyObj);
758-
decrRefCount(keyObj);
759-
return GETF_EXPIRED_HASH;
761+
res = GETF_EXPIRED_HASH;
760762
}
761-
762-
return GETF_EXPIRED;
763+
signalModifiedKey(NULL, db, keyObj);
764+
decrRefCount(keyObj);
765+
return res;
763766
}
764767

765768
/* Like hashTypeGetValue() but returns a Redis object, which is useful for
@@ -1155,10 +1158,12 @@ void hashTypeSetExDone(HashTypeSetEx *ex) {
11551158
if (ex->fieldDeleted && hashTypeLength(ex->hashObj, 0) == 0) {
11561159
dbDelete(ex->db,ex->key);
11571160
signalModifiedKey(ex->c, ex->db, ex->key);
1161+
notifyKeyspaceEvent(NOTIFY_HASH, "hexpired", ex->key, ex->db->id);
11581162
notifyKeyspaceEvent(NOTIFY_GENERIC,"del",ex->key, ex->db->id);
11591163
} else {
11601164
signalModifiedKey(ex->c, ex->db, ex->key);
1161-
notifyKeyspaceEvent(NOTIFY_HASH, "hexpire", ex->key, ex->db->id);
1165+
notifyKeyspaceEvent(NOTIFY_HASH, ex->fieldDeleted ? "hexpired" : "hexpire",
1166+
ex->key, ex->db->id);
11621167

11631168
/* If minimum HFE of the hash is smaller than expiration time of the
11641169
* specified fields in the command as well as it is smaller or equal
@@ -1819,16 +1824,23 @@ static ExpireAction hashTypeActiveExpire(eItem _hashObj, void *ctx) {
18191824
/* Update quota left */
18201825
activeExpireCtx->fieldsToExpireQuota -= info.itemsExpired;
18211826

1822-
/* If hash has no more fields to expire, remove it from HFE DB */
1823-
if (info.nextExpireTime == EB_EXPIRE_TIME_INVALID) {
1827+
/* In some cases, a field might have been deleted without updating the global DS.
1828+
* As a result, active-expire might not expire any fields, in such cases,
1829+
* we don't need to send notifications or perform other operations for this key. */
1830+
if (info.itemsExpired) {
1831+
robj *key = createStringObject(keystr, sdslen(keystr));
1832+
notifyKeyspaceEvent(NOTIFY_HASH,"hexpired",key,activeExpireCtx->db->id);
18241833
if (hashTypeLength(hashObj, 0) == 0) {
1825-
robj *key = createStringObject(keystr, sdslen(keystr));
18261834
dbDelete(activeExpireCtx->db, key);
1827-
notifyKeyspaceEvent(NOTIFY_GENERIC,"del",key, activeExpireCtx->db->id);
1828-
server.dirty++;
1829-
signalModifiedKey(NULL, &server.db[0], key);
1830-
decrRefCount(key);
1835+
notifyKeyspaceEvent(NOTIFY_GENERIC,"del",key,activeExpireCtx->db->id);
18311836
}
1837+
server.dirty++;
1838+
signalModifiedKey(NULL, activeExpireCtx->db, key);
1839+
decrRefCount(key);
1840+
}
1841+
1842+
/* If hash has no more fields to expire, remove it from HFE DB */
1843+
if (info.nextExpireTime == EB_EXPIRE_TIME_INVALID) {
18321844
return ACT_REMOVE_EXP_ITEM;
18331845
} else {
18341846
/* Hash has more fields to expire. Update next expiration time of the hash
@@ -2164,7 +2176,7 @@ void hincrbyfloatCommand(client *c) {
21642176
decrRefCount(newobj);
21652177
}
21662178

2167-
static GetFieldRes addHashFieldToReply(client *c, robj *o, sds field) {
2179+
static GetFieldRes addHashFieldToReply(client *c, robj *o, sds field, int hfeFlags) {
21682180
if (o == NULL) {
21692181
addReplyNull(c);
21702182
return GETF_NOT_FOUND;
@@ -2174,8 +2186,7 @@ static GetFieldRes addHashFieldToReply(client *c, robj *o, sds field) {
21742186
unsigned int vlen = UINT_MAX;
21752187
long long vll = LLONG_MAX;
21762188

2177-
GetFieldRes res = hashTypeGetValue(c->db, o, field, &vstr, &vlen, &vll,
2178-
HFE_LAZY_EXPIRE);
2189+
GetFieldRes res = hashTypeGetValue(c->db, o, field, &vstr, &vlen, &vll, hfeFlags);
21792190
if (res == GETF_OK) {
21802191
if (vstr) {
21812192
addReplyBulkCBuffer(c, vstr, vlen);
@@ -2194,13 +2205,14 @@ void hgetCommand(client *c) {
21942205
if ((o = lookupKeyReadOrReply(c,c->argv[1],shared.null[c->resp])) == NULL ||
21952206
checkType(c,o,OBJ_HASH)) return;
21962207

2197-
addHashFieldToReply(c, o, c->argv[2]->ptr);
2208+
addHashFieldToReply(c, o, c->argv[2]->ptr, HFE_LAZY_EXPIRE);
21982209
}
21992210

22002211
void hmgetCommand(client *c) {
22012212
GetFieldRes res = GETF_OK;
22022213
robj *o;
22032214
int i;
2215+
int expired = 0, deleted = 0;
22042216

22052217
/* Don't abort when the key cannot be found. Non-existing keys are empty
22062218
* hashes, where HMGET should respond with a series of null bulks. */
@@ -2209,17 +2221,22 @@ void hmgetCommand(client *c) {
22092221

22102222
addReplyArrayLen(c, c->argc-2);
22112223
for (i = 2; i < c->argc ; i++) {
2212-
2213-
res = addHashFieldToReply(c, o, c->argv[i]->ptr);
2214-
2215-
/* If hash got lazy expired since all fields are expired (o is invalid),
2216-
* then fill the rest with trivial nulls and return */
2217-
if (res == GETF_EXPIRED_HASH) {
2218-
while (++i < c->argc)
2219-
addReplyNull(c);
2220-
return;
2224+
if (!deleted) {
2225+
res = addHashFieldToReply(c, o, c->argv[i]->ptr, HFE_LAZY_NO_NOTIFICATION);
2226+
expired += (res == GETF_EXPIRED);
2227+
deleted += (res == GETF_EXPIRED_HASH);
2228+
} else {
2229+
/* If hash got lazy expired since all fields are expired (o is invalid),
2230+
* then fill the rest with trivial nulls and return. */
2231+
addReplyNull(c);
22212232
}
22222233
}
2234+
2235+
if (expired) {
2236+
notifyKeyspaceEvent(NOTIFY_HASH, "hexpired", c->argv[1], c->db->id);
2237+
if (deleted)
2238+
notifyKeyspaceEvent(NOTIFY_GENERIC, "del", c->argv[1], c->db->id);
2239+
}
22232240
}
22242241

22252242
void hdelCommand(client *c) {

tests/unit/pubsub.tcl

+34-3
Original file line numberDiff line numberDiff line change
@@ -356,25 +356,56 @@ start_server {tags {"pubsub network"}} {
356356
foreach {type max_lp_entries} {listpackex 512 hashtable 0} {
357357
test "Keyspace notifications: hash events test ($type)" {
358358
r config set hash-max-listpack-entries $max_lp_entries
359-
r config set notify-keyspace-events Kh
359+
r config set notify-keyspace-events Khg
360360
r del myhash
361361
set rd1 [redis_deferring_client]
362362
assert_equal {1} [psubscribe $rd1 *]
363-
r hmset myhash yes 1 no 0
363+
r hmset myhash yes 1 no 0 f1 1 f2 2 f3_hdel 3
364364
r hincrby myhash yes 10
365365
r hexpire myhash 999999 FIELDS 1 yes
366366
r hexpireat myhash [expr {[clock seconds] + 999999}] NX FIELDS 1 no
367367
r hpexpire myhash 999999 FIELDS 1 yes
368368
r hpersist myhash FIELDS 1 yes
369+
r hpexpire myhash 0 FIELDS 1 yes
369370
assert_encoding $type myhash
370371
assert_equal "pmessage * __keyspace@${db}__:myhash hset" [$rd1 read]
371372
assert_equal "pmessage * __keyspace@${db}__:myhash hincrby" [$rd1 read]
372373
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
373374
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
374375
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
375376
assert_equal "pmessage * __keyspace@${db}__:myhash hpersist" [$rd1 read]
377+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpired" [$rd1 read]
378+
379+
# Test that we will get `hexpired` notification when
380+
# a hash field is removed by active expire.
381+
r hpexpire myhash 10 FIELDS 1 no
382+
after 100 ;# Wait for active expire
383+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
384+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpired" [$rd1 read]
385+
386+
# Test that when a field with TTL is deleted by commands like hdel without
387+
# updating the global DS, active expire will not send a notification.
388+
r hpexpire myhash 100 FIELDS 1 f3_hdel
389+
r hdel myhash f3_hdel
390+
after 200 ;# Wait for active expire
391+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
392+
assert_equal "pmessage * __keyspace@${db}__:myhash hdel" [$rd1 read]
393+
394+
# Test that we will get `hexpired` notification when
395+
# a hash field is removed by lazy expire.
396+
r debug set-active-expire 0
397+
r hpexpire myhash 10 FIELDS 2 f1 f2
398+
after 20
399+
r hmget myhash f1 f2 ;# Trigger lazy expire
400+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpire" [$rd1 read]
401+
# We should get only one `hexpired` notification even two fields was expired.
402+
assert_equal "pmessage * __keyspace@${db}__:myhash hexpired" [$rd1 read]
403+
# We should get a `del` notification after all fields were expired.
404+
assert_equal "pmessage * __keyspace@${db}__:myhash del" [$rd1 read]
405+
r debug set-active-expire 1
406+
376407
$rd1 close
377-
}
408+
} {0} {needs:debug}
378409
} ;# foreach
379410

380411
test "Keyspace notifications: stream events test" {

0 commit comments

Comments
 (0)