See
After installation, you can cache the result of any method of any guice managed bean by adding the @CacheResult
annotation.
@CacheResult(cacheName = "CinemaUtil-sortedMovies")
public List<Map.Entry<Movie, Set<RoleType>>> sortedMovies(Person person) {
...
}
In this case in the magnolia cache configuration automaticly a cache 'CinemaUtil-sortedMovies' will appear.
The cache values may be null
and Optional
. This implementation will arrange that no nulls are stored in the underlying magnolia cache. If the value is Optional
, the value of the Optional
will be serialized.
Non serializable values are only possible if the underlying eh-cache is configured not to store to disk.
Sadly, "model classes are not instantiated by guice, but by Magnolia itself", so they cannot be proxied by guice.
Download the most recent jar from: https://oss.sonatype.org/content/repositories/snapshots/nl/vpro/jsr107-magnolia and install it like you’d normally would.
Or you can add this to your pom.xml
<dependency>
<groupId>nl.vpro</groupId>
<artifactId>jsr107-magnolia</artifactId>
<version>1.19</version>
</dependency>
For versions older then 1.14 caches were configured like so in the JCR-tree: From 1.14 onwards Magnolia 5.5.4 uses ehcache3 so the configuration has changed and looks like this:
Cache-configurations can be automaticly created using tasks on the version handler of your module. E.g. like this:
@Slf4j
public class CinemaVersionHandler extends DefaultModuleVersionHandler {
@Override
protected List<Task> getBasicInstallTasks(InstallContext installContext) {
List<Task> tasks = super.getBasicInstallTasks(installContext);
tasks.addAll(CreateConfigurationTasks.createConfigurationTasks(CinemaUtilWithCaching.class));
log.info("Created tasks {}", tasks);
return tasks;
}
}
Default settings could be configured using the nl.vpro.magnolia.jsr107.DefaultCacheSettings
annotation:
@CacheResult(cacheName = "CinemaUtil-scheduleForChannel")
@DefaultCacheSettings(blockingTimeout = 30000)
List<ScheduleItem> scheduleForChannel(String channel, LocalDate date) {
log.info("Getting movies for {} {}", channel, date);
MediaSearch search = new MediaSearch();
....
If you use an 'exception cache' too, you may want to configure this separately. You need to wrap a @nl.vpro.magnolia.jsr107.Defaults
then.
@CacheResult(cacheKeyGenerator = ImageCacheKey.class, cacheName = ASSET_LINKS_CACHE, exceptionCacheName = ASSET_LINKS_CACHE + "-exceptions")
@Defaults(
overrideOnUpdate = true,
exceptionCacheSettings = @DefaultCacheSettings(maxElementsInMemory = 200, timeToLiveSeconds = 300),
cacheSettings = @DefaultCacheSettings(maxElementsInMemory = 2000, timeToLiveSeconds = 3600)
)
@Override
public String getAssetLink(Image image, String variation) {
Actually the code can also be accessed if you want to configure a cache programmaticly for some other reason. This more or less eliminates the need to configure cache outside code altogether. The cache settings are in this way still visible in the JCR-tree, and can be modified and viewed via JMX, but they can be maintained in the code of your application.
// Create browser cache for api clients
setInstallOrUpdateTask(CreateCacheConfigurationTask.builder()
.name(CACHE)
.settings(CacheSettings.builder()
.eternal(true)
.overflowToDisk(true)
.diskSpoolBufferSizeMB(500)
.maxElementsInMemory(200)
.diskExpiryThreadInterval(Duration.ofHours(24))
)
.overrideOnUpdate(true)
.build());
The nl.vpro.magnolia.jsr107.MgnlCacheManager
implementation of javax.cache.CacheManager
contains a few utilities which may come in useful when interacting with caches. E.g. utilities to get existing values from the caches, or to retrieve all keys, which can be used when activily refreshing entries in the cache (e.g. in conjection with @javax.cache.annotation.CachePut
)
A MgnlCacheManager
can simply be obtained using @Inject
.
Since version 1.16 we will also support cache listening. E.g.
@Inject
public MediaPlayerPageCache(
Provider<VtkUtil> vtkUtil,
ServerConfiguration serverConfiguration,
Provider<MgnlCacheManager> mgnlCacheManager,
@Named(SystemEventBus.NAME) EventBus systemEventBus
) {
this.vtkUtil = vtkUtil;
this.serverConfiguration = serverConfiguration;
this.mgnlCacheManager = mgnlCacheManager;
systemEventBus.addHandler(ModulesStartedEvent.class, this::registerCacheEntryListener);
}
protected void registerCacheEntryListener(ModulesStartedEvent event) {
log.info("{}", event);
mgnlCacheManager.get().getCache(CACHE_NAME).registerCacheEntryListener(new MutableCacheEntryListenerConfiguration<>(
new FactoryBuilder.SingletonFactory<>(new Listener()),
new FactoryBuilder.SingletonFactory<>(e -> true),
false,
true));
}