Open Source Repository

Home /hibernate/hibernate-3.2.7.ga | Repository Home



org/hibernate/transaction/JDBCTransaction.java
//$Id: JDBCTransaction.java 9595 2006-03-10 18:14:21Z [email protected] $
package org.hibernate.transaction;

import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;

import javax.transaction.Status;
import javax.transaction.Synchronization;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.hibernate.HibernateException;
import org.hibernate.Transaction;
import org.hibernate.TransactionException;
import org.hibernate.jdbc.JDBCContext;

/**
 * Implements a basic transaction strategy for JDBC connections.This is the
 * default <tt>Transaction</tt> implementation used if none is explicitly
 * specified.
 @author Anton van Straaten, Gavin King
 */
public class JDBCTransaction implements Transaction {

  private static final Log log = LogFactory.getLog(JDBCTransaction.class);

  private final JDBCContext jdbcContext;
  private final TransactionFactory.Context transactionContext;

  private boolean toggleAutoCommit;
  private boolean begun;
  private boolean rolledBack;
  private boolean committed;
  private boolean commitFailed;
  private List synchronizations;
  private boolean callback;
  private int timeout = -1;

  public JDBCTransaction(JDBCContext jdbcContext, TransactionFactory.Context transactionContext) {
    this.jdbcContext = jdbcContext;
    this.transactionContext = transactionContext;
  }

  public void begin() throws HibernateException {
    if (begun) {
      return;
    }
    if (commitFailed) {
      throw new TransactionException("cannot re-start transaction after failed commit");
    }

    log.debug("begin");

    try {
      toggleAutoCommit = jdbcContext.connection().getAutoCommit();
      if log.isDebugEnabled() ) {
        log.debug("current autocommit status: " + toggleAutoCommit);
      }
      if (toggleAutoCommit) {
        log.debug("disabling autocommit");
        jdbcContext.connection().setAutoCommit(false);
      }
    }
    catch (SQLException e) {
      log.error("JDBC begin failed", e);
      throw new TransactionException("JDBC begin failed: ", e);
    }

    callback = jdbcContext.registerCallbackIfNecessary();

    begun = true;
    committed = false;
    rolledBack = false;

    if timeout>) {
      jdbcContext.getConnectionManager()
          .getBatcher()
          .setTransactionTimeout(timeout);
    }

    jdbcContext.afterTransactionBegin(this);
  }

  private void closeIfRequired() throws HibernateException {
    if callback && transactionContext.shouldAutoClose() && !transactionContext.isClosed() ) {
      try {
        transactionContext.managedClose();
      }
      catch (HibernateException he) {
        log.error("Could not close session", he);
        //swallow, the transaction was finished
      }
    }
  }

  public void commit() throws HibernateException {
    if (!begun) {
      throw new TransactionException("Transaction not successfully started");
    }

    log.debug("commit");

    if !transactionContext.isFlushModeNever() && callback ) {
      transactionContext.managedFlush()//if an exception occurs during flush, user must call rollback()
    }

    notifyLocalSynchsBeforeTransactionCompletion();
    if callback ) {
      jdbcContext.beforeTransactionCompletionthis );
    }

    try {
      commitAndResetAutoCommit();
      log.debug("committed JDBC Connection");
      committed = true;
      if callback ) {
        jdbcContext.afterTransactionCompletiontrue, this );
      }
      notifyLocalSynchsAfterTransactionCompletionStatus.STATUS_COMMITTED );
    }
    catch (SQLException e) {
      log.error("JDBC commit failed", e);
      commitFailed = true;
      if callback ) {
        jdbcContext.afterTransactionCompletionfalse, this );
      }
      notifyLocalSynchsAfterTransactionCompletionStatus.STATUS_UNKNOWN );
      throw new TransactionException("JDBC commit failed", e);
    }
    finally {
      closeIfRequired();
    }
  }

  private void commitAndResetAutoCommit() throws SQLException {
    try {
      jdbcContext.connection().commit();
    }
    finally {
      toggleAutoCommit();
    }
  }

  public void rollback() throws HibernateException {

    if (!begun && !commitFailed) {
      throw new TransactionException("Transaction not successfully started");
    }

    log.debug("rollback");

    if (!commitFailed) {

      /*notifyLocalSynchsBeforeTransactionCompletion();
      if ( callback ) {
        jdbcContext.notifyLocalSynchsBeforeTransactionCompletion( this );
      }*/

      try {
        rollbackAndResetAutoCommit();
        log.debug("rolled back JDBC Connection");
        rolledBack = true;
        notifyLocalSynchsAfterTransactionCompletion(Status.STATUS_ROLLEDBACK);
      }
      catch (SQLException e) {
        log.error("JDBC rollback failed", e);
        notifyLocalSynchsAfterTransactionCompletion(Status.STATUS_UNKNOWN);
        throw new TransactionException("JDBC rollback failed", e);
      }
      finally {
        if callback ) {
          jdbcContext.afterTransactionCompletionfalse, this );
        }
        closeIfRequired();
      }
    }
  }

  private void rollbackAndResetAutoCommit() throws SQLException {
    try {
      jdbcContext.connection().rollback();
    }
    finally {
      toggleAutoCommit();
    }
  }

  private void toggleAutoCommit() {
    try {
      if (toggleAutoCommit) {
        log.debug("re-enabling autocommit");
        jdbcContext.connection().setAutoCommittrue );
      }
    }
    catch (Exception sqle) {
      log.error("Could not toggle autocommit", sqle);
      //swallow it (the transaction _was_ successful or successfully rolled back)
    }
  }

  public boolean wasRolledBack() {
    return rolledBack;
  }

  public boolean wasCommitted() {
    return committed;
  }

  public boolean isActive() {
    return begun && ! rolledBack || committed | commitFailed );
  }

  public void registerSynchronization(Synchronization syncthrows HibernateException {
    if (sync==nullthrow new NullPointerException("null Synchronization");
    if (synchronizations==null) {
      synchronizations = new ArrayList();
    }
    synchronizations.add(sync);
  }

  private void notifyLocalSynchsBeforeTransactionCompletion() {
    if (synchronizations!=null) {
      for int i=0; i<synchronizations.size(); i++ ) {
        Synchronization sync = (Synchronizationsynchronizations.get(i);
        try {
          sync.beforeCompletion();
        }
        catch (Throwable t) {
          log.error("exception calling user Synchronization", t);
        }
      }
    }
  }

  private void notifyLocalSynchsAfterTransactionCompletion(int status) {
    begun = false;
    if (synchronizations!=null) {
      for int i=0; i<synchronizations.size(); i++ ) {
        Synchronization sync = (Synchronizationsynchronizations.get(i);
        try {
          sync.afterCompletion(status);
        }
        catch (Throwable t) {
          log.error("exception calling user Synchronization", t);
        }
      }
    }
  }

  public void setTimeout(int seconds) {
    timeout = seconds;
  }
}