HHH-14608 Merge causes StackOverflow when JPA proxy compliance is enabled

This commit is contained in:
Andrea Boriero 2021-05-16 09:35:06 +02:00 committed by gbadner
parent 8dcf6f983b
commit 8b02aaf5a8
19 changed files with 47 additions and 31 deletions

View File

@ -639,7 +639,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
private void reassociateProxy(LazyInitializer li, HibernateProxy proxy) { private void reassociateProxy(LazyInitializer li, HibernateProxy proxy) {
if ( li.getSession() != this.getSession() ) { if ( li.getSession() != this.getSession() ) {
final EntityPersister persister = session.getFactory().getMetamodel().entityPersister( li.getEntityName() ); final EntityPersister persister = session.getFactory().getMetamodel().entityPersister( li.getEntityName() );
final EntityKey key = session.generateEntityKey( li.getIdentifier(), persister ); final EntityKey key = session.generateEntityKey( li.getInternalIdentifier(), persister );
// any earlier proxy takes precedence // any earlier proxy takes precedence
getOrInitializeProxiesByKey().putIfAbsent( key, proxy ); getOrInitializeProxiesByKey().putIfAbsent( key, proxy );
proxy.getHibernateLazyInitializer().setSession( session ); proxy.getHibernateLazyInitializer().setSession( session );
@ -1348,7 +1348,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
); );
} }
if ( found ) { if ( found ) {
return proxy.getHibernateLazyInitializer().getIdentifier(); return proxy.getHibernateLazyInitializer().getInternalIdentifier();
} }
} }
} }

View File

