HHH-7640 Improve single node Infinispan 2LC performance
* Use an Infinispan cache to maintain pending puts, which avoids the need to control it's memory consumption in the critical path of putFromLoad. * This cache is shared by all regions, and it's configured with aggressive expiration settings to avoid piling up pending put operations. * Added a 2LC stress test that tests behaivour and performance of 2LC under multiple situations, such as entity inserts, updates, find via PK, find via query and deletes. * Some other minor performance enhancements, such as avoiding classloader aware cache wrapper if using 2LC locally. * Remove cache adapter to reduce construction of useless objects. * Cache flagged caches in order to avoid recomputing decorated caches all the time, which reduces memory consumption. * Skip locking for timestamp updates and separate timestamp region implementations for local vs clustered scenarios.
This commit is contained in:
parent
2b4097aa49
commit
a074d3244d
|
@ -32,3 +32,10 @@ bin
|
||||||
# Miscellaneous
|
# Miscellaneous
|
||||||
*.log
|
*.log
|
||||||
.clover
|
.clover
|
||||||
|
|
||||||
|
# JBoss Transactions
|
||||||
|
ObjectStore
|
||||||
|
|
||||||
|
# Profiler and heap dumps
|
||||||
|
*.jps
|
||||||
|
*.hprof
|
||||||
|
|
|
@ -61,6 +61,8 @@ public class StandardQueryCache implements QueryCache {
|
||||||
StandardQueryCache.class.getName()
|
StandardQueryCache.class.getName()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private static final boolean tracing = LOG.isTraceEnabled();
|
||||||
|
|
||||||
private QueryResultsRegion cacheRegion;
|
private QueryResultsRegion cacheRegion;
|
||||||
private UpdateTimestampsCache updateTimestampsCache;
|
private UpdateTimestampsCache updateTimestampsCache;
|
||||||
|
|
||||||
|
@ -241,7 +243,7 @@ public class StandardQueryCache implements QueryCache {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void logCachedResultRowDetails(Type[] returnTypes, Object[] tuple) {
|
private static void logCachedResultRowDetails(Type[] returnTypes, Object[] tuple) {
|
||||||
if ( !LOG.isTraceEnabled() ) {
|
if ( !tracing ) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if ( tuple == null ) {
|
if ( tuple == null ) {
|
||||||
|
|
|
@ -94,6 +94,8 @@ public class StatefulPersistenceContext implements PersistenceContext {
|
||||||
|
|
||||||
private static final CoreMessageLogger LOG = Logger.getMessageLogger( CoreMessageLogger.class, StatefulPersistenceContext.class.getName() );
|
private static final CoreMessageLogger LOG = Logger.getMessageLogger( CoreMessageLogger.class, StatefulPersistenceContext.class.getName() );
|
||||||
|
|
||||||
|
private static final boolean tracing = LOG.isTraceEnabled();
|
||||||
|
|
||||||
public static final Object NO_ROW = new MarkerObject( "NO_ROW" );
|
public static final Object NO_ROW = new MarkerObject( "NO_ROW" );
|
||||||
|
|
||||||
private static final int INIT_COLL_SIZE = 8;
|
private static final int INIT_COLL_SIZE = 8;
|
||||||
|
@ -1004,7 +1006,9 @@ public class StatefulPersistenceContext implements PersistenceContext {
|
||||||
@Override
|
@Override
|
||||||
public void initializeNonLazyCollections() throws HibernateException {
|
public void initializeNonLazyCollections() throws HibernateException {
|
||||||
if ( loadCounter == 0 ) {
|
if ( loadCounter == 0 ) {
|
||||||
LOG.debug( "Initializing non-lazy collections" );
|
if (tracing)
|
||||||
|
LOG.trace( "Initializing non-lazy collections" );
|
||||||
|
|
||||||
//do this work only at the very highest level of the load
|
//do this work only at the very highest level of the load
|
||||||
loadCounter++; //don't let this method be called recursively
|
loadCounter++; //don't let this method be called recursively
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -181,6 +181,8 @@ public final class SessionImpl extends AbstractSessionImpl implements EventSourc
|
||||||
|
|
||||||
private static final CoreMessageLogger LOG = Logger.getMessageLogger(CoreMessageLogger.class, SessionImpl.class.getName());
|
private static final CoreMessageLogger LOG = Logger.getMessageLogger(CoreMessageLogger.class, SessionImpl.class.getName());
|
||||||
|
|
||||||
|
private static final boolean tracing = LOG.isTraceEnabled();
|
||||||
|
|
||||||
private transient long timestamp;
|
private transient long timestamp;
|
||||||
|
|
||||||
private transient SessionOwner sessionOwner;
|
private transient SessionOwner sessionOwner;
|
||||||
|
@ -310,7 +312,8 @@ public final class SessionImpl extends AbstractSessionImpl implements EventSourc
|
||||||
factory.getStatisticsImplementor().openSession();
|
factory.getStatisticsImplementor().openSession();
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG.debugf( "Opened session at timestamp: %s", timestamp );
|
if (tracing)
|
||||||
|
LOG.tracef( "Opened session at timestamp: %s", timestamp );
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -13,6 +13,7 @@ dependencies {
|
||||||
testCompile( libraries.jnp_client )
|
testCompile( libraries.jnp_client )
|
||||||
testCompile( libraries.jnp_server )
|
testCompile( libraries.jnp_server )
|
||||||
testCompile( libraries.rhq )
|
testCompile( libraries.rhq )
|
||||||
|
testCompile ('mysql:mysql-connector-java:5.1.17')
|
||||||
}
|
}
|
||||||
|
|
||||||
test {
|
test {
|
||||||
|
@ -27,3 +28,15 @@ test {
|
||||||
enabled = true
|
enabled = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
task packageTests(type: Jar) {
|
||||||
|
from sourceSets.test.output
|
||||||
|
classifier = 'tests'
|
||||||
|
}
|
||||||
|
|
||||||
|
task sourcesTestJar(type: Jar, dependsOn:classes) {
|
||||||
|
from sourceSets.test.allSource
|
||||||
|
classifier = 'test-sources'
|
||||||
|
}
|
||||||
|
|
||||||
|
artifacts.archives packageTests
|
||||||
|
artifacts.archives sourcesTestJar
|
||||||
|
|
|
@ -10,15 +10,20 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import javax.transaction.TransactionManager;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.timestamp.ClusteredTimestampsRegionImpl;
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
import org.infinispan.AdvancedCache;
|
import org.infinispan.AdvancedCache;
|
||||||
import org.infinispan.Cache;
|
|
||||||
import org.infinispan.commands.module.ModuleCommandFactory;
|
import org.infinispan.commands.module.ModuleCommandFactory;
|
||||||
import org.infinispan.config.Configuration;
|
import org.infinispan.config.Configuration;
|
||||||
|
import org.infinispan.configuration.cache.CacheMode;
|
||||||
|
import org.infinispan.configuration.cache.ConfigurationBuilder;
|
||||||
import org.infinispan.factories.GlobalComponentRegistry;
|
import org.infinispan.factories.GlobalComponentRegistry;
|
||||||
import org.infinispan.manager.DefaultCacheManager;
|
import org.infinispan.manager.DefaultCacheManager;
|
||||||
import org.infinispan.manager.EmbeddedCacheManager;
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
|
import org.infinispan.transaction.TransactionMode;
|
||||||
|
import org.infinispan.util.concurrent.IsolationLevel;
|
||||||
import org.infinispan.util.logging.Log;
|
import org.infinispan.util.logging.Log;
|
||||||
import org.infinispan.util.logging.LogFactory;
|
import org.infinispan.util.logging.LogFactory;
|
||||||
|
|
||||||
|
@ -34,8 +39,6 @@ import org.hibernate.cache.infinispan.query.QueryResultsRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.timestamp.TimestampTypeOverrides;
|
import org.hibernate.cache.infinispan.timestamp.TimestampTypeOverrides;
|
||||||
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.tm.HibernateTransactionManagerLookup;
|
import org.hibernate.cache.infinispan.tm.HibernateTransactionManagerLookup;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapterImpl;
|
|
||||||
import org.hibernate.cache.spi.CollectionRegion;
|
import org.hibernate.cache.spi.CollectionRegion;
|
||||||
import org.hibernate.cache.spi.EntityRegion;
|
import org.hibernate.cache.spi.EntityRegion;
|
||||||
import org.hibernate.cache.spi.NaturalIdRegion;
|
import org.hibernate.cache.spi.NaturalIdRegion;
|
||||||
|
@ -166,6 +169,11 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
*/
|
*/
|
||||||
public static final boolean DEF_USE_SYNCHRONIZATION = true;
|
public static final boolean DEF_USE_SYNCHRONIZATION = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Name of the pending puts cache.
|
||||||
|
*/
|
||||||
|
public static final String PENDING_PUTS_CACHE_NAME = "pending-puts";
|
||||||
|
|
||||||
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>();
|
||||||
|
@ -174,8 +182,6 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
|
|
||||||
private org.infinispan.transaction.lookup.TransactionManagerLookup transactionManagerlookup;
|
private org.infinispan.transaction.lookup.TransactionManagerLookup transactionManagerlookup;
|
||||||
|
|
||||||
private TransactionManager transactionManager;
|
|
||||||
|
|
||||||
private List<String> regionNames = new ArrayList<String>();
|
private List<String> regionNames = new ArrayList<String>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -197,8 +203,8 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
public CollectionRegion buildCollectionRegion(String regionName, Properties properties, CacheDataDescription metadata) throws CacheException {
|
public CollectionRegion buildCollectionRegion(String regionName, Properties properties, CacheDataDescription metadata) throws CacheException {
|
||||||
if (log.isDebugEnabled()) log.debug("Building collection cache region [" + regionName + "]");
|
if (log.isDebugEnabled()) log.debug("Building collection cache region [" + regionName + "]");
|
||||||
AdvancedCache cache = getCache(regionName, COLLECTION_KEY, properties);
|
AdvancedCache cache = getCache(regionName, COLLECTION_KEY, properties);
|
||||||
CacheAdapter cacheAdapter = CacheAdapterImpl.newInstance(cache);
|
CollectionRegionImpl region = new CollectionRegionImpl(
|
||||||
CollectionRegionImpl region = new CollectionRegionImpl(cacheAdapter, regionName, metadata, transactionManager, this);
|
cache, regionName, metadata, this);
|
||||||
startRegion(region, regionName);
|
startRegion(region, regionName);
|
||||||
return region;
|
return region;
|
||||||
}
|
}
|
||||||
|
@ -207,8 +213,8 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
public EntityRegion buildEntityRegion(String regionName, Properties properties, CacheDataDescription metadata) throws CacheException {
|
public EntityRegion buildEntityRegion(String regionName, Properties properties, CacheDataDescription metadata) throws CacheException {
|
||||||
if (log.isDebugEnabled()) log.debug("Building entity cache region [" + regionName + "]");
|
if (log.isDebugEnabled()) log.debug("Building entity cache region [" + regionName + "]");
|
||||||
AdvancedCache cache = getCache(regionName, ENTITY_KEY, properties);
|
AdvancedCache cache = getCache(regionName, ENTITY_KEY, properties);
|
||||||
CacheAdapter cacheAdapter = CacheAdapterImpl.newInstance(cache);
|
EntityRegionImpl region = new EntityRegionImpl(
|
||||||
EntityRegionImpl region = new EntityRegionImpl(cacheAdapter, regionName, metadata, transactionManager, this);
|
cache, regionName, metadata, this);
|
||||||
startRegion(region, regionName);
|
startRegion(region, regionName);
|
||||||
return region;
|
return region;
|
||||||
}
|
}
|
||||||
|
@ -216,19 +222,13 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
@Override
|
@Override
|
||||||
public NaturalIdRegion buildNaturalIdRegion(String regionName, Properties properties, CacheDataDescription metadata)
|
public NaturalIdRegion buildNaturalIdRegion(String regionName, Properties properties, CacheDataDescription metadata)
|
||||||
throws CacheException {
|
throws CacheException {
|
||||||
if ( log.isDebugEnabled() ) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug( "Building natural id cache region [" + regionName + "]" );
|
log.debug("Building natural id cache region [" + regionName + "]");
|
||||||
}
|
}
|
||||||
AdvancedCache cache = getCache( regionName, NATURAL_ID_KEY, properties );
|
AdvancedCache cache = getCache(regionName, NATURAL_ID_KEY, properties);
|
||||||
CacheAdapter cacheAdapter = CacheAdapterImpl.newInstance( cache );
|
|
||||||
NaturalIdRegionImpl region = new NaturalIdRegionImpl(
|
NaturalIdRegionImpl region = new NaturalIdRegionImpl(
|
||||||
cacheAdapter,
|
cache, regionName, metadata, this);
|
||||||
regionName,
|
startRegion(region, regionName);
|
||||||
metadata,
|
|
||||||
transactionManager,
|
|
||||||
this
|
|
||||||
);
|
|
||||||
startRegion( region, regionName );
|
|
||||||
return region;
|
return region;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -244,8 +244,8 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
cacheName = regionName;
|
cacheName = regionName;
|
||||||
|
|
||||||
AdvancedCache cache = getCache(cacheName, QUERY_KEY, properties);
|
AdvancedCache cache = getCache(cacheName, QUERY_KEY, properties);
|
||||||
CacheAdapter cacheAdapter = CacheAdapterImpl.newInstance(cache);
|
QueryResultsRegionImpl region = new QueryResultsRegionImpl(
|
||||||
QueryResultsRegionImpl region = new QueryResultsRegionImpl(cacheAdapter, regionName, properties, transactionManager, this);
|
cache, regionName, this);
|
||||||
startRegion(region, regionName);
|
startRegion(region, regionName);
|
||||||
return region;
|
return region;
|
||||||
}
|
}
|
||||||
|
@ -257,14 +257,17 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
throws CacheException {
|
throws CacheException {
|
||||||
if (log.isDebugEnabled()) log.debug("Building timestamps cache region [" + regionName + "]");
|
if (log.isDebugEnabled()) log.debug("Building timestamps cache region [" + regionName + "]");
|
||||||
AdvancedCache cache = getCache(regionName, TIMESTAMPS_KEY, properties);
|
AdvancedCache cache = getCache(regionName, TIMESTAMPS_KEY, properties);
|
||||||
CacheAdapter cacheAdapter = CacheAdapterImpl.newInstance(cache);
|
TimestampsRegionImpl region = createTimestampsRegion(cache, regionName);
|
||||||
TimestampsRegionImpl region = createTimestampsRegion(cacheAdapter, regionName);
|
|
||||||
startRegion(region, regionName);
|
startRegion(region, regionName);
|
||||||
return region;
|
return region;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected TimestampsRegionImpl createTimestampsRegion(CacheAdapter cacheAdapter, String regionName) {
|
protected TimestampsRegionImpl createTimestampsRegion(
|
||||||
return new TimestampsRegionImpl(cacheAdapter, regionName, transactionManager, this);
|
AdvancedCache cache, String regionName) {
|
||||||
|
if (Caches.isClustered(cache))
|
||||||
|
return new ClusteredTimestampsRegionImpl(cache, regionName, this);
|
||||||
|
else
|
||||||
|
return new TimestampsRegionImpl(cache, regionName, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -301,7 +304,6 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
log.debug("Starting Infinispan region factory");
|
log.debug("Starting Infinispan region factory");
|
||||||
try {
|
try {
|
||||||
transactionManagerlookup = createTransactionManagerLookup(settings, properties);
|
transactionManagerlookup = createTransactionManagerLookup(settings, properties);
|
||||||
transactionManager = transactionManagerlookup.getTransactionManager();
|
|
||||||
manager = createCacheManager(properties);
|
manager = createCacheManager(properties);
|
||||||
initGenericDataTypeOverrides();
|
initGenericDataTypeOverrides();
|
||||||
Enumeration keys = properties.propertyNames();
|
Enumeration keys = properties.propertyNames();
|
||||||
|
@ -313,6 +315,7 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
defineGenericDataTypeCacheConfigurations(settings, properties);
|
defineGenericDataTypeCacheConfigurations(settings, properties);
|
||||||
|
definePendingPutsCache();
|
||||||
} catch (CacheException ce) {
|
} catch (CacheException ce) {
|
||||||
throw ce;
|
throw ce;
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
|
@ -320,6 +323,22 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void definePendingPutsCache() {
|
||||||
|
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(
|
||||||
Settings settings, Properties properties) {
|
Settings settings, Properties properties) {
|
||||||
return new HibernateTransactionManagerLookup(settings, properties);
|
return new HibernateTransactionManagerLookup(settings, properties);
|
||||||
|
@ -336,7 +355,8 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
|
|
||||||
protected void stopCacheRegions() {
|
protected void stopCacheRegions() {
|
||||||
log.debug("Clear region references");
|
log.debug("Clear region references");
|
||||||
getCacheCommandFactory(manager.getCache()).clearRegions(regionNames);
|
getCacheCommandFactory(manager.getCache().getAdvancedCache())
|
||||||
|
.clearRegions(regionNames);
|
||||||
regionNames.clear();
|
regionNames.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -376,8 +396,7 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
|
|
||||||
private void startRegion(BaseRegion region, String regionName) {
|
private void startRegion(BaseRegion region, String regionName) {
|
||||||
regionNames.add(regionName);
|
regionNames.add(regionName);
|
||||||
getCacheCommandFactory(region.getCacheAdapter().getCache())
|
getCacheCommandFactory(region.getCache()).addRegion(regionName, region);
|
||||||
.addRegion(regionName, region);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Map<String, TypeOverrides> initGenericDataTypeOverrides() {
|
private Map<String, TypeOverrides> initGenericDataTypeOverrides() {
|
||||||
|
@ -487,11 +506,14 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
return createCacheWrapper(cache);
|
return createCacheWrapper(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
private CacheCommandFactory getCacheCommandFactory(Cache cache) {
|
private CacheCommandFactory getCacheCommandFactory(AdvancedCache cache) {
|
||||||
GlobalComponentRegistry globalCr = cache.getAdvancedCache()
|
GlobalComponentRegistry globalCr = cache.getComponentRegistry()
|
||||||
.getComponentRegistry().getGlobalComponentRegistry();
|
.getGlobalComponentRegistry();
|
||||||
|
|
||||||
Map<Byte, ModuleCommandFactory> factories =
|
Map<Byte, ModuleCommandFactory> factories =
|
||||||
(Map<Byte, ModuleCommandFactory>) globalCr.getComponent("org.infinispan.modules.command.factories");
|
(Map<Byte, ModuleCommandFactory>) globalCr
|
||||||
|
.getComponent("org.infinispan.modules.command.factories");
|
||||||
|
|
||||||
for (ModuleCommandFactory factory : factories.values()) {
|
for (ModuleCommandFactory factory : factories.values()) {
|
||||||
if (factory instanceof CacheCommandFactory)
|
if (factory instanceof CacheCommandFactory)
|
||||||
return (CacheCommandFactory) factory;
|
return (CacheCommandFactory) factory;
|
||||||
|
@ -503,7 +525,11 @@ public class InfinispanRegionFactory implements RegionFactory {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected AdvancedCache createCacheWrapper(AdvancedCache cache) {
|
protected AdvancedCache createCacheWrapper(AdvancedCache cache) {
|
||||||
return new ClassLoaderAwareCache(cache, Thread.currentThread().getContextClassLoader());
|
if (Caches.isClustered(cache))
|
||||||
|
return new ClassLoaderAwareCache(cache,
|
||||||
|
Thread.currentThread().getContextClassLoader());
|
||||||
|
|
||||||
|
return cache;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Configuration configureTransactionManager(Configuration regionOverrides, String templateCacheName, Properties properties) {
|
private Configuration configureTransactionManager(Configuration regionOverrides, String templateCacheName, Properties properties) {
|
||||||
|
|
|
@ -23,7 +23,6 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.cache.infinispan.access;
|
package org.hibernate.cache.infinispan.access;
|
||||||
|
|
||||||
import java.lang.ref.WeakReference;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
@ -38,6 +37,9 @@ import javax.transaction.Transaction;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.manager.EmbeddedCacheManager;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Encapsulates logic to allow a {@link TransactionalAccessDelegate} to determine
|
* Encapsulates logic to allow a {@link TransactionalAccessDelegate} to determine
|
||||||
|
@ -91,42 +93,19 @@ public class PutFromLoadValidator {
|
||||||
*/
|
*/
|
||||||
public static final long NAKED_PUT_INVALIDATION_PERIOD = TimeUnit.SECONDS.toMillis(20);
|
public static final long NAKED_PUT_INVALIDATION_PERIOD = TimeUnit.SECONDS.toMillis(20);
|
||||||
|
|
||||||
/** Period (in ms) after which a pending put is placed in the over-age queue */
|
|
||||||
private static final long PENDING_PUT_OVERAGE_PERIOD = TimeUnit.SECONDS.toMillis(5);
|
|
||||||
|
|
||||||
/** Period (in ms) before which we stop trying to clean out pending puts */
|
|
||||||
private static final long PENDING_PUT_RECENT_PERIOD = TimeUnit.SECONDS.toMillis(2);
|
|
||||||
|
|
||||||
/** Period (in ms) after which a pending put is never expected to come in and should be cleaned */
|
|
||||||
private static final long MAX_PENDING_PUT_DELAY = TimeUnit.SECONDS.toMillis(2 * 60);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 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;
|
private final long nakedPutInvalidationPeriod;
|
||||||
private final long pendingPutOveragePeriod;
|
|
||||||
private final long pendingPutRecentPeriod;
|
|
||||||
private final long maxPendingPutDelay;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 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
|
||||||
* is not a "naked put" and is allowed to proceed.
|
* is not a "naked put" and is allowed to proceed.
|
||||||
*/
|
*/
|
||||||
private final ConcurrentMap<Object, PendingPutMap> pendingPuts = new ConcurrentHashMap<Object, PendingPutMap>();
|
private final ConcurrentMap<Object, PendingPutMap> pendingPuts;
|
||||||
/**
|
|
||||||
* List of pending puts. Used to ensure we don't leak memory via the pendingPuts map
|
|
||||||
*/
|
|
||||||
private final List<WeakReference<PendingPut>> pendingQueue = new LinkedList<WeakReference<PendingPut>>();
|
|
||||||
/**
|
|
||||||
* Separate list of pending puts that haven't been resolved within PENDING_PUT_OVERAGE_PERIOD.
|
|
||||||
* Used to ensure we don't leak memory via the pendingPuts map. Tracked separately from more
|
|
||||||
* recent pending puts for efficiency reasons.
|
|
||||||
*/
|
|
||||||
private final List<WeakReference<PendingPut>> overagePendingQueue = new LinkedList<WeakReference<PendingPut>>();
|
|
||||||
/** Lock controlling access to pending put queues */
|
|
||||||
private final Lock pendingLock = new ReentrantLock();
|
|
||||||
private final ConcurrentMap<Object, Long> recentRemovals = new ConcurrentHashMap<Object, Long>();
|
private final ConcurrentMap<Object, Long> recentRemovals = new ConcurrentHashMap<Object, Long>();
|
||||||
/**
|
/**
|
||||||
* List of recent removals. Used to ensure we don't leak memory via the recentRemovals map
|
* List of recent removals. Used to ensure we don't leak memory via the recentRemovals map
|
||||||
|
@ -148,27 +127,26 @@ public class PutFromLoadValidator {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new PutFromLoadValidator.
|
* Creates a new PutFromLoadValidator.
|
||||||
*
|
|
||||||
* @param transactionManager
|
|
||||||
* transaction manager to use to associate changes with a transaction; may be
|
|
||||||
* <code>null</code>
|
|
||||||
*/
|
*/
|
||||||
public PutFromLoadValidator(TransactionManager transactionManager) {
|
public PutFromLoadValidator(AdvancedCache cache) {
|
||||||
this(transactionManager, NAKED_PUT_INVALIDATION_PERIOD, PENDING_PUT_OVERAGE_PERIOD,
|
this(cache, NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
PENDING_PUT_RECENT_PERIOD, MAX_PENDING_PUT_DELAY);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor variant for use by unit tests; allows control of various timeouts by the test.
|
* Constructor variant for use by unit tests; allows control of various timeouts by the test.
|
||||||
*/
|
*/
|
||||||
protected PutFromLoadValidator(TransactionManager transactionManager,
|
public PutFromLoadValidator(AdvancedCache cache,
|
||||||
long nakedPutInvalidationPeriod, long pendingPutOveragePeriod,
|
long nakedPutInvalidationPeriod) {
|
||||||
long pendingPutRecentPeriod, long maxPendingPutDelay) {
|
this(cache.getCacheManager(), cache.getTransactionManager(),
|
||||||
this.transactionManager = transactionManager;
|
nakedPutInvalidationPeriod);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PutFromLoadValidator(EmbeddedCacheManager cacheManager,
|
||||||
|
TransactionManager tm, long nakedPutInvalidationPeriod) {
|
||||||
|
this.pendingPuts = cacheManager
|
||||||
|
.getCache(InfinispanRegionFactory.PENDING_PUTS_CACHE_NAME);
|
||||||
|
this.transactionManager = tm;
|
||||||
this.nakedPutInvalidationPeriod = nakedPutInvalidationPeriod;
|
this.nakedPutInvalidationPeriod = nakedPutInvalidationPeriod;
|
||||||
this.pendingPutOveragePeriod = pendingPutOveragePeriod;
|
|
||||||
this.pendingPutRecentPeriod = pendingPutRecentPeriod;
|
|
||||||
this.maxPendingPutDelay = maxPendingPutDelay;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ----------------------------------------------------------------- Public
|
// ----------------------------------------------------------------- Public
|
||||||
|
@ -191,10 +169,6 @@ public class PutFromLoadValidator {
|
||||||
boolean locked = false;
|
boolean locked = false;
|
||||||
long now = System.currentTimeMillis();
|
long now = System.currentTimeMillis();
|
||||||
|
|
||||||
// Important: Do cleanup before we acquire any locks so we
|
|
||||||
// don't deadlock with invalidateRegion
|
|
||||||
cleanOutdatedPendingPuts(now, true);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
PendingPutMap pending = pendingPuts.get(key);
|
PendingPutMap pending = pendingPuts.get(key);
|
||||||
if (pending != null) {
|
if (pending != null) {
|
||||||
|
@ -233,9 +207,6 @@ public class PutFromLoadValidator {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Throwable t) {
|
catch (Throwable t) {
|
||||||
|
|
||||||
valid = false;
|
|
||||||
|
|
||||||
if (locked) {
|
if (locked) {
|
||||||
PendingPutMap toRelease = pendingPuts.get(key);
|
PendingPutMap toRelease = pendingPuts.get(key);
|
||||||
if (toRelease != null) {
|
if (toRelease != null) {
|
||||||
|
@ -283,7 +254,6 @@ public class PutFromLoadValidator {
|
||||||
* caller should treat as an exception condition)
|
* caller should treat as an exception condition)
|
||||||
*/
|
*/
|
||||||
public boolean invalidateKey(Object key) {
|
public boolean invalidateKey(Object key) {
|
||||||
|
|
||||||
boolean success = true;
|
boolean success = true;
|
||||||
|
|
||||||
// Invalidate any pending puts
|
// Invalidate any pending puts
|
||||||
|
@ -330,7 +300,7 @@ public class PutFromLoadValidator {
|
||||||
Long cleaned = recentRemovals.get(toClean.key);
|
Long cleaned = recentRemovals.get(toClean.key);
|
||||||
if (cleaned != null && cleaned.equals(toClean.timestamp)) {
|
if (cleaned != null && cleaned.equals(toClean.timestamp)) {
|
||||||
cleaned = recentRemovals.remove(toClean.key);
|
cleaned = recentRemovals.remove(toClean.key);
|
||||||
if (cleaned != null && cleaned.equals(toClean.timestamp) == false) {
|
if (cleaned != null && !cleaned.equals(toClean.timestamp)) {
|
||||||
// Oops; removed the wrong timestamp; restore it
|
// Oops; removed the wrong timestamp; restore it
|
||||||
recentRemovals.putIfAbsent(toClean.key, cleaned);
|
recentRemovals.putIfAbsent(toClean.key, cleaned);
|
||||||
}
|
}
|
||||||
|
@ -405,13 +375,14 @@ public class PutFromLoadValidator {
|
||||||
* @param key key that will be used for subsequent cache put
|
* @param key key that will be used for subsequent cache put
|
||||||
*/
|
*/
|
||||||
public void registerPendingPut(Object key) {
|
public void registerPendingPut(Object key) {
|
||||||
PendingPut pendingPut = new PendingPut(key, getOwnerForPut());
|
PendingPut pendingPut = new PendingPut(getOwnerForPut());
|
||||||
PendingPutMap pendingForKey = new PendingPutMap(pendingPut);
|
PendingPutMap pendingForKey = new PendingPutMap(pendingPut);
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
PendingPutMap existing = pendingPuts.putIfAbsent(key, pendingForKey);
|
PendingPutMap existing = pendingPuts.putIfAbsent(key, pendingForKey);
|
||||||
if (existing != null) {
|
if (existing != null) {
|
||||||
if (existing.acquireLock(10, TimeUnit.SECONDS)) {
|
if (existing.acquireLock(10, TimeUnit.SECONDS)) {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
existing.put(pendingPut);
|
existing.put(pendingPut);
|
||||||
PendingPutMap doublecheck = pendingPuts.putIfAbsent(key, existing);
|
PendingPutMap doublecheck = pendingPuts.putIfAbsent(key, existing);
|
||||||
|
@ -432,33 +403,10 @@ public class PutFromLoadValidator {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Guard against memory leaks
|
|
||||||
preventOutdatedPendingPuts(pendingPut);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// -------------------------------------------------------------- Protected
|
// -------------------------------------------------------------- Protected
|
||||||
|
|
||||||
/** Only for use by unit tests; may be removed at any time */
|
|
||||||
protected int getPendingPutQueueLength() {
|
|
||||||
pendingLock.lock();
|
|
||||||
try {
|
|
||||||
return pendingQueue.size();
|
|
||||||
} finally {
|
|
||||||
pendingLock.unlock();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Only for use by unit tests; may be removed at any time */
|
|
||||||
protected int getOveragePendingPutQueueLength() {
|
|
||||||
pendingLock.lock();
|
|
||||||
try {
|
|
||||||
return overagePendingQueue.size();
|
|
||||||
} finally {
|
|
||||||
pendingLock.unlock();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Only for use by unit tests; may be removed at any time */
|
/** Only for use by unit tests; may be removed at any time */
|
||||||
protected int getRemovalQueueLength() {
|
protected int getRemovalQueueLength() {
|
||||||
removalsLock.lock();
|
removalsLock.lock();
|
||||||
|
@ -484,119 +432,6 @@ public class PutFromLoadValidator {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void preventOutdatedPendingPuts(PendingPut pendingPut) {
|
|
||||||
pendingLock.lock();
|
|
||||||
try {
|
|
||||||
pendingQueue.add(new WeakReference<PendingPut>(pendingPut));
|
|
||||||
if (pendingQueue.size() > 1) {
|
|
||||||
cleanOutdatedPendingPuts(pendingPut.timestamp, false);
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
pendingLock.unlock();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void cleanOutdatedPendingPuts(long now, boolean lock) {
|
|
||||||
|
|
||||||
PendingPut toClean = null;
|
|
||||||
if (lock) {
|
|
||||||
pendingLock.lock();
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
// Clean items out of the basic queue
|
|
||||||
long overaged = now - this.pendingPutOveragePeriod;
|
|
||||||
long recent = now - this.pendingPutRecentPeriod;
|
|
||||||
|
|
||||||
int pos = 0;
|
|
||||||
while (pendingQueue.size() > pos) {
|
|
||||||
WeakReference<PendingPut> ref = pendingQueue.get(pos);
|
|
||||||
PendingPut item = ref.get();
|
|
||||||
if (item == null || item.completed) {
|
|
||||||
pendingQueue.remove(pos);
|
|
||||||
} else if (item.timestamp < overaged) {
|
|
||||||
// Potential leak; move to the overaged queued
|
|
||||||
pendingQueue.remove(pos);
|
|
||||||
overagePendingQueue.add(ref);
|
|
||||||
} else if (item.timestamp >= recent) {
|
|
||||||
// Don't waste time on very recent items
|
|
||||||
break;
|
|
||||||
} else if (pos > 2) {
|
|
||||||
// Don't spend too much time getting nowhere
|
|
||||||
break;
|
|
||||||
} else {
|
|
||||||
// Move on to the next item
|
|
||||||
pos++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Process the overage queue until we find an item to clean
|
|
||||||
// or an incomplete item that hasn't aged out
|
|
||||||
long mustCleanTime = now - this.maxPendingPutDelay;
|
|
||||||
|
|
||||||
while (overagePendingQueue.size() > 0) {
|
|
||||||
WeakReference<PendingPut> ref = overagePendingQueue.get(0);
|
|
||||||
PendingPut item = ref.get();
|
|
||||||
if (item == null || item.completed) {
|
|
||||||
overagePendingQueue.remove(0);
|
|
||||||
} else {
|
|
||||||
if (item.timestamp < mustCleanTime) {
|
|
||||||
overagePendingQueue.remove(0);
|
|
||||||
toClean = item;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (lock) {
|
|
||||||
pendingLock.unlock();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// We've found a pendingPut that never happened; clean it up
|
|
||||||
if (toClean != null) {
|
|
||||||
PendingPutMap map = pendingPuts.get(toClean.key);
|
|
||||||
if (map != null) {
|
|
||||||
if (map.acquireLock(100, TimeUnit.MILLISECONDS)) {
|
|
||||||
try {
|
|
||||||
PendingPut cleaned = map.remove(toClean.owner);
|
|
||||||
if (toClean.equals(cleaned) == false) {
|
|
||||||
if (cleaned != null) {
|
|
||||||
// Oops. Restore it.
|
|
||||||
map.put(cleaned);
|
|
||||||
}
|
|
||||||
} else if (map.size() == 0) {
|
|
||||||
pendingPuts.remove(toClean.key, map);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally {
|
|
||||||
map.releaseLock();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Something's gone wrong and the lock isn't being released.
|
|
||||||
// We removed toClean from the queue and need to restore it
|
|
||||||
// TODO this is pretty dodgy
|
|
||||||
restorePendingPut(toClean);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private void restorePendingPut(PendingPut toRestore) {
|
|
||||||
pendingLock.lock();
|
|
||||||
try {
|
|
||||||
// Give it a new lease on life so it's not out of order. We could
|
|
||||||
// scan the queue and put toRestore back at the front, but then
|
|
||||||
// we'll just immediately try removing it again; instead we
|
|
||||||
// let it cycle through the queue again
|
|
||||||
toRestore.refresh();
|
|
||||||
pendingQueue.add(new WeakReference<PendingPut>(toRestore));
|
|
||||||
}
|
|
||||||
finally {
|
|
||||||
pendingLock.unlock();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lazy-initialization map for PendingPut. Optimized for the expected usual case where only a
|
* Lazy-initialization map for PendingPut. Optimized for the expected usual case where only a
|
||||||
* single put is pending for a given key.
|
* single put is pending for a given key.
|
||||||
|
@ -677,19 +512,12 @@ public class PutFromLoadValidator {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class PendingPut {
|
private static class PendingPut {
|
||||||
private final Object key;
|
|
||||||
private final Object owner;
|
private final Object owner;
|
||||||
private long timestamp = System.currentTimeMillis();
|
|
||||||
private volatile boolean completed;
|
private volatile boolean completed;
|
||||||
|
|
||||||
private PendingPut(Object key, Object owner) {
|
private PendingPut(Object owner) {
|
||||||
this.key = key;
|
|
||||||
this.owner = owner;
|
this.owner = owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void refresh() {
|
|
||||||
timestamp = System.currentTimeMillis();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class RecentRemoval {
|
private static class RecentRemoval {
|
||||||
|
|
|
@ -25,13 +25,14 @@ package org.hibernate.cache.infinispan.access;
|
||||||
|
|
||||||
import javax.transaction.Transaction;
|
import javax.transaction.Transaction;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
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.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseRegion;
|
import org.hibernate.cache.infinispan.impl.BaseRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
|
||||||
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
||||||
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
||||||
import org.hibernate.cache.spi.access.SoftLock;
|
import org.hibernate.cache.spi.access.SoftLock;
|
||||||
|
@ -49,20 +50,26 @@ import org.hibernate.cache.spi.access.SoftLock;
|
||||||
public class TransactionalAccessDelegate {
|
public class TransactionalAccessDelegate {
|
||||||
private static final Log log = LogFactory.getLog(TransactionalAccessDelegate.class);
|
private static final Log log = LogFactory.getLog(TransactionalAccessDelegate.class);
|
||||||
private static final boolean isTrace = log.isTraceEnabled();
|
private static final boolean isTrace = log.isTraceEnabled();
|
||||||
protected final CacheAdapter cacheAdapter;
|
private final AdvancedCache cache;
|
||||||
protected final BaseRegion region;
|
private final BaseRegion region;
|
||||||
protected final PutFromLoadValidator putValidator;
|
private final PutFromLoadValidator putValidator;
|
||||||
|
private final AdvancedCache writeCache;
|
||||||
|
private final AdvancedCache putFromLoadCache;
|
||||||
|
|
||||||
public TransactionalAccessDelegate(BaseRegion region, PutFromLoadValidator validator) {
|
public TransactionalAccessDelegate(BaseRegion region, PutFromLoadValidator validator) {
|
||||||
this.region = region;
|
this.region = region;
|
||||||
this.cacheAdapter = region.getCacheAdapter();
|
this.cache = region.getCache();
|
||||||
this.putValidator = validator;
|
this.putValidator = validator;
|
||||||
|
this.writeCache = Caches.isInvalidationCache(cache) ?
|
||||||
|
Caches.ignoreReturnValuesCache(cache, Flag.CACHE_MODE_LOCAL) :
|
||||||
|
Caches.ignoreReturnValuesCache(cache);
|
||||||
|
this.putFromLoadCache = Caches.ignoreReturnValuesCache(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Object get(Object key, long txTimestamp) throws CacheException {
|
public Object get(Object key, long txTimestamp) throws CacheException {
|
||||||
if (!region.checkValid())
|
if (!region.checkValid())
|
||||||
return null;
|
return null;
|
||||||
Object val = cacheAdapter.get(key);
|
Object val = cache.get(key);
|
||||||
if (val == null)
|
if (val == null)
|
||||||
putValidator.registerPendingPut(key);
|
putValidator.registerPendingPut(key);
|
||||||
return val;
|
return val;
|
||||||
|
@ -84,7 +91,7 @@ public class TransactionalAccessDelegate {
|
||||||
// without https://issues.jboss.org/browse/ISPN-1986, it's impossible to
|
// without https://issues.jboss.org/browse/ISPN-1986, it's impossible to
|
||||||
// know whether the put actually occurred. Knowing this is crucial so
|
// know whether the put actually occurred. Knowing this is crucial so
|
||||||
// that Hibernate can expose accurate statistics.
|
// that Hibernate can expose accurate statistics.
|
||||||
if (minimalPutOverride && cacheAdapter.containsKey(key))
|
if (minimalPutOverride && cache.containsKey(key))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!putValidator.acquirePutFromLoadLock(key)) {
|
if (!putValidator.acquirePutFromLoadLock(key)) {
|
||||||
|
@ -93,7 +100,7 @@ public class TransactionalAccessDelegate {
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
cacheAdapter.putForExternalRead(key, value);
|
putFromLoadCache.putForExternalRead(key, value);
|
||||||
} finally {
|
} finally {
|
||||||
putValidator.releasePutFromLoadLock(key);
|
putValidator.releasePutFromLoadLock(key);
|
||||||
}
|
}
|
||||||
|
@ -119,11 +126,7 @@ public class TransactionalAccessDelegate {
|
||||||
if (!region.checkValid())
|
if (!region.checkValid())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (cacheAdapter.isClusteredInvalidation())
|
writeCache.put(key, value);
|
||||||
cacheAdapter.withFlags(FlagAdapter.CACHE_MODE_LOCAL).put(key, value);
|
|
||||||
else
|
|
||||||
cacheAdapter.put(key, value);
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,7 +138,7 @@ public class TransactionalAccessDelegate {
|
||||||
// We update whether or not the region is valid. Other nodes
|
// We update whether or not the region is valid. Other nodes
|
||||||
// may have already restored the region so they need to
|
// may have already restored the region so they need to
|
||||||
// be informed of the change.
|
// be informed of the change.
|
||||||
cacheAdapter.put(key, value);
|
writeCache.put(key, value);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -151,21 +154,21 @@ public class TransactionalAccessDelegate {
|
||||||
// We update whether or not the region is valid. Other nodes
|
// We update whether or not the region is valid. Other nodes
|
||||||
// may have already restored the region so they need to
|
// may have already restored the region so they need to
|
||||||
// be informed of the change.
|
// be informed of the change.
|
||||||
cacheAdapter.remove(key);
|
writeCache.remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void removeAll() throws CacheException {
|
public void removeAll() throws CacheException {
|
||||||
if (!putValidator.invalidateRegion()) {
|
if (!putValidator.invalidateRegion()) {
|
||||||
throw new CacheException("Failed to invalidate pending putFromLoad calls for region " + region.getName());
|
throw new CacheException("Failed to invalidate pending putFromLoad calls for region " + region.getName());
|
||||||
}
|
}
|
||||||
cacheAdapter.clear();
|
cache.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evict(Object key) throws CacheException {
|
public void evict(Object key) throws CacheException {
|
||||||
if (!putValidator.invalidateKey(key)) {
|
if (!putValidator.invalidateKey(key)) {
|
||||||
throw new CacheException("Failed to invalidate pending putFromLoad calls for key " + key + " from region " + region.getName());
|
throw new CacheException("Failed to invalidate pending putFromLoad calls for key " + key + " from region " + region.getName());
|
||||||
}
|
}
|
||||||
cacheAdapter.remove(key);
|
writeCache.remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evictAll() throws CacheException {
|
public void evictAll() throws CacheException {
|
||||||
|
@ -175,9 +178,10 @@ public class TransactionalAccessDelegate {
|
||||||
Transaction tx = region.suspend();
|
Transaction tx = region.suspend();
|
||||||
try {
|
try {
|
||||||
region.invalidateRegion(); // Invalidate the local region and then go remote
|
region.invalidateRegion(); // Invalidate the local region and then go remote
|
||||||
cacheAdapter.broadcastEvictAll();
|
Caches.broadcastEvictAll(cache);
|
||||||
} finally {
|
} finally {
|
||||||
region.resume(tx);
|
region.resume(tx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,16 +1,14 @@
|
||||||
package org.hibernate.cache.infinispan.collection;
|
package org.hibernate.cache.infinispan.collection;
|
||||||
|
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.CollectionRegion;
|
import org.hibernate.cache.spi.CollectionRegion;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Chris Bredesen
|
* @author Chris Bredesen
|
||||||
|
@ -19,9 +17,9 @@ import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
||||||
*/
|
*/
|
||||||
public class CollectionRegionImpl extends BaseTransactionalDataRegion implements CollectionRegion {
|
public class CollectionRegionImpl extends BaseTransactionalDataRegion implements CollectionRegion {
|
||||||
|
|
||||||
public CollectionRegionImpl(CacheAdapter cacheAdapter, String name, CacheDataDescription metadata,
|
public CollectionRegionImpl(AdvancedCache cache, String name,
|
||||||
TransactionManager transactionManager, RegionFactory factory) {
|
CacheDataDescription metadata, RegionFactory factory) {
|
||||||
super(cacheAdapter, name, metadata, transactionManager, factory);
|
super(cache, name, metadata, factory);
|
||||||
}
|
}
|
||||||
|
|
||||||
public CollectionRegionAccessStrategy buildAccessStrategy(AccessType accessType) throws CacheException {
|
public CollectionRegionAccessStrategy buildAccessStrategy(AccessType accessType) throws CacheException {
|
||||||
|
@ -33,6 +31,7 @@ public class CollectionRegionImpl extends BaseTransactionalDataRegion implements
|
||||||
}
|
}
|
||||||
|
|
||||||
public PutFromLoadValidator getPutFromLoadValidator() {
|
public PutFromLoadValidator getPutFromLoadValidator() {
|
||||||
return new PutFromLoadValidator(transactionManager);
|
return new PutFromLoadValidator(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,16 +1,14 @@
|
||||||
package org.hibernate.cache.infinispan.entity;
|
package org.hibernate.cache.infinispan.entity;
|
||||||
|
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.EntityRegion;
|
import org.hibernate.cache.spi.EntityRegion;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Chris Bredesen
|
* @author Chris Bredesen
|
||||||
|
@ -19,9 +17,9 @@ import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
||||||
*/
|
*/
|
||||||
public class EntityRegionImpl extends BaseTransactionalDataRegion implements EntityRegion {
|
public class EntityRegionImpl extends BaseTransactionalDataRegion implements EntityRegion {
|
||||||
|
|
||||||
public EntityRegionImpl(CacheAdapter cacheAdapter, String name, CacheDataDescription metadata,
|
public EntityRegionImpl(AdvancedCache cache, String name,
|
||||||
TransactionManager transactionManager, RegionFactory factory) {
|
CacheDataDescription metadata, RegionFactory factory) {
|
||||||
super(cacheAdapter, name, metadata, transactionManager, factory);
|
super(cache, name, metadata, factory);
|
||||||
}
|
}
|
||||||
|
|
||||||
public EntityRegionAccessStrategy buildAccessStrategy(AccessType accessType) throws CacheException {
|
public EntityRegionAccessStrategy buildAccessStrategy(AccessType accessType) throws CacheException {
|
||||||
|
@ -34,6 +32,7 @@ public class EntityRegionImpl extends BaseTransactionalDataRegion implements Ent
|
||||||
}
|
}
|
||||||
|
|
||||||
public PutFromLoadValidator getPutFromLoadValidator() {
|
public PutFromLoadValidator getPutFromLoadValidator() {
|
||||||
return new PutFromLoadValidator(transactionManager);
|
return new PutFromLoadValidator(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -3,9 +3,10 @@ package org.hibernate.cache.infinispan.impl;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
import org.hibernate.cache.spi.GeneralDataRegion;
|
import org.hibernate.cache.spi.GeneralDataRegion;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Support for Infinispan {@link GeneralDataRegion} implementors.
|
* Support for Infinispan {@link GeneralDataRegion} implementors.
|
||||||
|
@ -16,24 +17,28 @@ import org.hibernate.cache.spi.RegionFactory;
|
||||||
*/
|
*/
|
||||||
public abstract class BaseGeneralDataRegion extends BaseRegion implements GeneralDataRegion {
|
public abstract class BaseGeneralDataRegion extends BaseRegion implements GeneralDataRegion {
|
||||||
|
|
||||||
public BaseGeneralDataRegion(CacheAdapter cacheAdapter, String name, TransactionManager transactionManager, RegionFactory factory) {
|
private final AdvancedCache putCache;
|
||||||
super(cacheAdapter, name, transactionManager, factory);
|
|
||||||
|
public BaseGeneralDataRegion(AdvancedCache cache, String name,
|
||||||
|
RegionFactory factory) {
|
||||||
|
super(cache, name, factory);
|
||||||
|
this.putCache = Caches.ignoreReturnValuesCache(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evict(Object key) throws CacheException {
|
public void evict(Object key) throws CacheException {
|
||||||
cacheAdapter.evict(key);
|
cache.evict(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evictAll() throws CacheException {
|
public void evictAll() throws CacheException {
|
||||||
cacheAdapter.clear();
|
cache.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Object get(Object key) throws CacheException {
|
public Object get(Object key) throws CacheException {
|
||||||
return cacheAdapter.get(key);
|
return cache.get(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void put(Object key, Object value) throws CacheException {
|
public void put(Object key, Object value) throws CacheException {
|
||||||
cacheAdapter.put(key, value);
|
putCache.put(key, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -8,13 +8,13 @@ import javax.transaction.SystemException;
|
||||||
import javax.transaction.Transaction;
|
import javax.transaction.Transaction;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
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.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.util.AddressAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
|
|
||||||
|
@ -29,37 +29,38 @@ import org.hibernate.cache.spi.RegionFactory;
|
||||||
*/
|
*/
|
||||||
public abstract class BaseRegion implements Region {
|
public abstract class BaseRegion implements Region {
|
||||||
|
|
||||||
private enum InvalidateState { INVALID, CLEARING, VALID };
|
|
||||||
private static final Log log = LogFactory.getLog(BaseRegion.class);
|
private static final Log log = LogFactory.getLog(BaseRegion.class);
|
||||||
|
|
||||||
|
private enum InvalidateState {
|
||||||
|
INVALID, CLEARING, VALID
|
||||||
|
}
|
||||||
|
|
||||||
private final String name;
|
private final String name;
|
||||||
protected final CacheAdapter cacheAdapter;
|
private final AdvancedCache regionClearCache;
|
||||||
protected final AddressAdapter address;
|
private final TransactionManager tm;
|
||||||
protected final TransactionManager transactionManager;
|
private final Object invalidationMutex = new Object();
|
||||||
protected final boolean replication;
|
private final AtomicReference<InvalidateState> invalidateState =
|
||||||
protected final Object invalidationMutex = new Object();
|
new AtomicReference<InvalidateState>(InvalidateState.VALID);
|
||||||
protected final AtomicReference<InvalidateState> invalidateState = new AtomicReference<InvalidateState>(InvalidateState.VALID);
|
|
||||||
private final RegionFactory factory;
|
private final RegionFactory factory;
|
||||||
|
|
||||||
public BaseRegion(CacheAdapter cacheAdapter, String name, TransactionManager transactionManager, RegionFactory factory) {
|
protected final AdvancedCache cache;
|
||||||
this.cacheAdapter = cacheAdapter;
|
|
||||||
|
public BaseRegion(AdvancedCache cache, String name, RegionFactory factory) {
|
||||||
|
this.cache = cache;
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.transactionManager = transactionManager;
|
this.tm = cache.getTransactionManager();
|
||||||
this.replication = cacheAdapter.isClusteredReplication();
|
|
||||||
this.address = this.cacheAdapter.getAddress();
|
|
||||||
this.factory = factory;
|
this.factory = factory;
|
||||||
|
this.regionClearCache = cache.withFlags(
|
||||||
|
Flag.CACHE_MODE_LOCAL, Flag.ZERO_LOCK_ACQUISITION_TIMEOUT);
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getName() {
|
public String getName() {
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
|
|
||||||
public CacheAdapter getCacheAdapter() {
|
|
||||||
return cacheAdapter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public long getElementCountInMemory() {
|
public long getElementCountInMemory() {
|
||||||
if (checkValid())
|
if (checkValid())
|
||||||
return cacheAdapter.size();
|
return cache.size();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -92,51 +93,46 @@ public abstract class BaseRegion implements Region {
|
||||||
|
|
||||||
public Map toMap() {
|
public Map toMap() {
|
||||||
if (checkValid())
|
if (checkValid())
|
||||||
return cacheAdapter.toMap();
|
return cache;
|
||||||
|
|
||||||
return Collections.EMPTY_MAP;
|
return Collections.EMPTY_MAP;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void destroy() throws CacheException {
|
public void destroy() throws CacheException {
|
||||||
try {
|
try {
|
||||||
cacheAdapter.stop();
|
cache.stop();
|
||||||
} finally {
|
} finally {
|
||||||
cacheAdapter.removeListener(this);
|
cache.removeListener(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean contains(Object key) {
|
public boolean contains(Object key) {
|
||||||
if (!checkValid())
|
return checkValid() && cache.containsKey(key);
|
||||||
return false;
|
|
||||||
// Reads are non-blocking in Infinispan, so not sure of the necessity of passing ZERO_LOCK_ACQUISITION_TIMEOUT
|
|
||||||
return cacheAdapter.withFlags(FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT).containsKey(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
public AddressAdapter getAddress() {
|
|
||||||
return address;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean checkValid() {
|
public boolean checkValid() {
|
||||||
boolean valid = isValid();
|
boolean valid = isValid();
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
synchronized (invalidationMutex) {
|
synchronized (invalidationMutex) {
|
||||||
if (invalidateState.compareAndSet(InvalidateState.INVALID, InvalidateState.CLEARING)) {
|
if (invalidateState.compareAndSet(
|
||||||
|
InvalidateState.INVALID, InvalidateState.CLEARING)) {
|
||||||
Transaction tx = suspend();
|
Transaction tx = suspend();
|
||||||
try {
|
try {
|
||||||
// Clear region in a separate transaction
|
// Clear region in a separate transaction
|
||||||
cacheAdapter.withinTx(new Callable<Void>() {
|
Caches.withinTx(cache, new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
cacheAdapter.withFlags(FlagAdapter.CACHE_MODE_LOCAL,
|
regionClearCache.clear();
|
||||||
FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT).clear();
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
invalidateState.compareAndSet(InvalidateState.CLEARING, InvalidateState.VALID);
|
invalidateState.compareAndSet(
|
||||||
|
InvalidateState.CLEARING, InvalidateState.VALID);
|
||||||
}
|
}
|
||||||
catch (Exception e) {
|
catch (Exception e) {
|
||||||
if (log.isTraceEnabled()) {
|
if (log.isTraceEnabled()) {
|
||||||
log.trace("Could not invalidate region: " + e.getLocalizedMessage());
|
log.trace("Could not invalidate region: "
|
||||||
|
+ e.getLocalizedMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
finally {
|
finally {
|
||||||
|
@ -150,44 +146,10 @@ public abstract class BaseRegion implements Region {
|
||||||
return valid;
|
return valid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
protected boolean isValid() {
|
protected boolean isValid() {
|
||||||
return invalidateState.get() == InvalidateState.VALID;
|
return invalidateState.get() == InvalidateState.VALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs a Infinispan <code>get(Fqn, Object)</code>
|
|
||||||
*
|
|
||||||
* @param key The key of the item to get
|
|
||||||
* @param suppressTimeout should any TimeoutException be suppressed?
|
|
||||||
* @param flagAdapters flags to add to the get invocation
|
|
||||||
* @return The retrieved object
|
|
||||||
* @throws CacheException issue managing transaction or talking to cache
|
|
||||||
*/
|
|
||||||
protected Object get(Object key, boolean suppressTimeout, FlagAdapter... flagAdapters) throws CacheException {
|
|
||||||
CacheAdapter localCacheAdapter = cacheAdapter;
|
|
||||||
if (flagAdapters != null && flagAdapters.length > 0)
|
|
||||||
localCacheAdapter = cacheAdapter.withFlags(flagAdapters);
|
|
||||||
|
|
||||||
if (suppressTimeout)
|
|
||||||
return localCacheAdapter.getAllowingTimeout(key);
|
|
||||||
else
|
|
||||||
return localCacheAdapter.get(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Object getOwnerForPut() {
|
|
||||||
Transaction tx = null;
|
|
||||||
try {
|
|
||||||
if (transactionManager != null) {
|
|
||||||
tx = transactionManager.getTransaction();
|
|
||||||
}
|
|
||||||
} catch (SystemException se) {
|
|
||||||
throw new CacheException("Could not obtain transaction", se);
|
|
||||||
}
|
|
||||||
return tx == null ? Thread.currentThread() : tx;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tell the TransactionManager to suspend any ongoing transaction.
|
* Tell the TransactionManager to suspend any ongoing transaction.
|
||||||
*
|
*
|
||||||
|
@ -197,8 +159,8 @@ public abstract class BaseRegion implements Region {
|
||||||
public Transaction suspend() {
|
public Transaction suspend() {
|
||||||
Transaction tx = null;
|
Transaction tx = null;
|
||||||
try {
|
try {
|
||||||
if (transactionManager != null) {
|
if (tm != null) {
|
||||||
tx = transactionManager.suspend();
|
tx = tm.suspend();
|
||||||
}
|
}
|
||||||
} catch (SystemException se) {
|
} catch (SystemException se) {
|
||||||
throw new CacheException("Could not suspend transaction", se);
|
throw new CacheException("Could not suspend transaction", se);
|
||||||
|
@ -215,7 +177,7 @@ public abstract class BaseRegion implements Region {
|
||||||
public void resume(Transaction tx) {
|
public void resume(Transaction tx) {
|
||||||
try {
|
try {
|
||||||
if (tx != null)
|
if (tx != null)
|
||||||
transactionManager.resume(tx);
|
tm.resume(tx);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new CacheException("Could not resume transaction", e);
|
throw new CacheException("Could not resume transaction", e);
|
||||||
}
|
}
|
||||||
|
@ -227,7 +189,17 @@ public abstract class BaseRegion implements Region {
|
||||||
}
|
}
|
||||||
|
|
||||||
public TransactionManager getTransactionManager() {
|
public TransactionManager getTransactionManager() {
|
||||||
return transactionManager;
|
return tm;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used to satisfy TransactionalDataRegion.isTransactionAware in subclasses
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public boolean isTransactionAware() {
|
||||||
|
return tm != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AdvancedCache getCache() {
|
||||||
|
return cache;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,9 @@
|
||||||
package org.hibernate.cache.infinispan.impl;
|
package org.hibernate.cache.infinispan.impl;
|
||||||
|
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.TransactionalDataRegion;
|
import org.hibernate.cache.spi.TransactionalDataRegion;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Support for Inifinispan {@link org.hibernate.cache.spi.TransactionalDataRegion} implementors.
|
* Support for Inifinispan {@link org.hibernate.cache.spi.TransactionalDataRegion} implementors.
|
||||||
|
@ -19,10 +17,9 @@ public abstract class BaseTransactionalDataRegion
|
||||||
|
|
||||||
private final CacheDataDescription metadata;
|
private final CacheDataDescription metadata;
|
||||||
|
|
||||||
public BaseTransactionalDataRegion(CacheAdapter cacheAdapter, String name,
|
public BaseTransactionalDataRegion(AdvancedCache cache, String name,
|
||||||
CacheDataDescription metadata, TransactionManager transactionManager,
|
CacheDataDescription metadata, RegionFactory factory) {
|
||||||
RegionFactory factory) {
|
super(cache, name, factory);
|
||||||
super(cacheAdapter, name, transactionManager, factory);
|
|
||||||
this.metadata = metadata;
|
this.metadata = metadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,8 +27,4 @@ public abstract class BaseTransactionalDataRegion
|
||||||
return metadata;
|
return metadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isTransactionAware() {
|
|
||||||
return transactionManager != null;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,25 +1,27 @@
|
||||||
package org.hibernate.cache.infinispan.naturalid;
|
package org.hibernate.cache.infinispan.naturalid;
|
||||||
|
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.NaturalIdRegion;
|
import org.hibernate.cache.spi.NaturalIdRegion;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
import org.hibernate.cache.spi.access.NaturalIdRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.NaturalIdRegionAccessStrategy;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* Natural ID cache region
|
||||||
|
*
|
||||||
* @author Strong Liu <stliu@hibernate.org>
|
* @author Strong Liu <stliu@hibernate.org>
|
||||||
|
* @author Galder Zamarreño
|
||||||
*/
|
*/
|
||||||
public class NaturalIdRegionImpl extends BaseTransactionalDataRegion implements NaturalIdRegion {
|
public class NaturalIdRegionImpl extends BaseTransactionalDataRegion
|
||||||
public NaturalIdRegionImpl(CacheAdapter cacheAdapter,
|
implements NaturalIdRegion {
|
||||||
String name, CacheDataDescription metadata,
|
|
||||||
TransactionManager transactionManager, RegionFactory factory) {
|
public NaturalIdRegionImpl(AdvancedCache cache, String name,
|
||||||
super( cacheAdapter, name, metadata, transactionManager, factory );
|
CacheDataDescription metadata, RegionFactory factory) {
|
||||||
|
super(cache, name, metadata, factory);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -33,6 +35,7 @@ public class NaturalIdRegionImpl extends BaseTransactionalDataRegion implements
|
||||||
}
|
}
|
||||||
|
|
||||||
public PutFromLoadValidator getPutFromLoadValidator() {
|
public PutFromLoadValidator getPutFromLoadValidator() {
|
||||||
return new PutFromLoadValidator(transactionManager);
|
return new PutFromLoadValidator(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,14 @@
|
||||||
package org.hibernate.cache.infinispan.query;
|
package org.hibernate.cache.infinispan.query;
|
||||||
|
|
||||||
import java.util.Properties;
|
|
||||||
import javax.transaction.Transaction;
|
import javax.transaction.Transaction;
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
import org.hibernate.cache.infinispan.impl.BaseTransactionalDataRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
|
||||||
import org.hibernate.cache.spi.QueryResultsRegion;
|
import org.hibernate.cache.spi.QueryResultsRegion;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Chris Bredesen
|
* @author Chris Bredesen
|
||||||
|
@ -17,27 +16,35 @@ import org.hibernate.cache.spi.RegionFactory;
|
||||||
* @since 3.5
|
* @since 3.5
|
||||||
*/
|
*/
|
||||||
public class QueryResultsRegionImpl extends BaseTransactionalDataRegion implements QueryResultsRegion {
|
public class QueryResultsRegionImpl extends BaseTransactionalDataRegion implements QueryResultsRegion {
|
||||||
private boolean localOnly;
|
|
||||||
|
|
||||||
public QueryResultsRegionImpl(CacheAdapter cacheAdapter, String name, Properties properties, TransactionManager transactionManager, RegionFactory factory) {
|
private final AdvancedCache evictCache;
|
||||||
super(cacheAdapter, name, null, transactionManager, factory);
|
private final AdvancedCache putCache;
|
||||||
|
private final AdvancedCache getCache;
|
||||||
|
|
||||||
|
public QueryResultsRegionImpl(AdvancedCache cache, String name, RegionFactory factory) {
|
||||||
|
super(cache, name, null, factory);
|
||||||
// If Infinispan is using INVALIDATION for query cache, we don't want to propagate changes.
|
// If Infinispan is using INVALIDATION for query cache, we don't want to propagate changes.
|
||||||
// We use the Timestamps cache to manage invalidation
|
// We use the Timestamps cache to manage invalidation
|
||||||
localOnly = cacheAdapter.isClusteredInvalidation();
|
boolean localOnly = Caches.isInvalidationCache(cache);
|
||||||
|
|
||||||
|
this.evictCache = localOnly ? Caches.localCache(cache) : cache;
|
||||||
|
|
||||||
|
this.putCache = localOnly ?
|
||||||
|
Caches.failSilentWriteCache(cache, Flag.CACHE_MODE_LOCAL) :
|
||||||
|
Caches.failSilentWriteCache(cache);
|
||||||
|
|
||||||
|
this.getCache = Caches.failSilentReadCache(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evict(Object key) throws CacheException {
|
public void evict(Object key) throws CacheException {
|
||||||
if (localOnly)
|
evictCache.remove(key);
|
||||||
cacheAdapter.withFlags(FlagAdapter.CACHE_MODE_LOCAL).remove(key);
|
|
||||||
else
|
|
||||||
cacheAdapter.remove(key);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evictAll() throws CacheException {
|
public void evictAll() throws CacheException {
|
||||||
Transaction tx = suspend();
|
Transaction tx = suspend();
|
||||||
try {
|
try {
|
||||||
invalidateRegion(); // Invalidate the local region and then go remote
|
invalidateRegion(); // Invalidate the local region and then go remote
|
||||||
cacheAdapter.broadcastEvictAll();
|
Caches.broadcastEvictAll(cache);
|
||||||
} finally {
|
} finally {
|
||||||
resume(tx);
|
resume(tx);
|
||||||
}
|
}
|
||||||
|
@ -60,9 +67,9 @@ public class QueryResultsRegionImpl extends BaseTransactionalDataRegion implemen
|
||||||
// Add a zero (or low) timeout option so we don't block
|
// Add a zero (or low) timeout option so we don't block
|
||||||
// waiting for tx's that did a put to commit
|
// waiting for tx's that did a put to commit
|
||||||
if (skipCacheStore)
|
if (skipCacheStore)
|
||||||
return get(key, true, FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT, FlagAdapter.SKIP_CACHE_STORE);
|
return getCache.withFlags(Flag.SKIP_CACHE_STORE).get(key);
|
||||||
else
|
else
|
||||||
return get(key, true, FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT);
|
return getCache.get(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void put(Object key, Object value) throws CacheException {
|
public void put(Object key, Object value) throws CacheException {
|
||||||
|
@ -82,12 +89,8 @@ public class QueryResultsRegionImpl extends BaseTransactionalDataRegion implemen
|
||||||
// any subsequent read will just see the old result with its
|
// any subsequent read will just see the old result with its
|
||||||
// out-of-date timestamp; that result will be discarded and the
|
// out-of-date timestamp; that result will be discarded and the
|
||||||
// db query performed again.
|
// db query performed again.
|
||||||
if (localOnly)
|
putCache.put(key, value);
|
||||||
cacheAdapter.withFlags(FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT, FlagAdapter.CACHE_MODE_LOCAL)
|
|
||||||
.putAllowingTimeout(key, value);
|
|
||||||
else
|
|
||||||
cacheAdapter.withFlags(FlagAdapter.ZERO_LOCK_ACQUISITION_TIMEOUT)
|
|
||||||
.putAllowingTimeout(key, value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -0,0 +1,152 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.cache.infinispan.timestamp;
|
||||||
|
|
||||||
|
import org.hibernate.cache.CacheException;
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
|
import org.infinispan.notifications.Listener;
|
||||||
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryModified;
|
||||||
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryRemoved;
|
||||||
|
import org.infinispan.notifications.cachelistener.event.CacheEntryModifiedEvent;
|
||||||
|
import org.infinispan.notifications.cachelistener.event.CacheEntryRemovedEvent;
|
||||||
|
|
||||||
|
import javax.transaction.Transaction;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Timestamp cache region for clustered environments.
|
||||||
|
*
|
||||||
|
* @author Galder Zamarreño
|
||||||
|
* @since 4.1
|
||||||
|
*/
|
||||||
|
@Listener
|
||||||
|
public class ClusteredTimestampsRegionImpl extends TimestampsRegionImpl {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maintains a local (authoritative) cache of timestamps along with the
|
||||||
|
* replicated cache held in Infinispan. It listens for changes in the
|
||||||
|
* cache and updates the local cache accordingly. This approach allows
|
||||||
|
* timestamp changes to be replicated asynchronously.
|
||||||
|
*/
|
||||||
|
private final Map localCache = new ConcurrentHashMap();
|
||||||
|
|
||||||
|
public ClusteredTimestampsRegionImpl(AdvancedCache cache,
|
||||||
|
String name, RegionFactory factory) {
|
||||||
|
super(cache, name, factory);
|
||||||
|
cache.addListener(this);
|
||||||
|
populateLocalCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected AdvancedCache getTimestampsPutCache(AdvancedCache cache) {
|
||||||
|
return Caches.asyncWriteCache(cache, Flag.SKIP_LOCKING);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object get(Object key) throws CacheException {
|
||||||
|
Object value = localCache.get(key);
|
||||||
|
|
||||||
|
// If the region is not valid, skip cache store to avoid going remote to retrieve the query.
|
||||||
|
// The aim of this is to maintain same logic/semantics as when state transfer was configured.
|
||||||
|
// TODO: Once https://issues.jboss.org/browse/ISPN-835 has been resolved, revert to state transfer and remove workaround
|
||||||
|
boolean skipCacheStore = false;
|
||||||
|
if (!isValid())
|
||||||
|
skipCacheStore = true;
|
||||||
|
|
||||||
|
if (value == null && checkValid()) {
|
||||||
|
if (skipCacheStore)
|
||||||
|
value = cache.withFlags(Flag.SKIP_CACHE_STORE).get(key);
|
||||||
|
else
|
||||||
|
value = cache.get(key);
|
||||||
|
|
||||||
|
if (value != null)
|
||||||
|
localCache.put(key, value);
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void evictAll() throws CacheException {
|
||||||
|
// TODO Is this a valid operation on a timestamps cache?
|
||||||
|
Transaction tx = suspend();
|
||||||
|
try {
|
||||||
|
invalidateRegion(); // Invalidate the local region and then go remote
|
||||||
|
Caches.broadcastEvictAll(cache);
|
||||||
|
} finally {
|
||||||
|
resume(tx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void invalidateRegion() {
|
||||||
|
super.invalidateRegion(); // Invalidate first
|
||||||
|
localCache.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroy() throws CacheException {
|
||||||
|
localCache.clear();
|
||||||
|
cache.removeListener(this);
|
||||||
|
super.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Brings all data from the distributed cache into our local cache.
|
||||||
|
*/
|
||||||
|
private void populateLocalCache() {
|
||||||
|
Set children = cache.keySet();
|
||||||
|
for (Object key : children)
|
||||||
|
get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Monitors cache events and updates the local cache
|
||||||
|
*
|
||||||
|
* @param event
|
||||||
|
*/
|
||||||
|
@CacheEntryModified
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public void nodeModified(CacheEntryModifiedEvent event) {
|
||||||
|
if (!event.isPre())
|
||||||
|
localCache.put(event.getKey(), event.getValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Monitors cache events and updates the local cache
|
||||||
|
*
|
||||||
|
* @param event
|
||||||
|
*/
|
||||||
|
@CacheEntryRemoved
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public void nodeRemoved(CacheEntryRemovedEvent event) {
|
||||||
|
if (event.isPre()) return;
|
||||||
|
localCache.remove(event.getKey());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,8 +4,10 @@ import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import javax.transaction.Transaction;
|
import javax.transaction.Transaction;
|
||||||
import javax.transaction.TransactionManager;
|
|
||||||
|
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
import org.infinispan.notifications.Listener;
|
import org.infinispan.notifications.Listener;
|
||||||
import org.infinispan.notifications.cachelistener.annotation.CacheEntryModified;
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryModified;
|
||||||
import org.infinispan.notifications.cachelistener.annotation.CacheEntryRemoved;
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryRemoved;
|
||||||
|
@ -14,8 +16,6 @@ import org.infinispan.notifications.cachelistener.event.CacheEntryRemovedEvent;
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
import org.hibernate.cache.CacheException;
|
||||||
import org.hibernate.cache.infinispan.impl.BaseGeneralDataRegion;
|
import org.hibernate.cache.infinispan.impl.BaseGeneralDataRegion;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.TimestampsRegion;
|
import org.hibernate.cache.spi.TimestampsRegion;
|
||||||
|
|
||||||
|
@ -26,110 +26,64 @@ import org.hibernate.cache.spi.TimestampsRegion;
|
||||||
* @author Galder Zamarreño
|
* @author Galder Zamarreño
|
||||||
* @since 3.5
|
* @since 3.5
|
||||||
*/
|
*/
|
||||||
@Listener
|
|
||||||
public class TimestampsRegionImpl extends BaseGeneralDataRegion implements TimestampsRegion {
|
public class TimestampsRegionImpl extends BaseGeneralDataRegion implements TimestampsRegion {
|
||||||
|
|
||||||
private Map localCache = new ConcurrentHashMap();
|
private final AdvancedCache removeCache;
|
||||||
|
private final AdvancedCache timestampsPutCache;
|
||||||
|
|
||||||
public TimestampsRegionImpl(CacheAdapter cacheAdapter, String name, TransactionManager transactionManager, RegionFactory factory) {
|
public TimestampsRegionImpl(AdvancedCache cache, String name,
|
||||||
super(cacheAdapter, name, transactionManager, factory);
|
RegionFactory factory) {
|
||||||
cacheAdapter.addListener(this);
|
super(cache, name, factory);
|
||||||
populateLocalCache();
|
this.removeCache = Caches.ignoreReturnValuesCache(cache);
|
||||||
|
|
||||||
|
// Skip locking when updating timestamps to provide better performance
|
||||||
|
// under highly concurrent insert scenarios, where update timestamps
|
||||||
|
// for an entity/collection type are constantly updated, creating
|
||||||
|
// contention.
|
||||||
|
//
|
||||||
|
// The worst it can happen is that an earlier an earlier timestamp
|
||||||
|
// (i.e. ts=1) will override a later on (i.e. ts=2), so it means that
|
||||||
|
// in highly concurrent environments, queries might be considered stale
|
||||||
|
// earlier in time. The upside is that inserts/updates are way faster
|
||||||
|
// in local set ups.
|
||||||
|
this.timestampsPutCache = getTimestampsPutCache(cache);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected AdvancedCache getTimestampsPutCache(AdvancedCache cache) {
|
||||||
|
return Caches.ignoreReturnValuesCache(cache, Flag.SKIP_LOCKING);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void evict(Object key) throws CacheException {
|
public void evict(Object key) throws CacheException {
|
||||||
// TODO Is this a valid operation on a timestamps cache?
|
// TODO Is this a valid operation on a timestamps cache?
|
||||||
cacheAdapter.remove(key);
|
removeCache.remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void evictAll() throws CacheException {
|
public void evictAll() throws CacheException {
|
||||||
// TODO Is this a valid operation on a timestamps cache?
|
// TODO Is this a valid operation on a timestamps cache?
|
||||||
Transaction tx = suspend();
|
Transaction tx = suspend();
|
||||||
try {
|
try {
|
||||||
invalidateRegion(); // Invalidate the local region and then go remote
|
invalidateRegion(); // Invalidate the local region
|
||||||
cacheAdapter.broadcastEvictAll();
|
|
||||||
} finally {
|
} finally {
|
||||||
resume(tx);
|
resume(tx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Object get(Object key) throws CacheException {
|
public Object get(Object key) throws CacheException {
|
||||||
Object value = localCache.get(key);
|
if (checkValid())
|
||||||
|
return cache.get(key);
|
||||||
|
|
||||||
// If the region is not valid, skip cache store to avoid going remote to retrieve the query.
|
return null;
|
||||||
// The aim of this is to maintain same logic/semantics as when state transfer was configured.
|
|
||||||
// TODO: Once https://issues.jboss.org/browse/ISPN-835 has been resolved, revert to state transfer and remove workaround
|
|
||||||
boolean skipCacheStore = false;
|
|
||||||
if (!isValid())
|
|
||||||
skipCacheStore = true;
|
|
||||||
|
|
||||||
if (value == null && checkValid()) {
|
|
||||||
if (skipCacheStore)
|
|
||||||
value = get(key, false, FlagAdapter.SKIP_CACHE_STORE);
|
|
||||||
else
|
|
||||||
value = get(key, false);
|
|
||||||
|
|
||||||
if (value != null)
|
|
||||||
localCache.put(key, value);
|
|
||||||
}
|
|
||||||
return value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void put(final Object key, final Object value) throws CacheException {
|
public void put(final Object key, final Object value) throws CacheException {
|
||||||
try {
|
try {
|
||||||
// We ensure ASYNC semantics (JBCACHE-1175) and make sure previous
|
// We ensure ASYNC semantics (JBCACHE-1175) and make sure previous
|
||||||
// value is not loaded from cache store cos it's not needed.
|
// value is not loaded from cache store cos it's not needed.
|
||||||
cacheAdapter.withFlags(FlagAdapter.FORCE_ASYNCHRONOUS).put(key, value);
|
timestampsPutCache.put(key, value);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new CacheException(e);
|
throw new CacheException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void destroy() throws CacheException {
|
|
||||||
localCache.clear();
|
|
||||||
cacheAdapter.removeListener(this);
|
|
||||||
super.destroy();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Monitors cache events and updates the local cache
|
|
||||||
*
|
|
||||||
* @param event
|
|
||||||
*/
|
|
||||||
@CacheEntryModified
|
|
||||||
@SuppressWarnings("unused")
|
|
||||||
public void nodeModified(CacheEntryModifiedEvent event) {
|
|
||||||
if (!event.isPre())
|
|
||||||
localCache.put(event.getKey(), event.getValue());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Monitors cache events and updates the local cache
|
|
||||||
*
|
|
||||||
* @param event
|
|
||||||
*/
|
|
||||||
@CacheEntryRemoved
|
|
||||||
@SuppressWarnings("unused")
|
|
||||||
public void nodeRemoved(CacheEntryRemovedEvent event) {
|
|
||||||
if (event.isPre()) return;
|
|
||||||
localCache.remove(event.getKey());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void invalidateRegion() {
|
|
||||||
super.invalidateRegion(); // Invalidate first
|
|
||||||
localCache.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Brings all data from the distributed cache into our local cache.
|
|
||||||
*/
|
|
||||||
private void populateLocalCache() {
|
|
||||||
Set children = cacheAdapter.keySet();
|
|
||||||
for (Object key : children)
|
|
||||||
get(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,33 +0,0 @@
|
||||||
/*
|
|
||||||
* JBoss, Home of Professional Open Source.
|
|
||||||
* Copyright 2009, Red Hat, Inc. and/or its affiliates, and
|
|
||||||
* individual contributors as indicated by the @author tags. See the
|
|
||||||
* copyright.txt file in the distribution for a full listing of
|
|
||||||
* individual contributors.
|
|
||||||
*
|
|
||||||
* This is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2.1 of
|
|
||||||
* the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this software; if not, write to the Free
|
|
||||||
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
||||||
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
|
||||||
*/
|
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* AddressAdapter.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
|
||||||
* @since 3.5
|
|
||||||
*/
|
|
||||||
public interface AddressAdapter {
|
|
||||||
}
|
|
|
@ -1,88 +0,0 @@
|
||||||
/*
|
|
||||||
* JBoss, Home of Professional Open Source.
|
|
||||||
* Copyright 2009, Red Hat, Inc. and/or its affiliates, and
|
|
||||||
* individual contributors as indicated by the @author tags. See the
|
|
||||||
* copyright.txt file in the distribution for a full listing of
|
|
||||||
* individual contributors.
|
|
||||||
*
|
|
||||||
* This is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2.1 of
|
|
||||||
* the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this software; if not, write to the Free
|
|
||||||
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
||||||
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
|
||||||
*/
|
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
import java.io.Externalizable;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.ObjectInput;
|
|
||||||
import java.io.ObjectOutput;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import org.infinispan.remoting.transport.Address;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* AddressAdapterImpl.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
|
||||||
* @since 3.5
|
|
||||||
*/
|
|
||||||
public class AddressAdapterImpl implements AddressAdapter, Externalizable {
|
|
||||||
|
|
||||||
private Address address;
|
|
||||||
|
|
||||||
// Required by Java Externalizable
|
|
||||||
public AddressAdapterImpl() {
|
|
||||||
}
|
|
||||||
|
|
||||||
public AddressAdapterImpl(Address address) {
|
|
||||||
this.address = address;
|
|
||||||
}
|
|
||||||
|
|
||||||
static AddressAdapter newInstance(Address address) {
|
|
||||||
return new AddressAdapterImpl(address);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static List<AddressAdapter> toAddressAdapter(List<Address> ispnAddresses) {
|
|
||||||
List<AddressAdapter> addresses = new ArrayList<AddressAdapter>(ispnAddresses.size());
|
|
||||||
for (Address address : ispnAddresses) {
|
|
||||||
addresses.add(AddressAdapterImpl.newInstance(address));
|
|
||||||
}
|
|
||||||
return addresses;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
|
|
||||||
address = (Address) in.readObject();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void writeExternal(ObjectOutput out) throws IOException {
|
|
||||||
out.writeObject(address);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean equals(Object obj) {
|
|
||||||
if (obj == this)
|
|
||||||
return true;
|
|
||||||
if (!(obj instanceof AddressAdapterImpl))
|
|
||||||
return false;
|
|
||||||
AddressAdapterImpl other = (AddressAdapterImpl) obj;
|
|
||||||
return other.address.equals(address);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int hashCode() {
|
|
||||||
int result = 17;
|
|
||||||
result = 31 * result + address.hashCode();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,228 +0,0 @@
|
||||||
/*
|
|
||||||
* JBoss, Home of Professional Open Source.
|
|
||||||
* Copyright 2009, Red Hat, Inc. and/or its affiliates, and
|
|
||||||
* individual contributors as indicated by the @author tags. See the
|
|
||||||
* copyright.txt file in the distribution for a full listing of
|
|
||||||
* individual contributors.
|
|
||||||
*
|
|
||||||
* This is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2.1 of
|
|
||||||
* the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this software; if not, write to the Free
|
|
||||||
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
||||||
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
|
||||||
*/
|
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.concurrent.Callable;
|
|
||||||
|
|
||||||
import org.infinispan.Cache;
|
|
||||||
import org.infinispan.config.Configuration;
|
|
||||||
import org.infinispan.util.concurrent.TimeoutException;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Infinispan cache abstraction.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
|
||||||
* @since 3.5
|
|
||||||
*/
|
|
||||||
public interface CacheAdapter {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Is this cache participating in a cluster with invalidation?
|
|
||||||
*
|
|
||||||
* @return true if the cache is configured for synchronous/asynchronous invalidation; false otherwise.
|
|
||||||
*/
|
|
||||||
boolean isClusteredInvalidation();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Is this cache participating in a cluster with replication?
|
|
||||||
*
|
|
||||||
* @return true if the cache is configured for synchronous/asynchronous invalidation; false otherwise.
|
|
||||||
*/
|
|
||||||
boolean isClusteredReplication();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Is this cache configured for synchronous communication?
|
|
||||||
*
|
|
||||||
* @return true if the cache is configured for synchronous communication; false otherwise.
|
|
||||||
*/
|
|
||||||
boolean isSynchronous();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set of keys of this cache.
|
|
||||||
*
|
|
||||||
* @return Set containing keys stored in this cache.
|
|
||||||
*/
|
|
||||||
Set keySet();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A builder-style method that adds flags to any cache API call.
|
|
||||||
*
|
|
||||||
* @param flagAdapters a set of flags to apply. See the {@link FlagAdapter} documentation.
|
|
||||||
* @return a cache on which a real operation is to be invoked.
|
|
||||||
*/
|
|
||||||
CacheAdapter withFlags(FlagAdapter... flagAdapters);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Method to check whether a certain key exists in this cache.
|
|
||||||
*
|
|
||||||
* @param key key to look up.
|
|
||||||
* @return true if key is present, false otherwise.
|
|
||||||
*/
|
|
||||||
boolean containsKey(Object key);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs an <code>get(Object)</code> on the cache, wrapping any exception in a {@link CacheException}.
|
|
||||||
*
|
|
||||||
* @param key key to retrieve
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
Object get(Object key) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs an <code>get(Object)</code> on the cache ignoring any {@link TimeoutException}
|
|
||||||
* and wrapping any other exception in a {@link CacheException}.
|
|
||||||
*
|
|
||||||
* @param key key to retrieve
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
Object getAllowingTimeout(Object key) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs a <code>put(Object, Object)</code> on the cache,
|
|
||||||
* wrapping any exception in a {@link CacheException}.
|
|
||||||
*
|
|
||||||
* @param key key whose value will be modified
|
|
||||||
* @param value data to store in the cache entry
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
void put(Object key, Object value) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs a <code>put(Object, Object)</code> on the cache ignoring
|
|
||||||
* any {@link TimeoutException} and wrapping any exception in a
|
|
||||||
* {@link CacheException}.
|
|
||||||
*
|
|
||||||
* @param key key whose value will be modified
|
|
||||||
* @param value data to store in the cache entry
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
void putAllowingTimeout(Object key, Object value) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* See {@link Cache#putForExternalRead(Object, Object)} for detailed documentation.
|
|
||||||
*
|
|
||||||
* @param key key with which the specified value is to be associated.
|
|
||||||
* @param value value to be associated with the specified key.
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
void putForExternalRead(Object key, Object value) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Performs a <code>remove(Object)</code>, wrapping any exception in
|
|
||||||
* a {@link CacheException}.
|
|
||||||
*
|
|
||||||
* @param key key to be removed
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
void remove(Object key) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Evict the given key from memory.
|
|
||||||
*
|
|
||||||
* @param key to evict.
|
|
||||||
*/
|
|
||||||
void evict(Object key) throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clear the cache.
|
|
||||||
*
|
|
||||||
* @throws CacheException
|
|
||||||
*/
|
|
||||||
void clear() throws CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stops the cache.
|
|
||||||
*/
|
|
||||||
void stop();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add listener to this cache.
|
|
||||||
*
|
|
||||||
* @param listener to be added to cache.
|
|
||||||
*/
|
|
||||||
void addListener(Object listener);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get local cluster address.
|
|
||||||
*
|
|
||||||
* @return Address representing local address.
|
|
||||||
*/
|
|
||||||
AddressAdapter getAddress();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get cluster members.
|
|
||||||
*
|
|
||||||
* @return List of cluster member Address instances
|
|
||||||
*/
|
|
||||||
List<AddressAdapter> getMembers();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Size of cache.
|
|
||||||
*
|
|
||||||
* @return number of cache entries.
|
|
||||||
*/
|
|
||||||
int size();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method returns a Map view of the cache.
|
|
||||||
*
|
|
||||||
* @return Map view of cache.
|
|
||||||
*/
|
|
||||||
Map toMap();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove listener from cache instance.
|
|
||||||
*
|
|
||||||
* @param listener to be removed.
|
|
||||||
*/
|
|
||||||
void removeListener(Object listener);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get cache configuration.
|
|
||||||
*
|
|
||||||
* @return Configuration instance associated with this cache.
|
|
||||||
*/
|
|
||||||
Configuration getConfiguration();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* TODO
|
|
||||||
*/
|
|
||||||
void broadcastEvictAll();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* TODO
|
|
||||||
*
|
|
||||||
* @param c
|
|
||||||
* @param <T>
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
<T> T withinTx(Callable<T> c) throws Exception;
|
|
||||||
|
|
||||||
Cache getCache();
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,255 +0,0 @@
|
||||||
/*
|
|
||||||
* JBoss, Home of Professional Open Source.
|
|
||||||
* Copyright 2009, Red Hat, Inc. and/or its affiliates, and
|
|
||||||
* individual contributors as indicated by the @author tags. See the
|
|
||||||
* copyright.txt file in the distribution for a full listing of
|
|
||||||
* individual contributors.
|
|
||||||
*
|
|
||||||
* This is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2.1 of
|
|
||||||
* the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this software; if not, write to the Free
|
|
||||||
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
||||||
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
|
||||||
*/
|
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.concurrent.Callable;
|
|
||||||
|
|
||||||
import org.infinispan.AdvancedCache;
|
|
||||||
import org.infinispan.Cache;
|
|
||||||
import org.infinispan.config.Configuration;
|
|
||||||
import org.infinispan.context.Flag;
|
|
||||||
import org.infinispan.remoting.rpc.RpcManager;
|
|
||||||
import org.infinispan.util.concurrent.TimeoutException;
|
|
||||||
import org.infinispan.util.logging.Log;
|
|
||||||
import org.infinispan.util.logging.LogFactory;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* CacheAdapterImpl.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
|
||||||
* @since 3.5
|
|
||||||
*/
|
|
||||||
public class CacheAdapterImpl implements CacheAdapter {
|
|
||||||
private static final Log log = LogFactory.getLog(CacheAdapterImpl.class);
|
|
||||||
|
|
||||||
private final AdvancedCache cache;
|
|
||||||
private final CacheCommandInitializer cacheCmdInitializer;
|
|
||||||
private final boolean isSync;
|
|
||||||
|
|
||||||
private CacheAdapterImpl(AdvancedCache cache) {
|
|
||||||
this.cache = cache;
|
|
||||||
this.cacheCmdInitializer = cache.getComponentRegistry()
|
|
||||||
.getComponent(CacheCommandInitializer.class);
|
|
||||||
this.isSync = isSynchronous(cache.getConfiguration().getCacheMode());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CacheAdapter newInstance(AdvancedCache cache) {
|
|
||||||
return new CacheAdapterImpl(cache);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isClusteredInvalidation() {
|
|
||||||
return isClusteredInvalidation(cache.getConfiguration().getCacheMode());
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isClusteredReplication() {
|
|
||||||
return isClusteredReplication(cache.getConfiguration().getCacheMode());
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isSynchronous() {
|
|
||||||
return isSync;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Set keySet() {
|
|
||||||
return cache.keySet();
|
|
||||||
}
|
|
||||||
|
|
||||||
public CacheAdapter withFlags(FlagAdapter... flagAdapters) {
|
|
||||||
Flag[] flags = FlagAdapter.toFlags(flagAdapters);
|
|
||||||
return newInstance(cache.withFlags(flags));
|
|
||||||
}
|
|
||||||
|
|
||||||
public Object get(Object key) throws CacheException {
|
|
||||||
try {
|
|
||||||
return cache.get(key);
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Object getAllowingTimeout(Object key) throws CacheException {
|
|
||||||
try {
|
|
||||||
return getFailSilentCache().get(key);
|
|
||||||
} catch (TimeoutException ignored) {
|
|
||||||
// ignore it
|
|
||||||
return null;
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void put(Object key, Object value) throws CacheException {
|
|
||||||
try {
|
|
||||||
// No previous value interest, so apply flags that avoid remote lookups.
|
|
||||||
getSkipRemoteGetLoadCache().put(key, value);
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void putAllowingTimeout(Object key, Object value) throws CacheException {
|
|
||||||
try {
|
|
||||||
// No previous value interest, so apply flags that avoid remote lookups.
|
|
||||||
getFailSilentCacheSkipRemotes().put(key, value);
|
|
||||||
} catch (TimeoutException allowed) {
|
|
||||||
// ignore it
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void putForExternalRead(Object key, Object value) throws CacheException {
|
|
||||||
try {
|
|
||||||
// No previous value interest, so apply flags that avoid remote lookups.
|
|
||||||
getFailSilentCacheSkipRemotes().putForExternalRead(key, value);
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void remove(Object key) throws CacheException {
|
|
||||||
try {
|
|
||||||
// No previous value interest, so apply flags that avoid remote lookups.
|
|
||||||
getSkipRemoteGetLoadCache().remove(key);
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void evict(Object key) throws CacheException {
|
|
||||||
try {
|
|
||||||
cache.evict(key);
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void clear() throws CacheException {
|
|
||||||
try {
|
|
||||||
cache.clear();
|
|
||||||
} catch (Exception e) {
|
|
||||||
throw new CacheException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void stop() {
|
|
||||||
if (log.isTraceEnabled())
|
|
||||||
log.trace("Stop " + cache);
|
|
||||||
cache.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean isClusteredInvalidation(Configuration.CacheMode cacheMode) {
|
|
||||||
return cacheMode == Configuration.CacheMode.INVALIDATION_ASYNC
|
|
||||||
|| cacheMode == Configuration.CacheMode.INVALIDATION_SYNC;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean isClusteredReplication(Configuration.CacheMode cacheMode) {
|
|
||||||
return cacheMode == Configuration.CacheMode.REPL_ASYNC
|
|
||||||
|| cacheMode == Configuration.CacheMode.REPL_SYNC;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean isSynchronous(Configuration.CacheMode cacheMode) {
|
|
||||||
return cacheMode == Configuration.CacheMode.REPL_SYNC
|
|
||||||
|| cacheMode == Configuration.CacheMode.INVALIDATION_SYNC
|
|
||||||
|| cacheMode == Configuration.CacheMode.DIST_SYNC;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void addListener(Object listener) {
|
|
||||||
cache.addListener(listener);
|
|
||||||
}
|
|
||||||
|
|
||||||
public AddressAdapter getAddress() {
|
|
||||||
RpcManager rpc = cache.getRpcManager();
|
|
||||||
if (rpc != null) {
|
|
||||||
return AddressAdapterImpl.newInstance(rpc.getTransport().getAddress());
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<AddressAdapter> getMembers() {
|
|
||||||
RpcManager rpc = cache.getRpcManager();
|
|
||||||
if (rpc != null) {
|
|
||||||
return AddressAdapterImpl.toAddressAdapter(rpc.getTransport().getMembers());
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int size() {
|
|
||||||
return cache.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map toMap() {
|
|
||||||
return cache;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void removeListener(Object listener) {
|
|
||||||
cache.removeListener(listener);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean containsKey(Object key) {
|
|
||||||
return cache.containsKey(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Configuration getConfiguration() {
|
|
||||||
return cache.getConfiguration();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void broadcastEvictAll() {
|
|
||||||
RpcManager rpcManager = cache.getRpcManager();
|
|
||||||
if (rpcManager != null) {
|
|
||||||
// Only broadcast evict all if it's clustered
|
|
||||||
EvictAllCommand cmd = cacheCmdInitializer.buildEvictAllCommand(cache.getName());
|
|
||||||
rpcManager.broadcastRpcCommand(cmd, isSync);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public <T> T withinTx(Callable<T> c) throws Exception {
|
|
||||||
return CacheHelper.withinTx(cache.getTransactionManager(), c);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Cache getCache() {
|
|
||||||
return cache;
|
|
||||||
}
|
|
||||||
|
|
||||||
private Cache getFailSilentCache() {
|
|
||||||
return cache.withFlags(Flag.FAIL_SILENTLY);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Cache getSkipRemoteGetLoadCache() {
|
|
||||||
return cache.withFlags(
|
|
||||||
Flag.SKIP_CACHE_LOAD, Flag.SKIP_REMOTE_LOOKUP);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Cache getFailSilentCacheSkipRemotes() {
|
|
||||||
return cache.withFlags(
|
|
||||||
Flag.FAIL_SILENTLY, Flag.SKIP_CACHE_LOAD, Flag.SKIP_REMOTE_LOOKUP);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
141
hibernate-infinispan/src/main/java/org/hibernate/cache/infinispan/util/Caches.java
vendored
Normal file
141
hibernate-infinispan/src/main/java/org/hibernate/cache/infinispan/util/Caches.java
vendored
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.cache.infinispan.util;
|
||||||
|
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
|
import org.infinispan.remoting.rpc.RpcManager;
|
||||||
|
|
||||||
|
import javax.transaction.Status;
|
||||||
|
import javax.transaction.TransactionManager;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper for dealing with Infinispan cache instances.
|
||||||
|
*
|
||||||
|
* @author Galder Zamarreño
|
||||||
|
* @since 4.1
|
||||||
|
*/
|
||||||
|
public class Caches {
|
||||||
|
|
||||||
|
private Caches() {
|
||||||
|
// Suppresses default constructor, ensuring non-instantiability.
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> T withinTx(AdvancedCache cache,
|
||||||
|
Callable<T> c) throws Exception {
|
||||||
|
// Retrieve transaction manager
|
||||||
|
return withinTx(cache.getTransactionManager(), c);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> T withinTx(TransactionManager tm,
|
||||||
|
Callable<T> c) throws Exception {
|
||||||
|
tm.begin();
|
||||||
|
try {
|
||||||
|
return c.call();
|
||||||
|
} catch (Exception e) {
|
||||||
|
tm.setRollbackOnly();
|
||||||
|
throw e;
|
||||||
|
} finally {
|
||||||
|
if (tm.getStatus() == Status.STATUS_ACTIVE) tm.commit();
|
||||||
|
else tm.rollback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache localCache(AdvancedCache cache) {
|
||||||
|
return cache.withFlags(Flag.CACHE_MODE_LOCAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache ignoreReturnValuesCache(AdvancedCache cache) {
|
||||||
|
return cache.withFlags(Flag.SKIP_CACHE_LOAD, Flag.SKIP_REMOTE_LOOKUP);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache ignoreReturnValuesCache(
|
||||||
|
AdvancedCache cache, Flag extraFlag) {
|
||||||
|
return cache.withFlags(
|
||||||
|
Flag.SKIP_CACHE_LOAD, Flag.SKIP_REMOTE_LOOKUP, extraFlag);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache asyncWriteCache(AdvancedCache cache,
|
||||||
|
Flag extraFlag) {
|
||||||
|
return cache.withFlags(
|
||||||
|
Flag.SKIP_CACHE_LOAD,
|
||||||
|
Flag.SKIP_REMOTE_LOOKUP,
|
||||||
|
Flag.FORCE_ASYNCHRONOUS,
|
||||||
|
extraFlag);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache failSilentWriteCache(AdvancedCache cache) {
|
||||||
|
return cache.withFlags(
|
||||||
|
Flag.FAIL_SILENTLY,
|
||||||
|
Flag.ZERO_LOCK_ACQUISITION_TIMEOUT,
|
||||||
|
Flag.SKIP_CACHE_LOAD,
|
||||||
|
Flag.SKIP_REMOTE_LOOKUP);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache failSilentWriteCache(AdvancedCache cache,
|
||||||
|
Flag extraFlag) {
|
||||||
|
return cache.withFlags(
|
||||||
|
Flag.FAIL_SILENTLY,
|
||||||
|
Flag.ZERO_LOCK_ACQUISITION_TIMEOUT,
|
||||||
|
Flag.SKIP_CACHE_LOAD,
|
||||||
|
Flag.SKIP_REMOTE_LOOKUP,
|
||||||
|
extraFlag);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AdvancedCache failSilentReadCache(AdvancedCache cache) {
|
||||||
|
return cache.withFlags(
|
||||||
|
Flag.FAIL_SILENTLY,
|
||||||
|
Flag.ZERO_LOCK_ACQUISITION_TIMEOUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void broadcastEvictAll(AdvancedCache cache) {
|
||||||
|
RpcManager rpcManager = cache.getRpcManager();
|
||||||
|
if (rpcManager != null) {
|
||||||
|
// Only broadcast evict all if it's clustered
|
||||||
|
CacheCommandInitializer factory = cache.getComponentRegistry()
|
||||||
|
.getComponent(CacheCommandInitializer.class);
|
||||||
|
boolean isSync = isSynchronousCache(cache);
|
||||||
|
|
||||||
|
EvictAllCommand cmd = factory.buildEvictAllCommand(cache.getName());
|
||||||
|
rpcManager.broadcastRpcCommand(cmd, isSync);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isInvalidationCache(AdvancedCache cache) {
|
||||||
|
return cache.getCacheConfiguration()
|
||||||
|
.clustering().cacheMode().isInvalidation();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isSynchronousCache(AdvancedCache cache) {
|
||||||
|
return cache.getCacheConfiguration()
|
||||||
|
.clustering().cacheMode().isSynchronous();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isClustered(AdvancedCache cache) {
|
||||||
|
return cache.getCacheConfiguration()
|
||||||
|
.clustering().cacheMode().isClustered();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,69 +0,0 @@
|
||||||
/*
|
|
||||||
* JBoss, Home of Professional Open Source.
|
|
||||||
* Copyright 2009, Red Hat, Inc. and/or its affiliates, and
|
|
||||||
* individual contributors as indicated by the @author tags. See the
|
|
||||||
* copyright.txt file in the distribution for a full listing of
|
|
||||||
* individual contributors.
|
|
||||||
*
|
|
||||||
* This is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as
|
|
||||||
* published by the Free Software Foundation; either version 2.1 of
|
|
||||||
* the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This software is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this software; if not, write to the Free
|
|
||||||
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
||||||
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
|
||||||
*/
|
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
import org.infinispan.context.Flag;
|
|
||||||
|
|
||||||
import org.hibernate.cache.CacheException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* FlagAdapter.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
|
||||||
* @since 3.5
|
|
||||||
*/
|
|
||||||
public enum FlagAdapter {
|
|
||||||
ZERO_LOCK_ACQUISITION_TIMEOUT,
|
|
||||||
CACHE_MODE_LOCAL,
|
|
||||||
FORCE_ASYNCHRONOUS,
|
|
||||||
FORCE_SYNCHRONOUS,
|
|
||||||
SKIP_CACHE_STORE,
|
|
||||||
SKIP_CACHE_LOAD;
|
|
||||||
|
|
||||||
Flag toFlag() {
|
|
||||||
switch(this) {
|
|
||||||
case ZERO_LOCK_ACQUISITION_TIMEOUT:
|
|
||||||
return Flag.ZERO_LOCK_ACQUISITION_TIMEOUT;
|
|
||||||
case CACHE_MODE_LOCAL:
|
|
||||||
return Flag.CACHE_MODE_LOCAL;
|
|
||||||
case FORCE_ASYNCHRONOUS:
|
|
||||||
return Flag.FORCE_ASYNCHRONOUS;
|
|
||||||
case FORCE_SYNCHRONOUS:
|
|
||||||
return Flag.FORCE_SYNCHRONOUS;
|
|
||||||
case SKIP_CACHE_STORE:
|
|
||||||
return Flag.SKIP_CACHE_STORE;
|
|
||||||
case SKIP_CACHE_LOAD:
|
|
||||||
return Flag.SKIP_CACHE_LOAD;
|
|
||||||
default:
|
|
||||||
throw new CacheException("Unmatched Infinispan flag " + this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static Flag[] toFlags(FlagAdapter[] adapters) {
|
|
||||||
Flag[] flags = new Flag[adapters.length];
|
|
||||||
for (int i = 0; i < adapters.length; i++) {
|
|
||||||
flags[i] = adapters[i].toFlag();
|
|
||||||
}
|
|
||||||
return flags;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -25,13 +25,13 @@ package org.hibernate.test.cache.infinispan;
|
||||||
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
||||||
import org.jboss.logging.Logger;
|
import org.jboss.logging.Logger;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.spi.GeneralDataRegion;
|
import org.hibernate.cache.spi.GeneralDataRegion;
|
||||||
import org.hibernate.cache.spi.QueryResultsRegion;
|
import org.hibernate.cache.spi.QueryResultsRegion;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
|
@ -152,7 +152,7 @@ public abstract class AbstractGeneralDataRegionTestCase extends AbstractRegionIm
|
||||||
cfg,
|
cfg,
|
||||||
getCacheTestSupport()
|
getCacheTestSupport()
|
||||||
);
|
);
|
||||||
CacheAdapter localCache = getInfinispanCache( regionFactory );
|
AdvancedCache localCache = getInfinispanCache( regionFactory );
|
||||||
|
|
||||||
// Sleep a bit to avoid concurrent FLUSH problem
|
// Sleep a bit to avoid concurrent FLUSH problem
|
||||||
avoidConcurrentFlush();
|
avoidConcurrentFlush();
|
||||||
|
@ -170,7 +170,7 @@ public abstract class AbstractGeneralDataRegionTestCase extends AbstractRegionIm
|
||||||
cfg,
|
cfg,
|
||||||
getCacheTestSupport()
|
getCacheTestSupport()
|
||||||
);
|
);
|
||||||
CacheAdapter remoteCache = getInfinispanCache( regionFactory );
|
AdvancedCache remoteCache = getInfinispanCache( regionFactory );
|
||||||
|
|
||||||
// Sleep a bit to avoid concurrent FLUSH problem
|
// Sleep a bit to avoid concurrent FLUSH problem
|
||||||
avoidConcurrentFlush();
|
avoidConcurrentFlush();
|
||||||
|
|
|
@ -26,11 +26,11 @@ package org.hibernate.test.cache.infinispan;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
import org.hibernate.internal.util.compare.ComparableComparator;
|
import org.hibernate.internal.util.compare.ComparableComparator;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Base class for tests of Region implementations.
|
* Base class for tests of Region implementations.
|
||||||
|
@ -40,7 +40,7 @@ import org.hibernate.internal.util.compare.ComparableComparator;
|
||||||
*/
|
*/
|
||||||
public abstract class AbstractRegionImplTestCase extends AbstractNonFunctionalTestCase {
|
public abstract class AbstractRegionImplTestCase extends AbstractNonFunctionalTestCase {
|
||||||
|
|
||||||
protected abstract CacheAdapter getInfinispanCache(InfinispanRegionFactory regionFactory);
|
protected abstract AdvancedCache getInfinispanCache(InfinispanRegionFactory regionFactory);
|
||||||
|
|
||||||
protected abstract Region createRegion(InfinispanRegionFactory regionFactory, String regionName, Properties properties, CacheDataDescription cdd);
|
protected abstract Region createRegion(InfinispanRegionFactory regionFactory, String regionName, Properties properties, CacheDataDescription cdd);
|
||||||
|
|
||||||
|
|
|
@ -24,6 +24,7 @@ package org.hibernate.test.cache.infinispan;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
import org.infinispan.config.Configuration;
|
import org.infinispan.config.Configuration;
|
||||||
import org.infinispan.config.Configuration.CacheMode;
|
import org.infinispan.config.Configuration.CacheMode;
|
||||||
import org.infinispan.eviction.EvictionStrategy;
|
import org.infinispan.eviction.EvictionStrategy;
|
||||||
|
@ -38,7 +39,6 @@ import org.hibernate.cache.infinispan.entity.EntityRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.query.QueryResultsRegionImpl;
|
import org.hibernate.cache.infinispan.query.QueryResultsRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.tm.HibernateTransactionManagerLookup;
|
import org.hibernate.cache.infinispan.tm.HibernateTransactionManagerLookup;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cfg.Settings;
|
import org.hibernate.cfg.Settings;
|
||||||
import org.hibernate.engine.transaction.jta.platform.internal.AbstractJtaPlatform;
|
import org.hibernate.engine.transaction.jta.platform.internal.AbstractJtaPlatform;
|
||||||
import org.hibernate.engine.transaction.jta.platform.internal.JBossStandAloneJtaPlatform;
|
import org.hibernate.engine.transaction.jta.platform.internal.JBossStandAloneJtaPlatform;
|
||||||
|
@ -137,13 +137,13 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertFalse(factory.getDefinedConfigurations().contains(person));
|
assertFalse(factory.getDefinedConfigurations().contains(person));
|
||||||
assertNotNull(factory.getTypeOverrides().get(addresses));
|
assertNotNull(factory.getTypeOverrides().get(addresses));
|
||||||
assertFalse(factory.getDefinedConfigurations().contains(addresses));
|
assertFalse(factory.getDefinedConfigurations().contains(addresses));
|
||||||
CacheAdapter cache = null;
|
AdvancedCache cache;
|
||||||
|
|
||||||
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion(person, p, null);
|
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion(person, p, null);
|
||||||
assertNotNull(factory.getTypeOverrides().get(person));
|
assertNotNull(factory.getTypeOverrides().get(person));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(person));
|
assertTrue(factory.getDefinedConfigurations().contains(person));
|
||||||
assertNull(factory.getTypeOverrides().get(address));
|
assertNull(factory.getTypeOverrides().get(address));
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(2000, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(2000, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -156,7 +156,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertNotNull(factory.getTypeOverrides().get(person));
|
assertNotNull(factory.getTypeOverrides().get(person));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(person));
|
assertTrue(factory.getDefinedConfigurations().contains(person));
|
||||||
assertNull(factory.getTypeOverrides().get(address));
|
assertNull(factory.getTypeOverrides().get(address));
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -167,7 +167,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertNotNull(factory.getTypeOverrides().get(person));
|
assertNotNull(factory.getTypeOverrides().get(person));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(person));
|
assertTrue(factory.getDefinedConfigurations().contains(person));
|
||||||
assertNull(factory.getTypeOverrides().get(address));
|
assertNull(factory.getTypeOverrides().get(address));
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -178,7 +178,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertNotNull(factory.getTypeOverrides().get(addresses));
|
assertNotNull(factory.getTypeOverrides().get(addresses));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(person));
|
assertTrue(factory.getDefinedConfigurations().contains(person));
|
||||||
assertNull(factory.getTypeOverrides().get(parts));
|
assertNull(factory.getTypeOverrides().get(parts));
|
||||||
cache = collectionRegion .getCacheAdapter();
|
cache = collectionRegion .getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(2500, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(2500, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -191,7 +191,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertNotNull(factory.getTypeOverrides().get(addresses));
|
assertNotNull(factory.getTypeOverrides().get(addresses));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(addresses));
|
assertTrue(factory.getDefinedConfigurations().contains(addresses));
|
||||||
assertNull(factory.getTypeOverrides().get(parts));
|
assertNull(factory.getTypeOverrides().get(parts));
|
||||||
cache = collectionRegion.getCacheAdapter();
|
cache = collectionRegion.getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -202,7 +202,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
assertNotNull(factory.getTypeOverrides().get(addresses));
|
assertNotNull(factory.getTypeOverrides().get(addresses));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(addresses));
|
assertTrue(factory.getDefinedConfigurations().contains(addresses));
|
||||||
assertNull(factory.getTypeOverrides().get(parts));
|
assertNull(factory.getTypeOverrides().get(parts));
|
||||||
cache = collectionRegion.getCacheAdapter();
|
cache = collectionRegion.getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -215,7 +215,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testBuildEntityCollectionRegionOverridesOnly() {
|
public void testBuildEntityCollectionRegionOverridesOnly() {
|
||||||
CacheAdapter cache;
|
AdvancedCache cache;
|
||||||
Properties p = new Properties();
|
Properties p = new Properties();
|
||||||
p.setProperty("hibernate.cache.infinispan.entity.eviction.strategy", "FIFO");
|
p.setProperty("hibernate.cache.infinispan.entity.eviction.strategy", "FIFO");
|
||||||
p.setProperty("hibernate.cache.infinispan.entity.eviction.wake_up_interval", "3000");
|
p.setProperty("hibernate.cache.infinispan.entity.eviction.wake_up_interval", "3000");
|
||||||
|
@ -228,7 +228,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
try {
|
try {
|
||||||
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
||||||
assertNull(factory.getTypeOverrides().get("com.acme.Address"));
|
assertNull(factory.getTypeOverrides().get("com.acme.Address"));
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -237,7 +237,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
|
|
||||||
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
||||||
assertNull(factory.getTypeOverrides().get("com.acme.Person.addresses"));
|
assertNull(factory.getTypeOverrides().get("com.acme.Person.addresses"));
|
||||||
cache = collectionRegion.getCacheAdapter();
|
cache = collectionRegion.getCache();
|
||||||
cacheCfg = cache.getConfiguration();
|
cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3500, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -267,7 +267,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion(person, p, null);
|
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion(person, p, null);
|
||||||
assertNotNull(factory.getTypeOverrides().get(person));
|
assertNotNull(factory.getTypeOverrides().get(person));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(person));
|
assertTrue(factory.getDefinedConfigurations().contains(person));
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.LRU, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(3000, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -305,7 +305,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
config.setFetchInMemoryState(false);
|
config.setFetchInMemoryState(false);
|
||||||
manager.defineConfiguration("timestamps", config);
|
manager.defineConfiguration("timestamps", config);
|
||||||
TimestampsRegionImpl region = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
TimestampsRegionImpl region = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.NONE, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.NONE, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(CacheMode.REPL_ASYNC, cacheCfg.getCacheMode());
|
assertEquals(CacheMode.REPL_ASYNC, cacheCfg.getCacheMode());
|
||||||
|
@ -331,7 +331,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
config.setCacheMode(CacheMode.REPL_SYNC);
|
config.setCacheMode(CacheMode.REPL_SYNC);
|
||||||
manager.defineConfiguration("unrecommended-timestamps", config);
|
manager.defineConfiguration("unrecommended-timestamps", config);
|
||||||
TimestampsRegionImpl region = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
TimestampsRegionImpl region = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.NONE, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.NONE, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(CacheMode.REPL_SYNC, cacheCfg.getCacheMode());
|
assertEquals(CacheMode.REPL_SYNC, cacheCfg.getCacheMode());
|
||||||
|
@ -401,7 +401,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
try {
|
try {
|
||||||
assertTrue(factory.getDefinedConfigurations().contains("local-query"));
|
assertTrue(factory.getDefinedConfigurations().contains("local-query"));
|
||||||
QueryResultsRegionImpl region = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
QueryResultsRegionImpl region = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(CacheMode.LOCAL, cacheCfg.getCacheMode());
|
assertEquals(CacheMode.LOCAL, cacheCfg.getCacheMode());
|
||||||
assertFalse(cacheCfg.isExposeJmxStatistics());
|
assertFalse(cacheCfg.isExposeJmxStatistics());
|
||||||
|
@ -425,7 +425,7 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
QueryResultsRegionImpl region = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(queryRegionName, p);
|
QueryResultsRegionImpl region = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(queryRegionName, p);
|
||||||
assertNotNull(factory.getTypeOverrides().get(queryRegionName));
|
assertNotNull(factory.getTypeOverrides().get(queryRegionName));
|
||||||
assertTrue(factory.getDefinedConfigurations().contains(queryRegionName));
|
assertTrue(factory.getDefinedConfigurations().contains(queryRegionName));
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
Configuration cacheCfg = cache.getConfiguration();
|
Configuration cacheCfg = cache.getConfiguration();
|
||||||
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
assertEquals(EvictionStrategy.FIFO, cacheCfg.getEvictionStrategy());
|
||||||
assertEquals(2222, cacheCfg.getEvictionWakeUpInterval());
|
assertEquals(2222, cacheCfg.getEvictionWakeUpInterval());
|
||||||
|
@ -449,18 +449,18 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
try {
|
try {
|
||||||
assertTrue(manager.getGlobalConfiguration().isExposeGlobalJmxStatistics());
|
assertTrue(manager.getGlobalConfiguration().isExposeGlobalJmxStatistics());
|
||||||
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
assertTrue(factory.getTypeOverrides().get("entity").isExposeStatistics());
|
assertTrue(factory.getTypeOverrides().get("entity").isExposeStatistics());
|
||||||
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Person", p, null);
|
region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Person", p, null);
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
assertTrue(factory.getTypeOverrides().get("com.acme.Person").isExposeStatistics());
|
assertTrue(factory.getTypeOverrides().get("com.acme.Person").isExposeStatistics());
|
||||||
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
final String query = "org.hibernate.cache.internal.StandardQueryCache";
|
final String query = "org.hibernate.cache.internal.StandardQueryCache";
|
||||||
QueryResultsRegionImpl queryRegion = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
QueryResultsRegionImpl queryRegion = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
||||||
cache = queryRegion.getCacheAdapter();
|
cache = queryRegion.getCache();
|
||||||
assertTrue(factory.getTypeOverrides().get("query").isExposeStatistics());
|
assertTrue(factory.getTypeOverrides().get("query").isExposeStatistics());
|
||||||
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
|
@ -469,12 +469,12 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
config.setFetchInMemoryState(false);
|
config.setFetchInMemoryState(false);
|
||||||
manager.defineConfiguration("timestamps", config);
|
manager.defineConfiguration("timestamps", config);
|
||||||
TimestampsRegionImpl timestampsRegion = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
TimestampsRegionImpl timestampsRegion = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
||||||
cache = timestampsRegion.getCacheAdapter();
|
cache = timestampsRegion.getCache();
|
||||||
assertTrue(factory.getTypeOverrides().get("timestamps").isExposeStatistics());
|
assertTrue(factory.getTypeOverrides().get("timestamps").isExposeStatistics());
|
||||||
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
||||||
cache = collectionRegion.getCacheAdapter();
|
cache = collectionRegion.getCache();
|
||||||
assertTrue(factory.getTypeOverrides().get("collection").isExposeStatistics());
|
assertTrue(factory.getTypeOverrides().get("collection").isExposeStatistics());
|
||||||
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
assertTrue(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -496,18 +496,18 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
try {
|
try {
|
||||||
assertFalse(manager.getGlobalConfiguration().isExposeGlobalJmxStatistics());
|
assertFalse(manager.getGlobalConfiguration().isExposeGlobalJmxStatistics());
|
||||||
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
EntityRegionImpl region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Address", p, null);
|
||||||
CacheAdapter cache = region.getCacheAdapter();
|
AdvancedCache cache = region.getCache();
|
||||||
assertFalse(factory.getTypeOverrides().get("entity").isExposeStatistics());
|
assertFalse(factory.getTypeOverrides().get("entity").isExposeStatistics());
|
||||||
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Person", p, null);
|
region = (EntityRegionImpl) factory.buildEntityRegion("com.acme.Person", p, null);
|
||||||
cache = region.getCacheAdapter();
|
cache = region.getCache();
|
||||||
assertFalse(factory.getTypeOverrides().get("com.acme.Person").isExposeStatistics());
|
assertFalse(factory.getTypeOverrides().get("com.acme.Person").isExposeStatistics());
|
||||||
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
final String query = "org.hibernate.cache.internal.StandardQueryCache";
|
final String query = "org.hibernate.cache.internal.StandardQueryCache";
|
||||||
QueryResultsRegionImpl queryRegion = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
QueryResultsRegionImpl queryRegion = (QueryResultsRegionImpl) factory.buildQueryResultsRegion(query, p);
|
||||||
cache = queryRegion.getCacheAdapter();
|
cache = queryRegion.getCache();
|
||||||
assertFalse(factory.getTypeOverrides().get("query").isExposeStatistics());
|
assertFalse(factory.getTypeOverrides().get("query").isExposeStatistics());
|
||||||
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
|
@ -516,12 +516,12 @@ public class InfinispanRegionFactoryTestCase {
|
||||||
config.setFetchInMemoryState(false);
|
config.setFetchInMemoryState(false);
|
||||||
manager.defineConfiguration("timestamps", config);
|
manager.defineConfiguration("timestamps", config);
|
||||||
TimestampsRegionImpl timestampsRegion = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
TimestampsRegionImpl timestampsRegion = (TimestampsRegionImpl) factory.buildTimestampsRegion(timestamps, p);
|
||||||
cache = timestampsRegion.getCacheAdapter();
|
cache = timestampsRegion.getCache();
|
||||||
assertFalse(factory.getTypeOverrides().get("timestamps").isExposeStatistics());
|
assertFalse(factory.getTypeOverrides().get("timestamps").isExposeStatistics());
|
||||||
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
|
|
||||||
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
CollectionRegionImpl collectionRegion = (CollectionRegionImpl) factory.buildCollectionRegion("com.acme.Person.addresses", p, null);
|
||||||
cache = collectionRegion.getCacheAdapter();
|
cache = collectionRegion.getCache();
|
||||||
assertFalse(factory.getTypeOverrides().get("collection").isExposeStatistics());
|
assertFalse(factory.getTypeOverrides().get("collection").isExposeStatistics());
|
||||||
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
assertFalse(cache.getConfiguration().isExposeJmxStatistics());
|
||||||
} finally {
|
} finally {
|
||||||
|
|
|
@ -31,13 +31,12 @@ import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.collection.CollectionRegionImpl;
|
import org.hibernate.cache.infinispan.collection.CollectionRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.entity.EntityRegionImpl;
|
import org.hibernate.cache.infinispan.entity.EntityRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cfg.Configuration;
|
import org.hibernate.cfg.Configuration;
|
||||||
import org.hibernate.boot.registry.internal.StandardServiceRegistryImpl;
|
import org.hibernate.boot.registry.internal.StandardServiceRegistryImpl;
|
||||||
import org.hibernate.test.cache.infinispan.util.CacheTestUtil;
|
import org.hibernate.test.cache.infinispan.util.CacheTestUtil;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
import static org.hibernate.cache.infinispan.util.CacheHelper.withinTx;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the environment for a node.
|
* Defines the environment for a node.
|
||||||
|
@ -120,27 +119,27 @@ public class NodeEnvironment {
|
||||||
public void release() throws Exception {
|
public void release() throws Exception {
|
||||||
if ( entityRegionMap != null ) {
|
if ( entityRegionMap != null ) {
|
||||||
for ( final EntityRegionImpl region : entityRegionMap.values() ) {
|
for ( final EntityRegionImpl region : entityRegionMap.values() ) {
|
||||||
withinTx(region.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(region.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
region.getCacheAdapter().withFlags(FlagAdapter.CACHE_MODE_LOCAL).clear();
|
region.getCache().withFlags(Flag.CACHE_MODE_LOCAL).clear();
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
region.getCacheAdapter().stop();
|
region.getCache().stop();
|
||||||
}
|
}
|
||||||
entityRegionMap.clear();
|
entityRegionMap.clear();
|
||||||
}
|
}
|
||||||
if ( collectionRegionMap != null ) {
|
if ( collectionRegionMap != null ) {
|
||||||
for ( final CollectionRegionImpl collectionRegion : collectionRegionMap.values() ) {
|
for ( final CollectionRegionImpl collectionRegion : collectionRegionMap.values() ) {
|
||||||
withinTx(collectionRegion.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(collectionRegion.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
collectionRegion.getCacheAdapter().withFlags( FlagAdapter.CACHE_MODE_LOCAL ).clear();
|
collectionRegion.getCache().withFlags(Flag.CACHE_MODE_LOCAL).clear();
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
collectionRegion.getCacheAdapter().stop();
|
collectionRegion.getCache().stop();
|
||||||
}
|
}
|
||||||
collectionRegionMap.clear();
|
collectionRegionMap.clear();
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,12 +32,11 @@ import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.TimeoutException;
|
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 javax.transaction.Transaction;
|
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
|
import org.infinispan.manager.DefaultCacheManager;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Ignore;
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
|
@ -56,7 +55,6 @@ import static org.junit.Assert.fail;
|
||||||
* @author Galder Zamarreño
|
* @author Galder Zamarreño
|
||||||
* @version $Revision: $
|
* @version $Revision: $
|
||||||
*/
|
*/
|
||||||
@Ignore
|
|
||||||
public class PutFromLoadValidatorUnitTestCase {
|
public class PutFromLoadValidatorUnitTestCase {
|
||||||
private Object KEY1 = "KEY1";
|
private Object KEY1 = "KEY1";
|
||||||
|
|
||||||
|
@ -77,6 +75,7 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
DualNodeJtaTransactionManagerImpl.cleanupTransactionManagers();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testNakedPut() throws Exception {
|
public void testNakedPut() throws Exception {
|
||||||
nakedPutTest(false);
|
nakedPutTest(false);
|
||||||
|
@ -87,7 +86,9 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void nakedPutTest(boolean transactional) throws Exception {
|
private void nakedPutTest(boolean transactional) throws Exception {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(transactional ? tm : null);
|
PutFromLoadValidator testee = new PutFromLoadValidator(
|
||||||
|
new DefaultCacheManager(), transactional ? tm : null,
|
||||||
|
PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
if (transactional) {
|
if (transactional) {
|
||||||
tm.begin();
|
tm.begin();
|
||||||
}
|
}
|
||||||
|
@ -111,8 +112,8 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void registeredPutTest(boolean transactional) throws Exception {
|
private void registeredPutTest(boolean transactional) throws Exception {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(
|
PutFromLoadValidator testee = new PutFromLoadValidator(new DefaultCacheManager(),
|
||||||
transactional ? tm : null);
|
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
if (transactional) {
|
if (transactional) {
|
||||||
tm.begin();
|
tm.begin();
|
||||||
}
|
}
|
||||||
|
@ -147,8 +148,8 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void nakedPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
private void nakedPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(
|
PutFromLoadValidator testee = new PutFromLoadValidator(new DefaultCacheManager(),
|
||||||
transactional ? tm : null);
|
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
if (removeRegion) {
|
if (removeRegion) {
|
||||||
testee.invalidateRegion();
|
testee.invalidateRegion();
|
||||||
} else {
|
} else {
|
||||||
|
@ -187,8 +188,8 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void registeredPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
private void registeredPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(
|
PutFromLoadValidator testee = new PutFromLoadValidator(new DefaultCacheManager(),
|
||||||
transactional ? tm : null);
|
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
if (removeRegion) {
|
if (removeRegion) {
|
||||||
testee.invalidateRegion();
|
testee.invalidateRegion();
|
||||||
} else {
|
} else {
|
||||||
|
@ -228,8 +229,8 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void registeredPutWithInterveningRemovalTest(boolean transactional, boolean removeRegion)
|
private void registeredPutWithInterveningRemovalTest(boolean transactional, boolean removeRegion)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
PutFromLoadValidator testee = new PutFromLoadValidator(
|
PutFromLoadValidator testee = new PutFromLoadValidator(new DefaultCacheManager(),
|
||||||
transactional ? tm : null);
|
transactional ? tm : null, PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
if (transactional) {
|
if (transactional) {
|
||||||
tm.begin();
|
tm.begin();
|
||||||
}
|
}
|
||||||
|
@ -269,7 +270,7 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
|
|
||||||
private void delayedNakedPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
private void delayedNakedPutAfterRemovalTest(boolean transactional, boolean removeRegion)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
PutFromLoadValidator testee = new TestValidator(transactional ? tm : null, 100, 1000, 500, 10000);
|
PutFromLoadValidator testee = new TestValidator(transactional ? tm : null, 100);
|
||||||
if (removeRegion) {
|
if (removeRegion) {
|
||||||
testee.invalidateRegion();
|
testee.invalidateRegion();
|
||||||
} else {
|
} else {
|
||||||
|
@ -300,7 +301,9 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void multipleRegistrationtest(final boolean transactional) throws Exception {
|
private void multipleRegistrationtest(final boolean transactional) throws Exception {
|
||||||
final PutFromLoadValidator testee = new PutFromLoadValidator(transactional ? tm : null);
|
final PutFromLoadValidator testee = new PutFromLoadValidator(
|
||||||
|
new DefaultCacheManager(), 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);
|
||||||
|
@ -356,14 +359,14 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testRemovalCleanup() throws Exception {
|
public void testRemovalCleanup() throws Exception {
|
||||||
TestValidator testee = new TestValidator(null, 200, 1000, 500, 10000);
|
TestValidator testee = new TestValidator(null, 200);
|
||||||
testee.invalidateKey("KEY1");
|
testee.invalidateKey("KEY1");
|
||||||
testee.invalidateKey("KEY2");
|
testee.invalidateKey("KEY2");
|
||||||
expectRemovalLenth(2, testee, 3000l);
|
expectRemovalLenth(2, testee, 60000l);
|
||||||
assertEquals(2, testee.getRemovalQueueLength());
|
assertEquals(2, testee.getRemovalQueueLength());
|
||||||
expectRemovalLenth(2, testee, 3000l);
|
expectRemovalLenth(2, testee, 60000l);
|
||||||
assertEquals(2, testee.getRemovalQueueLength());
|
assertEquals(2, testee.getRemovalQueueLength());
|
||||||
expectRemovalLenth( 2, testee, 3000l );
|
expectRemovalLenth( 2, testee, 60000l );
|
||||||
}
|
}
|
||||||
|
|
||||||
private void expectRemovalLenth(int expectedLength, TestValidator testee, long timeout) throws InterruptedException {
|
private void expectRemovalLenth(int expectedLength, TestValidator testee, long timeout) throws InterruptedException {
|
||||||
|
@ -383,132 +386,20 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Very much a white box test of the logic for ensuring pending put registrations get cleaned up.
|
|
||||||
*
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testPendingPutCleanup() throws Exception {
|
|
||||||
TestValidator testee = new TestValidator(tm, 5000, 600, 300, 900);
|
|
||||||
|
|
||||||
// Start with a regionRemoval so we can confirm at the end that all
|
|
||||||
// registrations have been cleaned out
|
|
||||||
testee.invalidateRegion();
|
|
||||||
|
|
||||||
testee.registerPendingPut("1");
|
|
||||||
testee.registerPendingPut("2");
|
|
||||||
testee.registerPendingPut("3");
|
|
||||||
testee.registerPendingPut("4");
|
|
||||||
testee.registerPendingPut("5");
|
|
||||||
testee.registerPendingPut("6");
|
|
||||||
testee.acquirePutFromLoadLock("6");
|
|
||||||
testee.releasePutFromLoadLock("6");
|
|
||||||
testee.acquirePutFromLoadLock("2");
|
|
||||||
testee.releasePutFromLoadLock("2");
|
|
||||||
// ppq = [1,2(c),3,4,5,6(c)]
|
|
||||||
assertEquals(6, testee.getPendingPutQueueLength());
|
|
||||||
assertEquals(0, testee.getOveragePendingPutQueueLength());
|
|
||||||
|
|
||||||
// Sleep past "pendingPutRecentPeriod"
|
|
||||||
Thread.sleep(310);
|
|
||||||
testee.registerPendingPut("7");
|
|
||||||
// White box -- should have cleaned out 2 (completed) but
|
|
||||||
// not gotten to 6 (also removed)
|
|
||||||
// ppq = [1,3,4,5,6(c),7]
|
|
||||||
assertEquals(0, testee.getOveragePendingPutQueueLength());
|
|
||||||
assertEquals(6, testee.getPendingPutQueueLength());
|
|
||||||
|
|
||||||
// Sleep past "pendingPutOveragePeriod"
|
|
||||||
Thread.sleep(310);
|
|
||||||
testee.registerPendingPut("8");
|
|
||||||
// White box -- should have cleaned out 6 (completed) and
|
|
||||||
// moved 1, 3, 4 and 5 to overage queue
|
|
||||||
// oppq = [1,3,4,5] ppq = [7,8]
|
|
||||||
assertEquals(4, testee.getOveragePendingPutQueueLength());
|
|
||||||
assertEquals(2, testee.getPendingPutQueueLength());
|
|
||||||
|
|
||||||
// Sleep past "maxPendingPutDelay"
|
|
||||||
Thread.sleep(310);
|
|
||||||
testee.acquirePutFromLoadLock("3");
|
|
||||||
testee.releasePutFromLoadLock("3");
|
|
||||||
// White box -- should have cleaned out 1 (overage) and
|
|
||||||
// moved 7 to overage queue
|
|
||||||
// oppq = [3(c),4,5,7] ppq=[8]
|
|
||||||
assertEquals(4, testee.getOveragePendingPutQueueLength());
|
|
||||||
assertEquals(1, testee.getPendingPutQueueLength());
|
|
||||||
|
|
||||||
// Sleep past "maxPendingPutDelay"
|
|
||||||
Thread.sleep(310);
|
|
||||||
tm.begin();
|
|
||||||
testee.registerPendingPut("7");
|
|
||||||
Transaction tx = tm.suspend();
|
|
||||||
|
|
||||||
// White box -- should have cleaned out 3 (completed)
|
|
||||||
// and 4 (overage) and moved 8 to overage queue
|
|
||||||
// We now have 5,7,8 in overage and 7tx in pending
|
|
||||||
// oppq = [5,7,8] ppq=[7tx]
|
|
||||||
assertEquals(3, testee.getOveragePendingPutQueueLength());
|
|
||||||
assertEquals(1, testee.getPendingPutQueueLength());
|
|
||||||
|
|
||||||
// Validate that only expected items can do puts, thus indirectly
|
|
||||||
// proving the others have been cleaned out of pendingPuts map
|
|
||||||
boolean locked = testee.acquirePutFromLoadLock("1");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
// 5 was overage, so should have been cleaned
|
|
||||||
assertEquals(2, testee.getOveragePendingPutQueueLength());
|
|
||||||
locked = testee.acquirePutFromLoadLock("2");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
// 7 was overage, so should have been cleaned
|
|
||||||
assertEquals(1, testee.getOveragePendingPutQueueLength());
|
|
||||||
locked = testee.acquirePutFromLoadLock("3");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
locked = testee.acquirePutFromLoadLock("4");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
locked = testee.acquirePutFromLoadLock("5");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
locked = testee.acquirePutFromLoadLock("1");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(testee.acquirePutFromLoadLock("6"));
|
|
||||||
locked = testee.acquirePutFromLoadLock("7");
|
|
||||||
if (locked) {
|
|
||||||
testee.releasePutFromLoadLock("1");
|
|
||||||
}
|
|
||||||
assertFalse(locked);
|
|
||||||
assertTrue(testee.acquirePutFromLoadLock("8"));
|
|
||||||
testee.releasePutFromLoadLock("8");
|
|
||||||
tm.resume(tx);
|
|
||||||
assertTrue(testee.acquirePutFromLoadLock("7"));
|
|
||||||
testee.releasePutFromLoadLock("7");
|
|
||||||
}
|
|
||||||
@Test
|
@Test
|
||||||
public void testInvalidateKeyBlocksForInProgressPut() throws Exception {
|
public void testInvalidateKeyBlocksForInProgressPut() throws Exception {
|
||||||
invalidationBlocksForInProgressPutTest(true);
|
invalidationBlocksForInProgressPutTest(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testInvalidateRegionBlocksForInProgressPut() throws Exception {
|
public void testInvalidateRegionBlocksForInProgressPut() throws Exception {
|
||||||
invalidationBlocksForInProgressPutTest(false);
|
invalidationBlocksForInProgressPutTest(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void invalidationBlocksForInProgressPutTest(final boolean keyOnly) throws Exception {
|
private void invalidationBlocksForInProgressPutTest(final boolean keyOnly) throws Exception {
|
||||||
final PutFromLoadValidator testee = new PutFromLoadValidator(null);
|
final PutFromLoadValidator testee = new PutFromLoadValidator(
|
||||||
|
new DefaultCacheManager(), null,
|
||||||
|
PutFromLoadValidator.NAKED_PUT_INVALIDATION_PERIOD);
|
||||||
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");
|
||||||
|
@ -566,22 +457,9 @@ public class PutFromLoadValidatorUnitTestCase {
|
||||||
private static class TestValidator extends PutFromLoadValidator {
|
private static class TestValidator extends PutFromLoadValidator {
|
||||||
|
|
||||||
protected TestValidator(TransactionManager transactionManager,
|
protected TestValidator(TransactionManager transactionManager,
|
||||||
long nakedPutInvalidationPeriod, long pendingPutOveragePeriod,
|
long nakedPutInvalidationPeriod) {
|
||||||
long pendingPutRecentPeriod, long maxPendingPutDelay) {
|
super(new DefaultCacheManager(),
|
||||||
super(transactionManager, nakedPutInvalidationPeriod, pendingPutOveragePeriod,
|
transactionManager, nakedPutInvalidationPeriod);
|
||||||
pendingPutRecentPeriod, maxPendingPutDelay);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getOveragePendingPutQueueLength() {
|
|
||||||
// TODO Auto-generated method stub
|
|
||||||
return super.getOveragePendingPutQueueLength();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getPendingPutQueueLength() {
|
|
||||||
// TODO Auto-generated method stub
|
|
||||||
return super.getPendingPutQueueLength();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -32,6 +32,8 @@ import java.util.concurrent.TimeUnit;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.transaction.TransactionManager;
|
||||||
|
|
||||||
import junit.framework.AssertionFailedError;
|
import junit.framework.AssertionFailedError;
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.infinispan.manager.DefaultCacheManager;
|
||||||
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
||||||
import org.jboss.logging.Logger;
|
import org.jboss.logging.Logger;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
|
@ -42,7 +44,6 @@ import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
import org.hibernate.cache.infinispan.access.PutFromLoadValidator;
|
||||||
import org.hibernate.cache.infinispan.access.TransactionalAccessDelegate;
|
import org.hibernate.cache.infinispan.access.TransactionalAccessDelegate;
|
||||||
import org.hibernate.cache.infinispan.collection.CollectionRegionImpl;
|
import org.hibernate.cache.infinispan.collection.CollectionRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.util.CacheHelper;
|
|
||||||
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
|
@ -103,8 +104,8 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
localCollectionRegion = localEnvironment.getCollectionRegion( REGION_NAME, getCacheDataDescription() );
|
localCollectionRegion = localEnvironment.getCollectionRegion( REGION_NAME, getCacheDataDescription() );
|
||||||
localAccessStrategy = localCollectionRegion.buildAccessStrategy( getAccessType() );
|
localAccessStrategy = localCollectionRegion.buildAccessStrategy( getAccessType() );
|
||||||
|
|
||||||
invalidation = localCollectionRegion.getCacheAdapter().isClusteredInvalidation();
|
invalidation = Caches.isInvalidationCache(localCollectionRegion.getCache());
|
||||||
synchronous = localCollectionRegion.getCacheAdapter().isSynchronous();
|
synchronous = Caches.isSynchronousCache(localCollectionRegion.getCache());
|
||||||
|
|
||||||
// Sleep a bit to avoid concurrent FLUSH problem
|
// Sleep a bit to avoid concurrent FLUSH problem
|
||||||
avoidConcurrentFlush();
|
avoidConcurrentFlush();
|
||||||
|
@ -161,7 +162,8 @@ 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();
|
||||||
PutFromLoadValidator validator = new PutFromLoadValidator(remoteTm) {
|
PutFromLoadValidator validator = new PutFromLoadValidator(
|
||||||
|
new DefaultCacheManager(), remoteTm, 20000) {
|
||||||
@Override
|
@Override
|
||||||
public boolean acquirePutFromLoadLock(Object key) {
|
public boolean acquirePutFromLoadLock(Object key) {
|
||||||
boolean acquired = super.acquirePutFromLoadLock( key );
|
boolean acquired = super.acquirePutFromLoadLock( key );
|
||||||
|
@ -195,7 +197,7 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
Callable<Void> removeCallable = new Callable<Void>() {
|
Callable<Void> removeCallable = new Callable<Void>() {
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
removeLatch.await();
|
removeLatch.await();
|
||||||
CacheHelper.withinTx(localTm, new Callable<Void>() {
|
Caches.withinTx(localTm, new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
delegate.remove("k1");
|
delegate.remove("k1");
|
||||||
|
@ -214,7 +216,7 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
pferFuture.get();
|
pferFuture.get();
|
||||||
removeFuture.get();
|
removeFuture.get();
|
||||||
|
|
||||||
assertFalse( localCollectionRegion.getCacheAdapter().containsKey( "k1" ) );
|
assertFalse(localCollectionRegion.getCache().containsKey("k1"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -394,7 +396,7 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
// Wait for async propagation
|
// Wait for async propagation
|
||||||
sleep( 250 );
|
sleep( 250 );
|
||||||
|
|
||||||
CacheHelper.withinTx(localCollectionRegion.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(localCollectionRegion.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
if (evict)
|
if (evict)
|
||||||
|
@ -414,9 +416,9 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
|
|
||||||
final String KEY = KEY_BASE + testCount++;
|
final String KEY = KEY_BASE + testCount++;
|
||||||
|
|
||||||
assertEquals( 0, getValidKeyCount( localCollectionRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localCollectionRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
assertEquals( 0, getValidKeyCount( remoteCollectionRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteCollectionRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
@ -429,7 +431,7 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
// Wait for async propagation
|
// Wait for async propagation
|
||||||
sleep( 250 );
|
sleep( 250 );
|
||||||
|
|
||||||
CacheHelper.withinTx(localCollectionRegion.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(localCollectionRegion.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
if (evict)
|
if (evict)
|
||||||
|
@ -443,19 +445,19 @@ public abstract class AbstractCollectionRegionAccessStrategyTestCase extends Abs
|
||||||
// This should re-establish the region root node
|
// This should re-establish the region root node
|
||||||
assertNull( localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
|
||||||
assertEquals( 0, getValidKeyCount( localCollectionRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localCollectionRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// 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( KEY, System.currentTimeMillis() ) );
|
assertEquals( null, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
|
||||||
assertEquals( 0, getValidKeyCount( remoteCollectionRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteCollectionRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// Test whether the get above messes up the optimistic version
|
// Test whether the get above messes up the optimistic version
|
||||||
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
||||||
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
|
||||||
assertEquals( 1, getValidKeyCount( remoteCollectionRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 1, getValidKeyCount( remoteCollectionRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// Wait for async propagation of the putFromLoad
|
// Wait for async propagation of the putFromLoad
|
||||||
sleep( 250 );
|
sleep( 250 );
|
||||||
|
|
|
@ -27,8 +27,6 @@ import java.util.Properties;
|
||||||
|
|
||||||
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.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapterImpl;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.CollectionRegion;
|
import org.hibernate.cache.spi.CollectionRegion;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
|
@ -36,6 +34,7 @@ import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
|
||||||
import org.hibernate.test.cache.infinispan.AbstractEntityCollectionRegionTestCase;
|
import org.hibernate.test.cache.infinispan.AbstractEntityCollectionRegionTestCase;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
import static org.junit.Assert.assertNull;
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
@ -70,8 +69,8 @@ public class CollectionRegionImplTestCase extends AbstractEntityCollectionRegion
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected CacheAdapter getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
protected AdvancedCache getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
||||||
return CacheAdapterImpl.newInstance(regionFactory.getCacheManager().getCache(InfinispanRegionFactory.DEF_ENTITY_RESOURCE).getAdvancedCache());
|
return regionFactory.getCacheManager().getCache(InfinispanRegionFactory.DEF_ENTITY_RESOURCE).getAdvancedCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -29,6 +29,7 @@ import java.util.concurrent.CountDownLatch;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import junit.framework.AssertionFailedError;
|
import junit.framework.AssertionFailedError;
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
import org.infinispan.Cache;
|
import org.infinispan.Cache;
|
||||||
import org.infinispan.test.TestingUtil;
|
import org.infinispan.test.TestingUtil;
|
||||||
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
import org.infinispan.transaction.tm.BatchModeTransactionManager;
|
||||||
|
@ -39,7 +40,6 @@ import org.junit.Test;
|
||||||
|
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.entity.EntityRegionImpl;
|
import org.hibernate.cache.infinispan.entity.EntityRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.util.CacheHelper;
|
|
||||||
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
import org.hibernate.cache.internal.CacheDataDescriptionImpl;
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
|
@ -97,8 +97,8 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
localEntityRegion = localEnvironment.getEntityRegion( REGION_NAME, getCacheDataDescription() );
|
localEntityRegion = localEnvironment.getEntityRegion( REGION_NAME, getCacheDataDescription() );
|
||||||
localAccessStrategy = localEntityRegion.buildAccessStrategy( getAccessType() );
|
localAccessStrategy = localEntityRegion.buildAccessStrategy( getAccessType() );
|
||||||
|
|
||||||
invalidation = localEntityRegion.getCacheAdapter().isClusteredInvalidation();
|
invalidation = Caches.isInvalidationCache(localEntityRegion.getCache());
|
||||||
synchronous = localEntityRegion.getCacheAdapter().isSynchronous();
|
synchronous = Caches.isSynchronousCache(localEntityRegion.getCache());
|
||||||
|
|
||||||
// Sleep a bit to avoid concurrent FLUSH problem
|
// Sleep a bit to avoid concurrent FLUSH problem
|
||||||
avoidConcurrentFlush();
|
avoidConcurrentFlush();
|
||||||
|
@ -109,8 +109,8 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
remoteEntityRegion = remoteEnvironment.getEntityRegion( REGION_NAME, getCacheDataDescription() );
|
remoteEntityRegion = remoteEnvironment.getEntityRegion( REGION_NAME, getCacheDataDescription() );
|
||||||
remoteAccessStrategy = remoteEntityRegion.buildAccessStrategy( getAccessType() );
|
remoteAccessStrategy = remoteEntityRegion.buildAccessStrategy( getAccessType() );
|
||||||
|
|
||||||
waitForClusterToForm(localEntityRegion.getCacheAdapter().getCache(),
|
waitForClusterToForm(localEntityRegion.getCache(),
|
||||||
remoteEntityRegion.getCacheAdapter().getCache());
|
remoteEntityRegion.getCache());
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void waitForClusterToForm(Cache... caches) {
|
protected void waitForClusterToForm(Cache... caches) {
|
||||||
|
@ -534,8 +534,8 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
|
|
||||||
private void evictOrRemoveTest(final boolean evict) throws Exception {
|
private void evictOrRemoveTest(final boolean evict) throws Exception {
|
||||||
final String KEY = KEY_BASE + testCount++;
|
final String KEY = KEY_BASE + testCount++;
|
||||||
assertEquals( 0, getValidKeyCount( localEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localEntityRegion.getCache().keySet() ) );
|
||||||
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
@ -545,7 +545,7 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
||||||
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
|
||||||
CacheHelper.withinTx(localEntityRegion.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(localEntityRegion.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
if ( evict )
|
if ( evict )
|
||||||
|
@ -556,15 +556,15 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
assertEquals(null, localAccessStrategy.get(KEY, System.currentTimeMillis()));
|
assertEquals(null, localAccessStrategy.get(KEY, System.currentTimeMillis()));
|
||||||
assertEquals( 0, getValidKeyCount( localEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localEntityRegion.getCache().keySet() ) );
|
||||||
assertEquals( null, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertEquals( null, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCache().keySet() ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
private void evictOrRemoveAllTest(final boolean evict) throws Exception {
|
private void evictOrRemoveAllTest(final boolean evict) throws Exception {
|
||||||
final String KEY = KEY_BASE + testCount++;
|
final String KEY = KEY_BASE + testCount++;
|
||||||
assertEquals( 0, getValidKeyCount( localEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localEntityRegion.getCache().keySet() ) );
|
||||||
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCache().keySet() ) );
|
||||||
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "local is clean", localAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertNull( "remote is clean", remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
|
|
||||||
|
@ -580,7 +580,7 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
// Wait for async propagation
|
// Wait for async propagation
|
||||||
sleep( 250 );
|
sleep( 250 );
|
||||||
|
|
||||||
CacheHelper.withinTx(localEntityRegion.getTransactionManager(), new Callable<Void>() {
|
Caches.withinTx(localEntityRegion.getTransactionManager(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
if (evict) {
|
if (evict) {
|
||||||
|
@ -595,17 +595,17 @@ public abstract class AbstractEntityRegionAccessStrategyTestCase extends Abstrac
|
||||||
|
|
||||||
// This should re-establish the region root node in the optimistic case
|
// This should re-establish the region root node in the optimistic case
|
||||||
assertNull(localAccessStrategy.get(KEY, System.currentTimeMillis()));
|
assertNull(localAccessStrategy.get(KEY, System.currentTimeMillis()));
|
||||||
assertEquals( 0, getValidKeyCount( localEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( localEntityRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// 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( KEY, System.currentTimeMillis() ) );
|
assertEquals( null, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 0, getValidKeyCount( remoteEntityRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// Test whether the get above messes up the optimistic version
|
// Test whether the get above messes up the optimistic version
|
||||||
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
remoteAccessStrategy.putFromLoad( KEY, VALUE1, System.currentTimeMillis(), new Integer( 1 ) );
|
||||||
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
assertEquals( VALUE1, remoteAccessStrategy.get( KEY, System.currentTimeMillis() ) );
|
||||||
assertEquals( 1, getValidKeyCount( remoteEntityRegion.getCacheAdapter().keySet() ) );
|
assertEquals( 1, getValidKeyCount( remoteEntityRegion.getCache().keySet() ) );
|
||||||
|
|
||||||
// Wait for async propagation
|
// Wait for async propagation
|
||||||
sleep( 250 );
|
sleep( 250 );
|
||||||
|
|
|
@ -27,14 +27,13 @@ import java.util.Properties;
|
||||||
|
|
||||||
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.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapterImpl;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.EntityRegion;
|
import org.hibernate.cache.spi.EntityRegion;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
import org.hibernate.cache.spi.access.AccessType;
|
import org.hibernate.cache.spi.access.AccessType;
|
||||||
import org.hibernate.test.cache.infinispan.AbstractEntityCollectionRegionTestCase;
|
import org.hibernate.test.cache.infinispan.AbstractEntityCollectionRegionTestCase;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
|
|
||||||
import static org.junit.Assert.assertNull;
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
@ -81,8 +80,9 @@ public class EntityRegionImplTestCase extends AbstractEntityCollectionRegionTest
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected CacheAdapter getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
protected AdvancedCache getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
||||||
return CacheAdapterImpl.newInstance(regionFactory.getCacheManager().getCache(InfinispanRegionFactory.DEF_ENTITY_RESOURCE).getAdvancedCache());
|
return regionFactory.getCacheManager().getCache(
|
||||||
|
InfinispanRegionFactory.DEF_ENTITY_RESOURCE).getAdvancedCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
/*
|
/*
|
||||||
* Hibernate, Relational Persistence for Idiomatic Java
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
*
|
*
|
||||||
* Copyright (c) 2007, Red Hat, Inc. and/or it's affiliates or third-party contributors as
|
* Copyright (c) 2012, Red Hat, Inc or third-party contributors as
|
||||||
* indicated by the @author tags or express copyright attribution
|
* indicated by the @author tags or express copyright attribution
|
||||||
* statements applied by the authors. All third-party contributions are
|
* statements applied by the authors. All third-party contributions are
|
||||||
* distributed under license by Red Hat, Inc. and/or it's affiliates.
|
* distributed under license by Red Hat Middleware LLC.
|
||||||
*
|
*
|
||||||
* This copyrighted material is made available to anyone wishing to use, modify,
|
* This copyrighted material is made available to anyone wishing to use, modify,
|
||||||
* copy, or redistribute it subject to the terms and conditions of the GNU
|
* copy, or redistribute it subject to the terms and conditions of the GNU
|
||||||
|
@ -21,37 +21,41 @@
|
||||||
* 51 Franklin Street, Fifth Floor
|
* 51 Franklin Street, Fifth Floor
|
||||||
* Boston, MA 02110-1301 USA
|
* Boston, MA 02110-1301 USA
|
||||||
*/
|
*/
|
||||||
package org.hibernate.cache.infinispan.util;
|
|
||||||
|
|
||||||
import java.util.concurrent.Callable;
|
package org.hibernate.test.cache.infinispan.functional;
|
||||||
import javax.transaction.Status;
|
import javax.persistence.Entity;
|
||||||
import javax.transaction.TransactionManager;
|
import javax.persistence.GeneratedValue;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.NamedQueries;
|
||||||
|
import javax.persistence.NamedQuery;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper for dealing with Infinisan cache instances.
|
|
||||||
*
|
|
||||||
* @author Galder Zamarreño
|
* @author Galder Zamarreño
|
||||||
* @since 3.5
|
|
||||||
*/
|
*/
|
||||||
public class CacheHelper {
|
@NamedQueries({@NamedQuery(name=Age.QUERY, query = "SELECT a FROM Age a")})
|
||||||
|
@Entity
|
||||||
|
public class Age {
|
||||||
|
|
||||||
/**
|
public static final String QUERY = "Age.findAll";
|
||||||
* Disallow external instantiation of CacheHelper.
|
|
||||||
*/
|
@Id
|
||||||
private CacheHelper() {
|
@GeneratedValue
|
||||||
|
private Integer id;
|
||||||
|
private Integer age;
|
||||||
|
|
||||||
|
public Integer getId() {
|
||||||
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static <T> T withinTx(TransactionManager tm, Callable<T> c) throws Exception {
|
public void setId(Integer id) {
|
||||||
tm.begin();
|
this.id = id;
|
||||||
try {
|
|
||||||
return c.call();
|
|
||||||
} catch (Exception e) {
|
|
||||||
tm.setRollbackOnly();
|
|
||||||
throw e;
|
|
||||||
} finally {
|
|
||||||
if (tm.getStatus() == Status.STATUS_ACTIVE) tm.commit();
|
|
||||||
else tm.rollback();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Integer getAge() {
|
||||||
|
return age;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAge(Integer age) {
|
||||||
|
this.age = age;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -29,6 +29,8 @@ import java.util.concurrent.CountDownLatch;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import junit.framework.AssertionFailedError;
|
import junit.framework.AssertionFailedError;
|
||||||
|
import org.hibernate.cache.infinispan.util.Caches;
|
||||||
|
import org.infinispan.AdvancedCache;
|
||||||
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;
|
||||||
|
@ -38,9 +40,6 @@ import org.jboss.logging.Logger;
|
||||||
|
|
||||||
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapterImpl;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheHelper;
|
|
||||||
import org.hibernate.cache.internal.StandardQueryCache;
|
import org.hibernate.cache.internal.StandardQueryCache;
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.GeneralDataRegion;
|
import org.hibernate.cache.spi.GeneralDataRegion;
|
||||||
|
@ -80,7 +79,7 @@ public class QueryRegionImplTestCase extends AbstractGeneralDataRegionTestCase {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void regionPut(final GeneralDataRegion region) throws Exception {
|
protected void regionPut(final GeneralDataRegion region) throws Exception {
|
||||||
CacheHelper.withinTx(BatchModeTransactionManager.getInstance(), new Callable<Void>() {
|
Caches.withinTx(BatchModeTransactionManager.getInstance(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
region.put(KEY, VALUE1);
|
region.put(KEY, VALUE1);
|
||||||
|
@ -91,7 +90,7 @@ public class QueryRegionImplTestCase extends AbstractGeneralDataRegionTestCase {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void regionEvict(final GeneralDataRegion region) throws Exception {
|
protected void regionEvict(final GeneralDataRegion region) throws Exception {
|
||||||
CacheHelper.withinTx(BatchModeTransactionManager.getInstance(), new Callable<Void>() {
|
Caches.withinTx(BatchModeTransactionManager.getInstance(), new Callable<Void>() {
|
||||||
@Override
|
@Override
|
||||||
public Void call() throws Exception {
|
public Void call() throws Exception {
|
||||||
region.evict(KEY);
|
region.evict(KEY);
|
||||||
|
@ -101,8 +100,8 @@ public class QueryRegionImplTestCase extends AbstractGeneralDataRegionTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected CacheAdapter getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
protected AdvancedCache getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
||||||
return CacheAdapterImpl.newInstance(regionFactory.getCacheManager().getCache( "local-query" ).getAdvancedCache());
|
return regionFactory.getCacheManager().getCache( "local-query" ).getAdvancedCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -230,7 +229,7 @@ public class QueryRegionImplTestCase extends AbstractGeneralDataRegionTestCase {
|
||||||
assertEquals( VALUE1, region.get( KEY ) );
|
assertEquals( VALUE1, region.get( KEY ) );
|
||||||
|
|
||||||
// final Fqn rootFqn = getRegionFqn(getStandardRegionName(REGION_PREFIX), REGION_PREFIX);
|
// final Fqn rootFqn = getRegionFqn(getStandardRegionName(REGION_PREFIX), REGION_PREFIX);
|
||||||
final CacheAdapter jbc = getInfinispanCache( regionFactory );
|
final AdvancedCache jbc = getInfinispanCache(regionFactory);
|
||||||
|
|
||||||
final CountDownLatch blockerLatch = new CountDownLatch( 1 );
|
final CountDownLatch blockerLatch = new CountDownLatch( 1 );
|
||||||
final CountDownLatch writerLatch = new CountDownLatch( 1 );
|
final CountDownLatch writerLatch = new CountDownLatch( 1 );
|
||||||
|
|
|
@ -0,0 +1,343 @@
|
||||||
|
package org.hibernate.test.cache.infinispan.stress;
|
||||||
|
|
||||||
|
import org.hibernate.Query;
|
||||||
|
import org.hibernate.Session;
|
||||||
|
import org.hibernate.SessionFactory;
|
||||||
|
import org.hibernate.boot.registry.internal.StandardServiceRegistryImpl;
|
||||||
|
import org.hibernate.cfg.Configuration;
|
||||||
|
import org.hibernate.cfg.Environment;
|
||||||
|
import org.hibernate.mapping.Collection;
|
||||||
|
import org.hibernate.mapping.PersistentClass;
|
||||||
|
import org.hibernate.test.cache.infinispan.functional.Age;
|
||||||
|
import org.hibernate.testing.ServiceRegistryBuilder;
|
||||||
|
import org.infinispan.util.logging.Log;
|
||||||
|
import org.infinispan.util.logging.LogFactory;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Ignore;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import javax.transaction.TransactionManager;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
import java.util.concurrent.CyclicBarrier;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
import static org.infinispan.test.TestingUtil.withTx;
|
||||||
|
import static org.junit.Assert.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A stress test for putFromLoad operations
|
||||||
|
*
|
||||||
|
* @author Galder Zamarreño
|
||||||
|
* @since 4.1
|
||||||
|
*/
|
||||||
|
@Ignore
|
||||||
|
public class PutFromLoadStressTestCase {
|
||||||
|
|
||||||
|
static final Log log = LogFactory.getLog(PutFromLoadStressTestCase.class);
|
||||||
|
static final boolean isTrace = log.isTraceEnabled();
|
||||||
|
static final int NUM_THREADS = 100;
|
||||||
|
static final int WARMUP_TIME_SECS = 10;
|
||||||
|
static final long RUNNING_TIME_SECS = Integer.getInteger("time", 60);
|
||||||
|
static final long LAUNCH_INTERVAL_MILLIS = 10;
|
||||||
|
|
||||||
|
static final int NUM_INSTANCES = 5000;
|
||||||
|
|
||||||
|
static SessionFactory sessionFactory;
|
||||||
|
static TransactionManager tm;
|
||||||
|
|
||||||
|
final AtomicBoolean run = new AtomicBoolean(true);
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void beforeClass() {
|
||||||
|
Configuration cfg = new Configuration();
|
||||||
|
cfg.setProperty(Environment.USE_SECOND_LEVEL_CACHE, "true");
|
||||||
|
cfg.setProperty(Environment.USE_QUERY_CACHE, "true");
|
||||||
|
// TODO: Tweak to have a fully local region factory (no transport, cache mode = local, no marshalling, ...etc)
|
||||||
|
cfg.setProperty(Environment.CACHE_REGION_FACTORY,
|
||||||
|
"org.hibernate.cache.infinispan.InfinispanRegionFactory");
|
||||||
|
cfg.setProperty(Environment.JTA_PLATFORM,
|
||||||
|
"org.hibernate.service.jta.platform.internal.JBossStandAloneJtaPlatform");
|
||||||
|
|
||||||
|
// Force minimal puts off to simplify stressing putFromLoad logic
|
||||||
|
cfg.setProperty(Environment.USE_MINIMAL_PUTS, "false");
|
||||||
|
|
||||||
|
// Mappings
|
||||||
|
configureMappings(cfg);
|
||||||
|
|
||||||
|
// // Database settings
|
||||||
|
// cfg.setProperty(Environment.DRIVER, "org.postgresql.Driver");
|
||||||
|
// cfg.setProperty(Environment.URL, "jdbc:postgresql://localhost/hibernate");
|
||||||
|
// cfg.setProperty(Environment.DIALECT, "org.hibernate.dialect.PostgreSQL82Dialect");
|
||||||
|
// cfg.setProperty(Environment.USER, "hbadmin");
|
||||||
|
// cfg.setProperty(Environment.PASS, "hbadmin");
|
||||||
|
|
||||||
|
// Create database schema in each run
|
||||||
|
cfg.setProperty(Environment.HBM2DDL_AUTO, "create-drop");
|
||||||
|
|
||||||
|
StandardServiceRegistryImpl registry =
|
||||||
|
ServiceRegistryBuilder.buildServiceRegistry(cfg.getProperties());
|
||||||
|
sessionFactory = cfg.buildSessionFactory(registry);
|
||||||
|
|
||||||
|
tm = com.arjuna.ats.jta.TransactionManager.transactionManager();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void configureMappings(Configuration cfg) {
|
||||||
|
String[] mappings = {
|
||||||
|
"cache/infinispan/functional/Item.hbm.xml",
|
||||||
|
"cache/infinispan/functional/Customer.hbm.xml",
|
||||||
|
"cache/infinispan/functional/Contact.hbm.xml"};
|
||||||
|
for (String mapping : mappings)
|
||||||
|
cfg.addResource("org/hibernate/test/" + mapping);
|
||||||
|
|
||||||
|
Class<?>[] annotatedClasses = getAnnotatedClasses();
|
||||||
|
if ( annotatedClasses != null ) {
|
||||||
|
for ( Class<?> annotatedClass : annotatedClasses ) {
|
||||||
|
cfg.addAnnotatedClass( annotatedClass );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg.buildMappings();
|
||||||
|
Iterator it = cfg.getClassMappings();
|
||||||
|
String cacheStrategy = "transactional";
|
||||||
|
while (it.hasNext()) {
|
||||||
|
PersistentClass clazz = (PersistentClass) it.next();
|
||||||
|
if (!clazz.isInherited()) {
|
||||||
|
cfg.setCacheConcurrencyStrategy(clazz.getEntityName(), cacheStrategy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it = cfg.getCollectionMappings();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Collection coll = (Collection) it.next();
|
||||||
|
cfg.setCollectionCacheConcurrencyStrategy( coll.getRole(), cacheStrategy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void afterClass() {
|
||||||
|
sessionFactory.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Class<Object>[] getAnnotatedClasses() {
|
||||||
|
return new Class[] {Age.class};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testQueryPerformance() throws Exception {
|
||||||
|
store();
|
||||||
|
// doTest(true);
|
||||||
|
// run.set(true); // Reset run
|
||||||
|
doTest(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void store() throws Exception {
|
||||||
|
for (int i = 0; i < NUM_INSTANCES; i++) {
|
||||||
|
final Age age = new Age();
|
||||||
|
age.setAge(i);
|
||||||
|
withTx(tm, new Callable<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
s.persist(age);
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void doTest(boolean warmup) throws Exception {
|
||||||
|
ExecutorService executor = Executors.newFixedThreadPool(NUM_THREADS);
|
||||||
|
try {
|
||||||
|
CyclicBarrier barrier = new CyclicBarrier(NUM_THREADS + 1);
|
||||||
|
List<Future<String>> futures = new ArrayList<Future<String>>(NUM_THREADS);
|
||||||
|
for (int i = 0; i < NUM_THREADS; i++) {
|
||||||
|
Future<String> future = executor.submit(
|
||||||
|
new SelectQueryRunner(barrier, warmup, i + 1));
|
||||||
|
futures.add(future);
|
||||||
|
Thread.sleep(LAUNCH_INTERVAL_MILLIS);
|
||||||
|
}
|
||||||
|
barrier.await(); // wait for all threads to be ready
|
||||||
|
|
||||||
|
long timeout = warmup ? WARMUP_TIME_SECS : RUNNING_TIME_SECS;
|
||||||
|
TimeUnit unit = TimeUnit.SECONDS;
|
||||||
|
|
||||||
|
Thread.sleep(unit.toMillis(timeout)); // Wait for the duration of the test
|
||||||
|
run.set(false); // Instruct tests to stop doing work
|
||||||
|
barrier.await(2, TimeUnit.MINUTES); // wait for all threads to finish
|
||||||
|
|
||||||
|
log.infof("[%s] All threads finished, check for exceptions", title(warmup));
|
||||||
|
for (Future<String> future : futures) {
|
||||||
|
String opsPerMS = future.get();
|
||||||
|
if (!warmup)
|
||||||
|
log.infof("[%s] Operations/ms: %s", title(warmup), opsPerMS);
|
||||||
|
}
|
||||||
|
log.infof("[%s] All future gets checked", title(warmup));
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.errorf(e, "Error in one of the execution threads during %s", title(warmup));
|
||||||
|
throw e;
|
||||||
|
} finally {
|
||||||
|
executor.shutdownNow();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String title(boolean warmup) {
|
||||||
|
return warmup ? "warmup" : "stress";
|
||||||
|
}
|
||||||
|
|
||||||
|
public class SelectQueryRunner implements Callable<String> {
|
||||||
|
|
||||||
|
final CyclicBarrier barrier;
|
||||||
|
final boolean warmup;
|
||||||
|
final Integer customerId;
|
||||||
|
|
||||||
|
public SelectQueryRunner(CyclicBarrier barrier, boolean warmup, Integer customerId) {
|
||||||
|
this.barrier = barrier;
|
||||||
|
this.warmup = warmup;
|
||||||
|
this.customerId = customerId;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String call() throws Exception {
|
||||||
|
try {
|
||||||
|
if (isTrace)
|
||||||
|
log.tracef("[%s] Wait for all executions paths to be ready to perform calls", title(warmup));
|
||||||
|
barrier.await();
|
||||||
|
|
||||||
|
long start = System.nanoTime();
|
||||||
|
int runs = 0;
|
||||||
|
if (isTrace)
|
||||||
|
log.tracef("[%s] Start time: %d", title(warmup), start);
|
||||||
|
|
||||||
|
// while (USE_TIME && PutFromLoadStressTestCase.this.run.get()) {
|
||||||
|
// if (runs % 100000 == 0)
|
||||||
|
// log.infof("[%s] Query run # %d", title(warmup), runs);
|
||||||
|
//
|
||||||
|
//// Customer customer = query();
|
||||||
|
//// deleteCached(customer);
|
||||||
|
|
||||||
|
queryItems();
|
||||||
|
// deleteCachedItems();
|
||||||
|
//
|
||||||
|
// runs++;
|
||||||
|
// }
|
||||||
|
long end = System.nanoTime();
|
||||||
|
long duration = end - start;
|
||||||
|
if (isTrace)
|
||||||
|
log.tracef("[%s] End time: %d, duration: %d, runs: %d",
|
||||||
|
title(warmup), start, duration, runs);
|
||||||
|
|
||||||
|
return opsPerMS(duration, runs);
|
||||||
|
} finally {
|
||||||
|
if (isTrace)
|
||||||
|
log.tracef("[%s] Wait for all execution paths to finish", title(warmup));
|
||||||
|
|
||||||
|
barrier.await();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void deleteCachedItems() throws Exception {
|
||||||
|
withTx(tm, new Callable<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
sessionFactory.getCache().evictEntityRegion(Age.class);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void queryItems() throws Exception {
|
||||||
|
withTx(tm, new Callable<Void>() {
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
Session s = sessionFactory.getCurrentSession();
|
||||||
|
Query query = s.getNamedQuery(Age.QUERY).setCacheable(true);
|
||||||
|
// Query query = s.createQuery("from Age").setCacheable(true);
|
||||||
|
List<Age> result = (List<Age>) query.list();
|
||||||
|
assertFalse(result.isEmpty());
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// private void deleteCachedItems() throws Exception {
|
||||||
|
// withTx(tm, new Callable<Void>() {
|
||||||
|
// @Override
|
||||||
|
// public Void call() throws Exception {
|
||||||
|
// sessionFactory.getCache().evictEntityRegion(Item.class);
|
||||||
|
// return null;
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// private void queryItems() throws Exception {
|
||||||
|
// withTx(tm, new Callable<Void>() {
|
||||||
|
// @Override
|
||||||
|
// public Void call() throws Exception {
|
||||||
|
// Session s = sessionFactory.getCurrentSession();
|
||||||
|
// Query query = s.createQuery("from Item").setCacheable(true);
|
||||||
|
// List<Item> result = (List<Item>) query.list();
|
||||||
|
// assertFalse(result.isEmpty());
|
||||||
|
// return null;
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
|
||||||
|
// private Customer query() throws Exception {
|
||||||
|
// return withTx(tm, new Callable<Customer>() {
|
||||||
|
// @Override
|
||||||
|
// public Customer call() throws Exception {
|
||||||
|
// Session s = sessionFactory.getCurrentSession();
|
||||||
|
// Customer customer = (Customer) s.load(Customer.class, customerId);
|
||||||
|
// assertNotNull(customer);
|
||||||
|
// Set<Contact> contacts = customer.getContacts();
|
||||||
|
// Contact contact = contacts.iterator().next();
|
||||||
|
// assertNotNull(contact);
|
||||||
|
// assertEquals("private contact", contact.getName());
|
||||||
|
//
|
||||||
|
//// Contact found = contacts.isEmpty() ? null : contacts.iterator().next();
|
||||||
|
//// Set<Contact> contacts = found.getContacts();
|
||||||
|
//// assertTrue(contacts + " not empty", contacts.isEmpty());
|
||||||
|
////
|
||||||
|
//// if (found != null && found.hashCode() == System.nanoTime()) {
|
||||||
|
//// System.out.print(" ");
|
||||||
|
//// } else if (found == null) {
|
||||||
|
//// throw new IllegalStateException("Contact cannot be null");
|
||||||
|
//// }
|
||||||
|
// return customer;
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
|
||||||
|
// private void deleteCached(final Customer customer) throws Exception {
|
||||||
|
// withTx(tm, new Callable<Void>() {
|
||||||
|
// @Override
|
||||||
|
// public Void call() throws Exception {
|
||||||
|
// sessionFactory.getCache().evictEntity(Customer.class, customer.getId());
|
||||||
|
// return null; // TODO: Customise this generated block
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
|
||||||
|
private String opsPerMS(long nanos, int ops) {
|
||||||
|
long totalMillis = TimeUnit.NANOSECONDS.toMillis(nanos);
|
||||||
|
if (totalMillis > 0)
|
||||||
|
return ops / totalMillis + " ops/ms";
|
||||||
|
else
|
||||||
|
return "NAN ops/ms";
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,626 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.test.cache.infinispan.stress;
|
||||||
|
|
||||||
|
import org.hibernate.Query;
|
||||||
|
import org.hibernate.Session;
|
||||||
|
import org.hibernate.SessionFactory;
|
||||||
|
import org.hibernate.boot.registry.internal.StandardServiceRegistryImpl;
|
||||||
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
|
import org.hibernate.cfg.Configuration;
|
||||||
|
import org.hibernate.cfg.Environment;
|
||||||
|
import org.hibernate.mapping.Collection;
|
||||||
|
import org.hibernate.mapping.PersistentClass;
|
||||||
|
import org.hibernate.test.cache.infinispan.stress.entities.Address;
|
||||||
|
import org.hibernate.test.cache.infinispan.stress.entities.Family;
|
||||||
|
import org.hibernate.test.cache.infinispan.stress.entities.Person;
|
||||||
|
import org.hibernate.testing.ServiceRegistryBuilder;
|
||||||
|
import org.infinispan.util.concurrent.ConcurrentHashSet;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Ignore;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import javax.transaction.TransactionManager;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Queue;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.concurrent.*;
|
||||||
|
|
||||||
|
import static org.infinispan.test.TestingUtil.withTx;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stress test for second level cache.
|
||||||
|
*
|
||||||
|
* TODO Various:
|
||||||
|
* - Switch to a JDBC connection pool to avoid too many connections created
|
||||||
|
* (as well as consuming memory, it's expensive to create)
|
||||||
|
* - Use barrier associated execution tasks at the beginning and end to track
|
||||||
|
* down start/end times for runs.
|
||||||
|
*
|
||||||
|
* @author Galder Zamarreño
|
||||||
|
* @since 4.1
|
||||||
|
*/
|
||||||
|
@Ignore
|
||||||
|
public class SecondLevelCacheStressTestCase {
|
||||||
|
|
||||||
|
static final int NUM_THREADS = 10;
|
||||||
|
static final long WARMUP_TIME = TimeUnit.SECONDS.toNanos(Integer.getInteger("warmup-time", 1) * 5);
|
||||||
|
static final long RUNNING_TIME = TimeUnit.SECONDS.toNanos(Integer.getInteger("time", 1) * 60);
|
||||||
|
static final boolean PROFILE = Boolean.getBoolean("profile");
|
||||||
|
static final boolean ALLOCATION = Boolean.getBoolean("allocation");
|
||||||
|
static final int RUN_COUNT_LIMIT = Integer.getInteger("count", 1000); // max number of runs per operation
|
||||||
|
static final Random RANDOM = new Random(12345);
|
||||||
|
|
||||||
|
String provider;
|
||||||
|
ConcurrentHashSet<Integer> updatedIds;
|
||||||
|
Queue<Integer> removeIds;
|
||||||
|
SessionFactory sessionFactory;
|
||||||
|
TransactionManager tm;
|
||||||
|
volatile int numEntities;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void beforeClass() {
|
||||||
|
provider = getProvider();
|
||||||
|
|
||||||
|
updatedIds = new ConcurrentHashSet<Integer>();
|
||||||
|
removeIds = new ConcurrentLinkedQueue<Integer>();
|
||||||
|
|
||||||
|
Configuration cfg = new Configuration();
|
||||||
|
cfg.setProperty(Environment.USE_SECOND_LEVEL_CACHE, "true");
|
||||||
|
cfg.setProperty(Environment.USE_QUERY_CACHE, "true");
|
||||||
|
configureCache(cfg);
|
||||||
|
|
||||||
|
// Mappings
|
||||||
|
configureMappings(cfg);
|
||||||
|
|
||||||
|
// Database settings
|
||||||
|
cfg.setProperty(Environment.DRIVER, "com.mysql.jdbc.Driver");
|
||||||
|
cfg.setProperty(Environment.URL, "jdbc:mysql://localhost:3306/hibernate");
|
||||||
|
cfg.setProperty(Environment.DIALECT, "org.hibernate.dialect.MySQL5InnoDBDialect");
|
||||||
|
cfg.setProperty(Environment.USER, "root");
|
||||||
|
cfg.setProperty(Environment.PASS, "password");
|
||||||
|
|
||||||
|
// Create database schema in each run
|
||||||
|
cfg.setProperty(Environment.HBM2DDL_AUTO, "create-drop");
|
||||||
|
|
||||||
|
StandardServiceRegistryImpl registry =
|
||||||
|
ServiceRegistryBuilder.buildServiceRegistry(cfg.getProperties());
|
||||||
|
sessionFactory = cfg.buildSessionFactory(registry);
|
||||||
|
|
||||||
|
tm = com.arjuna.ats.jta.TransactionManager.transactionManager();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected String getProvider() {
|
||||||
|
return "infinispan";
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void configureCache(Configuration cfg) {
|
||||||
|
cfg.setProperty(Environment.CACHE_REGION_FACTORY,
|
||||||
|
"org.hibernate.cache.infinispan.InfinispanRegionFactory");
|
||||||
|
cfg.setProperty(Environment.JTA_PLATFORM,
|
||||||
|
"org.hibernate.service.jta.platform.internal.JBossStandAloneJtaPlatform");
|
||||||
|
cfg.setProperty(InfinispanRegionFactory.INFINISPAN_CONFIG_RESOURCE_PROP,
|
||||||
|
"stress-local-infinispan.xml");
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void afterClass() {
|
||||||
|
sessionFactory.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEntityLifecycle() throws InterruptedException {
|
||||||
|
if (!PROFILE) {
|
||||||
|
System.out.printf("[provider=%s] Warming up\n", provider);
|
||||||
|
doEntityLifecycle(true);
|
||||||
|
|
||||||
|
// Recreate session factory cleaning everything
|
||||||
|
afterClass();
|
||||||
|
beforeClass();
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.printf("[provider=%s] Testing...\n", provider);
|
||||||
|
doEntityLifecycle(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
void doEntityLifecycle(boolean isWarmup) {
|
||||||
|
long runningTimeout = isWarmup ? WARMUP_TIME : RUNNING_TIME;
|
||||||
|
TotalStats insertPerf = runEntityInsert(runningTimeout);
|
||||||
|
numEntities = countEntities().intValue();
|
||||||
|
printResult(isWarmup, "[provider=%s] Inserts/s %10.2f (%d entities)\n",
|
||||||
|
provider, insertPerf.getOpsPerSec("INSERT"), numEntities);
|
||||||
|
|
||||||
|
TotalStats updatePerf = runEntityUpdate(runningTimeout);
|
||||||
|
List<Integer> updateIdsSeq = new ArrayList<Integer>(updatedIds);
|
||||||
|
printResult(isWarmup, "[provider=%s] Updates/s %10.2f (%d updates)\n",
|
||||||
|
provider, updatePerf.getOpsPerSec("UPDATE"), updateIdsSeq.size());
|
||||||
|
|
||||||
|
TotalStats findUpdatedPerf =
|
||||||
|
runEntityFindUpdated(runningTimeout, updateIdsSeq);
|
||||||
|
printResult(isWarmup, "[provider=%s] Updated finds/s %10.2f\n",
|
||||||
|
provider, findUpdatedPerf.getOpsPerSec("FIND_UPDATED"));
|
||||||
|
|
||||||
|
TotalStats findQueryPerf = runEntityFindQuery(runningTimeout, isWarmup);
|
||||||
|
printResult(isWarmup, "[provider=%s] Query finds/s %10.2f\n",
|
||||||
|
provider, findQueryPerf.getOpsPerSec("FIND_QUERY"));
|
||||||
|
|
||||||
|
TotalStats findRandomPerf = runEntityFindRandom(runningTimeout);
|
||||||
|
printResult(isWarmup, "[provider=%s] Random finds/s %10.2f\n",
|
||||||
|
provider, findRandomPerf.getOpsPerSec("FIND_RANDOM"));
|
||||||
|
|
||||||
|
// Get all entity ids
|
||||||
|
List<Integer> entityIds = new ArrayList<Integer>();
|
||||||
|
for (int i = 1; i <= numEntities; i++) entityIds.add(i);
|
||||||
|
|
||||||
|
// Shuffle them
|
||||||
|
Collections.shuffle(entityIds);
|
||||||
|
|
||||||
|
// Add them to the queue delete consumption
|
||||||
|
removeIds.addAll(entityIds);
|
||||||
|
|
||||||
|
TotalStats deletePerf = runEntityDelete(runningTimeout);
|
||||||
|
printResult(isWarmup, "[provider=%s] Deletes/s %10.2f\n",
|
||||||
|
provider, deletePerf.getOpsPerSec("DELETE"));
|
||||||
|
|
||||||
|
// TODO Print 2LC statistics...
|
||||||
|
}
|
||||||
|
|
||||||
|
static void printResult(boolean isWarmup, String format, Object... args) {
|
||||||
|
if (!isWarmup) System.out.printf(format, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
Long countEntities() {
|
||||||
|
try {
|
||||||
|
return withTx(tm, new Callable<Long>() {
|
||||||
|
@Override
|
||||||
|
public Long call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
Query query = s.createQuery("select count(*) from Family");
|
||||||
|
Object result = query.list().get(0);
|
||||||
|
s.close();
|
||||||
|
return (Long) result;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TotalStats runEntityInsert(long runningTimeout) {
|
||||||
|
return runSingleWork(runningTimeout, "insert", insertOperation());
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runEntityUpdate(long runningTimeout) {
|
||||||
|
return runSingleWork(runningTimeout, "update", updateOperation());
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runEntityFindUpdated(long runningTimeout,
|
||||||
|
List<Integer> updatedIdsSeq) {
|
||||||
|
return runSingleWork(runningTimeout, "find-updated", findUpdatedOperation(updatedIdsSeq));
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runEntityFindQuery(long runningTimeout, boolean warmup) {
|
||||||
|
return runSingleWork(runningTimeout, "find-query", findQueryOperation(warmup));
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runEntityFindRandom(long runningTimeout) {
|
||||||
|
return runSingleWork(runningTimeout, "find-random", findRandomOperation());
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runEntityDelete(long runningTimeout) {
|
||||||
|
return runSingleWork(runningTimeout, "remove", deleteOperation());
|
||||||
|
}
|
||||||
|
|
||||||
|
TotalStats runSingleWork(long runningTimeout, final String name, Operation op) {
|
||||||
|
final TotalStats perf = new TotalStats();
|
||||||
|
|
||||||
|
ExecutorService exec = Executors.newFixedThreadPool(
|
||||||
|
NUM_THREADS, new ThreadFactory() {
|
||||||
|
volatile int i = 0;
|
||||||
|
@Override
|
||||||
|
public Thread newThread(Runnable r) {
|
||||||
|
return new Thread(new ThreadGroup(name),
|
||||||
|
r, "worker-" + name + "-" + i++);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
List<Future<Void>> futures = new ArrayList<Future<Void>>(NUM_THREADS);
|
||||||
|
CyclicBarrier barrier = new CyclicBarrier(NUM_THREADS + 1);
|
||||||
|
|
||||||
|
for (int i = 0; i < NUM_THREADS; i++)
|
||||||
|
futures.add(exec.submit(
|
||||||
|
new WorkerThread(runningTimeout, perf, op, barrier)));
|
||||||
|
|
||||||
|
try {
|
||||||
|
barrier.await(); // wait for all threads to be ready
|
||||||
|
barrier.await(); // wait for all threads to finish
|
||||||
|
|
||||||
|
// Now check whether anything went wrong...
|
||||||
|
for (Future<Void> future : futures) future.get();
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return perf;
|
||||||
|
} finally {
|
||||||
|
exec.shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
<T> T captureThrowables(Callable<T> task) throws Exception {
|
||||||
|
try {
|
||||||
|
return task.call();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
t.printStackTrace();
|
||||||
|
if (t instanceof Exception)
|
||||||
|
throw (Exception) t;
|
||||||
|
else
|
||||||
|
throw new RuntimeException(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Operation insertOperation() {
|
||||||
|
return new Operation("INSERT") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
return withTx(tm, new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
|
||||||
|
String name = "Zamarreño-" + run;
|
||||||
|
Family family = new Family(name);
|
||||||
|
s.persist(family);
|
||||||
|
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
Operation updateOperation() {
|
||||||
|
return new Operation("UPDATE") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
return withTx(tm, new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
|
||||||
|
// Update random entity that has been inserted
|
||||||
|
int id = RANDOM.nextInt(numEntities) + 1;
|
||||||
|
Family family = (Family) s.load(Family.class, id);
|
||||||
|
String newSecondName = "Arrizabalaga-" + run;
|
||||||
|
family.setSecondName(newSecondName);
|
||||||
|
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
// Cache updated entities for later read
|
||||||
|
updatedIds.add(id);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
Operation findUpdatedOperation(final List<Integer> updatedIdsSeq) {
|
||||||
|
return new Operation("FIND_UPDATED") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
|
||||||
|
int id = updatedIdsSeq.get(RANDOM.nextInt(
|
||||||
|
updatedIdsSeq.size()));
|
||||||
|
Family family = (Family) s.load(Family.class, id);
|
||||||
|
String secondName = family.getSecondName();
|
||||||
|
assertNotNull(secondName);
|
||||||
|
assertTrue("Second name not expected: " + secondName,
|
||||||
|
secondName.startsWith("Arrizabalaga"));
|
||||||
|
|
||||||
|
s.close();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private Operation findQueryOperation(final boolean isWarmup) {
|
||||||
|
return new Operation("FIND_QUERY") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
|
||||||
|
Query query = s.createQuery("from Family")
|
||||||
|
.setCacheable(true);
|
||||||
|
int maxResults = isWarmup ? 10 : 100;
|
||||||
|
query.setMaxResults(maxResults);
|
||||||
|
List<Family> result = (List<Family>) query.list();
|
||||||
|
assertEquals(maxResults, result.size());
|
||||||
|
|
||||||
|
s.close();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private Operation findRandomOperation() {
|
||||||
|
return new Operation("FIND_RANDOM") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
|
||||||
|
int id = RANDOM.nextInt(numEntities) + 1;
|
||||||
|
Family family = (Family) s.load(Family.class, id);
|
||||||
|
String familyName = family.getName();
|
||||||
|
// Skip ñ check in order to avoid issues...
|
||||||
|
assertTrue("Unexpected family: " + familyName ,
|
||||||
|
familyName.startsWith("Zamarre"));
|
||||||
|
|
||||||
|
s.close();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private Operation deleteOperation() {
|
||||||
|
return new Operation("DELETE") {
|
||||||
|
@Override
|
||||||
|
boolean call(final int run) throws Exception {
|
||||||
|
return captureThrowables(new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
return withTx(tm, new Callable<Boolean>() {
|
||||||
|
@Override
|
||||||
|
public Boolean call() throws Exception {
|
||||||
|
Session s = sessionFactory.openSession();
|
||||||
|
s.getTransaction().begin();
|
||||||
|
|
||||||
|
// Get each id and remove it
|
||||||
|
int id = removeIds.poll();
|
||||||
|
Family family = (Family) s.load(Family.class, id);
|
||||||
|
String familyName = family.getName();
|
||||||
|
// Skip ñ check in order to avoid issues...
|
||||||
|
assertTrue("Unexpected family: " + familyName ,
|
||||||
|
familyName.startsWith("Zamarre"));
|
||||||
|
s.delete(family);
|
||||||
|
|
||||||
|
s.getTransaction().commit();
|
||||||
|
s.close();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Class<Object>[] getAnnotatedClasses() {
|
||||||
|
return new Class[] {Family.class, Person.class, Address.class};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void configureMappings(Configuration cfg) {
|
||||||
|
Class<?>[] annotatedClasses = getAnnotatedClasses();
|
||||||
|
if ( annotatedClasses != null ) {
|
||||||
|
for ( Class<?> annotatedClass : annotatedClasses ) {
|
||||||
|
cfg.addAnnotatedClass( annotatedClass );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg.buildMappings();
|
||||||
|
Iterator it = cfg.getClassMappings();
|
||||||
|
String cacheStrategy = "transactional";
|
||||||
|
while (it.hasNext()) {
|
||||||
|
PersistentClass clazz = (PersistentClass) it.next();
|
||||||
|
if (!clazz.isInherited()) {
|
||||||
|
cfg.setCacheConcurrencyStrategy(clazz.getEntityName(), cacheStrategy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it = cfg.getCollectionMappings();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Collection coll = (Collection) it.next();
|
||||||
|
cfg.setCollectionCacheConcurrencyStrategy(coll.getRole(), cacheStrategy);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static abstract class Operation {
|
||||||
|
final String name;
|
||||||
|
|
||||||
|
Operation(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract boolean call(int run) throws Exception;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private class WorkerThread implements Callable<Void> {
|
||||||
|
private final long runningTimeout;
|
||||||
|
private final TotalStats perf;
|
||||||
|
private final Operation op;
|
||||||
|
private final CyclicBarrier barrier;
|
||||||
|
|
||||||
|
public WorkerThread(long runningTimeout, TotalStats perf,
|
||||||
|
Operation op, CyclicBarrier barrier) {
|
||||||
|
this.runningTimeout = runningTimeout;
|
||||||
|
this.perf = perf;
|
||||||
|
this.op = op;
|
||||||
|
this.barrier = barrier;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Void call() throws Exception {
|
||||||
|
// TODO: Extend barrier to capture start time
|
||||||
|
barrier.await();
|
||||||
|
try {
|
||||||
|
long startNanos = System.nanoTime();
|
||||||
|
long endNanos = startNanos + runningTimeout;
|
||||||
|
int runs = 0;
|
||||||
|
long missCount = 0;
|
||||||
|
while (callOperation(endNanos, runs)) {
|
||||||
|
boolean hit = op.call(runs);
|
||||||
|
if (!hit) missCount++;
|
||||||
|
runs++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Extend barrier to capture end time
|
||||||
|
perf.addStats(op.name, runs,
|
||||||
|
System.nanoTime() - startNanos, missCount);
|
||||||
|
} finally {
|
||||||
|
barrier.await();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean callOperation(long endNanos, int runs) {
|
||||||
|
if (ALLOCATION) {
|
||||||
|
return runs < RUN_COUNT_LIMIT;
|
||||||
|
} else {
|
||||||
|
return (runs & 0x400) != 0 || System.nanoTime() < endNanos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TotalStats {
|
||||||
|
private ConcurrentHashMap<String, OpStats> statsMap =
|
||||||
|
new ConcurrentHashMap<String, OpStats>();
|
||||||
|
|
||||||
|
public void addStats(String opName, long opCount,
|
||||||
|
long runningTime, long missCount) {
|
||||||
|
OpStats s = new OpStats(opName, opCount, runningTime, missCount);
|
||||||
|
OpStats old = statsMap.putIfAbsent(opName, s);
|
||||||
|
boolean replaced = old == null;
|
||||||
|
while (!replaced) {
|
||||||
|
old = statsMap.get(opName);
|
||||||
|
s = new OpStats(old, opCount, runningTime, missCount);
|
||||||
|
replaced = statsMap.replace(opName, old, s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public double getOpsPerSec(String opName) {
|
||||||
|
OpStats s = statsMap.get(opName);
|
||||||
|
if (s == null) return 0;
|
||||||
|
return s.opCount * 1000000000. / s.runningTime * s.threadCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double getTotalOpsPerSec() {
|
||||||
|
long totalOpCount = 0;
|
||||||
|
long totalRunningTime = 0;
|
||||||
|
long totalThreadCount = 0;
|
||||||
|
for (Map.Entry<String, OpStats> e : statsMap.entrySet()) {
|
||||||
|
OpStats s = e.getValue();
|
||||||
|
totalOpCount += s.opCount;
|
||||||
|
totalRunningTime += s.runningTime;
|
||||||
|
totalThreadCount += s.threadCount;
|
||||||
|
}
|
||||||
|
return totalOpCount * 1000000000. / totalRunningTime * totalThreadCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double getHitRatio(String opName) {
|
||||||
|
OpStats s = statsMap.get(opName);
|
||||||
|
if (s == null) return 0;
|
||||||
|
return 1 - 1. * s.missCount / s.opCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double getTotalHitRatio() {
|
||||||
|
long totalOpCount = 0;
|
||||||
|
long totalMissCount = 0;
|
||||||
|
for (Map.Entry<String, OpStats> e : statsMap.entrySet()) {
|
||||||
|
OpStats s = e.getValue();
|
||||||
|
totalOpCount += s.opCount;
|
||||||
|
totalMissCount += s.missCount;
|
||||||
|
}
|
||||||
|
return 1 - 1. * totalMissCount / totalOpCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class OpStats {
|
||||||
|
public final String opName;
|
||||||
|
public final int threadCount;
|
||||||
|
public final long opCount;
|
||||||
|
public final long runningTime;
|
||||||
|
public final long missCount;
|
||||||
|
|
||||||
|
private OpStats(String opName, long opCount,
|
||||||
|
long runningTime, long missCount) {
|
||||||
|
this.opName = opName;
|
||||||
|
this.threadCount = 1;
|
||||||
|
this.opCount = opCount;
|
||||||
|
this.runningTime = runningTime;
|
||||||
|
this.missCount = missCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
private OpStats(OpStats base, long opCount,
|
||||||
|
long runningTime, long missCount) {
|
||||||
|
this.opName = base.opName;
|
||||||
|
this.threadCount = base.threadCount + 1;
|
||||||
|
this.opCount = base.opCount + opCount;
|
||||||
|
this.runningTime = base.runningTime + runningTime;
|
||||||
|
this.missCount = base.missCount + missCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
216
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Address.java
vendored
Normal file
216
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Address.java
vendored
Normal file
|
@ -0,0 +1,216 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.test.cache.infinispan.stress.entities;
|
||||||
|
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.GeneratedValue;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.OneToMany;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
public final class Address {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue
|
||||||
|
private int id;
|
||||||
|
private int streetNumber;
|
||||||
|
private String streetName;
|
||||||
|
private String cityName;
|
||||||
|
private String countryName;
|
||||||
|
private String zipCode;
|
||||||
|
@OneToMany
|
||||||
|
private Set<Person> inhabitants;
|
||||||
|
private int version;
|
||||||
|
|
||||||
|
public Address(int streetNumber, String streetName, String cityName, String countryName) {
|
||||||
|
this.streetNumber = streetNumber;
|
||||||
|
this.streetName = streetName;
|
||||||
|
this.cityName = cityName;
|
||||||
|
this.countryName = countryName;
|
||||||
|
this.zipCode = null;
|
||||||
|
this.inhabitants = new HashSet<Person>();
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Address() {
|
||||||
|
this.streetNumber = 0;
|
||||||
|
this.streetName = null;
|
||||||
|
this.cityName = null;
|
||||||
|
this.countryName = null;
|
||||||
|
this.zipCode = null;
|
||||||
|
this.inhabitants = new HashSet<Person>();
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getStreetNumber() {
|
||||||
|
return streetNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStreetName() {
|
||||||
|
return streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCityName() {
|
||||||
|
return cityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCountryName() {
|
||||||
|
return countryName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getZipCode() {
|
||||||
|
return zipCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setZipCode(String zipCode) {
|
||||||
|
this.zipCode = zipCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Set<Person> getInhabitants() {
|
||||||
|
return inhabitants;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean addInhabitant(Person inhabitant) {
|
||||||
|
boolean done = false;
|
||||||
|
if (inhabitants.add(inhabitant)) {
|
||||||
|
inhabitant.setAddress(this);
|
||||||
|
done = true;
|
||||||
|
}
|
||||||
|
return done;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean remInhabitant(Person inhabitant) {
|
||||||
|
boolean done = false;
|
||||||
|
if (inhabitants.remove(inhabitant)) {
|
||||||
|
inhabitant.setAddress(null);
|
||||||
|
done = true;
|
||||||
|
}
|
||||||
|
return done;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(int id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void removeAllInhabitants() {
|
||||||
|
// inhabitants relation is not CASCADED, we must delete the relation on other side by ourselves
|
||||||
|
for (Iterator<Person> iterator = inhabitants.iterator(); iterator.hasNext(); ) {
|
||||||
|
Person p = iterator.next();
|
||||||
|
p.setAddress(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setStreetNumber(int streetNumber) {
|
||||||
|
this.streetNumber = streetNumber;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setStreetName(String streetName) {
|
||||||
|
this.streetName = streetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setCityName(String cityName) {
|
||||||
|
this.cityName = cityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setCountryName(String countryName) {
|
||||||
|
this.countryName = countryName;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setInhabitants(Set<Person> inhabitants) {
|
||||||
|
if (inhabitants == null) {
|
||||||
|
this.inhabitants = new HashSet<Person>();
|
||||||
|
} else {
|
||||||
|
this.inhabitants = inhabitants;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setVersion(Integer version) {
|
||||||
|
this.version = version;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (this == o) return true;
|
||||||
|
if (o == null || getClass() != o.getClass()) return false;
|
||||||
|
|
||||||
|
Address address = (Address) o;
|
||||||
|
|
||||||
|
if (id != address.id) return false;
|
||||||
|
if (streetNumber != address.streetNumber) return false;
|
||||||
|
if (version != address.version) return false;
|
||||||
|
if (cityName != null ? !cityName.equals(address.cityName) : address.cityName != null)
|
||||||
|
return false;
|
||||||
|
if (countryName != null ? !countryName.equals(address.countryName) : address.countryName != null)
|
||||||
|
return false;
|
||||||
|
if (inhabitants != null ? !inhabitants.equals(address.inhabitants) : address.inhabitants != null)
|
||||||
|
return false;
|
||||||
|
if (streetName != null ? !streetName.equals(address.streetName) : address.streetName != null)
|
||||||
|
return false;
|
||||||
|
if (zipCode != null ? !zipCode.equals(address.zipCode) : address.zipCode != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int result = streetNumber;
|
||||||
|
result = 31 * result + (streetName != null ? streetName.hashCode() : 0);
|
||||||
|
result = 31 * result + (cityName != null ? cityName.hashCode() : 0);
|
||||||
|
result = 31 * result + (countryName != null ? countryName.hashCode() : 0);
|
||||||
|
result = 31 * result + (zipCode != null ? zipCode.hashCode() : 0);
|
||||||
|
result = 31 * result + (inhabitants != null ? inhabitants.hashCode() : 0);
|
||||||
|
result = 31 * result + id;
|
||||||
|
result = 31 * result + version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Address{" +
|
||||||
|
"cityName='" + cityName + '\'' +
|
||||||
|
", streetNumber=" + streetNumber +
|
||||||
|
", streetName='" + streetName + '\'' +
|
||||||
|
", countryName='" + countryName + '\'' +
|
||||||
|
", zipCode='" + zipCode + '\'' +
|
||||||
|
", inhabitants=" + inhabitants +
|
||||||
|
", id=" + id +
|
||||||
|
", version=" + version +
|
||||||
|
'}';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
149
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Family.java
vendored
Normal file
149
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Family.java
vendored
Normal file
|
@ -0,0 +1,149 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.test.cache.infinispan.stress.entities;
|
||||||
|
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.GeneratedValue;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.OneToMany;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
public final class Family {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue
|
||||||
|
private int id;
|
||||||
|
private String name;
|
||||||
|
private String secondName;
|
||||||
|
@OneToMany
|
||||||
|
private Set<Person> members;
|
||||||
|
private int version;
|
||||||
|
|
||||||
|
public Family(String name) {
|
||||||
|
this.name = name;
|
||||||
|
this.secondName = null;
|
||||||
|
this.members = new HashSet<Person>();
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Family() {
|
||||||
|
this.name = null;
|
||||||
|
this.secondName = null;
|
||||||
|
this.members = new HashSet<Person>();
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Set<Person> getMembers() {
|
||||||
|
return members;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSecondName() {
|
||||||
|
return secondName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSecondName(String secondName) {
|
||||||
|
this.secondName = secondName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(int id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMembers(Set<Person> members) {
|
||||||
|
if (members == null) {
|
||||||
|
this.members = new HashSet<Person>();
|
||||||
|
} else {
|
||||||
|
this.members = members;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setVersion(Integer version) {
|
||||||
|
this.version = version;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean addMember(Person member) {
|
||||||
|
return members.add(member);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (this == o) return true;
|
||||||
|
if (o == null || getClass() != o.getClass()) return false;
|
||||||
|
|
||||||
|
Family family = (Family) o;
|
||||||
|
|
||||||
|
if (id != family.id) return false;
|
||||||
|
if (version != family.version) return false;
|
||||||
|
if (members != null ? !members.equals(family.members) : family.members != null)
|
||||||
|
return false;
|
||||||
|
if (name != null ? !name.equals(family.name) : family.name != null)
|
||||||
|
return false;
|
||||||
|
if (secondName != null ? !secondName.equals(family.secondName) : family.secondName != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int result = name != null ? name.hashCode() : 0;
|
||||||
|
result = 31 * result + (secondName != null ? secondName.hashCode() : 0);
|
||||||
|
result = 31 * result + (members != null ? members.hashCode() : 0);
|
||||||
|
result = 31 * result + id;
|
||||||
|
result = 31 * result + version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Family{" +
|
||||||
|
"id=" + id +
|
||||||
|
", name='" + name + '\'' +
|
||||||
|
", secondName='" + secondName + '\'' +
|
||||||
|
", members=" + members +
|
||||||
|
", version=" + version +
|
||||||
|
'}';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
178
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Person.java
vendored
Normal file
178
hibernate-infinispan/src/test/java/org/hibernate/test/cache/infinispan/stress/entities/Person.java
vendored
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2012 Red Hat Inc. and/or its affiliates and other
|
||||||
|
* contributors as indicated by the @author tags. All rights reserved.
|
||||||
|
* See the copyright.txt in the distribution for a full listing of
|
||||||
|
* individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.hibernate.test.cache.infinispan.stress.entities;
|
||||||
|
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.GeneratedValue;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.ManyToOne;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
public class Person {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue
|
||||||
|
private int id;
|
||||||
|
private String firstName;
|
||||||
|
@ManyToOne
|
||||||
|
private Family family;
|
||||||
|
private Date birthDate;
|
||||||
|
@ManyToOne
|
||||||
|
private Address address;
|
||||||
|
private boolean checked;
|
||||||
|
private int version;
|
||||||
|
|
||||||
|
public Person(String firstName, Family family) {
|
||||||
|
this.firstName = firstName;
|
||||||
|
this.family = family;
|
||||||
|
this.birthDate = null;
|
||||||
|
this.address = null;
|
||||||
|
this.checked = false;
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
this.family.addMember(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Person() {
|
||||||
|
this.firstName = null;
|
||||||
|
this.family = null;
|
||||||
|
this.birthDate = null;
|
||||||
|
this.address = null;
|
||||||
|
this.checked = false;
|
||||||
|
this.id = 0;
|
||||||
|
this.version = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFirstName() {
|
||||||
|
return firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Family getFamily() {
|
||||||
|
return family;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getBirthDate() {
|
||||||
|
return birthDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBirthDate(Date birthDate) {
|
||||||
|
this.birthDate = birthDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Address getAddress() {
|
||||||
|
return address;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAddress(Address address) {
|
||||||
|
// To skip Hibernate BUG with access.PROPERTY : the rest should be done in DAO
|
||||||
|
// this.address = address;
|
||||||
|
// Hibernate BUG : if we update the relation on 2 sides
|
||||||
|
if (this.address != address) {
|
||||||
|
if (this.address != null) this.address.remInhabitant(this);
|
||||||
|
this.address = address;
|
||||||
|
if (this.address != null) this.address.addInhabitant(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isChecked() {
|
||||||
|
return checked;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setChecked(boolean checked) {
|
||||||
|
this.checked = checked;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(int id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setFirstName(String firstName) {
|
||||||
|
this.firstName = firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setFamily(Family family) {
|
||||||
|
this.family = family;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setVersion(Integer version) {
|
||||||
|
this.version = version;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (this == o) return true;
|
||||||
|
if (o == null || getClass() != o.getClass()) return false;
|
||||||
|
|
||||||
|
Person person = (Person) o;
|
||||||
|
|
||||||
|
if (checked != person.checked) return false;
|
||||||
|
if (id != person.id) return false;
|
||||||
|
if (version != person.version) return false;
|
||||||
|
if (address != null ? !address.equals(person.address) : person.address != null)
|
||||||
|
return false;
|
||||||
|
if (birthDate != null ? !birthDate.equals(person.birthDate) : person.birthDate != null)
|
||||||
|
return false;
|
||||||
|
if (family != null ? !family.equals(person.family) : person.family != null)
|
||||||
|
return false;
|
||||||
|
if (firstName != null ? !firstName.equals(person.firstName) : person.firstName != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int result = firstName != null ? firstName.hashCode() : 0;
|
||||||
|
result = 31 * result + (family != null ? family.hashCode() : 0);
|
||||||
|
result = 31 * result + (birthDate != null ? birthDate.hashCode() : 0);
|
||||||
|
result = 31 * result + (address != null ? address.hashCode() : 0);
|
||||||
|
result = 31 * result + (checked ? 1 : 0);
|
||||||
|
result = 31 * result + id;
|
||||||
|
result = 31 * result + version;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Person{" +
|
||||||
|
"address=" + address +
|
||||||
|
", firstName='" + firstName + '\'' +
|
||||||
|
", family=" + family +
|
||||||
|
", birthDate=" + birthDate +
|
||||||
|
", checked=" + checked +
|
||||||
|
", id=" + id +
|
||||||
|
", version=" + version +
|
||||||
|
'}';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -26,6 +26,7 @@ package org.hibernate.test.cache.infinispan.timestamp;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
import org.infinispan.AdvancedCache;
|
import org.infinispan.AdvancedCache;
|
||||||
|
import org.infinispan.context.Flag;
|
||||||
import org.infinispan.notifications.Listener;
|
import org.infinispan.notifications.Listener;
|
||||||
import org.infinispan.notifications.cachelistener.annotation.CacheEntryActivated;
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryActivated;
|
||||||
import org.infinispan.notifications.cachelistener.annotation.CacheEntryCreated;
|
import org.infinispan.notifications.cachelistener.annotation.CacheEntryCreated;
|
||||||
|
@ -42,9 +43,6 @@ import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
||||||
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
import org.hibernate.cache.infinispan.InfinispanRegionFactory;
|
||||||
import org.hibernate.cache.infinispan.impl.ClassLoaderAwareCache;
|
import org.hibernate.cache.infinispan.impl.ClassLoaderAwareCache;
|
||||||
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
import org.hibernate.cache.infinispan.timestamp.TimestampsRegionImpl;
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapter;
|
|
||||||
import org.hibernate.cache.infinispan.util.CacheAdapterImpl;
|
|
||||||
import org.hibernate.cache.infinispan.util.FlagAdapter;
|
|
||||||
import org.hibernate.cache.spi.CacheDataDescription;
|
import org.hibernate.cache.spi.CacheDataDescription;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
import org.hibernate.cache.spi.UpdateTimestampsCache;
|
import org.hibernate.cache.spi.UpdateTimestampsCache;
|
||||||
|
@ -75,8 +73,8 @@ public class TimestampsRegionImplTestCase extends AbstractGeneralDataRegionTestC
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected CacheAdapter getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
protected AdvancedCache getInfinispanCache(InfinispanRegionFactory regionFactory) {
|
||||||
return CacheAdapterImpl.newInstance(regionFactory.getCacheManager().getCache("timestamps").getAdvancedCache());
|
return regionFactory.getCacheManager().getCache("timestamps").getAdvancedCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testClearTimestampsRegionInIsolated() throws Exception {
|
public void testClearTimestampsRegionInIsolated() throws Exception {
|
||||||
|
@ -108,7 +106,7 @@ public class TimestampsRegionImplTestCase extends AbstractGeneralDataRegionTestC
|
||||||
|
|
||||||
Account acct = new Account();
|
Account acct = new Account();
|
||||||
acct.setAccountHolder(new AccountHolder());
|
acct.setAccountHolder(new AccountHolder());
|
||||||
region.getCacheAdapter().withFlags(FlagAdapter.FORCE_SYNCHRONOUS).put(acct, "boo");
|
region.getCache().withFlags(Flag.FORCE_SYNCHRONOUS).put(acct, "boo");
|
||||||
|
|
||||||
// region.put(acct, "boo");
|
// region.put(acct, "boo");
|
||||||
//
|
//
|
||||||
|
|
|
@ -137,6 +137,7 @@ public class XaTransactionImpl implements Transaction {
|
||||||
if (synchronizations != null) {
|
if (synchronizations != null) {
|
||||||
for (int i = 0; i < synchronizations.size(); i++) {
|
for (int i = 0; i < synchronizations.size(); i++) {
|
||||||
Synchronization s = (Synchronization) synchronizations.get(i);
|
Synchronization s = (Synchronization) synchronizations.get(i);
|
||||||
|
if (s != null)
|
||||||
s.afterCompletion(status);
|
s.afterCompletion(status);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,4 +9,4 @@ log4j.logger.org.hibernate.test=info
|
||||||
log4j.logger.org.hibernate.cache=info
|
log4j.logger.org.hibernate.cache=info
|
||||||
|
|
||||||
# SQL Logging - HHH-6833
|
# SQL Logging - HHH-6833
|
||||||
log4j.logger.org.hibernate.SQL=debug
|
log4j.logger.org.hibernate.SQL=warn
|
|
@ -0,0 +1,47 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
|
||||||
|
<!-- Infinispan configuration based on the AS7 standalone, single node, set up -->
|
||||||
|
<infinispan xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xmlns="urn:infinispan:config:5.1"
|
||||||
|
xsi:schemaLocation="urn:infinispan:config:5.1 http://www.infinispan.org/schemas/infinispan-config-5.1.xsd">
|
||||||
|
<global>
|
||||||
|
<!-- No JGroups transport -->
|
||||||
|
</global>
|
||||||
|
|
||||||
|
<default>
|
||||||
|
<!-- Used to register JMX statistics in any available MBean server -->
|
||||||
|
<jmxStatistics enabled="false"/>
|
||||||
|
</default>
|
||||||
|
|
||||||
|
<!-- Default configuration is appropriate for entity/collection caching. -->
|
||||||
|
<namedCache name="entity">
|
||||||
|
<locking isolationLevel="READ_COMMITTED" concurrencyLevel="1000"
|
||||||
|
lockAcquisitionTimeout="15000" useLockStriping="false"/>
|
||||||
|
<eviction maxEntries="140000" strategy="LRU"/>
|
||||||
|
<expiration maxIdle="1200000" wakeUpInterval="60000"/>
|
||||||
|
<!-- No lazy deserialization or store as binary for local caches -->
|
||||||
|
<transaction transactionMode="TRANSACTIONAL" autoCommit="false"
|
||||||
|
lockingMode="OPTIMISTIC" useSynchronization="true">
|
||||||
|
<recovery enabled="false"/>
|
||||||
|
</transaction>
|
||||||
|
</namedCache>
|
||||||
|
|
||||||
|
<!-- A config appropriate for query caching. Does not replicate queries. -->
|
||||||
|
<namedCache name="local-query">
|
||||||
|
<locking isolationLevel="READ_COMMITTED" concurrencyLevel="1000"
|
||||||
|
lockAcquisitionTimeout="15000" useLockStriping="false"/>
|
||||||
|
<eviction maxEntries="140000" strategy="LRU"/>
|
||||||
|
<expiration maxIdle="1200000" wakeUpInterval="60000"/>
|
||||||
|
<transaction transactionMode="NON_TRANSACTIONAL" autoCommit="false" />
|
||||||
|
</namedCache>
|
||||||
|
|
||||||
|
<!-- Optimized for timestamp caching. -->
|
||||||
|
<namedCache name="timestamps">
|
||||||
|
<locking isolationLevel="READ_COMMITTED" concurrencyLevel="1000"
|
||||||
|
lockAcquisitionTimeout="15000" useLockStriping="false"/>
|
||||||
|
<eviction strategy="NONE"/>
|
||||||
|
<expiration wakeUpInterval="0"/>
|
||||||
|
<transaction transactionMode="NON_TRANSACTIONAL"/>
|
||||||
|
</namedCache>
|
||||||
|
|
||||||
|
</infinispan>
|
Loading…
Reference in New Issue