Package org.jboss.jca.core.connectionmanager.transaction

Source Code of org.jboss.jca.core.connectionmanager.transaction.TransactionSynchronizer

/*
* JBoss, Home of Professional Open Source.
* Copyright 2006, Red Hat Middleware LLC, and individual contributors
* as indicated by the @author tags. See the copyright.txt file in the
* distribution for a full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software 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 GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.jca.core.connectionmanager.transaction;

import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

import javax.transaction.RollbackException;
import javax.transaction.Synchronization;
import javax.transaction.SystemException;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;

import org.jboss.logging.Logger;
import org.jboss.tm.TransactionLocal;
import org.jboss.util.NestedRuntimeException;

/**
* Organizes transaction synchronization done by JCA.
*
* <p>
* This class exists to make sure all Tx synchronizations
* are invoked before the cached connection manager closes any
* closed connections.
*
* @author <a href="mailto:adrian@jboss.org">Adrian Brock</a>
* @author gurkanerdogdu
* @version $Rev: 109096 $
*/
public class TransactionSynchronizer implements Synchronization
{
   /** The logger */
   private static Logger log = Logger.getLogger(TransactionSynchronizer.class);

   /** The transaction synchronizations */
   private static TransactionLocal txSynchs;
  
   /** The transaction */
   private Transaction tx;
  
   /** The enlisting thread */
   private Thread enlistingThread;
  
   /** Unenlisted */
   private CopyOnWriteArrayList<Synchronization> unenlisted = new CopyOnWriteArrayList<Synchronization>();
  
   /** Enlisted */
   private CopyOnWriteArrayList<Synchronization> enlisted = new CopyOnWriteArrayList<Synchronization>();
  
   /** The cached connection manager synchronization */
   private Synchronization ccmSynch;
  
   /**Lock*/
   private ReentrantLock lockObject = new ReentrantLock(true);
  
   /**Condition*/
   private Condition condition = this.lockObject.newCondition();

   /**
    * Initialization.
    * @param tm transaction manager
    */
   public static void setTransactionManager(TransactionManager tm)
   {
      txSynchs = new TransactionLocal(tm);
   }
  
   /**
    * Create a new transaction synchronizer
    *
    * @param tx the transaction to synchronize with
    */
   private TransactionSynchronizer(Transaction tx)
   {
      this.tx = tx;
   }
  
   /**
    * Add a new Tx synchronization that has not been enlisted
    *
    * @param synch the synchronization
    */
   public void addUnenlisted(Synchronization synch)
   {
      unenlisted.add(synch);
   }
  
   /**
    * Get the unenlisted synchronizations
    * and say we are enlisting if some are returned.
    *
    * @return the unenlisted synchronizations
    */
   public List<Synchronization> getUnenlisted()
   {
      Thread currentThread = Thread.currentThread();
     
      while (enlistingThread != null && enlistingThread != currentThread)
      {
         boolean interrupted = false;
         try
         {
            this.lockObject.lock();
           
            this.condition.await();
         }
         catch (InterruptedException e)
         {
            interrupted = true;
         }        
         finally
         {
            this.lockObject.unlock();
         }
        
         if (interrupted)
         {
            currentThread.interrupt()
         }
      }

      CopyOnWriteArrayList<Synchronization> result = unenlisted;
     
      unenlisted = null;
     
      if (result != null)
      {
         enlistingThread = currentThread; 
      }
      return result;
   }
  
   /**
    * The synchronization is now enlisted
    *
    * @param synch the synchronization
    */
   public void addEnlisted(Synchronization synch)
   {
      enlisted.add(synch);
   }
  
   /**
    * Remove an enlisted synchronization
    *
    * @param synch the synchronization
    * @return true when the synchronization was enlisted
    */
   public boolean removeEnlisted(Synchronization synch)
   {
      return enlisted.remove(synch);
   }
  
