Skip to content

Commit 0b71d3b

Browse files
committed
introduce SnapshotIsolationException to fix tests on Maria
1 parent c88201d commit 0b71d3b

File tree

4 files changed

+35
-3
lines changed

4 files changed

+35
-3
lines changed

hibernate-core/src/main/java/org/hibernate/dialect/MariaDBDialect.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
import org.hibernate.exception.ConstraintViolationException;
2828
import org.hibernate.exception.ConstraintViolationException.ConstraintKind;
2929
import org.hibernate.exception.LockAcquisitionException;
30+
import org.hibernate.exception.SnapshotIsolationException;
3031
import org.hibernate.exception.LockTimeoutException;
3132
import org.hibernate.exception.spi.SQLExceptionConversionDelegate;
3233
import org.hibernate.exception.spi.TemplatedViolatedConstraintNameExtractor;
@@ -348,6 +349,7 @@ public ViolatedConstraintNameExtractor getViolatedConstraintNameExtractor() {
348349
case 1048 -> extractUsingTemplate( "Column '", "'", sqle.getMessage() );
349350
default -> null;
350351
} );
352+
351353
@Override
352354
public SQLExceptionConversionDelegate buildSQLExceptionConversionDelegate() {
353355
return (sqlException, message, sql) -> {
@@ -357,7 +359,8 @@ public SQLExceptionConversionDelegate buildSQLExceptionConversionDelegate() {
357359
case 1020:
358360
// If @@innodb_snapshot_isolation is set (default since 11.6.2),
359361
// and an attempt to acquire a lock on a record that does not exist
360-
// in the current read view is made, error DB_RECORD_CHANGED is raised.
362+
// in the current read view is made, error DB_RECORD_CHANGED is raised
363+
return new SnapshotIsolationException( message, sqlException, sql );
361364
case 3572: // ER_LOCK_NOWAIT
362365
case 1207: // ER_READ_ONLY_TRANSACTION
363366
case 1206: // ER_LOCK_TABLE_FULL
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
/*
2+
* SPDX-License-Identifier: Apache-2.0
3+
* Copyright Red Hat Inc. and Hibernate Authors
4+
*/
5+
package org.hibernate.exception;
6+
7+
import org.hibernate.Incubating;
8+
import org.hibernate.JDBCException;
9+
10+
import java.sql.SQLException;
11+
12+
/**
13+
* A {@link org.hibernate.JDBCException} indicating that a request failed due to snapshot isolation.
14+
* This is a condition which indicates an <em>optimistic</em> failure.
15+
*
16+
* @apiNote At present, this is only used to represent {@code DB_RECORD_CHANGED} on MariaDB.
17+
*
18+
* @see jakarta.persistence.OptimisticLockException
19+
*/
20+
@Incubating
21+
public class SnapshotIsolationException extends JDBCException {
22+
public SnapshotIsolationException(String string, SQLException root, String sql) {
23+
super( string, root, sql );
24+
}
25+
}

hibernate-core/src/main/java/org/hibernate/internal/ExceptionConverterImpl.java

+4
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
import org.hibernate.dialect.lock.PessimisticEntityLockException;
2323
import org.hibernate.engine.spi.ExceptionConverter;
2424
import org.hibernate.engine.spi.SharedSessionContractImplementor;
25+
import org.hibernate.exception.SnapshotIsolationException;
2526
import org.hibernate.exception.TransactionSerializationException;
2627
import org.hibernate.loader.MultipleBagFetchException;
2728
import org.hibernate.query.IllegalQueryOperationException;
@@ -90,6 +91,9 @@ else if ( exception instanceof LockingStrategyException lockingStrategyException
9091
rollbackIfNecessary( converted );
9192
return converted;
9293
}
94+
else if ( exception instanceof SnapshotIsolationException ) {
95+
return new OptimisticLockException( exception.getMessage(), exception );
96+
}
9397
else if ( exception instanceof org.hibernate.QueryTimeoutException ) {
9498
final QueryTimeoutException converted = new QueryTimeoutException( exception.getMessage(), exception );
9599
rollbackIfNecessary( converted );

hibernate-core/src/test/java/org/hibernate/orm/test/batch/BatchUpdateAndVersionTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
import org.hibernate.cfg.AvailableSettings;
1515

1616
import org.hibernate.dialect.MariaDBDialect;
17-
import org.hibernate.exception.LockAcquisitionException;
17+
import org.hibernate.exception.SnapshotIsolationException;
1818
import org.hibernate.exception.TransactionSerializationException;
1919
import org.hibernate.testing.orm.junit.JiraKey;
2020
import org.hibernate.testing.orm.junit.DomainModel;
@@ -141,7 +141,7 @@ public void testFailedUpdate(SessionFactoryScope scope) {
141141
catch (OptimisticLockException ole) {
142142
if (getDialect() instanceof MariaDBDialect && getDialect().getVersion().isAfter( 11, 6, 2 )) {
143143
// if @@innodb_snapshot_isolation is set, database throw an exception if record is not available anymore
144-
assertTrue( ole.getCause() instanceof LockAcquisitionException );
144+
assertTrue( ole.getCause() instanceof SnapshotIsolationException );
145145
} else {
146146
assertTrue( ole.getCause() instanceof StaleObjectStateException );
147147
}

0 commit comments

Comments
 (0)