io.ebeaninternal.server.transaction.TableModState Maven / Gradle / Ivy
package io.ebeaninternal.server.transaction;
import io.ebean.cache.QueryCacheEntry;
import io.ebean.cache.QueryCacheEntryValidate;
import io.ebean.cache.ServerCacheNotification;
import io.ebean.cache.ServerCacheNotify;
import io.ebeaninternal.server.core.ClockService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
/**
* Holds timestamp of last modification per table.
*
* This information is used to validate entries in the L2 query caches.
*
*/
public class TableModState implements QueryCacheEntryValidate, ServerCacheNotify {
private static final Logger log = LoggerFactory.getLogger("io.ebean.cache.TABLEMOD");
private final ClockService clockService;
private Map tableModStamp = new ConcurrentHashMap<>();
public TableModState(ClockService clockService) {
this.clockService = clockService;
}
/**
* Set the modified timestamp on the tables that have been touched.
*/
void touch(Set touchedTables, long modTimestamp) {
for (String tableName : touchedTables) {
tableModStamp.put(tableName, modTimestamp);
}
if (log.isDebugEnabled()) {
log.debug("TableModState updated - touched:{} modTimestamp:{}", touchedTables, modTimestamp);
}
}
/**
* Return true if all the tables are valid based on timestamp comparison.
*/
boolean isValid(Set tables, long sinceTimestamp) {
for (String tableName : tables) {
Long modTime = tableModStamp.get(tableName);
if (modTime != null && modTime >= sinceTimestamp ) {
if (log.isTraceEnabled()) {
log.trace("Invalidate on table:{}", tableName);
}
return false;
}
}
return true;
}
@Override
public boolean isValid(QueryCacheEntry entry) {
Set dependentTables = entry.getDependentTables();
if (dependentTables != null && !dependentTables.isEmpty()) {
return isValid(dependentTables, entry.getTimestamp());
}
return true;
}
/**
* Update the table modification timestamps based on remote table modification events.
*
* Generally this is used with distributed caches (Hazelcast, Ignite etc) via topic.
*
*/
@Override
public void notify(ServerCacheNotification notification) {
// use local clock - for slightly more aggressive invalidation (as later)
// that removes any concern regarding clock syncing across cluster
if (log.isDebugEnabled()) {
log.debug("ServerCacheNotification:{}", notification);
}
touch(notification.getDependentTables(), clockService.nowMillis());
}
/**
* Update from Remote transaction event.
*
* Generally this is used with Clustering (ebean-cluster, k8scache).
*
*/
public void notify(RemoteTableMod tableMod) {
// use local clock - for slightly more aggressive invalidation (as later)
// that removes any concern regarding clock syncing across cluster
if (log.isDebugEnabled()) {
log.debug("RemoteTableMod:{}", tableMod);
}
touch(tableMod.getTables(), clockService.nowMillis());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy