HHH-18023 fix bugs with versioned LockModes
- NPE at commit time for OPTIMISTIC on entity with no version - LockMode.OPTIMISTIC treated as equivalent to LockModeType.OPTIMISTIC Signed-off-by: Gavin King <gavin@hibernate.org>
This commit is contained in:
parent
39a6f9880b
commit
789cc12b8e
|
@ -10,6 +10,8 @@ import org.hibernate.jpa.internal.util.LockModeTypeHelper;
|
|||
|
||||
import jakarta.persistence.LockModeType;
|
||||
|
||||
import java.util.Locale;
|
||||
|
||||
/**
|
||||
* Instances represent a lock mode for a row of a relational
|
||||
* database table. It is not intended that users spend much time
|
||||
|
@ -17,6 +19,18 @@ import jakarta.persistence.LockModeType;
|
|||
* the right lock level automatically. Some "advanced" users may
|
||||
* wish to explicitly specify lock levels.
|
||||
* <p>
|
||||
* A partial order of lock modes is defined such that every
|
||||
* optimistic lock mode is considered a weaker sort of lock than
|
||||
* evey pessimistic lock mode. If a session already holds a
|
||||
* stronger lock level on a given entity when a weaker lock level
|
||||
* is requested, then the request for the weaker lock level has
|
||||
* no effect.
|
||||
* <p>
|
||||
* Note that, in particular, a request for the optimistic lock
|
||||
* level {@link #OPTIMISTIC_FORCE_INCREMENT} on an entity for
|
||||
* which any pessimistic lock is already held has no effect,
|
||||
* and does not force a version increment.
|
||||
* <p>
|
||||
* This enumeration of lock modes competes with the JPA-defined
|
||||
* {@link LockModeType}, but offers additional options, including
|
||||
* {@link #UPGRADE_NOWAIT} and {@link #UPGRADE_SKIPLOCKED}.
|
||||
|
@ -42,19 +56,17 @@ public enum LockMode {
|
|||
*
|
||||
* @see LockModeType#NONE
|
||||
*/
|
||||
NONE( 0, "none" ),
|
||||
NONE,
|
||||
|
||||
/**
|
||||
* A shared lock. Objects in this lock mode were read from the
|
||||
* database in the current transaction, rather than being pulled
|
||||
* from a cache.
|
||||
* <p>
|
||||
* Note that this lock mode is the same as the JPA-defined modes
|
||||
* {@link LockModeType#READ} and {@link LockModeType#OPTIMISTIC}.
|
||||
*
|
||||
* @see LockModeType#OPTIMISTIC
|
||||
* Note that, despite the similar names this lock mode is not the
|
||||
* same as the JPA-defined mode {@link LockModeType#READ}.
|
||||
*/
|
||||
READ( 5, "read" ),
|
||||
READ,
|
||||
|
||||
/**
|
||||
* A shared optimistic lock. Assumes that the current transaction
|
||||
|
@ -62,10 +74,14 @@ public enum LockMode {
|
|||
* version will be checked near the end of the transaction, to
|
||||
* verify that this was indeed the case.
|
||||
* <p>
|
||||
* Note that, despite the similar names this lock mode is not the
|
||||
* same as the JPA-defined mode {@link LockModeType#OPTIMISTIC}.
|
||||
* Only legal for versioned entity types.
|
||||
* <p>
|
||||
* Note that this lock mode is the same as the JPA-defined modes
|
||||
* {@link LockModeType#READ} and {@link LockModeType#OPTIMISTIC}.
|
||||
*
|
||||
* @see LockModeType#OPTIMISTIC
|
||||
*/
|
||||
OPTIMISTIC( 6, "optimistic" ),
|
||||
OPTIMISTIC,
|
||||
|
||||
/**
|
||||
* A kind of exclusive optimistic lock. Assumes that the current
|
||||
|
@ -74,10 +90,12 @@ public enum LockMode {
|
|||
* end of the transaction, to verify that this was indeed the
|
||||
* case, and to signal to concurrent optimistic readers that their
|
||||
* optimistic locks have failed.
|
||||
* <p>
|
||||
* Only legal for versioned entity types.
|
||||
*
|
||||
* @see LockModeType#OPTIMISTIC_FORCE_INCREMENT
|
||||
*/
|
||||
OPTIMISTIC_FORCE_INCREMENT( 7, "optimistic_force_increment" ),
|
||||
OPTIMISTIC_FORCE_INCREMENT,
|
||||
|
||||
/**
|
||||
* An exclusive write lock. Objects in this lock mode were updated
|
||||
|
@ -93,7 +111,7 @@ public enum LockMode {
|
|||
* the same as the JPA-defined mode {@link LockModeType#WRITE}.
|
||||
*/
|
||||
@Internal
|
||||
WRITE( 10, "write" ),
|
||||
WRITE,
|
||||
|
||||
/**
|
||||
* A pessimistic upgrade lock, obtained using an Oracle-style
|
||||
|
@ -102,7 +120,7 @@ public enum LockMode {
|
|||
* as {@link #PESSIMISTIC_WRITE}. If the lock is not immediately
|
||||
* available, an exception occurs.
|
||||
*/
|
||||
UPGRADE_NOWAIT( 10, "upgrade-nowait" ),
|
||||
UPGRADE_NOWAIT,
|
||||
|
||||
/**
|
||||
* A pessimistic upgrade lock, obtained using an Oracle-style
|
||||
|
@ -112,7 +130,7 @@ public enum LockMode {
|
|||
* immediately available, no exception occurs, but the locked
|
||||
* row is not returned from the database.
|
||||
*/
|
||||
UPGRADE_SKIPLOCKED( 10, "upgrade-skiplocked" ),
|
||||
UPGRADE_SKIPLOCKED,
|
||||
|
||||
/**
|
||||
* A pessimistic shared lock, which prevents concurrent
|
||||
|
@ -126,7 +144,7 @@ public enum LockMode {
|
|||
*
|
||||
* @see LockModeType#PESSIMISTIC_READ
|
||||
*/
|
||||
PESSIMISTIC_READ( 12, "pessimistic_read" ),
|
||||
PESSIMISTIC_READ,
|
||||
|
||||
/**
|
||||
* A pessimistic upgrade lock, which prevents concurrent
|
||||
|
@ -135,33 +153,39 @@ public enum LockMode {
|
|||
*
|
||||
* @see LockModeType#PESSIMISTIC_WRITE
|
||||
*/
|
||||
PESSIMISTIC_WRITE( 13, "pessimistic_write" ),
|
||||
PESSIMISTIC_WRITE,
|
||||
|
||||
/**
|
||||
* A pessimistic write lock which immediately increments
|
||||
* the version of the locked object. Obtained by immediate
|
||||
* execution of an {@code update} statement.
|
||||
* <p>
|
||||
* Only legal for versioned entity types.
|
||||
*
|
||||
* @see LockModeType#PESSIMISTIC_FORCE_INCREMENT
|
||||
*/
|
||||
PESSIMISTIC_FORCE_INCREMENT( 17, "pessimistic_force_increment" );
|
||||
|
||||
private final int level;
|
||||
private final String externalForm;
|
||||
|
||||
LockMode(int level, String externalForm) {
|
||||
this.level = level;
|
||||
this.externalForm = externalForm;
|
||||
}
|
||||
PESSIMISTIC_FORCE_INCREMENT;
|
||||
|
||||
/**
|
||||
* @return an instance with the same semantics as the given JPA
|
||||
* {@link LockModeType}.
|
||||
*/
|
||||
public static LockMode fromJpaLockMode(LockModeType lockMode) {
|
||||
return LockModeTypeHelper.getLockMode( lockMode );
|
||||
}
|
||||
|
||||
/**
|
||||
* @return an instance of the JPA-defined {@link LockModeType}
|
||||
* with similar semantics to the given {@code LockMode}.
|
||||
*/
|
||||
public static LockModeType toJpaLockMode(LockMode lockMode) {
|
||||
return LockModeTypeHelper.getLockModeType( lockMode );
|
||||
}
|
||||
|
||||
/**
|
||||
* @return an instance of the JPA-defined {@link LockModeType}
|
||||
* with similar semantics to this {@code LockMode}.
|
||||
*/
|
||||
public LockModeType toJpaLockMode() {
|
||||
return LockModeTypeHelper.getLockModeType( this );
|
||||
}
|
||||
|
@ -174,7 +198,7 @@ public enum LockMode {
|
|||
* @return true if this lock mode is more restrictive than given lock mode
|
||||
*/
|
||||
public boolean greaterThan(LockMode mode) {
|
||||
return level > mode.level;
|
||||
return level() > mode.level();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -185,11 +209,61 @@ public enum LockMode {
|
|||
* @return true if this lock mode is less restrictive than given lock mode
|
||||
*/
|
||||
public boolean lessThan(LockMode mode) {
|
||||
return level < mode.level;
|
||||
return level() < mode.level();
|
||||
}
|
||||
|
||||
/**
|
||||
* Does this lock mode require a {@linkplain jakarta.persistence.Version version}?
|
||||
*
|
||||
* @return {@code true} if this lock mode only applies to versioned entities
|
||||
*/
|
||||
public boolean requiresVersion() {
|
||||
return this == OPTIMISTIC
|
||||
|| this == OPTIMISTIC_FORCE_INCREMENT
|
||||
|| this == PESSIMISTIC_FORCE_INCREMENT;
|
||||
}
|
||||
|
||||
public String toExternalForm() {
|
||||
return externalForm;
|
||||
final String externalForm = toString().toLowerCase(Locale.ROOT);
|
||||
return this == UPGRADE_NOWAIT || this == UPGRADE_SKIPLOCKED
|
||||
? externalForm.replace('_', '-')
|
||||
: externalForm;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines a partial order on the lock modes,
|
||||
* based on how "exclusive" the lock is.
|
||||
* <p>
|
||||
* Note that {@link #PESSIMISTIC_READ} is, quite
|
||||
* arbitrarily, treated as more exclusive than
|
||||
* {@link #OPTIMISTIC_FORCE_INCREMENT}. Thus, if
|
||||
* the program holds any pessimistic lock on an
|
||||
* instance, and requests
|
||||
* {@code OPTIMISTIC_FORCE_INCREMENT}, then no
|
||||
* forced version increment will occur.
|
||||
*/
|
||||
private int level() {
|
||||
switch (this) {
|
||||
case NONE:
|
||||
return 0;
|
||||
case READ:
|
||||
return 1;
|
||||
case OPTIMISTIC:
|
||||
return 2;
|
||||
case OPTIMISTIC_FORCE_INCREMENT:
|
||||
return 3;
|
||||
case PESSIMISTIC_READ:
|
||||
return 4;
|
||||
case UPGRADE_NOWAIT:
|
||||
case UPGRADE_SKIPLOCKED:
|
||||
case PESSIMISTIC_WRITE:
|
||||
return 5;
|
||||
case PESSIMISTIC_FORCE_INCREMENT:
|
||||
case WRITE:
|
||||
return 6;
|
||||
default:
|
||||
throw new AssertionFailure( "Unrecognized LockMode: " + this );
|
||||
}
|
||||
}
|
||||
|
||||
public static LockMode fromExternalForm(String externalForm) {
|
||||
|
@ -197,8 +271,8 @@ public enum LockMode {
|
|||
return NONE;
|
||||
}
|
||||
|
||||
for ( LockMode lockMode : LockMode.values() ) {
|
||||
if ( lockMode.externalForm.equals( externalForm ) ) {
|
||||
for ( LockMode lockMode : values() ) {
|
||||
if ( lockMode.toExternalForm().equalsIgnoreCase( externalForm ) ) {
|
||||
return lockMode;
|
||||
}
|
||||
}
|
||||
|
@ -215,8 +289,6 @@ public enum LockMode {
|
|||
* all other settings defaulted.
|
||||
*/
|
||||
public LockOptions toLockOptions() {
|
||||
// we have to do this in a big switch to
|
||||
// avoid circularities in the constructor
|
||||
switch (this) {
|
||||
case NONE:
|
||||
return LockOptions.NONE;
|
||||
|
@ -239,7 +311,7 @@ public enum LockMode {
|
|||
case WRITE:
|
||||
throw new UnsupportedOperationException("WRITE is not a valid LockMode as an argument");
|
||||
default:
|
||||
throw new IllegalStateException( "Unexpected value: " + this );
|
||||
throw new AssertionFailure( "Unrecognized LockMode: " + this );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -10,7 +10,6 @@ import org.hibernate.action.spi.BeforeTransactionCompletionProcess;
|
|||
import org.hibernate.dialect.lock.OptimisticEntityLockException;
|
||||
import org.hibernate.engine.spi.EntityEntry;
|
||||
import org.hibernate.engine.spi.SessionImplementor;
|
||||
import org.hibernate.persister.entity.EntityPersister;
|
||||
import org.hibernate.pretty.MessageHelper;
|
||||
|
||||
/**
|
||||
|
|
|
@ -42,7 +42,7 @@ public class OptimisticLockingStrategy implements LockingStrategy {
|
|||
@Override
|
||||
public void lock(Object id, Object version, Object object, int timeout, EventSource session) {
|
||||
if ( !lockable.isVersioned() ) {
|
||||
throw new OptimisticEntityLockException( object, "[" + lockMode + "] not supported for non-versioned entities [" + lockable.getEntityName() + "]" );
|
||||
throw new HibernateException( "[" + lockMode + "] not supported for non-versioned entities [" + lockable.getEntityName() + "]" );
|
||||
}
|
||||
// Register the EntityVerifyVersionProcess action to run just prior to transaction commit.
|
||||
session.getActionQueue().registerProcess( new EntityVerifyVersionProcess( object ) );
|
||||
|
|
|
@ -8,6 +8,8 @@ package org.hibernate.event.internal;
|
|||
|
||||
|
||||
import org.hibernate.AssertionFailure;
|
||||
import org.hibernate.HibernateException;
|
||||
import org.hibernate.LockMode;
|
||||
import org.hibernate.action.internal.EntityIncrementVersionProcess;
|
||||
import org.hibernate.action.internal.EntityVerifyVersionProcess;
|
||||
import org.hibernate.classic.Lifecycle;
|
||||
|
@ -17,6 +19,7 @@ import org.hibernate.event.spi.PostLoadEvent;
|
|||
import org.hibernate.event.spi.PostLoadEventListener;
|
||||
import org.hibernate.jpa.event.spi.CallbackRegistry;
|
||||
import org.hibernate.jpa.event.spi.CallbackRegistryConsumer;
|
||||
import org.hibernate.persister.entity.EntityPersister;
|
||||
|
||||
/**
|
||||
* We do two things here:
|
||||
|
@ -48,10 +51,14 @@ public class DefaultPostLoadEventListener implements PostLoadEventListener, Call
|
|||
throw new AssertionFailure( "possible non-threadsafe access to the session" );
|
||||
}
|
||||
|
||||
switch ( entry.getLockMode() ) {
|
||||
final LockMode lockMode = entry.getLockMode();
|
||||
if ( lockMode.requiresVersion() ) {
|
||||
final EntityPersister persister = entry.getPersister();
|
||||
if ( persister.isVersioned() ) {
|
||||
switch ( lockMode ) {
|
||||
case PESSIMISTIC_FORCE_INCREMENT:
|
||||
final Object nextVersion = entry.getPersister()
|
||||
.forceVersionIncrement( entry.getId(), entry.getVersion(), false, session );
|
||||
final Object nextVersion =
|
||||
persister.forceVersionIncrement( entry.getId(), entry.getVersion(), false, session );
|
||||
entry.forceLocked( entity, nextVersion );
|
||||
break;
|
||||
case OPTIMISTIC_FORCE_INCREMENT:
|
||||
|
@ -61,9 +68,14 @@ public class DefaultPostLoadEventListener implements PostLoadEventListener, Call
|
|||
session.getActionQueue().registerProcess( new EntityVerifyVersionProcess( entity ) );
|
||||
break;
|
||||
}
|
||||
}
|
||||
else {
|
||||
throw new HibernateException("[" + lockMode
|
||||
+ "] not supported for non-versioned entities [" + persister.getEntityName() + "]");
|
||||
}
|
||||
}
|
||||
|
||||
invokeLoadLifecycle( event, session );
|
||||
|
||||
}
|
||||
|
||||
protected void invokeLoadLifecycle(PostLoadEvent event, EventSource session) {
|
||||
|
|
|
@ -28,28 +28,27 @@ public final class LockModeConverter {
|
|||
* @return The JPA {@link LockModeType}
|
||||
*/
|
||||
public static LockModeType convertToLockModeType(LockMode lockMode) {
|
||||
if ( lockMode == LockMode.NONE ) {
|
||||
switch (lockMode) {
|
||||
case NONE:
|
||||
case READ: // no exact equivalent in JPA
|
||||
return LockModeType.NONE;
|
||||
}
|
||||
else if ( lockMode == LockMode.OPTIMISTIC || lockMode == LockMode.READ ) {
|
||||
case OPTIMISTIC:
|
||||
return LockModeType.OPTIMISTIC;
|
||||
}
|
||||
else if ( lockMode == LockMode.OPTIMISTIC_FORCE_INCREMENT || lockMode == LockMode.WRITE ) {
|
||||
case WRITE: // no exact equivalent in JPA
|
||||
case OPTIMISTIC_FORCE_INCREMENT:
|
||||
return LockModeType.OPTIMISTIC_FORCE_INCREMENT;
|
||||
}
|
||||
else if ( lockMode == LockMode.PESSIMISTIC_READ ) {
|
||||
case PESSIMISTIC_READ:
|
||||
return LockModeType.PESSIMISTIC_READ;
|
||||
}
|
||||
else if ( lockMode == LockMode.PESSIMISTIC_WRITE
|
||||
|| lockMode == LockMode.UPGRADE_NOWAIT
|
||||
|| lockMode == LockMode.UPGRADE_SKIPLOCKED) {
|
||||
case PESSIMISTIC_WRITE:
|
||||
case UPGRADE_NOWAIT:
|
||||
case UPGRADE_SKIPLOCKED:
|
||||
return LockModeType.PESSIMISTIC_WRITE;
|
||||
}
|
||||
else if ( lockMode == LockMode.PESSIMISTIC_FORCE_INCREMENT ) {
|
||||
case PESSIMISTIC_FORCE_INCREMENT:
|
||||
return LockModeType.PESSIMISTIC_FORCE_INCREMENT;
|
||||
}
|
||||
default:
|
||||
throw new AssertionFailure( "unhandled lock mode " + lockMode );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
|
@ -60,29 +59,22 @@ public final class LockModeConverter {
|
|||
*/
|
||||
public static LockMode convertToLockMode(LockModeType lockMode) {
|
||||
switch ( lockMode ) {
|
||||
case READ:
|
||||
case OPTIMISTIC: {
|
||||
return LockMode.OPTIMISTIC;
|
||||
}
|
||||
case OPTIMISTIC_FORCE_INCREMENT:
|
||||
case WRITE: {
|
||||
return LockMode.OPTIMISTIC_FORCE_INCREMENT;
|
||||
}
|
||||
case PESSIMISTIC_READ: {
|
||||
return LockMode.PESSIMISTIC_READ;
|
||||
}
|
||||
case PESSIMISTIC_WRITE: {
|
||||
return LockMode.PESSIMISTIC_WRITE;
|
||||
}
|
||||
case PESSIMISTIC_FORCE_INCREMENT: {
|
||||
return LockMode.PESSIMISTIC_FORCE_INCREMENT;
|
||||
}
|
||||
case NONE: {
|
||||
case NONE:
|
||||
return LockMode.NONE;
|
||||
}
|
||||
default: {
|
||||
case READ:
|
||||
case OPTIMISTIC:
|
||||
return LockMode.OPTIMISTIC;
|
||||
case WRITE:
|
||||
case OPTIMISTIC_FORCE_INCREMENT:
|
||||
return LockMode.OPTIMISTIC_FORCE_INCREMENT;
|
||||
case PESSIMISTIC_READ:
|
||||
return LockMode.PESSIMISTIC_READ;
|
||||
case PESSIMISTIC_WRITE:
|
||||
return LockMode.PESSIMISTIC_WRITE;
|
||||
case PESSIMISTIC_FORCE_INCREMENT:
|
||||
return LockMode.PESSIMISTIC_FORCE_INCREMENT;
|
||||
default:
|
||||
throw new AssertionFailure( "Unknown LockModeType: " + lockMode );
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -103,7 +103,8 @@ public class SingleIdEntityLoaderStandardImpl<T> extends SingleIdEntityLoaderSup
|
|||
return loadPlanCreator.apply( lockOptions, loadQueryInfluencers );
|
||||
}
|
||||
else if ( loadQueryInfluencers.hasEnabledCascadingFetchProfile()
|
||||
&& LockMode.WRITE.greaterThan( lockOptions.getLockMode() ) ) {
|
||||
// and if it's a non-exclusive (optimistic) lock
|
||||
&& LockMode.PESSIMISTIC_READ.greaterThan( lockOptions.getLockMode() ) ) {
|
||||
return getInternalCascadeLoadPlan(
|
||||
lockOptions,
|
||||
loadQueryInfluencers,
|
||||
|
|
|
@ -15,6 +15,7 @@ import java.util.concurrent.FutureTask;
|
|||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
|
||||
import org.hibernate.HibernateException;
|
||||
import org.hibernate.LockOptions;
|
||||
import org.hibernate.Session;
|
||||
import org.hibernate.TransactionException;
|
||||
|
@ -485,9 +486,9 @@ public class LockTest extends BaseEntityManagerFunctionalTestCase {
|
|||
// To get the same functionality as prior release, change the LockModeType.READ lock to:
|
||||
// em.lock(lock,LockModeType.PESSIMISTIC_READ);
|
||||
em.lock( _lock, LockModeType.READ );
|
||||
fail( "expected OptimisticLockException exception" );
|
||||
fail( "expected HibernateException exception" );
|
||||
}
|
||||
catch ( OptimisticLockException expected ) {
|
||||
catch ( HibernateException expected ) {
|
||||
}
|
||||
} );
|
||||
|
||||
|
|
Loading…
Reference in New Issue