HHH-9881 Pending put needs to be invalidated on update on remote node
* This could lead to performance degradation since new EndInvalidatingCommand needs to be send after transaction is committed
This commit is contained in:
parent
4b2a78785e
commit
fa7265ff0e
|
@ -214,6 +214,20 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
*/
|
*/
|
||||||
public static final String PENDING_PUTS_CACHE_NAME = "pending-puts";
|
public static final String PENDING_PUTS_CACHE_NAME = "pending-puts";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A local, lightweight cache for pending puts, which is
|
||||||
|
* non-transactional and has aggressive expiration settings.
|
||||||
|
* Locking is still required since the putFromLoad validator
|
||||||
|
* code uses conditional operations (i.e. putIfAbsent)
|
||||||
|
*/
|
||||||
|
public static final Configuration PENDING_PUTS_CACHE_CONFIGURATION = new ConfigurationBuilder()
|
||||||
|
.clustering().cacheMode(CacheMode.LOCAL)
|
||||||
|
.transaction().transactionMode(TransactionMode.NON_TRANSACTIONAL)
|
||||||
|
.expiration().maxIdle(TimeUnit.SECONDS.toMillis(60))
|
||||||
|
.storeAsBinary().enabled(false)
|
||||||
|
.locking().isolationLevel(IsolationLevel.READ_COMMITTED)
|
||||||
|
.jmxStatistics().disable().build();
|
||||||
|
|
||||||
private EmbeddedCacheManager manager;
|
private EmbeddedCacheManager manager;
|
||||||
|
|
||||||
private final Map<String, TypeOverrides> typeOverrides = new HashMap<String, TypeOverrides>();
|
private final Map<String, TypeOverrides> typeOverrides = new HashMap<String, TypeOverrides>();
|
||||||
|
@ -345,7 +359,7 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long nextTimestamp() {
|
public long nextTimestamp() {
|
||||||
return System.currentTimeMillis() / 100;
|
return System.currentTimeMillis();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setCacheManager(EmbeddedCacheManager manager) {
|
public void setCacheManager(EmbeddedCacheManager manager) {
|
||||||
|
@ -374,7 +388,7 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
defineGenericDataTypeCacheConfigurations( properties );
|
defineGenericDataTypeCacheConfigurations( properties );
|
||||||
definePendingPutsCache();
|
manager.defineConfiguration( PENDING_PUTS_CACHE_NAME, PENDING_PUTS_CACHE_CONFIGURATION );
|
||||||
}
|
}
|
||||||
catch (CacheException ce) {
|
catch (CacheException ce) {
|
||||||
throw ce;
|
throw ce;
|
||||||
|
@ -384,22 +398,6 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void definePendingPutsCache() {
|
|
||||||
final ConfigurationBuilder builder = new ConfigurationBuilder();
|
|
||||||
// A local, lightweight cache for pending puts, which is
|
|
||||||
// non-transactional and has aggressive expiration settings.
|
|
||||||
// Locking is still required since the putFromLoad validator
|
|
||||||
// code uses conditional operations (i.e. putIfAbsent).
|
|
||||||
builder.clustering().cacheMode( CacheMode.LOCAL )
|
|
||||||
.transaction().transactionMode( TransactionMode.NON_TRANSACTIONAL )
|
|
||||||
.expiration().maxIdle( TimeUnit.SECONDS.toMillis( 60 ) )
|
|
||||||
.storeAsBinary().enabled( false )
|
|
||||||
.locking().isolationLevel( IsolationLevel.READ_COMMITTED )
|
|
||||||
.jmxStatistics().disable();
|
|
||||||
|
|
||||||
manager.defineConfiguration( PENDING_PUTS_CACHE_NAME, builder.build() );
|
|
||||||
}
|
|
||||||
|
|
||||||
protected org.infinispan.transaction.lookup.TransactionManagerLookup createTransactionManagerLookup(
|
protected org.infinispan.transaction.lookup.TransactionManagerLookup createTransactionManagerLookup(
|
||||||
SessionFactoryOptions settings, Properties properties) {
|
SessionFactoryOptions settings, Properties properties) {
|
||||||
return new HibernateTransactionManagerLookup( settings, properties );
|
return new HibernateTransactionManagerLookup( settings, properties );
|
||||||
|
|
|
@ -6,25 +6,42 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.cache.infinispan.access;
|
package org.hibernate.cache.infinispan.access;
|
||||||
|
|
||||||
|
import javax.transaction.Status;
|
||||||
import javax.transaction.SystemException;
|
import javax.transaction.SystemException;
|
||||||
import javax.transaction.Transaction;
|
import javax.transaction.Transaction;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ConcurrentMap;
|
import java.util.concurrent.ConcurrentMap;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
|
|
||||||
import java.util.concurrent.locks.ReentrantLock;
|
import java.util.concurrent.locks.ReentrantLock;
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
|
import org.hibernate.cache.infinispan.util.CacheCommandInitializer;
|
||||||
|
import org.hibernate.cache.infinispan.util.EndInvalidationCommand;
|
||||||
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.infinispan.AdvancedCache;
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.commands.tx.CommitCommand;
|
||||||
|
import org.infinispan.commands.tx.PrepareCommand;
|
||||||
|
import org.infinispan.commands.write.InvalidateCommand;
|
||||||
|
import org.infinispan.commands.write.WriteCommand;
|
||||||
import org.infinispan.configuration.cache.Configuration;
|
import org.infinispan.configuration.cache.Configuration;
|
||||||
import org.infinispan.configuration.cache.ConfigurationBuilder;
|
import org.infinispan.configuration.cache.ConfigurationBuilder;
|
||||||
|
import org.infinispan.context.impl.TxInvocationContext;
|
||||||
|
import org.infinispan.interceptors.EntryWrappingInterceptor;
|
||||||
|
import org.infinispan.interceptors.base.BaseRpcInterceptor;
|
||||||
|
import org.infinispan.interceptors.base.CommandInterceptor;
|
||||||
import org.infinispan.manager.EmbeddedCacheManager;
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
|
import org.infinispan.remoting.inboundhandler.DeliverOrder;
|
||||||
|
import org.infinispan.remoting.rpc.RpcManager;
|
||||||
|
import org.infinispan.transaction.TransactionMode;
|
||||||
|
import org.infinispan.util.concurrent.ConcurrentHashSet;
|
||||||
|
import org.infinispan.util.logging.Log;
|
||||||
|
import org.infinispan.util.logging.LogFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Encapsulates logic to allow a {@link TransactionalAccessDelegate} to determine
|
* Encapsulates logic to allow a {@link TransactionalAccessDelegate} to determine
|
||||||
|
@ -38,9 +55,9 @@ import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
* not find data is:
|
* not find data is:
|
||||||
* <p/>
|
* <p/>
|
||||||
* <ol>
|
* <ol>
|
||||||
* <li> Call {@link #registerPendingPut(Object)}</li>
|
* <li> Call {@link #registerPendingPut(Object, long)}</li>
|
||||||
* <li> Read the database</li>
|
* <li> Read the database</li>
|
||||||
* <li> Call {@link #acquirePutFromLoadLock(Object)}
|
* <li> Call {@link #acquirePutFromLoadLock(Object, long)}
|
||||||
* <li> if above returns <code>null</code>, the thread should not cache the data;
|
* <li> if above returns <code>null</code>, the thread should not cache the data;
|
||||||
* only if above returns instance of <code>AcquiredLock</code>, put data in the cache and...</li>
|
* only if above returns instance of <code>AcquiredLock</code>, put data in the cache and...</li>
|
||||||
* <li> then call {@link #releasePutFromLoadLock(Object, Lock)}</li>
|
* <li> then call {@link #releasePutFromLoadLock(Object, Lock)}</li>
|
||||||
|
@ -54,7 +71,8 @@ import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
* <p/>
|
* <p/>
|
||||||
* <ul>
|
* <ul>
|
||||||
* <li> {@link #beginInvalidatingKey(Object)} (for a single key invalidation)</li>
|
* <li> {@link #beginInvalidatingKey(Object)} (for a single key invalidation)</li>
|
||||||
* <li>or {@link #invalidateRegion()} (for a general invalidation all pending puts)</li>
|
* <li>or {@link #beginInvalidatingRegion()} followed by {@link #endInvalidatingRegion()}
|
||||||
|
* (for a general invalidation all pending puts)</li>
|
||||||
* </ul>
|
* </ul>
|
||||||
* After transaction commit (when the DB is updated) {@link #endInvalidatingKey(Object)} should
|
* After transaction commit (when the DB is updated) {@link #endInvalidatingKey(Object)} should
|
||||||
* be called in order to allow further attempts to cache entry.
|
* be called in order to allow further attempts to cache entry.
|
||||||
|
@ -62,30 +80,31 @@ import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
* <p/>
|
* <p/>
|
||||||
* <p>
|
* <p>
|
||||||
* This class also supports the concept of "naked puts", which are calls to
|
* This class also supports the concept of "naked puts", which are calls to
|
||||||
* {@link #acquirePutFromLoadLock(Object)} without a preceding {@link #registerPendingPut(Object)}.
|
* {@link #acquirePutFromLoadLock(Object, long)} without a preceding {@link #registerPendingPut(Object, long)}.
|
||||||
* Besides not acquiring lock in {@link #registerPendingPut(Object)} this can happen when collection
|
* Besides not acquiring lock in {@link #registerPendingPut(Object, long)} this can happen when collection
|
||||||
* elements are loaded after the collection has not been found in the cache, where the elements
|
* elements are loaded after the collection has not been found in the cache, where the elements
|
||||||
* don't have their own table but can be listed as 'select ... from Element where collection_id = ...'.
|
* don't have their own table but can be listed as 'select ... from Element where collection_id = ...'.
|
||||||
|
* Naked puts are handled according to txTimestamp obtained by calling {@link RegionFactory#nextTimestamp()}
|
||||||
|
* before the transaction is started. The timestamp is compared with timestamp of last invalidation end time
|
||||||
|
* and the write to the cache is denied if it is lower or equal.
|
||||||
* </p>
|
* </p>
|
||||||
*
|
*
|
||||||
* @author Brian Stansberry
|
* @author Brian Stansberry
|
||||||
* @version $Revision: $
|
* @version $Revision: $
|
||||||
*/
|
*/
|
||||||
public class PutFromLoadValidator {
|
public class PutFromLoadValidator {
|
||||||
/**
|
private static final Log log = LogFactory.getLog(PutFromLoadValidator.class);
|
||||||
* Period (in ms) after a removal during which a call to
|
private static final boolean trace = log.isTraceEnabled();
|
||||||
* {@link #acquirePutFromLoadLock(Object)} that hasn't been
|
|
||||||
* {@link #registerPendingPut(Object) pre-registered} (aka a "naked put")
|
|
||||||
* will return false.
|
|
||||||
*/
|
|
||||||
public static final long NAKED_PUT_INVALIDATION_PERIOD = TimeUnit.SECONDS.toMillis( 20 );
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Used to determine whether the owner of a pending put is a thread or a transaction
|
* Used to determine whether the owner of a pending put is a thread or a transaction
|
||||||
*/
|
*/
|
||||||
private final TransactionManager transactionManager;
|
private final TransactionManager transactionManager;
|
||||||
|
|
||||||
private final long nakedPutInvalidationPeriod;
|
/**
|
||||||
|
* Period after which ongoing invalidation is removed. Value is retrieved from cache configuration.
|
||||||
|
*/
|
||||||
|
private final long expirationPeriod;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Registry of expected, future, isPutValid calls. If a key+owner is registered in this map, it
|
* Registry of expected, future, isPutValid calls. If a key+owner is registered in this map, it
|
||||||
|
@ -94,14 +113,26 @@ public class PutFromLoadValidator {
|
||||||
private final ConcurrentMap<Object, PendingPutMap> pendingPuts;
|
private final ConcurrentMap<Object, PendingPutMap> pendingPuts;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The time of the last call to {@link #invalidateRegion()}, plus NAKED_PUT_INVALIDATION_PERIOD. All naked
|
* Main cache where the entities/collections are stored. This is not modified from within this class.
|
||||||
* puts will be rejected until the current time is greater than this value.
|
|
||||||
* NOTE: update only through {@link #invalidationUpdater}!
|
|
||||||
*/
|
*/
|
||||||
private volatile long invalidationTimestamp = Long.MIN_VALUE;
|
private final AdvancedCache cache;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The time of the last call to {@link #endInvalidatingRegion()}. Puts from transactions started after
|
||||||
|
* this timestamp are denied.
|
||||||
|
*/
|
||||||
|
private volatile long regionInvalidationTimestamp = Long.MIN_VALUE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Number of ongoing concurrent invalidations.
|
||||||
|
*/
|
||||||
|
private int regionInvalidations = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transactions that invalidate the region. Entries are removed during next invalidation based on transaction status.
|
||||||
|
*/
|
||||||
|
private final ConcurrentHashSet<Transaction> regionInvalidators = new ConcurrentHashSet<Transaction>();
|
||||||
|
|
||||||
private static final AtomicLongFieldUpdater<PutFromLoadValidator> invalidationUpdater
|
|
||||||
= AtomicLongFieldUpdater.newUpdater(PutFromLoadValidator.class, "invalidationTimestamp");
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new put from load validator instance.
|
* Creates a new put from load validator instance.
|
||||||
|
@ -110,23 +141,7 @@ public class PutFromLoadValidator {
|
||||||
* @param transactionManager Transaction manager
|
* @param transactionManager Transaction manager
|
||||||
*/
|
*/
|
||||||
public PutFromLoadValidator(AdvancedCache cache, TransactionManager transactionManager) {
|
public PutFromLoadValidator(AdvancedCache cache, TransactionManager transactionManager) {
|
||||||
this( cache, transactionManager, NAKED_PUT_INVALIDATION_PERIOD );
|
this( cache, cache.getCacheManager(), transactionManager);
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor variant for use by unit tests; allows control of various timeouts by the test.
|
|
||||||
*
|
|
||||||
* @param cache Cache instance on which to store pending put information.
|
|
||||||
* @param transactionManager Transaction manager
|
|
||||||
* @param nakedPutInvalidationPeriod Period (in ms) after a removal during which a call to
|
|
||||||
* {@link #acquirePutFromLoadLock(Object)} that hasn't been
|
|
||||||
* {@link #registerPendingPut(Object) pre-registered} (aka a "naked put")
|
|
||||||
* will return false.
|
|
||||||
*/
|
|
||||||
public PutFromLoadValidator(
|
|
||||||
AdvancedCache cache, TransactionManager transactionManager,
|
|
||||||
long nakedPutInvalidationPeriod) {
|
|
||||||
this(cache, cache.getCacheManager(), transactionManager, nakedPutInvalidationPeriod);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -135,37 +150,68 @@ public class PutFromLoadValidator {
|
||||||
* @param cache Cache instance on which to store pending put information.
|
* @param cache Cache instance on which to store pending put information.
|
||||||
* @param cacheManager where to find a cache to store pending put information
|
* @param cacheManager where to find a cache to store pending put information
|
||||||
* @param tm transaction manager
|
* @param tm transaction manager
|
||||||
* @param nakedPutInvalidationPeriod Period (in ms) after a removal during which a call to
|
|
||||||
* {@link #acquirePutFromLoadLock(Object)} that hasn't been
|
|
||||||
* {@link #registerPendingPut(Object) pre-registered} (aka a "naked put")
|
|
||||||
* will return false.
|
|
||||||
*/
|
*/
|
||||||
public PutFromLoadValidator(AdvancedCache cache,
|
public PutFromLoadValidator(AdvancedCache cache,
|
||||||
EmbeddedCacheManager cacheManager,
|
EmbeddedCacheManager cacheManager, TransactionManager tm) {
|
||||||
TransactionManager tm, long nakedPutInvalidationPeriod) {
|
|
||||||
|
|
||||||
Configuration cacheConfiguration = cache.getCacheConfiguration();
|
Configuration cacheConfiguration = cache.getCacheConfiguration();
|
||||||
Configuration pendingPutsConfiguration = cacheManager.getCacheConfiguration(InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME);
|
Configuration pendingPutsConfiguration = cacheManager.getCacheConfiguration(InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME);
|
||||||
ConfigurationBuilder configurationBuilder = new ConfigurationBuilder();
|
ConfigurationBuilder configurationBuilder = new ConfigurationBuilder();
|
||||||
if (pendingPutsConfiguration != null) {
|
configurationBuilder.read(pendingPutsConfiguration);
|
||||||
configurationBuilder.read(pendingPutsConfiguration);
|
|
||||||
}
|
|
||||||
configurationBuilder.dataContainer().keyEquivalence(cacheConfiguration.dataContainer().keyEquivalence());
|
configurationBuilder.dataContainer().keyEquivalence(cacheConfiguration.dataContainer().keyEquivalence());
|
||||||
String pendingPutsName = cache.getName() + "-" + InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME;
|
String pendingPutsName = cache.getName() + "-" + InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME;
|
||||||
cacheManager.defineConfiguration(pendingPutsName, configurationBuilder.build());
|
cacheManager.defineConfiguration(pendingPutsName, configurationBuilder.build());
|
||||||
|
|
||||||
|
if (pendingPutsConfiguration.expiration() != null && pendingPutsConfiguration.expiration().maxIdle() > 0) {
|
||||||
|
this.expirationPeriod = pendingPutsConfiguration.expiration().maxIdle();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new IllegalArgumentException("Pending puts cache needs to have maxIdle expiration set!");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Since we need to intercept both invalidations of entries that are in the cache and those
|
||||||
|
// that are not, we need to use custom interceptor, not listeners (which fire only for present entries).
|
||||||
|
if (cacheConfiguration.clustering().cacheMode().isClustered()) {
|
||||||
|
RpcManager rpcManager = cache.getComponentRegistry().getComponent(RpcManager.class);
|
||||||
|
CacheCommandInitializer cacheCommandInitializer = cache.getComponentRegistry().getComponent(CacheCommandInitializer.class);
|
||||||
|
// Note that invalidation does *NOT* acquire locks; therefore, we have to start invalidating before
|
||||||
|
// wrapping the entry, since if putFromLoad was invoked between wrap and beginInvalidatingKey, the invalidation
|
||||||
|
// would not commit the entry removal (as during wrap the entry was not in cache)
|
||||||
|
cache.addInterceptorBefore(new PutFromLoadInterceptor(cache.getName(), rpcManager, cacheCommandInitializer), EntryWrappingInterceptor.class);
|
||||||
|
cacheCommandInitializer.addPutFromLoadValidator(cache.getName(), this);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.cache = cache;
|
||||||
this.pendingPuts = cacheManager.getCache(pendingPutsName);
|
this.pendingPuts = cacheManager.getCache(pendingPutsName);
|
||||||
this.transactionManager = tm;
|
this.transactionManager = tm;
|
||||||
this.nakedPutInvalidationPeriod = nakedPutInvalidationPeriod;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This methods should be called only from tests; it removes existing validator from the cache structures
|
||||||
|
* in order to replace it with new one.
|
||||||
|
*
|
||||||
|
* @param cache
|
||||||
|
*/
|
||||||
|
public static void removeFromCache(AdvancedCache cache) {
|
||||||
|
List<CommandInterceptor> interceptorChain = cache.getInterceptorChain();
|
||||||
|
int index = 0;
|
||||||
|
for (; index < interceptorChain.size(); ++index) {
|
||||||
|
if (interceptorChain.get(index).getClass().getName().startsWith(PutFromLoadValidator.class.getName())) {
|
||||||
|
cache.removeInterceptor(index);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
CacheCommandInitializer cci = cache.getComponentRegistry().getComponent(CacheCommandInitializer.class);
|
||||||
|
cci.removePutFromLoadValidator(cache.getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
// ----------------------------------------------------------------- Public
|
// ----------------------------------------------------------------- Public
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Marker for lock acquired in {@link #acquirePutFromLoadLock(Object)}
|
* Marker for lock acquired in {@link #acquirePutFromLoadLock(Object, long)}
|
||||||
*/
|
*/
|
||||||
public static class Lock {
|
public static abstract class Lock {
|
||||||
protected Lock() {}
|
private Lock() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -178,13 +224,16 @@ public class PutFromLoadValidator {
|
||||||
*
|
*
|
||||||
* @param key the key
|
* @param key the key
|
||||||
*
|
*
|
||||||
|
* @param txTimestamp
|
||||||
* @return <code>AcquiredLock</code> if the lock is acquired and the cache put
|
* @return <code>AcquiredLock</code> if the lock is acquired and the cache put
|
||||||
* can proceed; <code>null</code> if the data should not be cached
|
* can proceed; <code>null</code> if the data should not be cached
|
||||||
*/
|
*/
|
||||||
public Lock acquirePutFromLoadLock(Object key) {
|
public Lock acquirePutFromLoadLock(Object key, long txTimestamp) {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("acquirePutFromLoadLock(%s#%s, %d)", cache.getName(), key, txTimestamp);
|
||||||
|
}
|
||||||
boolean valid = false;
|
boolean valid = false;
|
||||||
boolean locked = false;
|
boolean locked = false;
|
||||||
long now = Long.MIN_VALUE;
|
|
||||||
|
|
||||||
PendingPutMap pending = pendingPuts.get( key );
|
PendingPutMap pending = pendingPuts.get( key );
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
@ -197,40 +246,43 @@ public class PutFromLoadValidator {
|
||||||
if (toCancel != null) {
|
if (toCancel != null) {
|
||||||
valid = !toCancel.completed;
|
valid = !toCancel.completed;
|
||||||
toCancel.completed = true;
|
toCancel.completed = true;
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
// this is a naked put
|
// this is a naked put
|
||||||
if (pending.hasInvalidator()) {
|
if (pending.hasInvalidator()) {
|
||||||
valid = false;
|
valid = false;
|
||||||
} else {
|
}
|
||||||
if (now == Long.MIN_VALUE) {
|
else {
|
||||||
now = System.currentTimeMillis();
|
// if this transaction started after last invalidation we can continue
|
||||||
}
|
valid = txTimestamp > pending.lastInvalidationEnd;
|
||||||
valid = now > pending.nakedPutsDeadline;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return valid ? pending : null;
|
return valid ? pending : null;
|
||||||
} finally {
|
}
|
||||||
|
finally {
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
pending.releaseLock();
|
pending.releaseLock();
|
||||||
locked = false;
|
locked = false;
|
||||||
}
|
}
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("acquirePutFromLoadLock(%s#%s, %d) ended with %s", cache.getName(), key, txTimestamp, pending);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("acquirePutFromLoadLock(%s#%s, %d) failed to lock", cache.getName(), key, txTimestamp);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// oops, we have leaked record for this owner, but we don't want to wait here
|
// oops, we have leaked record for this owner, but we don't want to wait here
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
// Key wasn't in pendingPuts, so either this is a "naked put"
|
else {
|
||||||
// or regionRemoved has been called. Check if we can proceed
|
if (txTimestamp <= regionInvalidationTimestamp) {
|
||||||
long invalidationTimestamp = this.invalidationTimestamp;
|
if (trace) {
|
||||||
if (invalidationTimestamp != Long.MIN_VALUE) {
|
log.tracef("acquirePutFromLoadLock(%s#%s, %d) failed due to invalidated region", cache.getName(), key, txTimestamp);
|
||||||
now = System.currentTimeMillis();
|
|
||||||
if (now > invalidationTimestamp) {
|
|
||||||
// time is +- monotonic se don't let other threads do the expensive currentTimeMillis()
|
|
||||||
invalidationUpdater.compareAndSet(this, invalidationTimestamp, Long.MIN_VALUE);
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
PendingPut pendingPut = new PendingPut(getOwnerForPut());
|
PendingPut pendingPut = new PendingPut(getOwnerForPut());
|
||||||
|
@ -241,16 +293,19 @@ public class PutFromLoadValidator {
|
||||||
}
|
}
|
||||||
// continue in next loop with lock acquisition
|
// continue in next loop with lock acquisition
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
}
|
||||||
|
catch (Throwable t) {
|
||||||
if (locked) {
|
if (locked) {
|
||||||
pending.releaseLock();
|
pending.releaseLock();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (t instanceof RuntimeException) {
|
if (t instanceof RuntimeException) {
|
||||||
throw (RuntimeException) t;
|
throw (RuntimeException) t;
|
||||||
} else if (t instanceof Error) {
|
}
|
||||||
|
else if (t instanceof Error) {
|
||||||
throw (Error) t;
|
throw (Error) t;
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
throw new RuntimeException(t);
|
throw new RuntimeException(t);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -259,11 +314,14 @@ public class PutFromLoadValidator {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Releases the lock previously obtained by a call to
|
* Releases the lock previously obtained by a call to
|
||||||
* {@link #acquirePutFromLoadLock(Object)} that returned <code>true</code>.
|
* {@link #acquirePutFromLoadLock(Object, long)}.
|
||||||
*
|
*
|
||||||
* @param key the key
|
* @param key the key
|
||||||
*/
|
*/
|
||||||
public void releasePutFromLoadLock(Object key, Lock lock) {
|
public void releasePutFromLoadLock(Object key, Lock lock) {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("releasePutFromLoadLock(%s#%s, %s)", cache.getName(), key, lock);
|
||||||
|
}
|
||||||
final PendingPutMap pending = (PendingPutMap) lock;
|
final PendingPutMap pending = (PendingPutMap) lock;
|
||||||
if ( pending != null ) {
|
if ( pending != null ) {
|
||||||
if ( pending.canRemove() ) {
|
if ( pending.canRemove() ) {
|
||||||
|
@ -274,29 +332,65 @@ public class PutFromLoadValidator {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Invalidates all {@link #registerPendingPut(Object) previously registered pending puts} ensuring a subsequent call to
|
* Invalidates all {@link #registerPendingPut(Object, long) previously registered pending puts} ensuring a subsequent call to
|
||||||
* {@link #acquirePutFromLoadLock(Object)} will return <code>false</code>. <p> This method will block until any
|
* {@link #acquirePutFromLoadLock(Object, long)} will return <code>false</code>. <p> This method will block until any
|
||||||
* concurrent thread that has {@link #acquirePutFromLoadLock(Object) acquired the putFromLoad lock} for the any key has
|
* concurrent thread that has {@link #acquirePutFromLoadLock(Object, long) acquired the putFromLoad lock} for the any key has
|
||||||
* released the lock. This allows the caller to be certain the putFromLoad will not execute after this method returns,
|
* released the lock. This allows the caller to be certain the putFromLoad will not execute after this method returns,
|
||||||
* possibly caching stale data. </p>
|
* possibly caching stale data. </p>
|
||||||
*
|
*
|
||||||
* @return <code>true</code> if the invalidation was successful; <code>false</code> if a problem occured (which the
|
* @return <code>true</code> if the invalidation was successful; <code>false</code> if a problem occured (which the
|
||||||
* caller should treat as an exception condition)
|
* caller should treat as an exception condition)
|
||||||
*/
|
*/
|
||||||
public boolean invalidateRegion() {
|
public boolean beginInvalidatingRegion() {
|
||||||
// TODO: not sure what happens with locks acquired *after* calling this method but before
|
if (trace) {
|
||||||
// the actual invalidation
|
log.trace("Started invalidating region " + cache.getName());
|
||||||
|
}
|
||||||
boolean ok = true;
|
boolean ok = true;
|
||||||
invalidationUpdater.set(this, System.currentTimeMillis() + nakedPutInvalidationPeriod);
|
long now = System.currentTimeMillis();
|
||||||
try {
|
// deny all puts until endInvalidatingRegion is called; at that time the region should be already
|
||||||
|
// in INVALID state, therefore all new requests should be blocked and ongoing should fail by timestamp
|
||||||
|
synchronized (this) {
|
||||||
|
regionInvalidationTimestamp = Long.MAX_VALUE;
|
||||||
|
regionInvalidations++;
|
||||||
|
}
|
||||||
|
if (transactionManager != null) {
|
||||||
|
// cleanup old transactions
|
||||||
|
for (Iterator<Transaction> it = regionInvalidators.iterator(); it.hasNext(); ) {
|
||||||
|
Transaction tx = it.next();
|
||||||
|
try {
|
||||||
|
switch (tx.getStatus()) {
|
||||||
|
case Status.STATUS_COMMITTED:
|
||||||
|
case Status.STATUS_ROLLEDBACK:
|
||||||
|
case Status.STATUS_UNKNOWN:
|
||||||
|
case Status.STATUS_NO_TRANSACTION:
|
||||||
|
it.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (SystemException e) {
|
||||||
|
log.error("Cannot retrieve transaction status", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// add this transaction
|
||||||
|
try {
|
||||||
|
Transaction tx = transactionManager.getTransaction();
|
||||||
|
if (tx != null) {
|
||||||
|
regionInvalidators.add(tx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (SystemException e) {
|
||||||
|
log.error("TransactionManager failed to provide transaction", e);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
// Acquire the lock for each entry to ensure any ongoing
|
// Acquire the lock for each entry to ensure any ongoing
|
||||||
// work associated with it is completed before we return
|
// work associated with it is completed before we return
|
||||||
for ( Iterator<PendingPutMap> it = pendingPuts.values().iterator(); it.hasNext(); ) {
|
for (Iterator<PendingPutMap> it = pendingPuts.values().iterator(); it.hasNext(); ) {
|
||||||
PendingPutMap entry = it.next();
|
PendingPutMap entry = it.next();
|
||||||
if ( entry.acquireLock( 60, TimeUnit.SECONDS ) ) {
|
if (entry.acquireLock(60, TimeUnit.SECONDS)) {
|
||||||
try {
|
try {
|
||||||
entry.invalidate();
|
entry.invalidate(now, expirationPeriod);
|
||||||
}
|
}
|
||||||
finally {
|
finally {
|
||||||
entry.releaseLock();
|
entry.releaseLock();
|
||||||
|
@ -307,24 +401,66 @@ public class PutFromLoadValidator {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
}
|
||||||
|
catch (Exception e) {
|
||||||
ok = false;
|
ok = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ok;
|
return ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called when the region invalidation is finished.
|
||||||
|
*/
|
||||||
|
public void endInvalidatingRegion() {
|
||||||
|
synchronized (this) {
|
||||||
|
if (--regionInvalidations == 0) {
|
||||||
|
regionInvalidationTimestamp = System.currentTimeMillis();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (trace) {
|
||||||
|
log.trace("Finished invalidating region " + cache.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Notifies this validator that it is expected that a database read followed by a subsequent {@link
|
* Notifies this validator that it is expected that a database read followed by a subsequent {@link
|
||||||
* #acquirePutFromLoadLock(Object)} call will occur. The intent is this method would be called following a cache miss
|
* #acquirePutFromLoadLock(Object, long)} call will occur. The intent is this method would be called following a cache miss
|
||||||
* wherein it is expected that a database read plus cache put will occur. Calling this method allows the validator to
|
* wherein it is expected that a database read plus cache put will occur. Calling this method allows the validator to
|
||||||
* treat the subsequent <code>acquirePutFromLoadLock</code> as if the database read occurred when this method was
|
* treat the subsequent <code>acquirePutFromLoadLock</code> as if the database read occurred when this method was
|
||||||
* invoked. This allows the validator to compare the timestamp of this call against the timestamp of subsequent removal
|
* invoked. This allows the validator to compare the timestamp of this call against the timestamp of subsequent removal
|
||||||
* notifications.
|
* notifications.
|
||||||
*
|
*
|
||||||
* @param key key that will be used for subsequent cache put
|
* @param key key that will be used for subsequent cache put
|
||||||
|
* @param txTimestamp
|
||||||
*/
|
*/
|
||||||
public void registerPendingPut(Object key) {
|
public void registerPendingPut(Object key, long txTimestamp) {
|
||||||
|
long invalidationTimestamp = this.regionInvalidationTimestamp;
|
||||||
|
if (txTimestamp <= invalidationTimestamp) {
|
||||||
|
boolean skip;
|
||||||
|
if (invalidationTimestamp == Long.MAX_VALUE) {
|
||||||
|
// there is ongoing invalidation of pending puts
|
||||||
|
skip = true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
Transaction tx = null;
|
||||||
|
if (transactionManager != null) {
|
||||||
|
try {
|
||||||
|
tx = transactionManager.getTransaction();
|
||||||
|
}
|
||||||
|
catch (SystemException e) {
|
||||||
|
log.error("TransactionManager failed to provide transaction", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
skip = tx == null || !regionInvalidators.contains(tx);
|
||||||
|
}
|
||||||
|
if (skip) {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("registerPendingPut(%s#%s, %d) skipped due to region invalidation (%d)", cache.getName(), key, txTimestamp, invalidationTimestamp);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
final PendingPut pendingPut = new PendingPut( getOwnerForPut() );
|
final PendingPut pendingPut = new PendingPut( getOwnerForPut() );
|
||||||
final PendingPutMap pendingForKey = new PendingPutMap( pendingPut );
|
final PendingPutMap pendingForKey = new PendingPutMap( pendingPut );
|
||||||
|
|
||||||
|
@ -335,21 +471,42 @@ public class PutFromLoadValidator {
|
||||||
if ( !existing.hasInvalidator() ) {
|
if ( !existing.hasInvalidator() ) {
|
||||||
existing.put(pendingPut);
|
existing.put(pendingPut);
|
||||||
}
|
}
|
||||||
} finally {
|
}
|
||||||
|
finally {
|
||||||
existing.releaseLock();
|
existing.releaseLock();
|
||||||
}
|
}
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("registerPendingPut(%s#%s, %d) ended with %s", cache.getName(), key, txTimestamp, existing);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("registerPendingPut(%s#%s, %d) failed to acquire lock", cache.getName(), key, txTimestamp);
|
||||||
|
}
|
||||||
// Can't get the lock; when we come back we'll be a "naked put"
|
// Can't get the lock; when we come back we'll be a "naked put"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("registerPendingPut(%s#%s, %d) registered using putIfAbsent: %s", cache.getName(), key, txTimestamp, pendingForKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Invalidates any {@link #registerPendingPut(Object) previously registered pending puts}
|
* Calls {@link #beginInvalidatingKey(Object, Object)} with current transaction or thread.
|
||||||
* and disables further registrations ensuring a subsequent call to {@link #acquirePutFromLoadLock(Object)}
|
* @param key
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean beginInvalidatingKey(Object key) {
|
||||||
|
return beginInvalidatingKey(key, getOwnerForPut());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Invalidates any {@link #registerPendingPut(Object, long) previously registered pending puts}
|
||||||
|
* and disables further registrations ensuring a subsequent call to {@link #acquirePutFromLoadLock(Object, long)}
|
||||||
* will return <code>false</code>. <p> This method will block until any concurrent thread that has
|
* will return <code>false</code>. <p> This method will block until any concurrent thread that has
|
||||||
* {@link #acquirePutFromLoadLock(Object) acquired the putFromLoad lock} for the given key
|
* {@link #acquirePutFromLoadLock(Object, long) acquired the putFromLoad lock} for the given key
|
||||||
* has released the lock. This allows the caller to be certain the putFromLoad will not execute after this method
|
* has released the lock. This allows the caller to be certain the putFromLoad will not execute after this method
|
||||||
* returns, possibly caching stale data. </p>
|
* returns, possibly caching stale data. </p>
|
||||||
* After this transaction completes, {@link #endInvalidatingKey(Object)} needs to be called }
|
* After this transaction completes, {@link #endInvalidatingKey(Object)} needs to be called }
|
||||||
|
@ -359,7 +516,7 @@ public class PutFromLoadValidator {
|
||||||
* @return <code>true</code> if the invalidation was successful; <code>false</code> if a problem occured (which the
|
* @return <code>true</code> if the invalidation was successful; <code>false</code> if a problem occured (which the
|
||||||
* caller should treat as an exception condition)
|
* caller should treat as an exception condition)
|
||||||
*/
|
*/
|
||||||
public boolean beginInvalidatingKey(Object key) {
|
public boolean beginInvalidatingKey(Object key, Object lockOwner) {
|
||||||
PendingPutMap pending = new PendingPutMap(null);
|
PendingPutMap pending = new PendingPutMap(null);
|
||||||
PendingPutMap prev = pendingPuts.putIfAbsent(key, pending);
|
PendingPutMap prev = pendingPuts.putIfAbsent(key, pending);
|
||||||
if (prev != null) {
|
if (prev != null) {
|
||||||
|
@ -367,39 +524,68 @@ public class PutFromLoadValidator {
|
||||||
}
|
}
|
||||||
if (pending.acquireLock(60, TimeUnit.SECONDS)) {
|
if (pending.acquireLock(60, TimeUnit.SECONDS)) {
|
||||||
try {
|
try {
|
||||||
pending.invalidate();
|
long now = System.currentTimeMillis();
|
||||||
pending.addInvalidator(getOwnerForPut(), System.currentTimeMillis() + nakedPutInvalidationPeriod);
|
pending.invalidate(now, expirationPeriod);
|
||||||
} finally {
|
pending.addInvalidator(lockOwner, now, expirationPeriod);
|
||||||
|
}
|
||||||
|
finally {
|
||||||
pending.releaseLock();
|
pending.releaseLock();
|
||||||
}
|
}
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("beginInvalidatingKey(%s#%s, %s) ends with %s", cache.getName(), key, lockOwner, pending);
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
|
log.tracef("beginInvalidatingKey(%s#%s, %s) failed to acquire lock", cache.getName(), key);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called after the transaction completes, allowing caching of entries. It is possible that this method
|
* Calls {@link #endInvalidatingKey(Object, Object)} with current transaction or thread.
|
||||||
* is called without previous invocation of {@link #beginInvalidatingKey(Object)}, then it should be noop.
|
|
||||||
*
|
|
||||||
* @param key
|
* @param key
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
public boolean endInvalidatingKey(Object key) {
|
public boolean endInvalidatingKey(Object key) {
|
||||||
|
return endInvalidatingKey(key, getOwnerForPut());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called after the transaction completes, allowing caching of entries. It is possible that this method
|
||||||
|
* is called without previous invocation of {@link #beginInvalidatingKey(Object)}, then it should be a no-op.
|
||||||
|
*
|
||||||
|
* @param key
|
||||||
|
* @param lockOwner owner of the invalidation - transaction or thread
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean endInvalidatingKey(Object key, Object lockOwner) {
|
||||||
PendingPutMap pending = pendingPuts.get(key);
|
PendingPutMap pending = pendingPuts.get(key);
|
||||||
if (pending == null) {
|
if (pending == null) {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("endInvalidatingKey(%s#%s, %s) could not find pending puts", cache.getName(), key, lockOwner);
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (pending.acquireLock(60, TimeUnit.SECONDS)) {
|
if (pending.acquireLock(60, TimeUnit.SECONDS)) {
|
||||||
try {
|
try {
|
||||||
pending.removeInvalidator(getOwnerForPut());
|
long now = System.currentTimeMillis();
|
||||||
|
pending.removeInvalidator(lockOwner, now);
|
||||||
// we can't remove the pending put yet because we wait for naked puts
|
// we can't remove the pending put yet because we wait for naked puts
|
||||||
// pendingPuts should be configured with maxIdle time so won't have memory leak
|
// pendingPuts should be configured with maxIdle time so won't have memory leak
|
||||||
return true;
|
return true;
|
||||||
} finally {
|
|
||||||
pending.releaseLock();
|
|
||||||
}
|
}
|
||||||
} else {
|
finally {
|
||||||
|
pending.releaseLock();
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("endInvalidatingKey(%s#%s, %s) ends with %s", cache.getName(), key, lockOwner, pending);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (trace) {
|
||||||
|
log.tracef("endInvalidatingKey(%s#%s, %s) failed to acquire lock", cache.getName(), key, lockOwner);
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -431,14 +617,61 @@ public class PutFromLoadValidator {
|
||||||
private PendingPut singlePendingPut;
|
private PendingPut singlePendingPut;
|
||||||
private Map<Object, PendingPut> fullMap;
|
private Map<Object, PendingPut> fullMap;
|
||||||
private final java.util.concurrent.locks.Lock lock = new ReentrantLock();
|
private final java.util.concurrent.locks.Lock lock = new ReentrantLock();
|
||||||
private Object singleInvalidator;
|
private Invalidator singleInvalidator;
|
||||||
private Set<Object> invalidators;
|
private Map<Object, Invalidator> invalidators;
|
||||||
private long nakedPutsDeadline = Long.MIN_VALUE;
|
private long lastInvalidationEnd = Long.MIN_VALUE;
|
||||||
|
|
||||||
PendingPutMap(PendingPut singleItem) {
|
PendingPutMap(PendingPut singleItem) {
|
||||||
this.singlePendingPut = singleItem;
|
this.singlePendingPut = singleItem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// toString should be called only for debugging purposes
|
||||||
|
public String toString() {
|
||||||
|
if (lock.tryLock()) {
|
||||||
|
try {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
sb.append("{ PendingPuts=");
|
||||||
|
if (singlePendingPut == null) {
|
||||||
|
if (fullMap == null) {
|
||||||
|
sb.append("[]");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append(fullMap.values());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append('[').append(singlePendingPut).append(']');
|
||||||
|
}
|
||||||
|
sb.append(", Invalidators=");
|
||||||
|
if (singleInvalidator == null) {
|
||||||
|
if (invalidators == null) {
|
||||||
|
sb.append("[]");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append(invalidators);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append('[').append(singleInvalidator).append(']');
|
||||||
|
}
|
||||||
|
sb.append(", LastInvalidationEnd=");
|
||||||
|
if (lastInvalidationEnd == Long.MIN_VALUE) {
|
||||||
|
sb.append("<none>");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append(lastInvalidationEnd);
|
||||||
|
}
|
||||||
|
return sb.append("}").toString();
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
lock.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return "PendingPutMap: <locked>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void put(PendingPut pendingPut) {
|
public void put(PendingPut pendingPut) {
|
||||||
if ( singlePendingPut == null ) {
|
if ( singlePendingPut == null ) {
|
||||||
if ( fullMap == null ) {
|
if ( fullMap == null ) {
|
||||||
|
@ -492,63 +725,167 @@ public class PutFromLoadValidator {
|
||||||
lock.unlock();
|
lock.unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void invalidate() {
|
public void invalidate(long now, long expirationPeriod) {
|
||||||
if ( singlePendingPut != null ) {
|
if ( singlePendingPut != null ) {
|
||||||
singlePendingPut.completed = true;
|
if (singlePendingPut.invalidate(now, expirationPeriod)) {
|
||||||
// Nullify to avoid leaking completed pending puts
|
singlePendingPut = null;
|
||||||
singlePendingPut = null;
|
}
|
||||||
}
|
}
|
||||||
else if ( fullMap != null ) {
|
else if ( fullMap != null ) {
|
||||||
for ( PendingPut pp : fullMap.values() ) {
|
for ( Iterator<PendingPut> it = fullMap.values().iterator(); it.hasNext(); ) {
|
||||||
pp.completed = true;
|
PendingPut pp = it.next();
|
||||||
|
if (pp.invalidate(now, expirationPeriod)) {
|
||||||
|
it.remove();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Nullify to avoid leaking completed pending puts
|
|
||||||
fullMap = null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addInvalidator(Object invalidator, long deadline) {
|
public void addInvalidator(Object owner, long now, long invalidatorTimeout) {
|
||||||
|
assert owner != null;
|
||||||
if (invalidators == null) {
|
if (invalidators == null) {
|
||||||
if (singleInvalidator == null) {
|
if (singleInvalidator == null) {
|
||||||
singleInvalidator = invalidator;
|
singleInvalidator = new Invalidator(owner, now);
|
||||||
} else {
|
}
|
||||||
invalidators = new HashSet<Object>();
|
else {
|
||||||
invalidators.add(singleInvalidator);
|
if (singleInvalidator.registeredTimestamp + invalidatorTimeout < now) {
|
||||||
invalidators.add(invalidator);
|
// remove leaked invalidator
|
||||||
|
singleInvalidator = new Invalidator(owner, now);
|
||||||
|
}
|
||||||
|
invalidators = new HashMap<Object, Invalidator>();
|
||||||
|
invalidators.put(singleInvalidator.owner, singleInvalidator);
|
||||||
|
invalidators.put(owner, new Invalidator(owner, now));
|
||||||
singleInvalidator = null;
|
singleInvalidator = null;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
invalidators.add(invalidator);
|
|
||||||
}
|
}
|
||||||
nakedPutsDeadline = Math.max(nakedPutsDeadline, deadline);
|
else {
|
||||||
|
long allowedRegistration = now - invalidatorTimeout;
|
||||||
|
// remove leaked invalidators
|
||||||
|
for (Iterator<Invalidator> it = invalidators.values().iterator(); it.hasNext(); ) {
|
||||||
|
if (it.next().registeredTimestamp < allowedRegistration) {
|
||||||
|
it.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
invalidators.put(owner, new Invalidator(owner, now));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean hasInvalidator() {
|
public boolean hasInvalidator() {
|
||||||
return singleInvalidator != null || (invalidators != null && !invalidators.isEmpty());
|
return singleInvalidator != null || (invalidators != null && !invalidators.isEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void removeInvalidator(Object invalidator) {
|
public void removeInvalidator(Object owner, long now) {
|
||||||
if (invalidators == null) {
|
if (invalidators == null) {
|
||||||
if (singleInvalidator != null && singleInvalidator.equals(invalidator)) {
|
if (singleInvalidator != null && singleInvalidator.owner.equals(owner)) {
|
||||||
singleInvalidator = null;
|
singleInvalidator = null;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
invalidators.remove(invalidator);
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
invalidators.remove(owner);
|
||||||
|
}
|
||||||
|
lastInvalidationEnd = Math.max(lastInvalidationEnd, now);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean canRemove() {
|
public boolean canRemove() {
|
||||||
return size() == 0 && !hasInvalidator() &&
|
return size() == 0 && !hasInvalidator() && lastInvalidationEnd == Long.MIN_VALUE;
|
||||||
(nakedPutsDeadline == Long.MIN_VALUE || nakedPutsDeadline < System.currentTimeMillis());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class PendingPut {
|
private static class PendingPut {
|
||||||
private final Object owner;
|
private final Object owner;
|
||||||
private volatile boolean completed;
|
private boolean completed;
|
||||||
|
// the timestamp is not filled during registration in order to avoid expensive currentTimeMillis() calls
|
||||||
|
private long registeredTimestamp = Long.MIN_VALUE;
|
||||||
|
|
||||||
private PendingPut(Object owner) {
|
private PendingPut(Object owner) {
|
||||||
this.owner = owner;
|
this.owner = owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String toString() {
|
||||||
|
return (completed ? "C@" : "R@") + owner;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean invalidate(long now, long expirationPeriod) {
|
||||||
|
completed = true;
|
||||||
|
if (registeredTimestamp == Long.MIN_VALUE) {
|
||||||
|
registeredTimestamp = now;
|
||||||
|
}
|
||||||
|
else if (registeredTimestamp + expirationPeriod < now){
|
||||||
|
return true; // this is a leaked pending put
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class Invalidator {
|
||||||
|
private final Object owner;
|
||||||
|
private final long registeredTimestamp;
|
||||||
|
|
||||||
|
private Invalidator(Object owner, long registeredTimestamp) {
|
||||||
|
this.owner = owner;
|
||||||
|
this.registeredTimestamp = registeredTimestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
final StringBuilder sb = new StringBuilder("{");
|
||||||
|
sb.append("Owner=").append(owner);
|
||||||
|
sb.append(", Timestamp=").append(registeredTimestamp);
|
||||||
|
sb.append('}');
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class PutFromLoadInterceptor extends BaseRpcInterceptor {
|
||||||
|
private final String cacheName;
|
||||||
|
private final RpcManager rpcManager;
|
||||||
|
private final CacheCommandInitializer cacheCommandInitializer;
|
||||||
|
|
||||||
|
public PutFromLoadInterceptor(String cacheName, RpcManager rpcManager, CacheCommandInitializer cacheCommandInitializer) {
|
||||||
|
this.cacheName = cacheName;
|
||||||
|
this.rpcManager = rpcManager;
|
||||||
|
this.cacheCommandInitializer = cacheCommandInitializer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We need to intercept PrepareCommand, not InvalidateCommand since the interception takes
|
||||||
|
// place before EntryWrappingInterceptor and the PrepareCommand is multiplexed into InvalidateCommands
|
||||||
|
// as part of EntryWrappingInterceptor
|
||||||
|
@Override
|
||||||
|
public Object visitPrepareCommand(TxInvocationContext ctx, PrepareCommand command) throws Throwable {
|
||||||
|
if (!ctx.isOriginLocal()) {
|
||||||
|
for (WriteCommand wc : command.getModifications()) {
|
||||||
|
if (wc instanceof InvalidateCommand) {
|
||||||
|
// InvalidateCommand does not correctly implement getAffectedKeys()
|
||||||
|
for (Object key : ((InvalidateCommand) wc).getKeys()) {
|
||||||
|
beginInvalidatingKey(key, ctx.getLockOwner());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
for (Object key : wc.getAffectedKeys()) {
|
||||||
|
beginInvalidatingKey(key, ctx.getLockOwner());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return invokeNextInterceptor(ctx, command);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object visitCommitCommand(TxInvocationContext ctx, CommitCommand command) throws Throwable {
|
||||||
|
try {
|
||||||
|
if (ctx.isOriginLocal()) {
|
||||||
|
// send async Commit
|
||||||
|
Set<Object> affectedKeys = ctx.getAffectedKeys();
|
||||||
|
if (!affectedKeys.isEmpty()) {
|
||||||
|
EndInvalidationCommand commitCommand = cacheCommandInitializer.buildEndInvalidationCommand(
|
||||||
|
cacheName, affectedKeys.toArray(), ctx.getGlobalTransaction());
|
||||||
|
rpcManager.invokeRemotely(null, commitCommand, rpcManager.getDefaultRpcOptions(false, DeliverOrder.NONE));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
return invokeNextInterceptor(ctx, command);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,7 +61,7 @@ public class TransactionalAccessDelegate {
|
||||||
}
|
}
|
||||||
final Object val = cache.get( key );
|
final Object val = cache.get( key );
|
||||||
if ( val == null ) {
|
if ( val == null ) {
|
||||||
putValidator.registerPendingPut( key );
|
putValidator.registerPendingPut( key, txTimestamp );
|
||||||
}
|
}
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
@ -110,7 +110,7 @@ public class TransactionalAccessDelegate {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
PutFromLoadValidator.Lock lock = putValidator.acquirePutFromLoadLock(key);
|
PutFromLoadValidator.Lock lock = putValidator.acquirePutFromLoadLock(key, txTimestamp);
|
||||||
if ( lock == null) {
|
if ( lock == null) {
|
||||||
if ( TRACE_ENABLED ) {
|
if ( TRACE_ENABLED ) {
|
||||||
log.tracef( "Put from load lock not acquired for key %s", key );
|
log.tracef( "Put from load lock not acquired for key %s", key );
|
||||||
|
@ -202,10 +202,15 @@ public class TransactionalAccessDelegate {
|
||||||
* @throws CacheException if eviction the region fails
|
* @throws CacheException if eviction the region fails
|
||||||
*/
|
*/
|
||||||
public void removeAll() throws CacheException {
|
public void removeAll() throws CacheException {
|
||||||
if ( !putValidator.invalidateRegion() ) {
|
try {
|
||||||
throw new CacheException( "Failed to invalidate pending putFromLoad calls for region " + region.getName() );
|
if (!putValidator.beginInvalidatingRegion()) {
|
||||||
|
throw new CacheException("Failed to invalidate pending putFromLoad calls for region " + region.getName());
|
||||||
|
}
|
||||||
|
Caches.removeAll(cache);
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
putValidator.endInvalidatingRegion();
|
||||||
}
|
}
|
||||||
Caches.removeAll( cache );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -231,13 +236,18 @@ public class TransactionalAccessDelegate {
|
||||||
* @throws CacheException if evicting items fails
|
* @throws CacheException if evicting items fails
|
||||||
*/
|
*/
|
||||||
public void evictAll() throws CacheException {
|
public void evictAll() throws CacheException {
|
||||||
if ( !putValidator.invalidateRegion() ) {
|
try {
|
||||||
throw new CacheException( "Failed to invalidate pending putFromLoad calls for region " + region.getName() );
|
if (!putValidator.beginInvalidatingRegion()) {
|
||||||
}
|
throw new CacheException("Failed to invalidate pending putFromLoad calls for region " + region.getName());
|
||||||
|
}
|
||||||
|
|
||||||
// Invalidate the local region and then go remote
|
// Invalidate the local region and then go remote
|
||||||
region.invalidateRegion();
|
region.invalidateRegion();
|
||||||
Caches.broadcastEvictAll( cache );
|
Caches.broadcastEvictAll(cache);
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
putValidator.endInvalidatingRegion();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -58,6 +58,7 @@ public class CacheCommandFactory implements ExtendedModuleCommandFactory {
|
||||||
public Map<Byte, Class<? extends ReplicableCommand>> getModuleCommands() {
|
public Map<Byte, Class<? extends ReplicableCommand>> getModuleCommands() {
|
||||||
final Map<Byte, Class<? extends ReplicableCommand>> map = new HashMap<Byte, Class<? extends ReplicableCommand>>( 3 );
|
final Map<Byte, Class<? extends ReplicableCommand>> map = new HashMap<Byte, Class<? extends ReplicableCommand>>( 3 );
|
||||||
map.put( CacheCommandIds.EVICT_ALL, EvictAllCommand.class );
|
map.put( CacheCommandIds.EVICT_ALL, EvictAllCommand.class );
|
||||||
|
map.put( CacheCommandIds.END_INVALIDATION, EndInvalidationCommand.class );
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -68,6 +69,9 @@ public class CacheCommandFactory implements ExtendedModuleCommandFactory {
|
||||||
case CacheCommandIds.EVICT_ALL:
|
case CacheCommandIds.EVICT_ALL:
|
||||||
c = new EvictAllCommand( cacheName, allRegions.get( cacheName ) );
|
c = new EvictAllCommand( cacheName, allRegions.get( cacheName ) );
|
||||||
break;
|
break;
|
||||||
|
case CacheCommandIds.END_INVALIDATION:
|
||||||
|
c = new EndInvalidationCommand(cacheName);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException( "Not registered to handle command id " + commandId );
|
throw new IllegalArgumentException( "Not registered to handle command id " + commandId );
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,12 @@ package org.hibernate.cache.infinispan.util;
|
||||||
*/
|
*/
|
||||||
public interface CacheCommandIds {
|
public interface CacheCommandIds {
|
||||||
/**
|
/**
|
||||||
* The "evict all" command id
|
* {@link EvictAllCommand} id
|
||||||
*/
|
*/
|
||||||
public static final byte EVICT_ALL = 120;
|
public static final byte EVICT_ALL = 120;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link EndInvalidationCommand} id
|
||||||
|
*/
|
||||||
|
public static final byte END_INVALIDATION = 121;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,9 +6,12 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.cache.infinispan.util;
|
package org.hibernate.cache.infinispan.util;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.infinispan.commands.ReplicableCommand;
|
import org.infinispan.commands.ReplicableCommand;
|
||||||
import org.infinispan.commands.module.ModuleCommandInitializer;
|
import org.infinispan.commands.module.ModuleCommandInitializer;
|
||||||
|
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Command initializer
|
* Command initializer
|
||||||
*
|
*
|
||||||
|
@ -17,7 +20,22 @@ import org.infinispan.commands.module.ModuleCommandInitializer;
|
||||||
*/
|
*/
|
||||||
public class CacheCommandInitializer implements ModuleCommandInitializer {
|
public class CacheCommandInitializer implements ModuleCommandInitializer {
|
||||||
|
|
||||||
/**
|
private final ConcurrentHashMap<String, PutFromLoadValidator> putFromLoadValidators
|
||||||
|
= new ConcurrentHashMap<String, PutFromLoadValidator>();
|
||||||
|
|
||||||
|
public void addPutFromLoadValidator(String cacheName, PutFromLoadValidator putFromLoadValidator) {
|
||||||
|
// there could be two instances of PutFromLoadValidator bound to the same cache when
|
||||||
|
// there are two JndiInfinispanRegionFactories bound to the same cacheManager via JNDI.
|
||||||
|
// In that case, as putFromLoadValidator does not really own the pendingPuts cache,
|
||||||
|
// it's safe to have more instances.
|
||||||
|
putFromLoadValidators.put(cacheName, putFromLoadValidator);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PutFromLoadValidator removePutFromLoadValidator(String cacheName) {
|
||||||
|
return putFromLoadValidators.remove(cacheName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
* Build an instance of {@link EvictAllCommand} for a given region.
|
* Build an instance of {@link EvictAllCommand} for a given region.
|
||||||
*
|
*
|
||||||
* @param regionName name of region for {@link EvictAllCommand}
|
* @param regionName name of region for {@link EvictAllCommand}
|
||||||
|
@ -31,9 +49,17 @@ public class CacheCommandInitializer implements ModuleCommandInitializer {
|
||||||
return new EvictAllCommand( regionName );
|
return new EvictAllCommand( regionName );
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public EndInvalidationCommand buildEndInvalidationCommand(String cacheName, Object[] keys, Object lockOwner) {
|
||||||
public void initializeReplicableCommand(ReplicableCommand c, boolean isRemote) {
|
return new EndInvalidationCommand( cacheName, keys, lockOwner );
|
||||||
// No need to initialize...
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void initializeReplicableCommand(ReplicableCommand c, boolean isRemote) {
|
||||||
|
switch (c.getCommandId()) {
|
||||||
|
case CacheCommandIds.END_INVALIDATION:
|
||||||
|
EndInvalidationCommand endInvalidationCommand = (EndInvalidationCommand) c;
|
||||||
|
endInvalidationCommand.setPutFromLoadValidator(putFromLoadValidators.get(endInvalidationCommand.getCacheName()));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,86 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||||
|
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||||
|
*/
|
||||||
|
package org.hibernate.cache.infinispan.util;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
|
import org.infinispan.commands.remote.BaseRpcCommand;
|
||||||
|
import org.infinispan.context.InvocationContext;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sent in commit phase (after DB commit) to remote nodes in order to stop invalidating
|
||||||
|
* putFromLoads.
|
||||||
|
*
|
||||||
|
* @author Radim Vansa <rvansa@redhat.com>
|
||||||
|
*/
|
||||||
|
public class EndInvalidationCommand extends BaseRpcCommand {
|
||||||
|
private Object[] keys;
|
||||||
|
private Object lockOwner;
|
||||||
|
private PutFromLoadValidator putFromLoadValidator;
|
||||||
|
|
||||||
|
public EndInvalidationCommand(String cacheName) {
|
||||||
|
this(cacheName, null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param cacheName name of the cache to evict
|
||||||
|
*/
|
||||||
|
public EndInvalidationCommand(String cacheName, Object[] keys, Object lockOwner) {
|
||||||
|
super(cacheName);
|
||||||
|
this.keys = keys;
|
||||||
|
this.lockOwner = lockOwner;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object perform(InvocationContext ctx) throws Throwable {
|
||||||
|
for (Object key : keys) {
|
||||||
|
putFromLoadValidator.endInvalidatingKey(key, lockOwner);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public byte getCommandId() {
|
||||||
|
return CacheCommandIds.END_INVALIDATION;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object[] getParameters() {
|
||||||
|
return new Object[] { keys, lockOwner };
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setParameters(int commandId, Object[] parameters) {
|
||||||
|
keys = (Object[]) parameters[0];
|
||||||
|
lockOwner = parameters[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isReturnValueExpected() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canBlock() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPutFromLoadValidator(PutFromLoadValidator putFromLoadValidator) {
|
||||||
|
this.putFromLoadValidator = putFromLoadValidator;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
final StringBuilder sb = new StringBuilder("EndInvalidationCommand{");
|
||||||
|
sb.append("cacheName=").append(cacheName);
|
||||||
|
sb.append(", keys=").append(Arrays.toString(keys));
|
||||||
|
sb.append(", lockOwner=").append(lockOwner);
|
||||||
|
sb.append('}');
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
}
|
|
@ -164,9 +164,9 @@ public abstract class AbstractGeneralDataRegionTestCase extends AbstractRegionIm
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void regionEvict(GeneralDataRegion region) throws Exception {
|
protected void regionEvict(GeneralDataRegion region) throws Exception {
|
||||||
region.evict(KEY);
|
region.evict(KEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract String getStandardRegionName(String regionPrefix);
|
protected abstract String getStandardRegionName(String regionPrefix);
|
||||||
|
|
||||||
|
|
|
@ -17,9 +17,11 @@ import java.util.concurrent.TimeoutException;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.hibernate.test.cache.infinispan.functional.cluster.DualNodeJtaTransactionManagerImpl;
|
import org.hibernate.test.cache.infinispan.functional.cluster.DualNodeJtaTransactionManagerImpl;
|
||||||
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
||||||
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
import org.infinispan.test.CacheManagerCallable;
|
import org.infinispan.test.CacheManagerCallable;
|
||||||
import org.infinispan.test.fwk.TestCacheManagerFactory;
|
import org.infinispan.test.fwk.TestCacheManagerFactory;
|
||||||
import org.infinispan.util.logging.Log;
|
import org.infinispan.util.logging.Log;
|
||||||
|
@ -70,7 +72,14 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
finally {
|
finally {
|
||||||
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static EmbeddedCacheManager createCacheManager() {
|
||||||
|
EmbeddedCacheManager cacheManager = TestCacheManagerFactory.createCacheManager(false);
|
||||||
|
cacheManager.defineConfiguration(InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME,
|
||||||
|
InfinispanRegionFactory.PENDING_PUTS_CACHE_CONFIGURATION);
|
||||||
|
return cacheManager;
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testNakedPut() throws Exception {
|
public void testNakedPut() throws Exception {
|
||||||
|
@ -82,12 +91,11 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void nakedPutTest(final boolean transactional) throws Exception {
|
private void nakedPutTest(final boolean transactional) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
transactional ? tm : null);
|
||||||
exec(transactional, new NakedPut(testee, true));
|
exec(transactional, new NakedPut(testee, true));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -103,12 +111,11 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void registeredPutTest(final boolean transactional) throws Exception {
|
private void registeredPutTest(final boolean transactional) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
transactional ? tm : null);
|
||||||
exec(transactional, new RegularPut(testee));
|
exec(transactional, new RegularPut(testee));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -133,14 +140,14 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void nakedPutAfterRemovalTest(final boolean transactional,
|
private void nakedPutAfterRemovalTest(final boolean transactional,
|
||||||
final boolean removeRegion) throws Exception {
|
final boolean removeRegion) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
transactional ? tm : null);
|
||||||
Invalidation invalidation = new Invalidation(testee, removeRegion);
|
Invalidation invalidation = new Invalidation(testee, removeRegion);
|
||||||
NakedPut nakedPut = new NakedPut(testee, false);
|
// the naked put can succeed because it has txTimestamp after invalidation
|
||||||
|
NakedPut nakedPut = new NakedPut(testee, true);
|
||||||
exec(transactional, invalidation, nakedPut);
|
exec(transactional, invalidation, nakedPut);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -166,12 +173,11 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void registeredPutAfterRemovalTest(final boolean transactional,
|
private void registeredPutAfterRemovalTest(final boolean transactional,
|
||||||
final boolean removeRegion) throws Exception {
|
final boolean removeRegion) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
transactional ? tm : null);
|
||||||
Invalidation invalidation = new Invalidation(testee, removeRegion);
|
Invalidation invalidation = new Invalidation(testee, removeRegion);
|
||||||
RegularPut regularPut = new RegularPut(testee);
|
RegularPut regularPut = new RegularPut(testee);
|
||||||
exec(transactional, invalidation, regularPut);
|
exec(transactional, invalidation, regularPut);
|
||||||
|
@ -199,24 +205,24 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
private void registeredPutWithInterveningRemovalTest(
|
private void registeredPutWithInterveningRemovalTest(
|
||||||
final boolean transactional, final boolean removeRegion)
|
final boolean transactional, final boolean removeRegion)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
transactional ? tm : null);
|
||||||
try {
|
try {
|
||||||
|
long txTimestamp = System.currentTimeMillis();
|
||||||
if (transactional) {
|
if (transactional) {
|
||||||
tm.begin();
|
tm.begin();
|
||||||
}
|
}
|
||||||
testee.registerPendingPut(KEY1);
|
testee.registerPendingPut(KEY1, txTimestamp);
|
||||||
if (removeRegion) {
|
if (removeRegion) {
|
||||||
testee.invalidateRegion();
|
testee.beginInvalidatingRegion();
|
||||||
} else {
|
} else {
|
||||||
testee.beginInvalidatingKey(KEY1);
|
testee.beginInvalidatingKey(KEY1);
|
||||||
}
|
}
|
||||||
|
|
||||||
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1);
|
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1, txTimestamp);
|
||||||
try {
|
try {
|
||||||
assertNull(lock);
|
assertNull(lock);
|
||||||
}
|
}
|
||||||
|
@ -224,59 +230,10 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
if (lock != null) {
|
if (lock != null) {
|
||||||
testee.releasePutFromLoadLock(KEY1, lock);
|
testee.releasePutFromLoadLock(KEY1, lock);
|
||||||
}
|
}
|
||||||
testee.endInvalidatingKey(KEY1);
|
if (removeRegion) {
|
||||||
}
|
testee.endInvalidatingRegion();
|
||||||
} catch (Exception e) {
|
} else {
|
||||||
throw new RuntimeException(e);
|
testee.endInvalidatingKey(KEY1);
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testDelayedNakedPutAfterKeyRemoval() throws Exception {
|
|
||||||
delayedNakedPutAfterRemovalTest(false, false);
|
|
||||||
}
|
|
||||||
@Test
|
|
||||||
public void testDelayedNakedPutAfterKeyRemovalTransactional() throws Exception {
|
|
||||||
delayedNakedPutAfterRemovalTest(true, false);
|
|
||||||
}
|
|
||||||
@Test
|
|
||||||
public void testDelayedNakedPutAfterRegionRemoval() throws Exception {
|
|
||||||
delayedNakedPutAfterRemovalTest(false, true);
|
|
||||||
}
|
|
||||||
@Test
|
|
||||||
public void testDelayedNakedPutAfterRegionRemovalTransactional() throws Exception {
|
|
||||||
delayedNakedPutAfterRemovalTest(true, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void delayedNakedPutAfterRemovalTest(
|
|
||||||
final boolean transactional, final boolean removeRegion)
|
|
||||||
throws Exception {
|
|
||||||
withCacheManager(new CacheManagerCallable(
|
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
|
||||||
public void call() {
|
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
|
||||||
transactional ? tm : null, 100);
|
|
||||||
if (removeRegion) {
|
|
||||||
testee.invalidateRegion();
|
|
||||||
} else {
|
|
||||||
testee.beginInvalidatingKey(KEY1);
|
|
||||||
testee.endInvalidatingKey(KEY1);
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
if (transactional) {
|
|
||||||
tm.begin();
|
|
||||||
}
|
|
||||||
Thread.sleep(110);
|
|
||||||
|
|
||||||
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1);
|
|
||||||
try {
|
|
||||||
assertNotNull(lock);
|
|
||||||
} finally {
|
|
||||||
if (lock != null) {
|
|
||||||
testee.releasePutFromLoadLock(KEY1, null);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
@ -297,13 +254,11 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void multipleRegistrationtest(final boolean transactional) throws Exception {
|
private void multipleRegistrationtest(final boolean transactional) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
final PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
final PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(), cm,
|
||||||
transactional ? tm : null,
|
transactional ? tm : null);
|
||||||
PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
|
||||||
|
|
||||||
final CountDownLatch registeredLatch = new CountDownLatch(3);
|
final CountDownLatch registeredLatch = new CountDownLatch(3);
|
||||||
final CountDownLatch finishedLatch = new CountDownLatch(3);
|
final CountDownLatch finishedLatch = new CountDownLatch(3);
|
||||||
|
@ -312,13 +267,14 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
Runnable r = new Runnable() {
|
Runnable r = new Runnable() {
|
||||||
public void run() {
|
public void run() {
|
||||||
try {
|
try {
|
||||||
|
long txTimestamp = System.currentTimeMillis();
|
||||||
if (transactional) {
|
if (transactional) {
|
||||||
tm.begin();
|
tm.begin();
|
||||||
}
|
}
|
||||||
testee.registerPendingPut(KEY1);
|
testee.registerPendingPut(KEY1, txTimestamp);
|
||||||
registeredLatch.countDown();
|
registeredLatch.countDown();
|
||||||
registeredLatch.await(5, TimeUnit.SECONDS);
|
registeredLatch.await(5, TimeUnit.SECONDS);
|
||||||
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1);
|
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1, txTimestamp);
|
||||||
if (lock != null) {
|
if (lock != null) {
|
||||||
try {
|
try {
|
||||||
log.trace("Put from load lock acquired for key = " + KEY1);
|
log.trace("Put from load lock acquired for key = " + KEY1);
|
||||||
|
@ -341,7 +297,13 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
// Start with a removal so the "isPutValid" calls will fail if
|
// Start with a removal so the "isPutValid" calls will fail if
|
||||||
// any of the concurrent activity isn't handled properly
|
// any of the concurrent activity isn't handled properly
|
||||||
|
|
||||||
testee.invalidateRegion();
|
testee.beginInvalidatingRegion();
|
||||||
|
testee.endInvalidatingRegion();
|
||||||
|
try {
|
||||||
|
Thread.sleep(10);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
Thread.currentThread().interrupt();
|
||||||
|
}
|
||||||
|
|
||||||
// Do the registration + isPutValid calls
|
// Do the registration + isPutValid calls
|
||||||
executor.execute(r);
|
executor.execute(r);
|
||||||
|
@ -370,20 +332,20 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void invalidationBlocksForInProgressPutTest(final boolean keyOnly) throws Exception {
|
private void invalidationBlocksForInProgressPutTest(final boolean keyOnly) throws Exception {
|
||||||
withCacheManager(new CacheManagerCallable(
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
TestCacheManagerFactory.createCacheManager(false)) {
|
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
final PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(),
|
final PutFromLoadValidator testee = new PutFromLoadValidator(cm.getCache().getAdvancedCache(),
|
||||||
cm, null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
cm, null);
|
||||||
final CountDownLatch removeLatch = new CountDownLatch(1);
|
final CountDownLatch removeLatch = new CountDownLatch(1);
|
||||||
final CountDownLatch pferLatch = new CountDownLatch(1);
|
final CountDownLatch pferLatch = new CountDownLatch(1);
|
||||||
final AtomicReference<Object> cache = new AtomicReference<Object>("INITIAL");
|
final AtomicReference<Object> cache = new AtomicReference<Object>("INITIAL");
|
||||||
|
|
||||||
Callable<Boolean> pferCallable = new Callable<Boolean>() {
|
Callable<Boolean> pferCallable = new Callable<Boolean>() {
|
||||||
public Boolean call() throws Exception {
|
public Boolean call() throws Exception {
|
||||||
testee.registerPendingPut(KEY1);
|
long txTimestamp = System.currentTimeMillis();
|
||||||
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1);
|
testee.registerPendingPut(KEY1, txTimestamp);
|
||||||
|
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1, txTimestamp);
|
||||||
if (lock != null) {
|
if (lock != null) {
|
||||||
try {
|
try {
|
||||||
removeLatch.countDown();
|
removeLatch.countDown();
|
||||||
|
@ -405,7 +367,7 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
if (keyOnly) {
|
if (keyOnly) {
|
||||||
testee.beginInvalidatingKey(KEY1);
|
testee.beginInvalidatingKey(KEY1);
|
||||||
} else {
|
} else {
|
||||||
testee.invalidateRegion();
|
testee.beginInvalidatingRegion();
|
||||||
}
|
}
|
||||||
cache.set(null);
|
cache.set(null);
|
||||||
return null;
|
return null;
|
||||||
|
@ -466,14 +428,18 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
if (removeRegion) {
|
if (removeRegion) {
|
||||||
boolean success = putFromLoadValidator.invalidateRegion();
|
boolean success = putFromLoadValidator.beginInvalidatingRegion();
|
||||||
assertTrue(success);
|
assertTrue(success);
|
||||||
|
putFromLoadValidator.endInvalidatingRegion();;
|
||||||
} else {
|
} else {
|
||||||
boolean success = putFromLoadValidator.beginInvalidatingKey(KEY1);
|
boolean success = putFromLoadValidator.beginInvalidatingKey(KEY1);
|
||||||
assertTrue(success);
|
assertTrue(success);
|
||||||
success = putFromLoadValidator.endInvalidatingKey(KEY1);
|
success = putFromLoadValidator.endInvalidatingKey(KEY1);
|
||||||
assertTrue(success);
|
assertTrue(success);
|
||||||
}
|
}
|
||||||
|
// if we go for the timestamp-based approach, invalidation in the same millisecond
|
||||||
|
// as the registerPendingPut/acquirePutFromLoad lock results in failure.
|
||||||
|
Thread.sleep(10);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -488,9 +454,10 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
try {
|
try {
|
||||||
putFromLoadValidator.registerPendingPut(KEY1);
|
long txTimestamp = System.currentTimeMillis(); // this should be acquired before UserTransaction.begin()
|
||||||
|
putFromLoadValidator.registerPendingPut(KEY1, txTimestamp);
|
||||||
|
|
||||||
PutFromLoadValidator.Lock lock = putFromLoadValidator.acquirePutFromLoadLock(KEY1);
|
PutFromLoadValidator.Lock lock = putFromLoadValidator.acquirePutFromLoadLock(KEY1, txTimestamp);
|
||||||
try {
|
try {
|
||||||
assertNotNull(lock);
|
assertNotNull(lock);
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -517,7 +484,8 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
try {
|
try {
|
||||||
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1);
|
long txTimestamp = System.currentTimeMillis(); // this should be acquired before UserTransaction.begin()
|
||||||
|
PutFromLoadValidator.Lock lock = testee.acquirePutFromLoadLock(KEY1, txTimestamp);
|
||||||
try {
|
try {
|
||||||
if (expectSuccess) {
|
if (expectSuccess) {
|
||||||
assertNotNull(lock);
|
assertNotNull(lock);
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
package org.hibernate.test.cache.infinispan.collection;
|
package org.hibernate.test.cache.infinispan.collection;
|
||||||
|
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
import java.util.concurrent.Callable;
|
import java.util.concurrent.Callable;
|
||||||
import java.util.concurrent.CountDownLatch;
|
import java.util.concurrent.CountDownLatch;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
|
@ -31,6 +32,8 @@ import org.hibernate.test.cache.infinispan.NodeEnvironment;
|
||||||
import org.hibernate.test.cache.infinispan.util.CacheTestUtil;
|
import org.hibernate.test.cache.infinispan.util.CacheTestUtil;
|
||||||
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
||||||
import org.hibernate.test.cache.infinispan.util.TestingKeyFactory;
|
import org.hibernate.test.cache.infinispan.util.TestingKeyFactory;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
import org.infinispan.test.CacheManagerCallable;
|
import org.infinispan.test.CacheManagerCallable;
|
||||||
import org.infinispan.test.fwk.TestCacheManagerFactory;
|
import org.infinispan.test.fwk.TestCacheManagerFactory;
|
||||||
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
||||||
|
@ -155,29 +158,10 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
final CountDownLatch pferLatch = new CountDownLatch( 1 );
|
final CountDownLatch pferLatch = new CountDownLatch( 1 );
|
||||||
final CountDownLatch removeLatch = new CountDownLatch( 1 );
|
final CountDownLatch removeLatch = new CountDownLatch( 1 );
|
||||||
final TransactionManager remoteTm = remoteCollectionRegion.getTransactionManager();
|
final TransactionManager remoteTm = remoteCollectionRegion.getTransactionManager();
|
||||||
withCacheManager(new CacheManagerCallable(TestCacheManagerFactory.createCacheManager(false)) {
|
withCacheManager(new CacheManagerCallable(createCacheManager()) {
|
||||||
@Override
|
@Override
|
||||||
public void call() {
|
public void call() {
|
||||||
PutFromLoadValidator validator = new PutFromLoadValidator(remoteCollectionRegion.getCache(), cm,
|
PutFromLoadValidator validator = getPutFromLoadValidator(remoteCollectionRegion.getCache(), cm, remoteTm, removeLatch, pferLatch);
|
||||||
remoteTm, 20000) {
|
|
||||||
@Override
|
|
||||||
public Lock acquirePutFromLoadLock(Object key) {
|
|
||||||
Lock lock = super.acquirePutFromLoadLock( key );
|
|
||||||
try {
|
|
||||||
removeLatch.countDown();
|
|
||||||
pferLatch.await( 2, TimeUnit.SECONDS );
|
|
||||||
}
|
|
||||||
catch (InterruptedException e) {
|
|
||||||
log.debug( "Interrupted" );
|
|
||||||
Thread.currentThread().interrupt();
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
log.error( "Error", e );
|
|
||||||
throw new RuntimeException( "Error", e );
|
|
||||||
}
|
|
||||||
return lock;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
final TransactionalAccessDelegate delegate =
|
final TransactionalAccessDelegate delegate =
|
||||||
new TransactionalAccessDelegate(localCollectionRegion, validator);
|
new TransactionalAccessDelegate(localCollectionRegion, validator);
|
||||||
|
@ -218,7 +202,40 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
|
|
||||||
assertFalse(localCollectionRegion.getCache().containsKey("k1"));
|
assertFalse(localCollectionRegion.getCache().containsKey("k1"));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private static EmbeddedCacheManager createCacheManager() {
|
||||||
|
EmbeddedCacheManager cacheManager = TestCacheManagerFactory.createCacheManager(false);
|
||||||
|
cacheManager.defineConfiguration(InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME,
|
||||||
|
InfinispanRegionFactory.PENDING_PUTS_CACHE_CONFIGURATION);
|
||||||
|
return cacheManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected PutFromLoadValidator getPutFromLoadValidator(AdvancedCache cache, EmbeddedCacheManager cm,
|
||||||
|
TransactionManager tm,
|
||||||
|
CountDownLatch removeLatch, CountDownLatch pferLatch) {
|
||||||
|
// remove the interceptor inserted by default PutFromLoadValidator, we're using different one
|
||||||
|
PutFromLoadValidator.removeFromCache(cache);
|
||||||
|
return new PutFromLoadValidator(cache, cm, tm) {
|
||||||
|
@Override
|
||||||
|
public Lock acquirePutFromLoadLock(Object key, long txTimestamp) {
|
||||||
|
Lock lock = super.acquirePutFromLoadLock( key, txTimestamp);
|
||||||
|
try {
|
||||||
|
removeLatch.countDown();
|
||||||
|
pferLatch.await( 2, TimeUnit.SECONDS );
|
||||||
|
}
|
||||||
|
catch (InterruptedException e) {
|
||||||
|
log.debug( "Interrupted" );
|
||||||
|
Thread.currentThread().interrupt();
|
||||||
|
}
|
||||||
|
catch (Exception e) {
|
||||||
|
log.error( "Error", e );
|
||||||
|
throw new RuntimeException( "Error", e );
|
||||||
|
}
|
||||||
|
return lock;
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -455,6 +472,9 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
|
|
||||||
assertEquals( 0, remoteCollectionRegion.getCache().size() );
|
assertEquals( 0, remoteCollectionRegion.getCache().size() );
|
||||||
|
|
||||||
|
// Wait for async propagation of EndInvalidationCommand
|
||||||
|
sleep( 250 );
|
||||||
|
|
||||||
// Test whether the get above messes up the optimistic version
|
// Test whether the get above messes up the optimistic version
|
||||||
remoteAccessStrategy.putFromLoad(null, KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
remoteAccessStrategy.putFromLoad(null, KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
||||||
assertEquals( VALUE1, remoteAccessStrategy.get(null, KEY, System.currentTimeMillis() ) );
|
assertEquals( VALUE1, remoteAccessStrategy.get(null, KEY, System.currentTimeMillis() ) );
|
||||||
|
|
|
@ -289,8 +289,8 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
assertEquals("Correct node1 value", VALUE2, localAccessStrategy.get(null, KEY, txTimestamp));
|
assertEquals("Correct node1 value", VALUE2, localAccessStrategy.get(null, KEY, txTimestamp));
|
||||||
|
|
||||||
if (isUsingInvalidation()) {
|
if (isUsingInvalidation()) {
|
||||||
// no data version to prevent the PFER; we count on db locks preventing this
|
// invalidation command invalidates pending put
|
||||||
assertEquals("Expected node2 value", VALUE1, remoteAccessStrategy.get(null, KEY, txTimestamp));
|
assertEquals("Expected node2 value", null, remoteAccessStrategy.get(null, KEY, txTimestamp));
|
||||||
} else {
|
} else {
|
||||||
// The node1 update is replicated, preventing the node2 PFER
|
// The node1 update is replicated, preventing the node2 PFER
|
||||||
assertEquals("Correct node2 value", VALUE2, remoteAccessStrategy.get(null, KEY, txTimestamp));
|
assertEquals("Correct node2 value", VALUE2, remoteAccessStrategy.get(null, KEY, txTimestamp));
|
||||||
|
@ -571,9 +571,12 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
|
|
||||||
// Re-establishing the region root on the local node doesn't
|
// Re-establishing the region root on the local node doesn't
|
||||||
// propagate it to other nodes. Do a get on the remote node to re-establish
|
// propagate it to other nodes. Do a get on the remote node to re-establish
|
||||||
assertEquals(null, remoteAccessStrategy.get(null, KEY, System.currentTimeMillis()));
|
assertNull(remoteAccessStrategy.get(null, KEY, System.currentTimeMillis()));
|
||||||
assertEquals(0, remoteEntityRegion.getCache().size());
|
assertEquals(0, remoteEntityRegion.getCache().size());
|
||||||
|
|
||||||
|
// Wait for async propagation of EndInvalidationCommand before executing naked put
|
||||||
|
sleep(250);
|
||||||
|
|
||||||
// Test whether the get above messes up the optimistic version
|
// Test whether the get above messes up the optimistic version
|
||||||
remoteAccessStrategy.putFromLoad(null, KEY, VALUE1, System.currentTimeMillis(), new Integer(1));
|
remoteAccessStrategy.putFromLoad(null, KEY, VALUE1, System.currentTimeMillis(), new Integer(1));
|
||||||
assertEquals(VALUE1, remoteAccessStrategy.get(null, KEY, System.currentTimeMillis()));
|
assertEquals(VALUE1, remoteAccessStrategy.get(null, KEY, System.currentTimeMillis()));
|
||||||
|
|
|
@ -117,6 +117,7 @@ public abstract class AbstractFunctionalTestCase extends SingleNodeTestCase {
|
||||||
log.info("Entry persisted, let's load and delete it.");
|
log.info("Entry persisted, let's load and delete it.");
|
||||||
|
|
||||||
cleanupCache();
|
cleanupCache();
|
||||||
|
Thread.sleep(10);
|
||||||
|
|
||||||
withTx(tm, new Callable<Void>() {
|
withTx(tm, new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -861,6 +861,7 @@ public class BasicTransactionalTestCase extends AbstractFunctionalTestCase {
|
||||||
|
|
||||||
// Clear the cache before the transaction begins
|
// Clear the cache before the transaction begins
|
||||||
BasicTransactionalTestCase.this.cleanupCache();
|
BasicTransactionalTestCase.this.cleanupCache();
|
||||||
|
Thread.sleep(10);
|
||||||
|
|
||||||
withTx(tm, new Callable<Void>() {
|
withTx(tm, new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -951,6 +952,7 @@ public class BasicTransactionalTestCase extends AbstractFunctionalTestCase {
|
||||||
|
|
||||||
// TODO: Clear caches manually via cache manager (it's faster!!)
|
// TODO: Clear caches manually via cache manager (it's faster!!)
|
||||||
this.cleanupCache();
|
this.cleanupCache();
|
||||||
|
Thread.sleep(10);
|
||||||
stats.setStatisticsEnabled( true );
|
stats.setStatisticsEnabled( true );
|
||||||
stats.clear();
|
stats.clear();
|
||||||
|
|
||||||
|
@ -1028,6 +1030,7 @@ public class BasicTransactionalTestCase extends AbstractFunctionalTestCase {
|
||||||
assertEquals(2, slcStats.getPutCount());
|
assertEquals(2, slcStats.getPutCount());
|
||||||
|
|
||||||
cache.evictEntityRegions();
|
cache.evictEntityRegions();
|
||||||
|
Thread.sleep(10);
|
||||||
|
|
||||||
assertEquals(0, slcStats.getElementCountInMemory());
|
assertEquals(0, slcStats.getElementCountInMemory());
|
||||||
assertFalse("2lc entity cache is expected to not contain Citizen id = " + citizens.get(0).getId(),
|
assertFalse("2lc entity cache is expected to not contain Citizen id = " + citizens.get(0).getId(),
|
||||||
|
@ -1052,6 +1055,46 @@ public class BasicTransactionalTestCase extends AbstractFunctionalTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMultipleEvictAll() throws Exception {
|
||||||
|
final List<Citizen> citizens = saveSomeCitizens();
|
||||||
|
|
||||||
|
withTx(tm, new Callable<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
Session s = openSession();
|
||||||
|
Transaction tx = s.beginTransaction();
|
||||||
|
Cache cache = s.getSessionFactory().getCache();
|
||||||
|
|
||||||
|
cache.evictEntityRegions();
|
||||||
|
cache.evictEntityRegions();
|
||||||
|
|
||||||
|
// cleanup
|
||||||
|
tx.commit();
|
||||||
|
s.close();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
withTx(tm, new Callable<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
Session s = openSession();
|
||||||
|
Transaction tx = s.beginTransaction();
|
||||||
|
Cache cache = s.getSessionFactory().getCache();
|
||||||
|
|
||||||
|
cache.evictEntityRegions();
|
||||||
|
|
||||||
|
s.delete(s.load(Citizen.class, citizens.get(0).getId()));
|
||||||
|
s.delete(s.load(Citizen.class, citizens.get(1).getId()));
|
||||||
|
|
||||||
|
// cleanup
|
||||||
|
tx.commit();
|
||||||
|
s.close();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
private List<Citizen> saveSomeCitizens() throws Exception {
|
private List<Citizen> saveSomeCitizens() throws Exception {
|
||||||
final Citizen c1 = new Citizen();
|
final Citizen c1 = new Citizen();
|
||||||
c1.setFirstname( "Emmanuel" );
|
c1.setFirstname( "Emmanuel" );
|
||||||
|
|
|
@ -25,6 +25,9 @@ import org.hibernate.service.spi.Configurable;
|
||||||
public class DualNodeJtaPlatformImpl implements JtaPlatform, Configurable {
|
public class DualNodeJtaPlatformImpl implements JtaPlatform, Configurable {
|
||||||
private String nodeId;
|
private String nodeId;
|
||||||
|
|
||||||
|
public DualNodeJtaPlatformImpl() {
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void configure(Map configurationValues) {
|
public void configure(Map configurationValues) {
|
||||||
nodeId = (String) configurationValues.get( DualNodeTestCase.NODE_ID_PROP );
|
nodeId = (String) configurationValues.get( DualNodeTestCase.NODE_ID_PROP );
|
||||||
|
|
|
@ -20,11 +20,10 @@ import org.hibernate.resource.transaction.backend.jta.internal.JtaTransactionCoo
|
||||||
|
|
||||||
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
import org.hibernate.test.cache.infinispan.util.InfinispanTestingSetup;
|
||||||
import org.hibernate.testing.junit4.BaseNonConfigCoreFunctionalTestCase;
|
import org.hibernate.testing.junit4.BaseNonConfigCoreFunctionalTestCase;
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Before;
|
|
||||||
|
|
||||||
import org.infinispan.util.logging.Log;
|
import org.infinispan.util.logging.Log;
|
||||||
import org.infinispan.util.logging.LogFactory;
|
import org.infinispan.util.logging.LogFactory;
|
||||||
|
import org.junit.Before;
|
||||||
import org.junit.ClassRule;
|
import org.junit.ClassRule;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -78,18 +77,20 @@ public abstract class DualNodeTestCase extends BaseNonConfigCoreFunctionalTestCa
|
||||||
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Before
|
@Override
|
||||||
public void prepare() throws Exception {
|
public void startUp() {
|
||||||
|
super.startUp();
|
||||||
// In some cases tests are multi-threaded, so they have to join the group
|
// In some cases tests are multi-threaded, so they have to join the group
|
||||||
infinispanTestIdentifier.joinContext();
|
infinispanTestIdentifier.joinContext();
|
||||||
secondNodeEnvironment = new SecondNodeEnvironment();
|
secondNodeEnvironment = new SecondNodeEnvironment();
|
||||||
}
|
}
|
||||||
|
|
||||||
@After
|
@Override
|
||||||
public void unPrepare() {
|
public void shutDown() {
|
||||||
if ( secondNodeEnvironment != null ) {
|
if ( secondNodeEnvironment != null ) {
|
||||||
secondNodeEnvironment.shutDown();
|
secondNodeEnvironment.shutDown();
|
||||||
}
|
}
|
||||||
|
super.shutDown();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected SecondNodeEnvironment secondNodeEnvironment() {
|
protected SecondNodeEnvironment secondNodeEnvironment() {
|
||||||
|
|
|
@ -10,13 +10,24 @@ import javax.transaction.TransactionManager;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.Phaser;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.TimeoutException;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
import org.hibernate.Session;
|
import org.hibernate.Session;
|
||||||
import org.hibernate.SessionFactory;
|
import org.hibernate.SessionFactory;
|
||||||
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.test.cache.infinispan.functional.Contact;
|
import org.hibernate.test.cache.infinispan.functional.Contact;
|
||||||
import org.hibernate.test.cache.infinispan.functional.Customer;
|
import org.hibernate.test.cache.infinispan.functional.Customer;
|
||||||
|
import org.hibernate.testing.TestForIssue;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
import org.infinispan.Cache;
|
import org.infinispan.Cache;
|
||||||
import org.infinispan.manager.CacheContainer;
|
import org.infinispan.commands.read.GetKeyValueCommand;
|
||||||
|
import org.infinispan.commons.util.Util;
|
||||||
|
import org.infinispan.context.InvocationContext;
|
||||||
|
import org.infinispan.interceptors.base.BaseCustomInterceptor;
|
||||||
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
import org.infinispan.notifications.Listener;
|
import org.infinispan.notifications.Listener;
|
||||||
import org.infinispan.notifications.cachelistener.annotation.CacheEntryVisited;
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryVisited;
|
||||||
import org.infinispan.notifications.cachelistener.event.CacheEntryVisitedEvent;
|
import org.infinispan.notifications.cachelistener.event.CacheEntryVisitedEvent;
|
||||||
|
@ -25,7 +36,9 @@ import org.infinispan.util.logging.LogFactory;
|
||||||
import org.jboss.util.collection.ConcurrentSet;
|
import org.jboss.util.collection.ConcurrentSet;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import static org.infinispan.test.TestingUtil.withTx;
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -42,96 +55,204 @@ public class EntityCollectionInvalidationTestCase extends DualNodeTestCase {
|
||||||
|
|
||||||
static int test = 0;
|
static int test = 0;
|
||||||
|
|
||||||
@Test
|
private EmbeddedCacheManager localManager, remoteManager;
|
||||||
public void testAll() throws Exception {
|
private Cache localCustomerCache, remoteCustomerCache;
|
||||||
log.info( "*** testAll()" );
|
private Cache localContactCache, remoteContactCache;
|
||||||
|
private Cache localCollectionCache, remoteCollectionCache;
|
||||||
|
private MyListener localListener, remoteListener;
|
||||||
|
private TransactionManager localTM, remoteTM;
|
||||||
|
private SessionFactory localFactory, remoteFactory;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void startUp() {
|
||||||
|
super.startUp();
|
||||||
// Bind a listener to the "local" cache
|
// Bind a listener to the "local" cache
|
||||||
// Our region factory makes its CacheManager available to us
|
// Our region factory makes its CacheManager available to us
|
||||||
CacheContainer localManager = ClusterAwareRegionFactory.getCacheManager( DualNodeTestCase.LOCAL );
|
localManager = ClusterAwareRegionFactory.getCacheManager( DualNodeTestCase.LOCAL );
|
||||||
// Cache localCache = localManager.getCache("entity");
|
// Cache localCache = localManager.getCache("entity");
|
||||||
Cache localCustomerCache = localManager.getCache( Customer.class.getName() );
|
localCustomerCache = localManager.getCache( Customer.class.getName() );
|
||||||
Cache localContactCache = localManager.getCache( Contact.class.getName() );
|
localContactCache = localManager.getCache( Contact.class.getName() );
|
||||||
Cache localCollectionCache = localManager.getCache( Customer.class.getName() + ".contacts" );
|
localCollectionCache = localManager.getCache( Customer.class.getName() + ".contacts" );
|
||||||
MyListener localListener = new MyListener( "local" );
|
localListener = new MyListener( "local" );
|
||||||
localCustomerCache.addListener( localListener );
|
localCustomerCache.addListener( localListener );
|
||||||
localContactCache.addListener( localListener );
|
localContactCache.addListener( localListener );
|
||||||
localCollectionCache.addListener( localListener );
|
localCollectionCache.addListener( localListener );
|
||||||
TransactionManager localTM = DualNodeJtaTransactionManagerImpl.getInstance( DualNodeTestCase.LOCAL );
|
|
||||||
|
|
||||||
// Bind a listener to the "remote" cache
|
// Bind a listener to the "remote" cache
|
||||||
CacheContainer remoteManager = ClusterAwareRegionFactory.getCacheManager( DualNodeTestCase.REMOTE );
|
remoteManager = ClusterAwareRegionFactory.getCacheManager( DualNodeTestCase.REMOTE );
|
||||||
Cache remoteCustomerCache = remoteManager.getCache( Customer.class.getName() );
|
remoteCustomerCache = remoteManager.getCache( Customer.class.getName() );
|
||||||
Cache remoteContactCache = remoteManager.getCache( Contact.class.getName() );
|
remoteContactCache = remoteManager.getCache( Contact.class.getName() );
|
||||||
Cache remoteCollectionCache = remoteManager.getCache( Customer.class.getName() + ".contacts" );
|
remoteCollectionCache = remoteManager.getCache( Customer.class.getName() + ".contacts" );
|
||||||
MyListener remoteListener = new MyListener( "remote" );
|
remoteListener = new MyListener( "remote" );
|
||||||
remoteCustomerCache.addListener( remoteListener );
|
remoteCustomerCache.addListener( remoteListener );
|
||||||
remoteContactCache.addListener( remoteListener );
|
remoteContactCache.addListener( remoteListener );
|
||||||
remoteCollectionCache.addListener( remoteListener );
|
remoteCollectionCache.addListener( remoteListener );
|
||||||
TransactionManager remoteTM = DualNodeJtaTransactionManagerImpl.getInstance( DualNodeTestCase.REMOTE );
|
|
||||||
|
|
||||||
SessionFactory localFactory = sessionFactory();
|
localFactory = sessionFactory();
|
||||||
SessionFactory remoteFactory = secondNodeEnvironment().getSessionFactory();
|
remoteFactory = secondNodeEnvironment().getSessionFactory();
|
||||||
|
|
||||||
try {
|
localTM = DualNodeJtaTransactionManagerImpl.getInstance( DualNodeTestCase.LOCAL );
|
||||||
assertTrue( remoteListener.isEmpty() );
|
remoteTM = DualNodeJtaTransactionManagerImpl.getInstance( DualNodeTestCase.REMOTE );
|
||||||
assertTrue( localListener.isEmpty() );
|
}
|
||||||
|
|
||||||
log.debug( "Create node 0" );
|
@Override
|
||||||
IdContainer ids = createCustomer( localFactory, localTM );
|
public void shutDown() {
|
||||||
|
cleanupTransactionManagement();
|
||||||
|
}
|
||||||
|
|
||||||
assertTrue( remoteListener.isEmpty() );
|
@Override
|
||||||
assertTrue( localListener.isEmpty() );
|
protected void cleanupTest() throws Exception {
|
||||||
|
cleanup(localFactory, localTM);
|
||||||
|
localListener.clear();
|
||||||
|
remoteListener.clear();
|
||||||
|
// do not call super.cleanupTest becasue we would clean transaction managers
|
||||||
|
}
|
||||||
|
|
||||||
// Sleep a bit to let async commit propagate. Really just to
|
@Test
|
||||||
// help keep the logs organized for debugging any issues
|
public void testAll() throws Exception {
|
||||||
sleep( SLEEP_TIME );
|
assertEmptyCaches();
|
||||||
|
assertTrue( remoteListener.isEmpty() );
|
||||||
|
assertTrue( localListener.isEmpty() );
|
||||||
|
|
||||||
log.debug( "Find node 0" );
|
log.debug( "Create node 0" );
|
||||||
// This actually brings the collection into the cache
|
IdContainer ids = createCustomer( localFactory, localTM );
|
||||||
getCustomer( ids.customerId, localFactory, localTM );
|
|
||||||
|
|
||||||
sleep( SLEEP_TIME );
|
assertTrue( remoteListener.isEmpty() );
|
||||||
|
assertTrue( localListener.isEmpty() );
|
||||||
|
|
||||||
// Now the collection is in the cache so, the 2nd "get"
|
// Sleep a bit to let async commit propagate. Really just to
|
||||||
// should read everything from the cache
|
// help keep the logs organized for debugging any issues
|
||||||
log.debug( "Find(2) node 0" );
|
sleep( SLEEP_TIME );
|
||||||
localListener.clear();
|
|
||||||
getCustomer( ids.customerId, localFactory, localTM );
|
|
||||||
|
|
||||||
// Check the read came from the cache
|
log.debug( "Find node 0" );
|
||||||
log.debug( "Check cache 0" );
|
// This actually brings the collection into the cache
|
||||||
assertLoadedFromCache( localListener, ids.customerId, ids.contactIds );
|
getCustomer( ids.customerId, localFactory, localTM );
|
||||||
|
|
||||||
log.debug( "Find node 1" );
|
sleep( SLEEP_TIME );
|
||||||
// This actually brings the collection into the cache since invalidation is in use
|
|
||||||
getCustomer( ids.customerId, remoteFactory, remoteTM );
|
|
||||||
|
|
||||||
// Now the collection is in the cache so, the 2nd "get"
|
// Now the collection is in the cache so, the 2nd "get"
|
||||||
// should read everything from the cache
|
// should read everything from the cache
|
||||||
log.debug( "Find(2) node 1" );
|
log.debug( "Find(2) node 0" );
|
||||||
remoteListener.clear();
|
localListener.clear();
|
||||||
getCustomer( ids.customerId, remoteFactory, remoteTM );
|
getCustomer( ids.customerId, localFactory, localTM );
|
||||||
|
|
||||||
// Check the read came from the cache
|
// Check the read came from the cache
|
||||||
log.debug( "Check cache 1" );
|
log.debug( "Check cache 0" );
|
||||||
assertLoadedFromCache( remoteListener, ids.customerId, ids.contactIds );
|
assertLoadedFromCache( localListener, ids.customerId, ids.contactIds );
|
||||||
|
|
||||||
// Modify customer in remote
|
log.debug( "Find node 1" );
|
||||||
remoteListener.clear();
|
// This actually brings the collection into the cache since invalidation is in use
|
||||||
ids = modifyCustomer( ids.customerId, remoteFactory, remoteTM );
|
getCustomer( ids.customerId, remoteFactory, remoteTM );
|
||||||
sleep( 250 );
|
|
||||||
assertLoadedFromCache( remoteListener, ids.customerId, ids.contactIds );
|
|
||||||
|
|
||||||
// After modification, local cache should have been invalidated and hence should be empty
|
// Now the collection is in the cache so, the 2nd "get"
|
||||||
assertEquals( 0, localCollectionCache.size() );
|
// should read everything from the cache
|
||||||
assertEquals( 0, localCustomerCache.size() );
|
log.debug( "Find(2) node 1" );
|
||||||
|
remoteListener.clear();
|
||||||
|
getCustomer( ids.customerId, remoteFactory, remoteTM );
|
||||||
|
|
||||||
|
// Check the read came from the cache
|
||||||
|
log.debug( "Check cache 1" );
|
||||||
|
assertLoadedFromCache( remoteListener, ids.customerId, ids.contactIds );
|
||||||
|
|
||||||
|
// Modify customer in remote
|
||||||
|
remoteListener.clear();
|
||||||
|
ids = modifyCustomer( ids.customerId, remoteFactory, remoteTM );
|
||||||
|
sleep( 250 );
|
||||||
|
assertLoadedFromCache( remoteListener, ids.customerId, ids.contactIds );
|
||||||
|
|
||||||
|
// After modification, local cache should have been invalidated and hence should be empty
|
||||||
|
assertEquals( 0, localCollectionCache.size() );
|
||||||
|
assertEquals( 0, localCustomerCache.size() );
|
||||||
|
}
|
||||||
|
|
||||||
|
@TestForIssue(jiraKey = "HHH-9881")
|
||||||
|
@Test
|
||||||
|
public void testConcurrentLoadAndRemoval() throws Exception {
|
||||||
|
AtomicReference<Exception> getException = new AtomicReference<>();
|
||||||
|
AtomicReference<Exception> deleteException = new AtomicReference<>();
|
||||||
|
|
||||||
|
Phaser getPhaser = new Phaser(2);
|
||||||
|
HookInterceptor hookInterceptor = new HookInterceptor(getException);
|
||||||
|
AdvancedCache remotePPCache = remoteCustomerCache.getCacheManager().getCache(
|
||||||
|
remoteCustomerCache.getName() + "-" + InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME).getAdvancedCache();
|
||||||
|
remotePPCache.getAdvancedCache().addInterceptor(hookInterceptor, 0);
|
||||||
|
|
||||||
|
IdContainer idContainer = new IdContainer();
|
||||||
|
withTx(localTM, () -> {
|
||||||
|
Session s = localFactory.getCurrentSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
Customer customer = new Customer();
|
||||||
|
customer.setName( "JBoss" );
|
||||||
|
s.persist(customer);
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
idContainer.customerId = customer.getId();
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
// start loading
|
||||||
|
|
||||||
|
Thread getThread = new Thread(() -> {
|
||||||
|
try {
|
||||||
|
withTx(remoteTM, () -> {
|
||||||
|
Session s = remoteFactory.getCurrentSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
s.get(Customer.class, idContainer.customerId);
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Failure to get customer", e);
|
||||||
|
getException.set(e);
|
||||||
|
}
|
||||||
|
}, "get-thread");
|
||||||
|
Thread deleteThread = new Thread(() -> {
|
||||||
|
try {
|
||||||
|
withTx(localTM, () -> {
|
||||||
|
Session s = localFactory.getCurrentSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
Customer customer = s.get(Customer.class, idContainer.customerId);
|
||||||
|
s.delete(customer);
|
||||||
|
s.getTransaction().commit();
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Failure to delete customer", e);
|
||||||
|
deleteException.set(e);
|
||||||
|
}
|
||||||
|
}, "delete-thread");
|
||||||
|
// get thread should block on the beginning of PutFromLoadValidator#acquirePutFromLoadLock
|
||||||
|
hookInterceptor.block(getPhaser, getThread);
|
||||||
|
getThread.start();
|
||||||
|
|
||||||
|
arriveAndAwait(getPhaser);
|
||||||
|
deleteThread.start();
|
||||||
|
deleteThread.join();
|
||||||
|
hookInterceptor.unblock();
|
||||||
|
arriveAndAwait(getPhaser);
|
||||||
|
getThread.join();
|
||||||
|
|
||||||
|
if (getException.get() != null) {
|
||||||
|
throw new IllegalStateException("get-thread failed", getException.get());
|
||||||
}
|
}
|
||||||
finally {
|
if (deleteException.get() != null) {
|
||||||
// cleanup the db
|
throw new IllegalStateException("delete-thread failed", deleteException.get());
|
||||||
log.debug( "Cleaning up" );
|
|
||||||
cleanup( localFactory, localTM );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Customer localCustomer = getCustomer(idContainer.customerId, localFactory, localTM);
|
||||||
|
assertNull(localCustomer);
|
||||||
|
Customer remoteCustomer = getCustomer(idContainer.customerId, remoteFactory, remoteTM);
|
||||||
|
assertNull(remoteCustomer);
|
||||||
|
assertTrue(remoteCustomerCache.isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void assertEmptyCaches() {
|
||||||
|
assertTrue( localCustomerCache.isEmpty() );
|
||||||
|
assertTrue( localContactCache.isEmpty() );
|
||||||
|
assertTrue( localCollectionCache.isEmpty() );
|
||||||
|
assertTrue( remoteCustomerCache.isEmpty() );
|
||||||
|
assertTrue( remoteContactCache.isEmpty() );
|
||||||
|
assertTrue( remoteCollectionCache.isEmpty() );
|
||||||
}
|
}
|
||||||
|
|
||||||
private IdContainer createCustomer(SessionFactory sessionFactory, TransactionManager tm)
|
private IdContainer createCustomer(SessionFactory sessionFactory, TransactionManager tm)
|
||||||
|
@ -211,10 +332,16 @@ public class EntityCollectionInvalidationTestCase extends DualNodeTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private Customer doGetCustomer(Integer id, Session session, TransactionManager tm) throws Exception {
|
private Customer doGetCustomer(Integer id, Session session, TransactionManager tm) throws Exception {
|
||||||
Customer customer = (Customer) session.get( Customer.class, id );
|
Customer customer = session.get( Customer.class, id );
|
||||||
|
if (customer == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
// Access all the contacts
|
// Access all the contacts
|
||||||
for ( Iterator it = customer.getContacts().iterator(); it.hasNext(); ) {
|
Set<Contact> contacts = customer.getContacts();
|
||||||
((Contact) it.next()).getName();
|
if (contacts != null) {
|
||||||
|
for (Iterator it = contacts.iterator(); it.hasNext(); ) {
|
||||||
|
((Contact) it.next()).getName();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return customer;
|
return customer;
|
||||||
}
|
}
|
||||||
|
@ -271,7 +398,10 @@ public class EntityCollectionInvalidationTestCase extends DualNodeTestCase {
|
||||||
c.setContacts( null );
|
c.setContacts( null );
|
||||||
session.delete( c );
|
session.delete( c );
|
||||||
}
|
}
|
||||||
|
// since we don't use orphan removal, some contacts may persist
|
||||||
|
for (Object contact : session.createCriteria(Contact.class).list()) {
|
||||||
|
session.delete(contact);
|
||||||
|
}
|
||||||
tm.commit();
|
tm.commit();
|
||||||
}
|
}
|
||||||
catch (Exception e) {
|
catch (Exception e) {
|
||||||
|
@ -313,6 +443,15 @@ public class EntityCollectionInvalidationTestCase extends DualNodeTestCase {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected static void arriveAndAwait(Phaser phaser) throws TimeoutException, InterruptedException {
|
||||||
|
try {
|
||||||
|
phaser.awaitAdvanceInterruptibly(phaser.arrive(), 10, TimeUnit.SECONDS);
|
||||||
|
} catch (TimeoutException e) {
|
||||||
|
log.error("Failed to progress: " + Util.threadDump());
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Listener
|
@Listener
|
||||||
public static class MyListener {
|
public static class MyListener {
|
||||||
private static final Log log = LogFactory.getLog( MyListener.class );
|
private static final Log log = LogFactory.getLog( MyListener.class );
|
||||||
|
@ -355,4 +494,45 @@ public class EntityCollectionInvalidationTestCase extends DualNodeTestCase {
|
||||||
Set<Integer> contactIds;
|
Set<Integer> contactIds;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class HookInterceptor extends BaseCustomInterceptor {
|
||||||
|
final AtomicReference<Exception> failure;
|
||||||
|
Phaser phaser;
|
||||||
|
Thread thread;
|
||||||
|
|
||||||
|
private HookInterceptor(AtomicReference<Exception> failure) {
|
||||||
|
this.failure = failure;
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void block(Phaser phaser, Thread thread) {
|
||||||
|
this.phaser = phaser;
|
||||||
|
this.thread = thread;
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void unblock() {
|
||||||
|
phaser = null;
|
||||||
|
thread = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object visitGetKeyValueCommand(InvocationContext ctx, GetKeyValueCommand command) throws Throwable {
|
||||||
|
try {
|
||||||
|
Phaser phaser;
|
||||||
|
Thread thread;
|
||||||
|
synchronized (this) {
|
||||||
|
phaser = this.phaser;
|
||||||
|
thread = this.thread;
|
||||||
|
}
|
||||||
|
if (phaser != null && Thread.currentThread() == thread) {
|
||||||
|
arriveAndAwait(phaser);
|
||||||
|
arriveAndAwait(phaser);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
failure.set(e);
|
||||||
|
throw e;
|
||||||
|
} finally {
|
||||||
|
return super.visitGetKeyValueCommand(ctx, command);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue