TransactionContext.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.dbcp2.managed;

  18. import java.lang.ref.WeakReference;
  19. import java.sql.Connection;
  20. import java.sql.SQLException;
  21. import java.util.Objects;

  22. import javax.transaction.RollbackException;
  23. import javax.transaction.Status;
  24. import javax.transaction.Synchronization;
  25. import javax.transaction.SystemException;
  26. import javax.transaction.Transaction;
  27. import javax.transaction.TransactionSynchronizationRegistry;
  28. import javax.transaction.xa.XAResource;

  29. /**
  30.  * TransactionContext represents the association between a single XAConnectionFactory and a Transaction. This context
  31.  * contains a single shared connection which should be used by all ManagedConnections for the XAConnectionFactory, the
  32.  * ability to listen for the transaction completion event, and a method to check the status of the transaction.
  33.  *
  34.  * @since 2.0
  35.  */
  36. public class TransactionContext {
  37.     private final TransactionRegistry transactionRegistry;
  38.     private final WeakReference<Transaction> transactionRef;
  39.     private final TransactionSynchronizationRegistry transactionSynchronizationRegistry;
  40.     private Connection sharedConnection;
  41.     private boolean transactionComplete;

  42.     /**
  43.      * Provided for backwards compatibility
  44.      *
  45.      * @param transactionRegistry the TransactionRegistry used to obtain the XAResource for the
  46.      * shared connection
  47.      * @param transaction the transaction
  48.      */
  49.     public TransactionContext(final TransactionRegistry transactionRegistry, final Transaction transaction) {
  50.         this (transactionRegistry, transaction, null);
  51.     }

  52.     /**
  53.      * Creates a TransactionContext for the specified Transaction and TransactionRegistry. The TransactionRegistry is
  54.      * used to obtain the XAResource for the shared connection when it is enlisted in the transaction.
  55.      *
  56.      * @param transactionRegistry
  57.      *            the TransactionRegistry used to obtain the XAResource for the shared connection
  58.      * @param transaction
  59.      *            the transaction
  60.      * @param transactionSynchronizationRegistry
  61.      *              The optional TSR to register synchronizations with
  62.      * @since 2.6.0
  63.      */
  64.     public TransactionContext(final TransactionRegistry transactionRegistry, final Transaction transaction,
  65.                               final TransactionSynchronizationRegistry transactionSynchronizationRegistry) {
  66.         Objects.requireNonNull(transactionRegistry, "transactionRegistry");
  67.         Objects.requireNonNull(transaction, "transaction");
  68.         this.transactionRegistry = transactionRegistry;
  69.         this.transactionRef = new WeakReference<>(transaction);
  70.         this.transactionComplete = false;
  71.         this.transactionSynchronizationRegistry = transactionSynchronizationRegistry;
  72.     }

  73.     /**
  74.      * Adds a listener for transaction completion events.
  75.      *
  76.      * @param listener
  77.      *            the listener to add
  78.      * @throws SQLException
  79.      *             if a problem occurs adding the listener to the transaction
  80.      */
  81.     public void addTransactionContextListener(final TransactionContextListener listener) throws SQLException {
  82.         try {
  83.             if (!isActive()) {
  84.                 final Transaction transaction = this.transactionRef.get();
  85.                 listener.afterCompletion(this, transaction != null && transaction.getStatus() == Status.STATUS_COMMITTED);
  86.                 return;
  87.             }
  88.             final Synchronization s = new SynchronizationAdapter() {
  89.                 @Override
  90.                 public void afterCompletion(final int status) {
  91.                     listener.afterCompletion(TransactionContext.this, status == Status.STATUS_COMMITTED);
  92.                 }
  93.             };
  94.             if (transactionSynchronizationRegistry != null) {
  95.                 transactionSynchronizationRegistry.registerInterposedSynchronization(s);
  96.             } else {
  97.                 getTransaction().registerSynchronization(s);
  98.             }
  99.         } catch (final RollbackException ignored) {
  100.             // JTA spec doesn't let us register with a transaction marked rollback only
  101.             // just ignore this and the tx state will be cleared another way.
  102.         } catch (final Exception e) {
  103.             throw new SQLException("Unable to register transaction context listener", e);
  104.         }
  105.     }

  106.     /**
  107.      * Sets the transaction complete flag to true.
  108.      *
  109.      * @since 2.4.0
  110.      */
  111.     public void completeTransaction() {
  112.         this.transactionComplete = true;
  113.     }

  114.     /**
  115.      * Gets the connection shared by all ManagedConnections in the transaction. Specifically, connection using the same
  116.      * XAConnectionFactory from which the TransactionRegistry was obtained.
  117.      *
  118.      * @return the shared connection for this transaction
  119.      */
  120.     public Connection getSharedConnection() {
  121.         return sharedConnection;
  122.     }

  123.     private Transaction getTransaction() throws SQLException {
  124.         final Transaction transaction = this.transactionRef.get();
  125.         if (transaction == null) {
  126.             throw new SQLException("Unable to enlist connection because the transaction has been garbage collected");
  127.         }
  128.         return transaction;
  129.     }

  130.     /**
  131.      * True if the transaction is active or marked for rollback only.
  132.      *
  133.      * @return true if the transaction is active or marked for rollback only; false otherwise
  134.      * @throws SQLException
  135.      *             if a problem occurs obtaining the transaction status
  136.      */
  137.     public boolean isActive() throws SQLException {
  138.         try {
  139.             final Transaction transaction = this.transactionRef.get();
  140.             if (transaction == null) {
  141.                 return false;
  142.             }
  143.             final int status = transaction.getStatus();
  144.             return status == Status.STATUS_ACTIVE || status == Status.STATUS_MARKED_ROLLBACK;
  145.         } catch (final SystemException e) {
  146.             throw new SQLException("Unable to get transaction status", e);
  147.         }
  148.     }

  149.     /**
  150.      * Gets the transaction complete flag to true.
  151.      *
  152.      * @return The transaction complete flag.
  153.      *
  154.      * @since 2.4.0
  155.      */
  156.     public boolean isTransactionComplete() {
  157.         return this.transactionComplete;
  158.     }

  159.     /**
  160.      * Sets the shared connection for this transaction. The shared connection is enlisted in the transaction.
  161.      *
  162.      * @param sharedConnection
  163.      *            the shared connection
  164.      * @throws SQLException
  165.      *             if a shared connection is already set, if XAResource for the connection could not be found in the
  166.      *             transaction registry, or if there was a problem enlisting the connection in the transaction
  167.      */
  168.     public void setSharedConnection(final Connection sharedConnection) throws SQLException {
  169.         if (this.sharedConnection != null) {
  170.             throw new IllegalStateException("A shared connection is already set");
  171.         }

  172.         // This is the first use of the connection in this transaction, so we must
  173.         // enlist it in the transaction
  174.         final Transaction transaction = getTransaction();
  175.         try {
  176.             final XAResource xaResource = transactionRegistry.getXAResource(sharedConnection);
  177.             if (!transaction.enlistResource(xaResource)) {
  178.                 throw new SQLException("Unable to enlist connection in transaction: enlistResource returns 'false'.");
  179.             }
  180.         } catch (final IllegalStateException e) {
  181.             // This can happen if the transaction is already timed out
  182.             throw new SQLException("Unable to enlist connection in the transaction", e);
  183.         } catch (final RollbackException ignored) {
  184.             // transaction was rolled back... proceed as if there never was a transaction
  185.         } catch (final SystemException e) {
  186.             throw new SQLException("Unable to enlist connection the transaction", e);
  187.         }

  188.         this.sharedConnection = sharedConnection;
  189.     }
  190. }