@@ -750,16 +750,19 @@ GetFieldRes hashTypeGetValue(redisDb *db, robj *o, sds field, unsigned char **vs
750
750
propagateHashFieldDeletion (db , key , field , sdslen (field ));
751
751
752
752
/* 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 );
753
757
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 );
756
760
dbDelete (db ,keyObj );
757
- signalModifiedKey (NULL , db , keyObj );
758
- decrRefCount (keyObj );
759
- return GETF_EXPIRED_HASH ;
761
+ res = GETF_EXPIRED_HASH ;
760
762
}
761
-
762
- return GETF_EXPIRED ;
763
+ signalModifiedKey (NULL , db , keyObj );
764
+ decrRefCount (keyObj );
765
+ return res ;
763
766
}
764
767
765
768
/* Like hashTypeGetValue() but returns a Redis object, which is useful for
@@ -1155,10 +1158,12 @@ void hashTypeSetExDone(HashTypeSetEx *ex) {
1155
1158
if (ex -> fieldDeleted && hashTypeLength (ex -> hashObj , 0 ) == 0 ) {
1156
1159
dbDelete (ex -> db ,ex -> key );
1157
1160
signalModifiedKey (ex -> c , ex -> db , ex -> key );
1161
+ notifyKeyspaceEvent (NOTIFY_HASH , "hexpired" , ex -> key , ex -> db -> id );
1158
1162
notifyKeyspaceEvent (NOTIFY_GENERIC ,"del" ,ex -> key , ex -> db -> id );
1159
1163
} else {
1160
1164
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 );
1162
1167
1163
1168
/* If minimum HFE of the hash is smaller than expiration time of the
1164
1169
* specified fields in the command as well as it is smaller or equal
@@ -1819,16 +1824,23 @@ static ExpireAction hashTypeActiveExpire(eItem _hashObj, void *ctx) {
1819
1824
/* Update quota left */
1820
1825
activeExpireCtx -> fieldsToExpireQuota -= info .itemsExpired ;
1821
1826
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 );
1824
1833
if (hashTypeLength (hashObj , 0 ) == 0 ) {
1825
- robj * key = createStringObject (keystr , sdslen (keystr ));
1826
1834
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 );
1831
1836
}
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 ) {
1832
1844
return ACT_REMOVE_EXP_ITEM ;
1833
1845
} else {
1834
1846
/* Hash has more fields to expire. Update next expiration time of the hash
@@ -2164,7 +2176,7 @@ void hincrbyfloatCommand(client *c) {
2164
2176
decrRefCount (newobj );
2165
2177
}
2166
2178
2167
- static GetFieldRes addHashFieldToReply (client * c , robj * o , sds field ) {
2179
+ static GetFieldRes addHashFieldToReply (client * c , robj * o , sds field , int hfeFlags ) {
2168
2180
if (o == NULL ) {
2169
2181
addReplyNull (c );
2170
2182
return GETF_NOT_FOUND ;
@@ -2174,8 +2186,7 @@ static GetFieldRes addHashFieldToReply(client *c, robj *o, sds field) {
2174
2186
unsigned int vlen = UINT_MAX ;
2175
2187
long long vll = LLONG_MAX ;
2176
2188
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 );
2179
2190
if (res == GETF_OK ) {
2180
2191
if (vstr ) {
2181
2192
addReplyBulkCBuffer (c , vstr , vlen );
@@ -2194,13 +2205,14 @@ void hgetCommand(client *c) {
2194
2205
if ((o = lookupKeyReadOrReply (c ,c -> argv [1 ],shared .null [c -> resp ])) == NULL ||
2195
2206
checkType (c ,o ,OBJ_HASH )) return ;
2196
2207
2197
- addHashFieldToReply (c , o , c -> argv [2 ]-> ptr );
2208
+ addHashFieldToReply (c , o , c -> argv [2 ]-> ptr , HFE_LAZY_EXPIRE );
2198
2209
}
2199
2210
2200
2211
void hmgetCommand (client * c ) {
2201
2212
GetFieldRes res = GETF_OK ;
2202
2213
robj * o ;
2203
2214
int i ;
2215
+ int expired = 0 , deleted = 0 ;
2204
2216
2205
2217
/* Don't abort when the key cannot be found. Non-existing keys are empty
2206
2218
* hashes, where HMGET should respond with a series of null bulks. */
@@ -2209,17 +2221,22 @@ void hmgetCommand(client *c) {
2209
2221
2210
2222
addReplyArrayLen (c , c -> argc - 2 );
2211
2223
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 );
2221
2232
}
2222
2233
}
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
+ }
2223
2240
}
2224
2241
2225
2242
void hdelCommand (client * c ) {
0 commit comments