   /**
    * This thread has finished enlisting.
    */
   public void enlisted()
   {
      try
      {
         this.lockObject.lock();
        
         Thread currentThread = Thread.currentThread();
        
         if (enlistingThread == null || enlistingThread != currentThread)
         {
            log.warn("Thread " + currentThread + " not the enlisting thread " +
                  enlistingThread, new Exception("STACKTRACE"));
           
            return;
         }
        
         enlistingThread = null;
        
         this.condition.signalAll();
      }
      finally
      {
         this.lockObject.unlock();
      }     
   }
  
   /**
    * Get a registered transaction synchronizer.
    *
    * @param tx the transaction
    * @throws SystemException sys. exception
    * @throws RollbackException rollback exception
    * @return the registered transaction synchronizer for this transaction
    */
   public static TransactionSynchronizer getRegisteredSynchronizer(Transaction tx)
      throws SystemException, RollbackException
   {
      TransactionSynchronizer result = (TransactionSynchronizer) txSynchs.get(tx);
      if (result == null)
      {
         result = new TransactionSynchronizer(tx);
         tx.registerSynchronization(result);
         txSynchs.set(tx, result);
      }
      return result;
   }
  
   /**
    * Check whether we have a CCM synchronization
    *
    * @param tx the transaction
    * @return synch
    */
   public static Synchronization getCCMSynchronization(Transaction tx)
   {
      TransactionSynchronizer ts = (TransactionSynchronizer) txSynchs.get(tx);
      if (ts != null)
      {
         return ts.ccmSynch; 
      }
      else
      {
         return null
      }
   }
  
   /**
    * Register a new CCM synchronization
    *
    * @param tx the transaction
    * @param synch the synchronization
    * @throws Exception e
    */
   public static void registerCCMSynchronization(Transaction tx, Synchronization synch)
      throws Exception
   {
      TransactionSynchronizer ts = getRegisteredSynchronizer(tx);
      ts.ccmSynch = synch;
   }
  
   /**
    * Lock for the given transaction
    *
    * @param tx the transaction
    */
   public static void lock(Transaction tx)
   {
      try
      {
         txSynchs.lock(tx);
      }
      catch (InterruptedException e)
      {
         throw new NestedRuntimeException("Unable to get synchronization", e);
      }
   }
  
   /**
    * Unlock for the given transaction
    *
    * @param tx the transaction
    */
   public static void unlock(Transaction tx)
   {
      txSynchs.unlock(tx);
   }

   /**
    * {@inheritDoc}
    */
   public void beforeCompletion()
   {
      if (enlisted != null)
      {
         int i = 0;
         while (i < enlisted.size())
         {
            Synchronization synch = enlisted.get(i);
            invokeBefore(synch);
            ++i;
         }
      }
     
      if (ccmSynch != null)
      {
         invokeBefore(ccmSynch)
      }
   }

   /**
    * {@inheritDoc}
    */
   public void afterCompletion(int status)
   {
      if (enlisted != null)
      {
         int i = 0;
         while (i < enlisted.size())
         {
            Synchronization synch = enlisted.get(i);
            invokeAfter(synch, status);
            ++i;
         }
      }
     
      if (ccmSynch != null)
      {
         invokeAfter(ccmSynch, status)
      }
   }

   /**
    * Invoke a beforeCompletion
    *
    * @param synch the synchronization
    */
   protected void invokeBefore(Synchronization synch)
   {
      try
      {
         synch.beforeCompletion();
      }
      catch (Throwable t)
      {
         log.warn("Transaction " + tx + " error in before completion " + synch, t);
      }
   }

   /**
    * Invoke an afterCompletion
    *
    * @param synch the synchronization
    * @param status the status of the transaction
    */
   protected void invokeAfter(Synchronization synch, int status)
   {
      try
      {
         synch.afterCompletion(status);
      }
      catch (Throwable t)
      {
         log.warn("Transaction " + tx + " error in after completion " + synch, t);
      }
   }  
  
}
TOP

Related Classes of org.jboss.jca.core.connectionmanager.transaction.TransactionSynchronizer

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.