HHH-10708 - Accessing a lazy collection in an enhanced class deletes it afterwards
(cherry picked from commit0e1b79d2b5
) Conflicts: hibernate-core/src/main/java/org/hibernate/engine/internal/Collections.java HHH-10708 : Corrections due to backporting (cherry picked from commit0f918b4d42
) Conflicts: hibernate-core/src/main/java/org/hibernate/engine/internal/AbstractEntityEntry.java
This commit is contained in:
parent
fffd2515a6
commit
33e16f2788
|
@ -17,6 +17,7 @@ import org.hibernate.HibernateException;
|
||||||
import org.hibernate.LockMode;
|
import org.hibernate.LockMode;
|
||||||
import org.hibernate.Session;
|
import org.hibernate.Session;
|
||||||
import org.hibernate.bytecode.instrumentation.spi.FieldInterceptor;
|
import org.hibernate.bytecode.instrumentation.spi.FieldInterceptor;
|
||||||
|
import org.hibernate.collection.spi.PersistentCollection;
|
||||||
import org.hibernate.engine.spi.CachedNaturalIdValueSource;
|
import org.hibernate.engine.spi.CachedNaturalIdValueSource;
|
||||||
import org.hibernate.engine.spi.EntityEntry;
|
import org.hibernate.engine.spi.EntityEntry;
|
||||||
import org.hibernate.engine.spi.EntityEntryExtraState;
|
import org.hibernate.engine.spi.EntityEntryExtraState;
|
||||||
|
@ -332,6 +333,15 @@ public abstract class AbstractEntityEntry implements Serializable, EntityEntry {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void overwriteLoadedStateCollectionValue(String propertyName, PersistentCollection collection) {
|
||||||
|
assert propertyName != null;
|
||||||
|
assert loadedState != null;
|
||||||
|
|
||||||
|
final int propertyIndex = ( (UniqueKeyLoadable) persister ).getPropertyIndex( propertyName );
|
||||||
|
loadedState[propertyIndex] = collection;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean requiresDirtyCheck(Object entity) {
|
public boolean requiresDirtyCheck(Object entity) {
|
||||||
return isModifiableEntity()
|
return isModifiableEntity()
|
||||||
|
|
|
@ -43,7 +43,7 @@ public final class Collections {
|
||||||
* @param session The session
|
* @param session The session
|
||||||
*/
|
*/
|
||||||
public static void processUnreachableCollection(PersistentCollection coll, SessionImplementor session) {
|
public static void processUnreachableCollection(PersistentCollection coll, SessionImplementor session) {
|
||||||
if ( coll.getOwner()==null ) {
|
if ( coll.getOwner() == null ) {
|
||||||
processNeverReferencedCollection( coll, session );
|
processNeverReferencedCollection( coll, session );
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -73,7 +73,7 @@ public final class Collections {
|
||||||
// the owning entity may have been deleted and its identifier unset due to
|
// the owning entity may have been deleted and its identifier unset due to
|
||||||
// identifier-rollback; in which case, try to look up its identifier from
|
// identifier-rollback; in which case, try to look up its identifier from
|
||||||
// the persistence context
|
// the persistence context
|
||||||
if ( session.getFactory().getSettings().isIdentifierRollbackEnabled() ) {
|
if ( session.getFactory().getSessionFactoryOptions().isIdentifierRollbackEnabled() ) {
|
||||||
final EntityEntry ownerEntry = persistenceContext.getEntry( coll.getOwner() );
|
final EntityEntry ownerEntry = persistenceContext.getEntry( coll.getOwner() );
|
||||||
if ( ownerEntry != null ) {
|
if ( ownerEntry != null ) {
|
||||||
ownerId = ownerEntry.getId();
|
ownerId = ownerEntry.getId();
|
||||||
|
@ -156,40 +156,73 @@ public final class Collections {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// The CollectionEntry.isReached() stuff is just to detect any silly users
|
|
||||||
// who set up circular or shared references between/to collections.
|
|
||||||
if ( ce.isReached() ) {
|
|
||||||
// We've been here before
|
|
||||||
throw new HibernateException(
|
|
||||||
"Found shared references to a collection: " + type.getRole()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
ce.setReached( true );
|
|
||||||
|
|
||||||
final SessionFactoryImplementor factory = session.getFactory();
|
final SessionFactoryImplementor factory = session.getFactory();
|
||||||
final CollectionPersister persister = factory.getCollectionPersister( type.getRole() );
|
final CollectionPersister persister = factory.getCollectionPersister( type.getRole() );
|
||||||
|
|
||||||
ce.setCurrentPersister( persister );
|
ce.setCurrentPersister( persister );
|
||||||
//TODO: better to pass the id in as an argument?
|
//TODO: better to pass the id in as an argument?
|
||||||
ce.setCurrentKey( type.getKeyOfOwner( entity, session ) );
|
ce.setCurrentKey( type.getKeyOfOwner( entity, session ) );
|
||||||
|
|
||||||
if ( LOG.isDebugEnabled() ) {
|
final boolean isBytecodeEnhanced = persister.getOwnerEntityPersister().getInstrumentationMetadata().isEnhancedForLazyLoading();
|
||||||
if ( collection.wasInitialized() ) {
|
if ( isBytecodeEnhanced && !collection.wasInitialized() ) {
|
||||||
LOG.debugf(
|
// skip it
|
||||||
"Collection found: %s, was: %s (initialized)",
|
LOG.debugf(
|
||||||
MessageHelper.collectionInfoString( persister, collection, ce.getCurrentKey(), session ),
|
"Skipping uninitialized bytecode-lazy collection: %s",
|
||||||
MessageHelper.collectionInfoString( ce.getLoadedPersister(), collection, ce.getLoadedKey(), session )
|
MessageHelper.collectionInfoString( persister, collection, ce.getCurrentKey(), session )
|
||||||
);
|
);
|
||||||
}
|
ce.setReached( true );
|
||||||
else {
|
ce.setProcessed( true );
|
||||||
LOG.debugf(
|
|
||||||
"Collection found: %s, was: %s (uninitialized)",
|
|
||||||
MessageHelper.collectionInfoString( persister, collection, ce.getCurrentKey(), session ),
|
|
||||||
MessageHelper.collectionInfoString( ce.getLoadedPersister(), collection, ce.getLoadedKey(), session )
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
// The CollectionEntry.isReached() stuff is just to detect any silly users
|
||||||
|
// who set up circular or shared references between/to collections.
|
||||||
|
if ( ce.isReached() ) {
|
||||||
|
// We've been here beforeQuery
|
||||||
|
throw new HibernateException(
|
||||||
|
"Found shared references to a collection: " + type.getRole()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
ce.setReached( true );
|
||||||
|
|
||||||
prepareCollectionForUpdate( collection, ce, factory );
|
if ( LOG.isDebugEnabled() ) {
|
||||||
|
if ( collection.wasInitialized() ) {
|
||||||
|
LOG.debugf(
|
||||||
|
"Collection found: %s, was: %s (initialized)",
|
||||||
|
MessageHelper.collectionInfoString(
|
||||||
|
persister,
|
||||||
|
collection,
|
||||||
|
ce.getCurrentKey(),
|
||||||
|
session
|
||||||
|
),
|
||||||
|
MessageHelper.collectionInfoString(
|
||||||
|
ce.getLoadedPersister(),
|
||||||
|
collection,
|
||||||
|
ce.getLoadedKey(),
|
||||||
|
session
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LOG.debugf(
|
||||||
|
"Collection found: %s, was: %s (uninitialized)",
|
||||||
|
MessageHelper.collectionInfoString(
|
||||||
|
persister,
|
||||||
|
collection,
|
||||||
|
ce.getCurrentKey(),
|
||||||
|
session
|
||||||
|
),
|
||||||
|
MessageHelper.collectionInfoString(
|
||||||
|
ce.getLoadedPersister(),
|
||||||
|
collection,
|
||||||
|
ce.getLoadedKey(),
|
||||||
|
session
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
prepareCollectionForUpdate( collection, ce, factory );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -169,37 +169,43 @@ public final class CollectionEntry implements Serializable {
|
||||||
loadedKey = collection.getKey();
|
loadedKey = collection.getKey();
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean nonMutableChange = collection.isDirty() &&
|
boolean nonMutableChange = collection.isDirty()
|
||||||
getLoadedPersister()!=null &&
|
&& getLoadedPersister() != null
|
||||||
!getLoadedPersister().isMutable();
|
&& !getLoadedPersister().isMutable();
|
||||||
if (nonMutableChange) {
|
if ( nonMutableChange ) {
|
||||||
throw new HibernateException(
|
throw new HibernateException(
|
||||||
"changed an immutable collection instance: " +
|
"changed an immutable collection instance: " +
|
||||||
MessageHelper.collectionInfoString( getLoadedPersister().getRole(), getLoadedKey() )
|
MessageHelper.collectionInfoString( getLoadedPersister().getRole(), getLoadedKey() )
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
dirty(collection);
|
dirty( collection );
|
||||||
|
|
||||||
if ( LOG.isDebugEnabled() && collection.isDirty() && getLoadedPersister() != null ) {
|
if ( LOG.isDebugEnabled() && collection.isDirty() && getLoadedPersister() != null ) {
|
||||||
LOG.debugf( "Collection dirty: %s",
|
LOG.debugf(
|
||||||
MessageHelper.collectionInfoString( getLoadedPersister().getRole(), getLoadedKey() ) );
|
"Collection dirty: %s",
|
||||||
|
MessageHelper.collectionInfoString( getLoadedPersister().getRole(), getLoadedKey() )
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
setDoupdate(false);
|
setReached( false );
|
||||||
setDoremove(false);
|
setProcessed( false );
|
||||||
setDorecreate(false);
|
|
||||||
setReached(false);
|
setDoupdate( false );
|
||||||
setProcessed(false);
|
setDoremove( false );
|
||||||
|
setDorecreate( false );
|
||||||
}
|
}
|
||||||
|
|
||||||
public void postInitialize(PersistentCollection collection) throws HibernateException {
|
public void postInitialize(PersistentCollection collection) throws HibernateException {
|
||||||
snapshot = getLoadedPersister().isMutable() ?
|
snapshot = getLoadedPersister().isMutable()
|
||||||
collection.getSnapshot( getLoadedPersister() ) :
|
? collection.getSnapshot( getLoadedPersister() )
|
||||||
null;
|
: null;
|
||||||
collection.setSnapshot(loadedKey, role, snapshot);
|
collection.setSnapshot(loadedKey, role, snapshot);
|
||||||
if (getLoadedPersister().getBatchSize() > 1) {
|
if ( getLoadedPersister().getBatchSize() > 1 ) {
|
||||||
((AbstractPersistentCollection) collection).getSession().getPersistenceContext().getBatchFetchQueue().removeBatchLoadableCollection(this);
|
( (AbstractPersistentCollection) collection ).getSession()
|
||||||
|
.getPersistenceContext()
|
||||||
|
.getBatchFetchQueue()
|
||||||
|
.removeBatchLoadableCollection( this );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -273,7 +279,7 @@ public final class CollectionEntry implements Serializable {
|
||||||
}
|
}
|
||||||
|
|
||||||
void afterDeserialize(SessionFactoryImplementor factory) {
|
void afterDeserialize(SessionFactoryImplementor factory) {
|
||||||
loadedPersister = ( factory == null ? null : factory.getCollectionPersister(role) );
|
loadedPersister = ( factory == null ? null : factory.getCollectionPersister( role ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean wasDereferenced() {
|
public boolean wasDereferenced() {
|
||||||
|
|
|
@ -11,6 +11,7 @@ import java.io.ObjectOutputStream;
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
|
||||||
import org.hibernate.LockMode;
|
import org.hibernate.LockMode;
|
||||||
|
import org.hibernate.collection.spi.PersistentCollection;
|
||||||
import org.hibernate.persister.entity.EntityPersister;
|
import org.hibernate.persister.entity.EntityPersister;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,6 +38,10 @@ public interface EntityEntry {
|
||||||
|
|
||||||
Object[] getLoadedState();
|
Object[] getLoadedState();
|
||||||
|
|
||||||
|
Object getLoadedValue(String propertyName);
|
||||||
|
|
||||||
|
void overwriteLoadedStateCollectionValue(String propertyName, PersistentCollection collection);
|
||||||
|
|
||||||
Object[] getDeletedState();
|
Object[] getDeletedState();
|
||||||
|
|
||||||
void setDeletedState(Object[] deletedState);
|
void setDeletedState(Object[] deletedState);
|
||||||
|
@ -86,8 +91,6 @@ public interface EntityEntry {
|
||||||
|
|
||||||
boolean isNullifiable(boolean earlyInsert, SessionImplementor session);
|
boolean isNullifiable(boolean earlyInsert, SessionImplementor session);
|
||||||
|
|
||||||
Object getLoadedValue(String propertyName);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Not sure this is the best method name, but the general idea here is to return {@code true} if the entity can
|
* Not sure this is the best method name, but the general idea here is to return {@code true} if the entity can
|
||||||
* possibly be dirty. This can only be the case if it is in a modifiable state (not read-only/deleted) and it
|
* possibly be dirty. This can only be the case if it is in a modifiable state (not read-only/deleted) and it
|
||||||
|
|
|
@ -928,6 +928,17 @@ public abstract class AbstractEntityPersister
|
||||||
session.getPersistenceContext().addCollectionHolder( collection );
|
session.getPersistenceContext().addCollectionHolder( collection );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// update the "state" of the entity's EntityEntry to over-write UNFETCHED_PROPERTY reference
|
||||||
|
// for the collection to the just loaded collection
|
||||||
|
final EntityEntry ownerEntry = session.getPersistenceContext().getEntry( entity );
|
||||||
|
if ( ownerEntry == null ) {
|
||||||
|
// not good
|
||||||
|
throw new AssertionFailure(
|
||||||
|
"Could not locate EntityEntry for the collection owner in the PersistenceContext"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
ownerEntry.overwriteLoadedStateCollectionValue( fieldName, collection );
|
||||||
|
|
||||||
// EARLY EXIT!!!
|
// EARLY EXIT!!!
|
||||||
return collection;
|
return collection;
|
||||||
}
|
}
|
||||||
|
|
|
@ -63,7 +63,7 @@ public class UnexpectedDeleteOneTestTask extends AbstractEnhancerTestTask {
|
||||||
Foo foo = s.get( Foo.class, fooId );
|
Foo foo = s.get( Foo.class, fooId );
|
||||||
|
|
||||||
// accessing the collection results in an exception
|
// accessing the collection results in an exception
|
||||||
foo.bar.size();
|
foo.bars.size();
|
||||||
|
|
||||||
s.flush();
|
s.flush();
|
||||||
s.getTransaction().commit();
|
s.getTransaction().commit();
|
||||||
|
@ -88,8 +88,9 @@ public class UnexpectedDeleteOneTestTask extends AbstractEnhancerTestTask {
|
||||||
@Id @GeneratedValue
|
@Id @GeneratedValue
|
||||||
int id;
|
int id;
|
||||||
|
|
||||||
@OneToMany(orphanRemoval = true, mappedBy = Bar.FOO, targetEntity = Bar.class) @Cascade(CascadeType.ALL)
|
@OneToMany(orphanRemoval = true, mappedBy = Bar.FOO, targetEntity = Bar.class)
|
||||||
Set<Bar> bar = new HashSet<>();
|
@Cascade(CascadeType.ALL)
|
||||||
|
Set<Bar> bars = new HashSet<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
Loading…
Reference in New Issue