com.j256.ormlite.misc.TransactionManager Maven / Gradle / Ivy
Show all versions of ormlite-core Show documentation
package com.j256.ormlite.misc;
import java.sql.SQLException;
import java.sql.Savepoint;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicInteger;
import com.j256.ormlite.db.DatabaseType;
import com.j256.ormlite.logger.Logger;
import com.j256.ormlite.logger.LoggerFactory;
import com.j256.ormlite.support.ConnectionSource;
import com.j256.ormlite.support.DatabaseConnection;
/**
*
* Provides basic transaction support for a {@link ConnectionSource}.
*
*
*
* NOTE: For transactions to work, the database being used must support the functionality.
*
*
*
* NOTE: If you are using the Spring type wiring in Java, {@link #initialize} should be called after all of the
* set methods. In Spring XML, init-method="initialize" should be used.
*
*
*
* You can call this as an instance with a new TransactionManager(dataSource); or you can call it as a static like the
* below example:
*
*
*
* TransactionManager.callInTransaction(dataSource, new Callable<Void>() {
* public Void call() throws Exception {
* // delete both objects but make sure that if either one fails, the transaction is rolled back
* // and both objects are "restored" to the database
* fooDao.delete(foo);
* barDao.delete(bar);
* return null;
* }
* });
*
*
*
* For Spring wiring of a Transaction Manager bean, we would do something like the following:
*
*
*
* <bean id="transactionManager" class="com.j256.ormlite.misc.TransactionManager" init-method="initialize">
* <property name="dataSource" ref="dataSource" />
* </bean>
*
*
*
* WARNING: For most of the methods in this class, it is up to you to properly synchronize them if multiple threads are
* using a single database connection -- this includes a connection-source which works gives out a single-connection.
* The reason why this is necessary is that multiple operations are performed on the connection and race-conditions will
* exist with multiple threads working on the same connection.
*
*
* @author graywatson
*/
public class TransactionManager {
private static final Logger logger = LoggerFactory.getLogger(TransactionManager.class);
private static final String SAVE_POINT_PREFIX = "ORMLITE";
private ConnectionSource connectionSource;
private static final AtomicInteger savePointCounter = new AtomicInteger();
/** used to track our transaction level so we know when we are in the commit-able outer transaction */
private static final ThreadLocal transactionLevelThreadLocal =
new ThreadLocal() {
@Override
protected TransactionLevel initialValue() {
return new TransactionLevel();
}
};
/**
* Constructor for Spring type wiring if you are using the set methods.
*/
public TransactionManager() {
// for spring wiring -- must call setDataSource()
}
/**
* Constructor for direct java code wiring.
*/
public TransactionManager(ConnectionSource connectionSource) {
this.connectionSource = connectionSource;
initialize();
}
/**
* If you are using the Spring type wiring, this should be called after all of the set methods.
*/
public void initialize() {
if (connectionSource == null) {
throw new IllegalStateException("dataSource was not set on " + getClass().getSimpleName());
}
}
/**
* Execute the {@link Callable} class inside of a transaction. If the callable returns then the transaction is
* committed. If the callable throws an exception then the transaction is rolled back and a {@link SQLException} is
* thrown by this method.
*
*
* NOTE: If your callable block really doesn't have a return object then use the Void class and return null
* from the call method.
*
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using a
* connection-source which works gives out a single-connection. The reason why this is necessary is that multiple
* operations are performed on the connection and race-conditions will exist with multiple threads working on the
* same connection.
*
*
* @param callable
* Callable to execute inside of the transaction.
* @return The object returned by the callable.
* @throws SQLException
* If the callable threw an exception then the transaction is rolled back and a SQLException wraps the
* callable exception and is thrown by this method.
*/
public T callInTransaction(final Callable callable) throws SQLException {
return callInTransaction(connectionSource, callable);
}
/**
* Same as {@link #callInTransaction(Callable)} except as a this has a table-name specified.
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using a
* connection-source which works gives out a single-connection. The reason why this is necessary is that multiple
* operations are performed on the connection and race-conditions will exist with multiple threads working on the
* same connection.
*
*/
public T callInTransaction(String tableName, final Callable callable) throws SQLException {
return callInTransaction(tableName, connectionSource, callable);
}
/**
* Same as {@link #callInTransaction(Callable)} except as a static method with a connection source.
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using a
* connection-source which works gives out a single-connection. The reason why this is necessary is that multiple
* operations are performed on the connection and race-conditions will exist with multiple threads working on the
* same connection.
*
*/
public static T callInTransaction(final ConnectionSource connectionSource, final Callable callable)
throws SQLException {
return callInTransaction(null, connectionSource, callable);
}
/**
* Same as {@link #callInTransaction(ConnectionSource, Callable)} except this has a table-name.
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using a
* connection-source which works gives out a single-connection. The reason why this is necessary is that multiple
* operations are performed on the connection and race-conditions will exist with multiple threads working on the
* same connection.
*
*/
public static T callInTransaction(String tableName, final ConnectionSource connectionSource,
final Callable callable) throws SQLException {
DatabaseConnection connection = connectionSource.getReadWriteConnection(tableName);
try {
boolean saved = connectionSource.saveSpecialConnection(connection);
return callInTransaction(connection, saved, connectionSource.getDatabaseType(), callable);
} finally {
// we should clear aggressively
connectionSource.clearSpecialConnection(connection);
connectionSource.releaseConnection(connection);
}
}
/**
* Same as {@link #callInTransaction(Callable)} except as a static method on a connection with database-type.
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using the same
* database connection or if your connection-source is single-connection. The reason why this is necessary is that
* multiple operations are performed on the connection and race-conditions will exist with multiple threads working
* on the same connection.
*
*/
public static T callInTransaction(final DatabaseConnection connection, final DatabaseType databaseType,
final Callable callable) throws SQLException {
return callInTransaction(connection, false, databaseType, callable);
}
/**
* Same as {@link #callInTransaction(Callable)} except as a static method on a connection with database-type.
*
*
* WARNING: it is up to you to properly synchronize around this method if multiple threads are using the same
* database connection or if your connection-source is single-connection. The reason why this is necessary is that
* multiple operations are performed on the connection and race-conditions will exist with multiple threads working
* on the same connection.
*
*/
public static T callInTransaction(final DatabaseConnection connection, boolean saved,
final DatabaseType databaseType, final Callable callable) throws SQLException {
boolean restoreAutoCommit = false;
TransactionLevel levelCount = transactionLevelThreadLocal.get();
try {
boolean hasSavePoint = false;
Savepoint savePoint = null;
if (saved || databaseType.isNestedSavePointsSupported()) {
if (connection.isAutoCommitSupported()) {
if (connection.isAutoCommit()) {
// disable auto-commit mode if supported and enabled at start
connection.setAutoCommit(false);
restoreAutoCommit = true;
logger.trace("had to set auto-commit to false");
}
}
savePoint = connection.setSavePoint(SAVE_POINT_PREFIX + savePointCounter.incrementAndGet());
if (savePoint == null) {
logger.trace("started savePoint transaction");
} else {
logger.trace("started savePoint transaction {}", savePoint.getSavepointName());
}
hasSavePoint = true;
}
try {
levelCount.incrementAndGet();
T result = callable.call();
int level = levelCount.decrementAndGet();
if (level <= 0) {
transactionLevelThreadLocal.remove();
levelCount = null;
}
if (hasSavePoint) {
// only commit if we have reached the end of our transaction stack
if (level <= 0) {
commit(connection, savePoint);
} else {
// otherwise we just release the savepoint
release(connection, savePoint);
}
}
return result;
} catch (Exception e) {
if (levelCount != null && levelCount.decrementAndGet() <= 0) {
transactionLevelThreadLocal.remove();
}
if (hasSavePoint) {
try {
rollBack(connection, savePoint);
} catch (SQLException e2) {
logger.error(e, "after commit exception, rolling back to save-point also threw exception");
// we continue to throw the commit exception
}
}
if (e instanceof SQLException) {
throw (SQLException) e;
} else {
throw SqlExceptionUtil.create("Transaction callable threw non-SQL exception", e);
}
}
} finally {
if (restoreAutoCommit) {
// try to restore if we are in auto-commit mode
connection.setAutoCommit(true);
logger.trace("restored auto-commit to true");
}
}
}
public void setConnectionSource(ConnectionSource connectionSource) {
this.connectionSource = connectionSource;
}
private static void commit(DatabaseConnection connection, Savepoint savePoint) throws SQLException {
String name = (savePoint == null ? null : savePoint.getSavepointName());
connection.commit(savePoint);
if (name == null) {
logger.trace("committed savePoint transaction");
} else {
logger.trace("committed savePoint transaction {}", name);
}
}
private static void release(DatabaseConnection connection, Savepoint savePoint) throws SQLException {
String name = (savePoint == null ? null : savePoint.getSavepointName());
connection.releaseSavePoint(savePoint);
if (name == null) {
logger.trace("released savePoint transaction");
} else {
logger.trace("released savePoint transaction {}", name);
}
}
private static void rollBack(DatabaseConnection connection, Savepoint savePoint) throws SQLException {
String name = (savePoint == null ? null : savePoint.getSavepointName());
connection.rollback(savePoint);
if (name == null) {
logger.trace("rolled back savePoint transaction");
} else {
logger.trace("rolled back savePoint transaction {}", name);
}
}
/**
* Mutable int for tracking transaction level.
*/
private static class TransactionLevel {
int counter;
int incrementAndGet() {
return ++counter;
}
int decrementAndGet() {
return --counter;
}
}
}