org.firebirdsql.jdbc.InternalTransactionCoordinator Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jaybird Show documentation
Show all versions of jaybird Show documentation
JDBC Driver for the Firebird RDBMS
/*
* Firebird Open Source JDBC Driver
*
* Distributable under LGPL license.
* You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* LGPL License for more details.
*
* This file was created by members of the firebird development team.
* All individual contributions remain the Copyright (C) of those
* individuals. Contributors to this file are either listed here or
* can be obtained from a source control history command.
*
* All rights reserved.
*/
package org.firebirdsql.jdbc;
import org.firebirdsql.gds.JaybirdErrorCodes;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.LockCloseable;
import org.firebirdsql.gds.ng.StatementType;
import org.firebirdsql.jaybird.util.SQLExceptionChainBuilder;
import org.firebirdsql.jaybird.xca.FBLocalTransaction;
import org.firebirdsql.jaybird.xca.FBManagedConnection;
import org.firebirdsql.util.InternalApi;
import java.sql.SQLException;
import java.util.*;
import static java.util.Objects.requireNonNull;
import static org.firebirdsql.jdbc.SQLStateConstants.SQL_STATE_GENERAL_ERROR;
/**
* Transaction coordinator for the {@link org.firebirdsql.jdbc.FBConnection} class.
*
* This class is internal API of Jaybird. Future versions may radically change, move, or make inaccessible this type.
*
*/
@InternalApi
public final class InternalTransactionCoordinator implements FBObjectListener.StatementListener,
FBObjectListener.BlobListener {
private final FBConnection connection;
private AbstractTransactionCoordinator coordinator;
InternalTransactionCoordinator(FBConnection connection) {
this.connection = requireNonNull(connection, "connection");
}
private LockCloseable withLock() {
return connection.withLock();
}
/**
* Switches the auto commit coordinator if currently in a different auto commit mode.
*
* @param autoCommit
* Value for auto commit
* @throws SQLException
* If {@code autoCommit} is {@code true} and the connection is currently enlisted in a
* distributed transaction
*/
void switchTransactionCoordinator(boolean autoCommit) throws SQLException {
if (coordinator != null && getAutoCommit() == autoCommit) {
return;
}
setTransactionCoordinator(false, autoCommit);
}
/**
* Sets the transaction coordinator.
*
* In most cases {@link #switchTransactionCoordinator(boolean)} should be called.
*
*
* @param managedConnection
* Whether this is a managed connection
* @param autoCommit
* Value for auto commit
* @throws SQLException
* If {@code autoCommit} is {@code true}, and {@code managedConnection} is {@code false} and
* the connection is currently enlisted in a distributed transaction
*/
void setTransactionCoordinator(boolean managedConnection, boolean autoCommit) throws SQLException {
FBManagedConnection mc = connection.getManagedConnection();
InternalTransactionCoordinator.AbstractTransactionCoordinator coordinator;
if (managedConnection && mc.inDistributedTransaction()) {
coordinator = new ManagedTransactionCoordinator(connection);
} else if (autoCommit) {
if (mc.inDistributedTransaction()) {
throw new SQLException("Connection enlisted in distributed transaction",
SQLStateConstants.SQL_STATE_INVALID_TX_STATE);
}
if (connection.isUseFirebirdAutoCommit()) {
coordinator = new FirebirdAutoCommitCoordinator(connection, connection.getLocalTransaction());
} else {
coordinator = new AutoCommitCoordinator(connection, connection.getLocalTransaction());
}
} else {
coordinator = new LocalTransactionCoordinator(connection, connection.getLocalTransaction());
}
setCoordinator(coordinator);
}
public boolean getAutoCommit() throws SQLException {
try (LockCloseable ignored = withLock()) {
return coordinator != null && coordinator.isAutoCommit();
}
}
@Override
public void executionStarted(AbstractStatement stmt) throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.executionStarted(stmt);
}
}
@Override
public FBConnection getConnection() throws SQLException {
return connection;
}
@Override
public void statementClosed(AbstractStatement stmt) throws SQLException {
coordinator.statementClosed(stmt);
}
@Override
public void statementCompleted(AbstractStatement stmt) throws SQLException {
statementCompleted(stmt, true);
}
@Override
public void statementCompleted(AbstractStatement stmt, boolean success) throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.statementCompleted(stmt, success);
}
}
@Override
public void executionCompleted(FirebirdBlob blob) throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.executionCompleted(blob);
}
}
@Override
public void executionStarted(FirebirdBlob blob) throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.executionStarted(blob);
}
}
public void ensureTransaction() throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.ensureTransaction();
}
}
/**
* Starts a new transaction using {@code setTransactionSql}.
*
* @param sql
* the {@code SET TRANSACTION} statement to execute
* @throws SQLException
* if executing transaction management statements is not allowed ({@code allowTxStmts=false}),
* the connection is in auto-commit mode, or if a transaction is currently active (including distributed
* transactions)
* @since 6
*/
void startSqlTransaction(String sql) throws SQLException {
if (!connection.isAllowTxStmts()) {
throw FbExceptionBuilder.toNonTransientException(JaybirdErrorCodes.jb_setTransactionStatementNotAllowed);
}
try (LockCloseable ignored = withLock()) {
coordinator.startSqlTransaction(sql);
}
}
public void commit() throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.commit();
}
}
public void rollback() throws SQLException {
try (LockCloseable ignored = withLock()) {
coordinator.rollback();
}
}
void handleConnectionClose() throws SQLException {
coordinator.handleConnectionClose();
}
void handleConnectionAbort() throws SQLException {
coordinator.handleConnectionAbort();
}
private void setCoordinator(AbstractTransactionCoordinator coordinator) throws SQLException {
try (LockCloseable ignored = withLock()) {
if (this.coordinator != null) {
var chain = new SQLExceptionChainBuilder();
try {
this.coordinator.completeStatements(CompletionReason.COMMIT);
} catch (SQLException ex) {
chain.append(ex);
} finally {
try {
this.coordinator.internalCommit();
} catch (SQLException ex) {
chain.append(ex);
}
}
chain.throwIfPresent();
coordinator.setStatements(this.coordinator.getStatements());
}
this.coordinator = coordinator;
}
}
/**
* Transaction coordinator base implementation.
*
* This class is internal API of Jaybird. Future versions may radically change, move, or make inaccessible this type.
*
*/
public abstract static class AbstractTransactionCoordinator implements FBObjectListener.StatementListener,
FBObjectListener.BlobListener {
protected final FBLocalTransaction localTransaction;
protected final FBConnection connection;
protected final Collection statements = new HashSet<>();
protected AbstractTransactionCoordinator(FBConnection connection, FBLocalTransaction localTransaction) {
this.localTransaction = localTransaction;
this.connection = connection;
}
/**
* Get the connection which owns this coordinator.
*
* @return instance of {@link FBConnection}
*/
@Override
public final FBConnection getConnection() throws SQLException {
return connection;
}
protected final Collection getStatements() {
return statements;
}
protected final void setStatements(Collection statements) {
this.statements.addAll(statements);
}
protected void completeStatements(CompletionReason reason) throws SQLException {
var chain = new SQLExceptionChainBuilder();
// we have to loop through the array, since the statement.completeStatement() call causes
// ConcurrentModificationException
for (AbstractStatement statement : statements.toArray(new AbstractStatement[0])) {
try {
statement.completeStatement(reason);
} catch (SQLException ex) {
chain.append(ex);
}
}
// clear the statements (usually needed only for those that
// were not removed in the statement.completeStatement() call
statements.clear();
chain.throwIfPresent();
}
final void internalCommit() throws SQLException {
if (localTransaction != null && localTransaction.inTransaction()) {
localTransaction.commit();
}
}
final void internalRollback() throws SQLException {
if (localTransaction != null && localTransaction.inTransaction()) {
localTransaction.rollback();
}
}
public void ensureTransaction() throws SQLException {
configureFirebirdAutoCommit();
configureReadOnly();
if (!localTransaction.inTransaction()) {
localTransaction.begin();
}
}
private void configureFirebirdAutoCommit() throws SQLException {
// Handle Firebird autocommit support
if (connection.isUseFirebirdAutoCommit()) {
connection.getManagedConnection().setTpbAutoCommit(connection.getAutoCommit());
}
}
private void configureReadOnly() throws SQLException {
connection.getManagedConnection().setTpbReadOnly(connection.isReadOnly());
}
public abstract void commit() throws SQLException;
public abstract void rollback() throws SQLException;
abstract void handleConnectionClose() throws SQLException;
void handleConnectionAbort() throws SQLException {
completeStatements(CompletionReason.CONNECTION_ABORT);
}
boolean isAutoCommit() throws SQLException {
return false;
}
/**
* @see InternalTransactionCoordinator#startSqlTransaction(String)
*/
abstract void startSqlTransaction(String setTransactionSql) throws SQLException;
@Override
public final void statementCompleted(AbstractStatement stmt) throws SQLException {
statementCompleted(stmt, true);
}
}
static final class AutoCommitCoordinator extends AbstractTransactionCoordinator {
public AutoCommitCoordinator(FBConnection connection, FBLocalTransaction localTransaction) {
super(connection, localTransaction);
}
@Override
@SuppressWarnings("resource")
public void executionStarted(AbstractStatement stmt) throws SQLException {
List tempList = new ArrayList<>(statements);
var chain = new SQLExceptionChainBuilder();
// complete all open statements for the connection (there should be only one anyway)
for (Iterator iter = tempList.iterator(); iter.hasNext(); ) {
AbstractStatement tempStatement = iter.next();
// enable re-entrancy for the same statement
if (tempStatement == stmt) {
iter.remove();
continue;
}
// Autocommit, so reason of completion is COMMIT
try {
tempStatement.completeStatement(CompletionReason.COMMIT);
} catch (SQLException e) {
chain.append(e);
}
}
statements.removeAll(tempList);
chain.throwIfPresent();
if (!statements.contains(stmt)) {
statements.add(stmt);
}
ensureTransaction();
}
@Override
public void statementClosed(AbstractStatement stmt) throws SQLException {
stmt.completeStatement();
connection.notifyStatementClosed(stmt);
}
@Override
public void statementCompleted(AbstractStatement stmt, boolean success) throws SQLException {
statements.remove(stmt);
try {
if (!localTransaction.inTransaction()) {
return;
}
if (success) {
localTransaction.commit();
} else {
localTransaction.rollback();
}
} catch (SQLException ex) {
try {
internalRollback();
} catch (SQLException ex2) {
ex.setNextException(ex2);
}
throw ex;
}
}
@Override
public void executionStarted(FirebirdBlob blob) throws SQLException {
ensureTransaction();
}
@Override
public void commit() throws SQLException {
throw new SQLException("Calling commit() in auto-commit mode is not allowed.", SQL_STATE_GENERAL_ERROR);
}
@Override
public void rollback() throws SQLException {
throw new SQLException("Calling rollback() in auto-commit mode is not allowed.", SQL_STATE_GENERAL_ERROR);
}
@Override
void handleConnectionClose() throws SQLException {
if (localTransaction.inTransaction()) {
try {
completeStatements(CompletionReason.COMMIT);
} finally {
internalCommit();
}
}
}
@Override
boolean isAutoCommit() {
return true;
}
@Override
void startSqlTransaction(String setTransactionSql) throws SQLException {
throw FbExceptionBuilder.toNonTransientException(JaybirdErrorCodes.jb_setTransactionNotAllowedInAutoCommit);
}
}
static sealed class LocalTransactionCoordinator extends AbstractTransactionCoordinator {
public LocalTransactionCoordinator(FBConnection connection, FBLocalTransaction localTransaction) {
super(connection, localTransaction);
}
@Override
public void commit() throws SQLException {
try {
completeStatements(CompletionReason.COMMIT);
} finally {
internalCommit();
}
}
@Override
public void rollback() throws SQLException {
try {
completeStatements(CompletionReason.ROLLBACK);
} finally {
internalRollback();
}
}
@Override
void handleConnectionClose() throws SQLException {
if (localTransaction.inTransaction()) {
rollback();
}
}
@Override
public void executionStarted(AbstractStatement stmt) throws SQLException {
if (!statements.contains(stmt)) {
statements.add(stmt);
}
ensureTransaction();
}
@Override
public void statementClosed(AbstractStatement stmt) throws SQLException {
stmt.completeStatement();
connection.notifyStatementClosed(stmt);
}
@Override
public void statementCompleted(AbstractStatement stmt, boolean success) throws SQLException {
statements.remove(stmt);
}
@Override
public void executionStarted(FirebirdBlob blob) throws SQLException {
ensureTransaction();
}
@Override
void startSqlTransaction(String sql) throws SQLException {
localTransaction.begin(sql);
}
}
static final class FirebirdAutoCommitCoordinator extends LocalTransactionCoordinator {
public FirebirdAutoCommitCoordinator(FBConnection connection, FBLocalTransaction localTransaction) {
super(connection, localTransaction);
}
@Override
@SuppressWarnings("resource")
public void executionStarted(AbstractStatement stmt) throws SQLException {
List tempList = new ArrayList<>(statements);
var chain = new SQLExceptionChainBuilder();
// complete all open statements for the connection (there should be only one anyway)
for (Iterator iter = tempList.iterator(); iter.hasNext(); ) {
AbstractStatement tempStatement = iter.next();
// enable re-entrancy for the same statement
if (tempStatement == stmt) {
iter.remove();
continue;
}
// Firebird Autocommit doesn't really commit, but we use CompletionReason.COMMIT to get desired behavior
// for holdable result sets, and the performance win of not having a server roundtrip to close
// the cursor. This might lead to a slight cursor 'leak' until statement or connection close
try {
tempStatement.completeStatement(CompletionReason.COMMIT);
} catch (SQLException e) {
chain.append(e);
}
}
statements.removeAll(tempList);
chain.throwIfPresent();
if (!statements.contains(stmt)) {
statements.add(stmt);
}
ensureTransaction();
}
@Override
public void statementCompleted(AbstractStatement stmt, boolean success) throws SQLException {
statements.remove(stmt);
if (stmt.getStatementType() == StatementType.DDL.getStatementTypeCode()) {
try {
if (!localTransaction.inTransaction()) {
return;
}
if (success) {
localTransaction.commit();
} else {
localTransaction.rollback();
}
} catch (SQLException ex) {
try {
internalRollback();
} catch (SQLException ex2) {
ex.setNextException(ex2);
}
throw ex;
}
}
}
@Override
public void commit() throws SQLException {
throw new SQLException("Calling commit() in auto-commit mode is not allowed.", SQL_STATE_GENERAL_ERROR);
}
@Override
public void rollback() throws SQLException {
throw new SQLException("Calling rollback() in auto-commit mode is not allowed.", SQL_STATE_GENERAL_ERROR);
}
@Override
void handleConnectionClose() throws SQLException {
if (localTransaction.inTransaction()) {
try {
completeStatements(CompletionReason.COMMIT);
} finally {
internalCommit();
}
}
}
@Override
boolean isAutoCommit() {
return true;
}
@Override
void startSqlTransaction(String sql) throws SQLException {
throw FbExceptionBuilder.toNonTransientException(JaybirdErrorCodes.jb_setTransactionNotAllowedInAutoCommit);
}
}
static final class ManagedTransactionCoordinator extends LocalTransactionCoordinator {
/**
* Create instance of this class for the specified connection.
*
* @param connection
* connection to coordinate.
*/
public ManagedTransactionCoordinator(FBConnection connection) {
super(connection, null);
}
@Override
public void ensureTransaction() {
// do nothing, we are in managed environment.
}
@Override
public void executionStarted(AbstractStatement stmt) {
// NO TRANSACTION MANAGEMENT HERE - empty method
}
@Override
public void commit() throws SQLException {
// do nothing, we are in managed environment.
}
@Override
public void rollback() throws SQLException {
// do nothing, we are in managed environment.
}
@Override
public void executionStarted(FirebirdBlob blob) {
// nothing to do
}
@Override
void handleConnectionClose() {
// do nothing, we are in a managed environment
}
@Override
void startSqlTransaction(String sql) throws SQLException {
throw FbExceptionBuilder.toException(JaybirdErrorCodes.jb_setTransactionNotAllowedActiveTx);
}
}
static final class MetaDataTransactionCoordinator extends AbstractTransactionCoordinator {
private final InternalTransactionCoordinator tc;
public MetaDataTransactionCoordinator(InternalTransactionCoordinator tc) {
super(tc.connection, tc.connection.getLocalTransaction());
this.tc = tc;
}
@Override
public void ensureTransaction() {
throw new UnsupportedOperationException("ensureTransaction");
}
@Override
public void commit() throws SQLException {
throw new UnsupportedOperationException("commit");
}
@Override
public void rollback() throws SQLException {
throw new UnsupportedOperationException("rollback");
}
@Override
void handleConnectionClose() {
throw new UnsupportedOperationException();
}
@Override
public void executionStarted(AbstractStatement stmt) throws SQLException {
tc.ensureTransaction();
}
@Override
public void statementClosed(AbstractStatement stmt) throws SQLException {
stmt.completeStatement();
}
@Override
public void statementCompleted(AbstractStatement stmt, boolean success) {
// nothing to do
}
@Override
boolean isAutoCommit() throws SQLException {
return tc.getAutoCommit();
}
@Override
void startSqlTransaction(String setTransactionSql) {
throw new UnsupportedOperationException("startSqlTransaction");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy