Merge remote-tracking branch 'upstream/main' into wip/6.0

This commit is contained in:
Andrea Boriero 2021-09-17 11:52:20 +02:00
commit 52c932cc1c
2 changed files with 109 additions and 21 deletions

View File

@ -133,7 +133,9 @@ public class EntityUpdateAction extends EntityAction {
final SharedSessionContractImplementor session = getSession(); final SharedSessionContractImplementor session = getSession();
final Object instance = getInstance(); final Object instance = getInstance();
final boolean veto = preUpdate(); if ( preUpdate() ) {
return;
}
final SessionFactoryImplementor factory = session.getFactory(); final SessionFactoryImplementor factory = session.getFactory();
Object previousVersion = this.previousVersion; Object previousVersion = this.previousVersion;
@ -158,8 +160,6 @@ public class EntityUpdateAction extends EntityAction {
else { else {
ck = null; ck = null;
} }
if ( !veto ) {
persister.update( persister.update(
id, id,
state, state,
@ -171,14 +171,13 @@ public class EntityUpdateAction extends EntityAction {
rowId, rowId,
session session
); );
}
final EntityEntry entry = session.getPersistenceContextInternal().getEntry( instance ); final EntityEntry entry = session.getPersistenceContextInternal().getEntry( instance );
if ( entry == null ) { if ( entry == null ) {
throw new AssertionFailure( "possible non-threadsafe access to session" ); throw new AssertionFailure( "possible non thread safe access to session" );
} }
if ( entry.getStatus()==Status.MANAGED || persister.isVersionPropertyGenerated() ) { if ( entry.getStatus() == Status.MANAGED || persister.isVersionPropertyGenerated() ) {
// get the updated snapshot of the entity state by cloning current state; // get the updated snapshot of the entity state by cloning current state;
// it is safe to copy in place, since by this time no-one else (should have) // it is safe to copy in place, since by this time no-one else (should have)
// has a reference to the array // has a reference to the array
@ -204,12 +203,12 @@ public class EntityUpdateAction extends EntityAction {
final StatisticsImplementor statistics = factory.getStatistics(); final StatisticsImplementor statistics = factory.getStatistics();
if ( persister.canWriteToCache() ) { if ( persister.canWriteToCache() ) {
if ( persister.isCacheInvalidationRequired() || entry.getStatus()!= Status.MANAGED ) { if ( persister.isCacheInvalidationRequired() || entry.getStatus() != Status.MANAGED ) {
persister.getCacheAccessStrategy().remove( session, ck); persister.getCacheAccessStrategy().remove( session, ck );
} }
else if ( session.getCacheMode().isPutEnabled() ) { else if ( session.getCacheMode().isPutEnabled() ) {
//TODO: inefficient if that cache is just going to ignore the updated state! //TODO: inefficient if that cache is just going to ignore the updated state!
final CacheEntry ce = persister.buildCacheEntry( instance,state, nextVersion, getSession() ); final CacheEntry ce = persister.buildCacheEntry( instance, state, nextVersion, getSession() );
cacheEntry = persister.getCacheEntryStructure().structure( ce ); cacheEntry = persister.getCacheEntryStructure().structure( ce );
final boolean put = cacheUpdate( persister, previousVersion, ck ); final boolean put = cacheUpdate( persister, previousVersion, ck );
@ -234,9 +233,10 @@ public class EntityUpdateAction extends EntityAction {
postUpdate(); postUpdate();
if ( statistics.isStatisticsEnabled() && !veto ) { if ( statistics.isStatisticsEnabled() ) {
statistics.updateEntity( getPersister().getEntityName() ); statistics.updateEntity( getPersister().getEntityName() );
} }
} }
protected boolean cacheUpdate(EntityPersister persister, Object previousVersion, Object ck) { protected boolean cacheUpdate(EntityPersister persister, Object previousVersion, Object ck) {

View File

@ -0,0 +1,88 @@
package org.hibernate.event;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Version;
import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.event.service.spi.EventListenerRegistry;
import org.hibernate.event.spi.EventType;
import org.hibernate.testing.TestForIssue;
import org.hibernate.testing.orm.junit.BaseSessionFactoryFunctionalTest;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;
/**
* @author Nathan Xu
* @author Tassilo Karge
*/
@TestForIssue(jiraKey = "HHH-14413")
public class PreUpdateEventListenerVetoTest extends BaseSessionFactoryFunctionalTest {
private static final Long EXAMPLE_ID_VALUE = 1L;
@Override
protected Class<?>[] getAnnotatedClasses() {
return new Class<?>[] { ExampleEntity.class };
}
@Override
protected void sessionFactoryBuilt(SessionFactoryImplementor factory) {
EventListenerRegistry registry = factory.getServiceRegistry().getService( EventListenerRegistry.class );
registry.appendListeners(
EventType.PRE_UPDATE,
event -> true
);
}
@BeforeEach
public void setUp() {
inTransaction( session -> {
ExampleEntity entity = new ExampleEntity();
entity.id = EXAMPLE_ID_VALUE;
entity.name = "old_name";
session.save( entity );
} );
}
@Test
public void testVersionNotChangedWhenPreUpdateEventVetoed() {
inTransaction( session -> {
ExampleEntity entity = session.byId( ExampleEntity.class ).load( EXAMPLE_ID_VALUE );
entity.name = "new_name";
session.update( entity );
final Long versionBeforeFlush = entity.version;
session.flush();
final Long versionAfterFlush = entity.version;
assertEquals(
versionBeforeFlush,
versionAfterFlush,
"The entity version must not change when update is vetoed"
);
} );
}
@Entity(name = "ExampleEntity")
public static class ExampleEntity {
@Id
Long id;
String name;
@Version
Long version;
}
}