Skip to content

Commit 772165f

Browse files
Merge pull request #99 from k-r-g/CacheMessageManager_MethodValueCache_Updates
Make CacheMessageManager support the EntityStore method value cache
2 parents f00a790 + 5ca11e7 commit 772165f

File tree

4 files changed

+77
-23
lines changed

4 files changed

+77
-23
lines changed

gemini/src/main/java/com/techempower/cache/EntityStore.java

+45-19
Original file line numberDiff line numberDiff line change
@@ -280,6 +280,45 @@ protected <T extends Identifiable> boolean isIndexed(Class<T> type, String metho
280280
}
281281
}
282282

283+
/**
284+
* Whether the provided class uses the @Indexed annotation and therefore uses
285+
* the method value cache.
286+
*/
287+
public boolean usesMethodValueCache(Class<? extends Identifiable> type)
288+
{
289+
return methodValueCaches.get(type) != null;
290+
}
291+
292+
/**
293+
* Update the specified method value cache.
294+
*/
295+
public void methodValueCacheUpdate(Class<? extends Identifiable> type, long... ids)
296+
{
297+
final MethodValueCache<?> methodValueCache = methodValueCaches.get(type);
298+
if (methodValueCache != null)
299+
{
300+
for (long id : ids)
301+
{
302+
methodValueCache.update(id);
303+
}
304+
}
305+
}
306+
307+
/**
308+
* Delete from the specified method value cache.
309+
*/
310+
public void methodValueCacheDelete(Class<? extends Identifiable> type, long... ids)
311+
{
312+
final MethodValueCache<?> methodValueCache = methodValueCaches.get(type);
313+
if (methodValueCache != null)
314+
{
315+
for (long id : ids)
316+
{
317+
methodValueCache.delete(id);
318+
}
319+
}
320+
}
321+
283322
/**
284323
* Reset all entity groups controlled by this controller.
285324
*/
@@ -1028,14 +1067,9 @@ public <T extends Identifiable> T get(Class<T> type, String methodName, Object v
10281067
public void refresh(Class<? extends Identifiable> type, long... ids)
10291068
{
10301069
getGroupSafe(type).refresh(ids);
1031-
1032-
final MethodValueCache<?> methodValueCache = methodValueCaches.get(type);
1033-
if (methodValueCache != null)
1034-
{
1035-
for (long id : ids) {
1036-
methodValueCache.update(id);
1037-
}
1038-
}
1070+
1071+
// Update index/methodValueCache.
1072+
methodValueCacheUpdate(type, ids);
10391073

10401074
// Notify the listeners.
10411075
notifyListenersCacheObjectExpired(true, type, ids);
@@ -1085,11 +1119,7 @@ public <T extends Identifiable> void put(T entity)
10851119
if (!useAffectedRows || rowsUpdated > 0)
10861120
{
10871121
// Update method value caches.
1088-
final MethodValueCache<?> methodValueCache = methodValueCaches.get(entity.getClass());
1089-
if (methodValueCache != null)
1090-
{
1091-
methodValueCache.update(entity.getId());
1092-
}
1122+
methodValueCacheUpdate(entity.getClass(), entity.getId());
10931123

10941124
// Notify the listeners.
10951125
final CacheListener[] toNotify = listeners;
@@ -1133,12 +1163,8 @@ public <T extends Identifiable> void remove(T entity)
11331163
}
11341164

11351165
// Update method value cache.
1136-
final MethodValueCache<?> methodValueCache = methodValueCaches.get(entity.getClass());
1137-
if (methodValueCache != null)
1138-
{
1139-
methodValueCache.delete(entity.getId());
1140-
}
1141-
1166+
methodValueCacheDelete(entity.getClass(), entity.getId());
1167+
11421168
// Notify the listeners.
11431169
final CacheListener[] toNotify = listeners;
11441170
for (CacheListener listener : toNotify)

gemini/src/main/java/com/techempower/cache/annotation/Indexed.java

+12-4
Original file line numberDiff line numberDiff line change
@@ -29,10 +29,18 @@
2929
import java.lang.annotation.*;
3030

3131
/**
32-
* This annotation is used to mark entity classes or methods as indexed.
33-
* Classes or methods marked as @Indexed should always cache method values,
34-
* even if method value caching is turned off.
32+
* This annotation is used to mark entity classes or methods as indexed. Classes
33+
* or methods marked as @Indexed should always cache method values, even if
34+
* method value caching is turned off.
35+
* <p>
36+
* <b>IMPORTANT</b>: If you use this annotation in an application that uses the
37+
* CacheMessageManager to distribute cache updates to other instances, it is
38+
* your responsibility to ensure that your corresponding EntityGroup
39+
* ("distribute" defaulting to false) or CacheGroup ("distribute" defaulting to
40+
* true) has the "distribute" flag set to true. Otherwise each instance will
41+
* risk having a stale method value cache and you'll get wrong answers from
42+
* EntityStore.get() and list() and honestly it won't be very fun.
3543
*/
36-
@Target({ElementType.METHOD, ElementType.TYPE})
44+
@Target({ ElementType.METHOD, ElementType.TYPE })
3745
@Retention(RetentionPolicy.RUNTIME)
3846
public @interface Indexed { }

gemini/src/main/java/com/techempower/data/EntityGroup.java

+14
Original file line numberDiff line numberDiff line change
@@ -2269,6 +2269,13 @@ public static class Builder<T extends Identifiable>
22692269
* need to notify DistributionListeners. However, if some instances use a
22702270
* CacheGroup for this entity, then it may be useful to set this to true so
22712271
* those instances can update their cache.
2272+
* <p>
2273+
* <b>IMPORTANT</b>: If you use the @Indexed annotation on this entity in an
2274+
* application that uses the CacheMessageManager to distribute cache updates to
2275+
* other instances, it is your responsibility to ensure that "distribute" is set
2276+
* to true. Otherwise each instance will risk having a stale method value cache
2277+
* and you'll get wrong answers from EntityStore.get() and list() and it
2278+
* honestly won't be very fun.
22722279
*/
22732280
protected boolean distribute = false;
22742281

@@ -2364,6 +2371,13 @@ public Builder<T> readOnly()
23642371
/**
23652372
* Specifies updates to the resulting EntityGroup should be passed to
23662373
* DistributionListeners.
2374+
* <p>
2375+
* <b>IMPORTANT</b>: If you use the @Indexed annotation on this entity in an
2376+
* application that uses the CacheMessageManager to distribute cache updates to
2377+
* other instances, it is your responsibility to ensure that "distribute" is set
2378+
* to true. Otherwise each instance will risk having a stale method value cache
2379+
* and you'll get wrong answers from EntityStore.get() and list() and it
2380+
* honestly won't be very fun.
23672381
*/
23682382
public Builder<T> distribute(boolean distribute)
23692383
{

gemini/src/main/java/com/techempower/gemini/cluster/jms/CacheMessageManager.java

+6
Original file line numberDiff line numberDiff line change
@@ -489,6 +489,9 @@ else if (group instanceof EntityGroup)
489489
log.info("Receiving 'cache object expired' but group id is invalid:{}, group: {}",
490490
cacheMessage, group);
491491
}
492+
// Now that the object is updated in the cache, update the method value cache if
493+
// needed.
494+
store.methodValueCacheUpdate(group.getType(), cacheMessage.getObjectId());
492495

493496
break;
494497
}
@@ -514,6 +517,9 @@ else if (group instanceof EntityGroup)
514517
"invalid: {}, group: {}, cacheMessage: {}",
515518
cacheMessage.getGroupId(), group, cacheMessage);
516519
}
520+
// Now that the object is deleted from the cache, also delete from the method
521+
// value cache if needed.
522+
store.methodValueCacheDelete(group.getType(), cacheMessage.getObjectId());
517523
break;
518524
}
519525
case (CacheMessage.ACTION_GROUP_RESET):

0 commit comments

Comments
 (0)