HHH-10418 : Fix inconsistency between SessionFactoryImpl and CacheImpl
This commit is contained in:
parent
317e0b0ece
commit
25bf9b643a
|
@ -8,6 +8,9 @@ package org.hibernate.internal;
|
||||||
|
|
||||||
import org.hibernate.HibernateException;
|
import org.hibernate.HibernateException;
|
||||||
import org.hibernate.boot.spi.SessionFactoryOptions;
|
import org.hibernate.boot.spi.SessionFactoryOptions;
|
||||||
|
import org.hibernate.cache.spi.CollectionRegion;
|
||||||
|
import org.hibernate.cache.spi.EntityRegion;
|
||||||
|
import org.hibernate.cache.spi.NaturalIdRegion;
|
||||||
import org.hibernate.cache.spi.QueryCache;
|
import org.hibernate.cache.spi.QueryCache;
|
||||||
import org.hibernate.cache.spi.Region;
|
import org.hibernate.cache.spi.Region;
|
||||||
import org.hibernate.cache.spi.RegionFactory;
|
import org.hibernate.cache.spi.RegionFactory;
|
||||||
|
@ -39,7 +42,10 @@ public class CacheImpl implements CacheImplementor {
|
||||||
private final transient RegionFactory regionFactory;
|
private final transient RegionFactory regionFactory;
|
||||||
private final transient UpdateTimestampsCache updateTimestampsCache;
|
private final transient UpdateTimestampsCache updateTimestampsCache;
|
||||||
private final transient ConcurrentMap<String, QueryCache> queryCaches;
|
private final transient ConcurrentMap<String, QueryCache> queryCaches;
|
||||||
private final transient ConcurrentMap<String, Region> allCacheRegions = new ConcurrentHashMap<String, Region>();
|
private final transient ConcurrentMap<String, EntityRegion> entityRegionMap = new ConcurrentHashMap<String, EntityRegion>();
|
||||||
|
private final transient ConcurrentMap<String, CollectionRegion> collectionRegionMap = new ConcurrentHashMap<String, CollectionRegion>();
|
||||||
|
private final transient ConcurrentMap<String, NaturalIdRegion> naturalIdRegionMap = new ConcurrentHashMap<String, NaturalIdRegion>();
|
||||||
|
private final transient ConcurrentMap<String, Region> otherRegionMap = new ConcurrentHashMap<String, Region>();
|
||||||
|
|
||||||
public CacheImpl(SessionFactoryImplementor sessionFactory) {
|
public CacheImpl(SessionFactoryImplementor sessionFactory) {
|
||||||
this.sessionFactory = sessionFactory;
|
this.sessionFactory = sessionFactory;
|
||||||
|
@ -56,8 +62,6 @@ public class CacheImpl implements CacheImplementor {
|
||||||
queryCache = settings.getQueryCacheFactory()
|
queryCache = settings.getQueryCacheFactory()
|
||||||
.getQueryCache( null, updateTimestampsCache, settings, sessionFactory.getProperties() );
|
.getQueryCache( null, updateTimestampsCache, settings, sessionFactory.getProperties() );
|
||||||
queryCaches = new ConcurrentHashMap<String, QueryCache>();
|
queryCaches = new ConcurrentHashMap<String, QueryCache>();
|
||||||
allCacheRegions.put( updateTimestampsCache.getRegion().getName(), updateTimestampsCache.getRegion() );
|
|
||||||
allCacheRegions.put( queryCache.getRegion().getName(), queryCache.getRegion() );
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
updateTimestampsCache = null;
|
updateTimestampsCache = null;
|
||||||
|
@ -227,7 +231,7 @@ public class CacheImpl implements CacheImplementor {
|
||||||
}
|
}
|
||||||
if ( sessionFactory.getSessionFactoryOptions().isQueryCacheEnabled() ) {
|
if ( sessionFactory.getSessionFactoryOptions().isQueryCacheEnabled() ) {
|
||||||
QueryCache namedQueryCache = queryCaches.get( regionName );
|
QueryCache namedQueryCache = queryCaches.get( regionName );
|
||||||
// TODO : cleanup entries in queryCaches + allCacheRegions ?
|
// TODO : cleanup entries in queryCaches ?
|
||||||
if ( namedQueryCache != null ) {
|
if ( namedQueryCache != null ) {
|
||||||
if ( LOG.isDebugEnabled() ) {
|
if ( LOG.isDebugEnabled() ) {
|
||||||
LOG.debugf( "Evicting query cache, region: %s", regionName );
|
LOG.debugf( "Evicting query cache, region: %s", regionName );
|
||||||
|
@ -283,7 +287,7 @@ public class CacheImpl implements CacheImplementor {
|
||||||
|
|
||||||
QueryCache currentQueryCache = queryCaches.get( regionName );
|
QueryCache currentQueryCache = queryCaches.get( regionName );
|
||||||
if ( currentQueryCache == null ) {
|
if ( currentQueryCache == null ) {
|
||||||
synchronized (allCacheRegions) {
|
synchronized (queryCaches) {
|
||||||
currentQueryCache = queryCaches.get( regionName );
|
currentQueryCache = queryCaches.get( regionName );
|
||||||
if ( currentQueryCache == null ) {
|
if ( currentQueryCache == null ) {
|
||||||
currentQueryCache = settings.getQueryCacheFactory()
|
currentQueryCache = settings.getQueryCacheFactory()
|
||||||
|
@ -294,7 +298,6 @@ public class CacheImpl implements CacheImplementor {
|
||||||
sessionFactory.getProperties()
|
sessionFactory.getProperties()
|
||||||
);
|
);
|
||||||
queryCaches.put( regionName, currentQueryCache );
|
queryCaches.put( regionName, currentQueryCache );
|
||||||
allCacheRegions.put( currentQueryCache.getRegion().getName(), currentQueryCache.getRegion() );
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return currentQueryCache;
|
return currentQueryCache;
|
||||||
|
@ -306,7 +309,19 @@ public class CacheImpl implements CacheImplementor {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addCacheRegion(String name, Region region) {
|
public void addCacheRegion(String name, Region region) {
|
||||||
allCacheRegions.put( name, region );
|
if ( EntityRegion.class.isInstance( region ) ) {
|
||||||
|
entityRegionMap.put( name, (EntityRegion) region );
|
||||||
|
}
|
||||||
|
else if ( CollectionRegion.class.isInstance( region ) ) {
|
||||||
|
collectionRegionMap.put( name, (CollectionRegion) region );
|
||||||
|
}
|
||||||
|
else if ( NaturalIdRegion.class.isInstance( region ) ) {
|
||||||
|
naturalIdRegionMap.put( name, (NaturalIdRegion) region );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// in case an application uses this method for some other type of Region
|
||||||
|
otherRegionMap.put( name, region );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -323,18 +338,46 @@ public class CacheImpl implements CacheImplementor {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Region getSecondLevelCacheRegion(String regionName) {
|
public Region getSecondLevelCacheRegion(String regionName) {
|
||||||
return allCacheRegions.get( regionName );
|
// If there is an EntityRegion and CollectionRegion with the same name,
|
||||||
|
// then the EntityRegion should be returned.
|
||||||
|
Region region = entityRegionMap.get(regionName);
|
||||||
|
if ( region != null ) {
|
||||||
|
return region;
|
||||||
|
}
|
||||||
|
region = collectionRegionMap.get( regionName );
|
||||||
|
if ( region != null ) {
|
||||||
|
return region;
|
||||||
|
}
|
||||||
|
// Check for another type of Region of name regionName
|
||||||
|
return getNonEntityNonCollectionRegions().get( regionName );
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Region getNaturalIdCacheRegion(String regionName) {
|
public Region getNaturalIdCacheRegion(String regionName) {
|
||||||
return allCacheRegions.get( regionName );
|
return naturalIdRegionMap.get( regionName );
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings({"unchecked"})
|
|
||||||
@Override
|
@Override
|
||||||
public Map<String, Region> getAllSecondLevelCacheRegions() {
|
public Map<String, Region> getAllSecondLevelCacheRegions() {
|
||||||
return new HashMap<String, Region>( allCacheRegions );
|
// Order is important!
|
||||||
|
// For example, if there is a CollectionRegion and an EntityRegion with the same name, then we
|
||||||
|
// want the EntityRegion to be in the Map that gets returned.
|
||||||
|
Map<String, Region> allCacheRegions = getNonEntityNonCollectionRegions();
|
||||||
|
allCacheRegions.putAll( collectionRegionMap );
|
||||||
|
allCacheRegions.putAll( entityRegionMap );
|
||||||
|
return allCacheRegions;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Map<String, Region> getNonEntityNonCollectionRegions() {
|
||||||
|
final Map<String, Region> allCacheRegions = new HashMap<String, Region>();
|
||||||
|
allCacheRegions.put( updateTimestampsCache.getRegion().getName(), updateTimestampsCache.getRegion() );
|
||||||
|
allCacheRegions.put( queryCache.getRegion().getName(), queryCache.getRegion() );
|
||||||
|
for ( QueryCache queryCacheValue : queryCaches.values() ) {
|
||||||
|
allCacheRegions.put( queryCacheValue.getRegion().getName(), queryCacheValue.getRegion() );
|
||||||
|
}
|
||||||
|
allCacheRegions.putAll( otherRegionMap );
|
||||||
|
allCacheRegions.putAll( naturalIdRegionMap );
|
||||||
|
return allCacheRegions;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,8 +19,6 @@ import org.hibernate.Hibernate;
|
||||||
import org.hibernate.annotations.Cache;
|
import org.hibernate.annotations.Cache;
|
||||||
import org.hibernate.annotations.CacheConcurrencyStrategy;
|
import org.hibernate.annotations.CacheConcurrencyStrategy;
|
||||||
import org.hibernate.cache.internal.DefaultCacheKeysFactory;
|
import org.hibernate.cache.internal.DefaultCacheKeysFactory;
|
||||||
import org.hibernate.cache.spi.EntityRegion;
|
|
||||||
import org.hibernate.cache.spi.Region;
|
|
||||||
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
import org.hibernate.cache.spi.access.EntityRegionAccessStrategy;
|
||||||
import org.hibernate.cache.spi.access.RegionAccessStrategy;
|
import org.hibernate.cache.spi.access.RegionAccessStrategy;
|
||||||
import org.hibernate.cfg.Environment;
|
import org.hibernate.cfg.Environment;
|
||||||
|
|
Loading…
Reference in New Issue