@ -54,7 +54,7 @@ public class DefaultEvictEventListener implements EvictEventListener {
if ( object instanceof HibernateProxy ) { if ( object instanceof HibernateProxy ) {
final LazyInitializer li = ( (HibernateProxy) object ).getHibernateLazyInitializer(); final LazyInitializer li = ( (HibernateProxy) object ).getHibernateLazyInitializer();
final Serializable id = li.getIdentifier(); final Serializable id = li.getInternalIdentifier();
if ( id == null ) { if ( id == null ) {
throw new IllegalArgumentException( "Could not determine identifier of proxy passed to evict()" ); throw new IllegalArgumentException( "Could not determine identifier of proxy passed to evict()" );
} }

View File

@ -103,7 +103,7 @@ public class DefaultMergeEventListener extends AbstractSaveEventListener impleme
LazyInitializer li = ( (HibernateProxy) original ).getHibernateLazyInitializer(); LazyInitializer li = ( (HibernateProxy) original ).getHibernateLazyInitializer();
if ( li.isUninitialized() ) { if ( li.isUninitialized() ) {
LOG.trace( "Ignoring uninitialized proxy" ); LOG.trace( "Ignoring uninitialized proxy" );
event.setResult( source.load( li.getEntityName(), li.getIdentifier() ) ); event.setResult( source.load( li.getEntityName(), li.getInternalIdentifier() ) );
//EARLY EXIT! //EARLY EXIT!
return; return;
} }

View File

@ -1666,7 +1666,7 @@ public class SessionImpl
if ( li.getSession() != this ) { if ( li.getSession() != this ) {
throw new TransientObjectException( "The proxy was not associated with this session" ); throw new TransientObjectException( "The proxy was not associated with this session" );
} }
return li.getIdentifier(); return li.getInternalIdentifier();
} }
else { else {
EntityEntry entry = persistenceContext.getEntry( object ); EntityEntry entry = persistenceContext.getEntry( object );
@ -1694,7 +1694,7 @@ public class SessionImpl
} }
private Serializable getProxyIdentifier(Object proxy) { private Serializable getProxyIdentifier(Object proxy) {
return ( (HibernateProxy) proxy ).getHibernateLazyInitializer().getIdentifier(); return ( (HibernateProxy) proxy ).getHibernateLazyInitializer().getInternalIdentifier();
} }
private FilterQueryPlan getFilterQueryPlan( private FilterQueryPlan getFilterQueryPlan(

View File

@ -68,7 +68,7 @@ public class PersistenceUnitUtilImpl implements PersistenceUnitUtil, Serializabl
} }
if ( entity instanceof HibernateProxy ) { if ( entity instanceof HibernateProxy ) {
return ((HibernateProxy) entity).getHibernateLazyInitializer().getIdentifier(); return ((HibernateProxy) entity).getHibernateLazyInitializer().getInternalIdentifier();
} }
else if ( entity instanceof ManagedEntity ) { else if ( entity instanceof ManagedEntity ) {
EntityEntry entityEntry = ((ManagedEntity) entity).$$_hibernate_getEntityEntry(); EntityEntry entityEntry = ((ManagedEntity) entity).$$_hibernate_getEntityEntry();

View File

@ -84,6 +84,11 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
return entityName; return entityName;
} }
@Override
public final Serializable getInternalIdentifier() {
return id;
}
@Override @Override
public final Serializable getIdentifier() { public final Serializable getIdentifier() {
if ( isUninitialized() && isInitializeProxyWhenAccessingIdentifier() ) { if ( isUninitialized() && isInitializeProxyWhenAccessingIdentifier() ) {
@ -93,7 +98,7 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
} }
private boolean isInitializeProxyWhenAccessingIdentifier() { private boolean isInitializeProxyWhenAccessingIdentifier() {
return session != null && session.getFactory() return getSession() != null && getSession().getFactory()
.getSessionFactoryOptions() .getSessionFactoryOptions()
.getJpaCompliance().isJpaProxyComplianceEnabled(); .getJpaCompliance().isJpaProxyComplianceEnabled();
} }
@ -260,7 +265,7 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
public final void initializeWithoutLoadIfPossible() { public final void initializeWithoutLoadIfPossible() {
if ( !initialized && session != null && session.isOpenOrWaitingForAutoClose() ) { if ( !initialized && session != null && session.isOpenOrWaitingForAutoClose() ) {
final EntityKey key = session.generateEntityKey( final EntityKey key = session.generateEntityKey(
getIdentifier(), getInternalIdentifier(),
session.getFactory().getMetamodel().entityPersister( getEntityName() ) session.getFactory().getMetamodel().entityPersister( getEntityName() )
); );
final Object entity = session.getPersistenceContextInternal().getEntity( key ); final Object entity = session.getPersistenceContextInternal().getEntity( key );
@ -305,7 +310,7 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
} }
private Object getProxyOrNull() { private Object getProxyOrNull() {
final EntityKey entityKey = generateEntityKeyOrNull( getIdentifier(), session, getEntityName() ); final EntityKey entityKey = generateEntityKeyOrNull( getInternalIdentifier(), session, getEntityName() );
if ( entityKey != null && session != null && session.isOpenOrWaitingForAutoClose() ) { if ( entityKey != null && session != null && session.isOpenOrWaitingForAutoClose() ) {
return session.getPersistenceContextInternal().getProxy( entityKey ); return session.getPersistenceContextInternal().getProxy( entityKey );
} }
@ -326,7 +331,7 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
@Override @Override
public final Object getImplementation(SharedSessionContractImplementor s) throws HibernateException { public final Object getImplementation(SharedSessionContractImplementor s) throws HibernateException {
final EntityKey entityKey = generateEntityKeyOrNull( getIdentifier(), s, getEntityName() ); final EntityKey entityKey = generateEntityKeyOrNull( getInternalIdentifier(), s, getEntityName() );
return ( entityKey == null ? null : s.getPersistenceContext().getEntity( entityKey ) ); return ( entityKey == null ? null : s.getPersistenceContext().getEntity( entityKey ) );
} }
@ -376,7 +381,7 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
} }
this.readOnly = readOnly; this.readOnly = readOnly;
if ( initialized ) { if ( initialized ) {
EntityKey key = generateEntityKeyOrNull( getIdentifier(), session, getEntityName() ); EntityKey key = generateEntityKeyOrNull( getInternalIdentifier(), session, getEntityName() );
final PersistenceContext persistenceContext = session.getPersistenceContext(); final PersistenceContext persistenceContext = session.getPersistenceContext();
if ( key != null && persistenceContext.containsEntity( key ) ) { if ( key != null && persistenceContext.containsEntity( key ) ) {
persistenceContext.setReadOnly( target, readOnly ); persistenceContext.setReadOnly( target, readOnly );

View File

@ -30,6 +30,17 @@ public interface LazyInitializer {
* *
* @return The identifier value. * @return The identifier value.
*/ */
default Serializable getInternalIdentifier() {
return getIdentifier();
}
/**
* Retrieve the identifier value for the entity our owning proxy represents.
*
* When JPA proxy compliance is enabled the proxy is initialized.
*
* @return The identifier value.
*/
Serializable getIdentifier(); Serializable getIdentifier();
/** /**

View File

@ -116,7 +116,7 @@ public class MapProxy implements HibernateProxy, Map, Serializable {
private Object serializableProxy() { private Object serializableProxy() {
return new SerializableMapProxy( return new SerializableMapProxy(
li.getEntityName(), li.getEntityName(),
li.getIdentifier(), li.getInternalIdentifier(),
( li.isReadOnlySettingAvailable() ? Boolean.valueOf( li.isReadOnly() ) : li.isReadOnlyBeforeAttachedToSession() ), ( li.isReadOnlySettingAvailable() ? Boolean.valueOf( li.isReadOnly() ) : li.isReadOnlyBeforeAttachedToSession() ),
li.getSessionFactoryUuid(), li.getSessionFactoryUuid(),
li.isAllowLoadOutsideTransaction() li.isAllowLoadOutsideTransaction()

View File

@ -85,7 +85,7 @@ public class ByteBuddyInterceptor extends BasicLazyInitializer implements ProxyC
getEntityName(), getEntityName(),
persistentClass, persistentClass,
interfaces, interfaces,
getIdentifier(), getInternalIdentifier(),
( isReadOnlySettingAvailable() ? Boolean.valueOf( isReadOnly() ) : isReadOnlyBeforeAttachedToSession() ), ( isReadOnlySettingAvailable() ? Boolean.valueOf( isReadOnly() ) : isReadOnlyBeforeAttachedToSession() ),
getSessionFactoryUuid(), getSessionFactoryUuid(),
isAllowLoadOutsideTransaction(), isAllowLoadOutsideTransaction(),

View File

@ -123,7 +123,7 @@ public class JavassistLazyInitializer extends BasicLazyInitializer implements Me
getEntityName(), getEntityName(),
persistentClass, persistentClass,
interfaces, interfaces,
getIdentifier(), getInternalIdentifier(),
( isReadOnlySettingAvailable() ? Boolean.valueOf( isReadOnly() ) : isReadOnlyBeforeAttachedToSession() ), ( isReadOnlySettingAvailable() ? Boolean.valueOf( isReadOnly() ) : isReadOnlyBeforeAttachedToSession() ),
getSessionFactoryUuid(), getSessionFactoryUuid(),
isAllowLoadOutsideTransaction(), isAllowLoadOutsideTransaction(),

View File

@ -215,7 +215,7 @@ public abstract class AbstractEntityTuplizer implements EntityTuplizer {
id = entity; id = entity;
} }
else if ( HibernateProxy.class.isInstance( entity ) ) { else if ( HibernateProxy.class.isInstance( entity ) ) {
id = ( (HibernateProxy) entity ).getHibernateLazyInitializer().getIdentifier(); id = ( (HibernateProxy) entity ).getHibernateLazyInitializer().getInternalIdentifier();
} }
else { else {
if ( idGetter == null ) { if ( idGetter == null ) {
@ -452,7 +452,7 @@ public abstract class AbstractEntityTuplizer implements EntityTuplizer {
if ( HibernateProxy.class.isInstance( entity ) ) { if ( HibernateProxy.class.isInstance( entity ) ) {
// entity is a proxy, so we know it is not transient; just return ID from proxy // entity is a proxy, so we know it is not transient; just return ID from proxy
return ( (HibernateProxy) entity ).getHibernateLazyInitializer().getIdentifier(); return ( (HibernateProxy) entity ).getHibernateLazyInitializer().getInternalIdentifier();
} }
if ( session != null ) { if ( session != null ) {

View File

@ -369,7 +369,7 @@ public abstract class EntityType extends AbstractType implements AssociationType
final Serializable id; final Serializable id;
if ( x instanceof HibernateProxy ) { if ( x instanceof HibernateProxy ) {
id = ( (HibernateProxy) x ).getHibernateLazyInitializer().getIdentifier(); id = ( (HibernateProxy) x ).getHibernateLazyInitializer().getInternalIdentifier();
} }
else { else {
final Class mappedClass = persister.getMappedClass(); final Class mappedClass = persister.getMappedClass();
@ -399,7 +399,7 @@ public abstract class EntityType extends AbstractType implements AssociationType
Serializable xid; Serializable xid;
if ( x instanceof HibernateProxy ) { if ( x instanceof HibernateProxy ) {
xid = ( (HibernateProxy) x ).getHibernateLazyInitializer() xid = ( (HibernateProxy) x ).getHibernateLazyInitializer()
.getIdentifier(); .getInternalIdentifier();
} }
else { else {
if ( mappedClass.isAssignableFrom( x.getClass() ) ) { if ( mappedClass.isAssignableFrom( x.getClass() ) ) {
@ -414,7 +414,7 @@ public abstract class EntityType extends AbstractType implements AssociationType
Serializable yid; Serializable yid;
if ( y instanceof HibernateProxy ) { if ( y instanceof HibernateProxy ) {
yid = ( (HibernateProxy) y ).getHibernateLazyInitializer() yid = ( (HibernateProxy) y ).getHibernateLazyInitializer()
.getIdentifier(); .getInternalIdentifier();
} }
else { else {
if ( mappedClass.isAssignableFrom( y.getClass() ) ) { if ( mappedClass.isAssignableFrom( y.getClass() ) ) {
@ -558,7 +558,7 @@ public abstract class EntityType extends AbstractType implements AssociationType
final Serializable id; final Serializable id;
if ( value instanceof HibernateProxy ) { if ( value instanceof HibernateProxy ) {
HibernateProxy proxy = (HibernateProxy) value; HibernateProxy proxy = (HibernateProxy) value;
id = proxy.getHibernateLazyInitializer().getIdentifier(); id = proxy.getHibernateLazyInitializer().getInternalIdentifier();
} }
else { else {
id = persister.getIdentifier( value ); id = persister.getIdentifier( value );

View File

@ -94,7 +94,7 @@ public abstract class BaseEnversEventListener implements EnversListener {
if ( value instanceof HibernateProxy ) { if ( value instanceof HibernateProxy ) {
final HibernateProxy hibernateProxy = (HibernateProxy) value; final HibernateProxy hibernateProxy = (HibernateProxy) value;
id = hibernateProxy.getHibernateLazyInitializer().getIdentifier(); id = hibernateProxy.getHibernateLazyInitializer().getInternalIdentifier();
// We've got to initialize the object from the proxy to later read its state. // We've got to initialize the object from the proxy to later read its state.
value = EntityTools.getTargetFromProxy( session.getFactory(), hibernateProxy ); value = EntityTools.getTargetFromProxy( session.getFactory(), hibernateProxy );
// HHH-7249 // HHH-7249

View File

@ -111,7 +111,7 @@ public class EntityInstantiator {
final HibernateProxy hibernateProxy = (HibernateProxy) value; final HibernateProxy hibernateProxy = (HibernateProxy) value;
final LazyInitializer initializer = hibernateProxy.getHibernateLazyInitializer(); final LazyInitializer initializer = hibernateProxy.getHibernateLazyInitializer();
final String entityName = initializer.getEntityName(); final String entityName = initializer.getEntityName();
final Serializable entityId = initializer.getIdentifier(); final Serializable entityId = initializer.getInternalIdentifier();
if ( enversService.getEntitiesConfigurations().isVersioned( entityName ) ) { if ( enversService.getEntitiesConfigurations().isVersioned( entityName ) ) {
final String entityClassName = enversService.getEntitiesConfigurations().get( entityName ).getEntityClassName(); final String entityClassName = enversService.getEntitiesConfigurations().get( entityName ).getEntityClassName();
final Class entityClass = ReflectionTools.loadClass( final Class entityClass = ReflectionTools.loadClass(

View File

@ -88,7 +88,7 @@ public class SingleIdMapper extends AbstractIdMapper implements SimpleIdMapperBu
if ( data instanceof HibernateProxy ) { if ( data instanceof HibernateProxy ) {
final HibernateProxy hibernateProxy = (HibernateProxy) data; final HibernateProxy hibernateProxy = (HibernateProxy) data;
return hibernateProxy.getHibernateLazyInitializer().getIdentifier(); return hibernateProxy.getHibernateLazyInitializer().getInternalIdentifier();
} }
else { else {
return AccessController.doPrivileged( return AccessController.doPrivileged(
@ -122,7 +122,7 @@ public class SingleIdMapper extends AbstractIdMapper implements SimpleIdMapperBu
else { else {
if ( obj instanceof HibernateProxy ) { if ( obj instanceof HibernateProxy ) {
final HibernateProxy hibernateProxy = (HibernateProxy) obj; final HibernateProxy hibernateProxy = (HibernateProxy) obj;
data.put( propertyData.getName(), hibernateProxy.getHibernateLazyInitializer().getIdentifier() ); data.put( propertyData.getName(), hibernateProxy.getHibernateLazyInitializer().getInternalIdentifier() );
} }
else { else {
final Object value = AccessController.doPrivileged( final Object value = AccessController.doPrivileged(

View File

@ -155,7 +155,7 @@ public class VirtualEntitySingleIdMapper extends SingleIdMapper {
else { else {
if ( obj instanceof HibernateProxy ) { if ( obj instanceof HibernateProxy ) {
final HibernateProxy proxy = (HibernateProxy) obj; final HibernateProxy proxy = (HibernateProxy) obj;
data.put( propertyData.getName(), proxy.getHibernateLazyInitializer().getIdentifier() ); data.put( propertyData.getName(), proxy.getHibernateLazyInitializer().getInternalIdentifier() );
} }
else { else {
final Object value = AccessController.doPrivileged( final Object value = AccessController.doPrivileged(

View File

@ -33,7 +33,7 @@ public abstract class EntityTools {
if ( obj instanceof HibernateProxy ) { if ( obj instanceof HibernateProxy ) {
final HibernateProxy hibernateProxy = (HibernateProxy) obj; final HibernateProxy hibernateProxy = (HibernateProxy) obj;
return hibernateProxy.getHibernateLazyInitializer().getIdentifier(); return hibernateProxy.getHibernateLazyInitializer().getInternalIdentifier();
} }
return session.getEntityPersister( entityName, obj ).getIdentifier( obj, session ); return session.getEntityPersister( entityName, obj ).getIdentifier( obj, session );
@ -51,7 +51,7 @@ public abstract class EntityTools {
try { try {
return tempSession.get( return tempSession.get(
proxy.getHibernateLazyInitializer().getEntityName(), proxy.getHibernateLazyInitializer().getEntityName(),
proxy.getHibernateLazyInitializer().getIdentifier() proxy.getHibernateLazyInitializer().getInternalIdentifier()
); );
} }
finally { finally {

View File

@ -80,7 +80,7 @@ public class RevisionsOfEntityQuery extends AbstractAuditQuery {
Object revisionInfoObject = ( (Map) versionsEntity.get( originalId ) ).get( revisionPropertyName ); Object revisionInfoObject = ( (Map) versionsEntity.get( originalId ) ).get( revisionPropertyName );
if ( revisionInfoObject instanceof HibernateProxy ) { if ( revisionInfoObject instanceof HibernateProxy ) {
return (Number) ( (HibernateProxy) revisionInfoObject ).getHibernateLazyInitializer().getIdentifier(); return (Number) ( (HibernateProxy) revisionInfoObject ).getHibernateLazyInitializer().getInternalIdentifier();
} }
else { else {
// Not a proxy - must be read from cache or with a join // Not a proxy - must be read from cache or with a join

View File

@ -114,8 +114,8 @@ public class ProxyIdentifier extends BaseEnversJPAFunctionalTestCase {
LazyInitializer lazyInitializer = proxyCreateByEnvers.getHibernateLazyInitializer(); LazyInitializer lazyInitializer = proxyCreateByEnvers.getHibernateLazyInitializer();
Assert.assertTrue( lazyInitializer.isUninitialized() ); Assert.assertTrue( lazyInitializer.isUninitialized() );
Assert.assertNotNull( lazyInitializer.getIdentifier() ); Assert.assertNotNull( lazyInitializer.getInternalIdentifier() );
Assert.assertEquals( tnae1.getId(), lazyInitializer.getIdentifier() ); Assert.assertEquals( tnae1.getId(), lazyInitializer.getInternalIdentifier() );
Assert.assertTrue( lazyInitializer.isUninitialized() ); Assert.assertTrue( lazyInitializer.isUninitialized() );
Assert.assertEquals( uste1.getId(), rev1.getReference().getId() ); Assert.assertEquals( uste1.getId(), rev1.getReference().getId() );