Package org.jboss.seam.ioc.spring

Source Code of org.jboss.seam.ioc.spring.SpringTransaction$JtaSpringSynchronizationAdapter

package org.jboss.seam.ioc.spring;

import static org.jboss.seam.annotations.Install.FRAMEWORK;

import javax.persistence.EntityManager;
import javax.transaction.HeuristicMixedException;
import javax.transaction.HeuristicRollbackException;
import javax.transaction.NotSupportedException;
import javax.transaction.RollbackException;
import javax.transaction.Status;
import javax.transaction.Synchronization;
import javax.transaction.SystemException;

import org.jboss.seam.ScopeType;
import org.jboss.seam.annotations.Destroy;
import org.jboss.seam.annotations.Install;
import org.jboss.seam.annotations.Name;
import org.jboss.seam.annotations.Scope;
import org.jboss.seam.annotations.intercept.BypassInterceptors;
import org.jboss.seam.core.Expressions.ValueExpression;
import org.jboss.seam.log.LogProvider;
import org.jboss.seam.log.Logging;
import org.jboss.seam.transaction.AbstractUserTransaction;
import org.springframework.orm.jpa.JpaTransactionManager;
import org.springframework.transaction.HeuristicCompletionException;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionException;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.TransactionSystemException;
import org.springframework.transaction.UnexpectedRollbackException;
import org.springframework.transaction.support.DefaultTransactionDefinition;
import org.springframework.transaction.support.TransactionSynchronizationAdapter;
import org.springframework.transaction.support.TransactionSynchronizationManager;

@Name("org.jboss.seam.transaction.transaction")
@Scope(ScopeType.EVENT)
@Install(value = false, precedence = FRAMEWORK)
@BypassInterceptors
public class SpringTransaction extends AbstractUserTransaction
{
   private static final LogProvider log = Logging.getLogProvider(SpringTransaction.class);

   private ValueExpression<PlatformTransactionManager> platformTransactionManager;

   private DefaultTransactionDefinition definition = new DefaultTransactionDefinition();

   private boolean conversationContextRequired = true;

   private TransactionStatus currentTransaction;

   private Boolean joinTransaction;

   @Override
   public void registerSynchronization(final Synchronization sync)
   {
      if (TransactionSynchronizationManager.isSynchronizationActive())
      {
         TransactionSynchronizationManager
                  .registerSynchronization(new JtaSpringSynchronizationAdapter(sync));
      }
      else
      {
         throw new IllegalStateException(
                  "TransactionSynchronization not available with this Spring Transaction Manager");
      }
   }

   public void begin() throws NotSupportedException, SystemException
   {
      if (TransactionSynchronizationManager.isActualTransactionActive())
      {
         throw new NotSupportedException("A Spring transaction is already active.");
      }
      currentTransaction = platformTransactionManager.getValue().getTransaction(definition);
   }

   public void commit() throws RollbackException, HeuristicMixedException,
            HeuristicRollbackException, SecurityException, IllegalStateException, SystemException
   {
      assertActive();
      try
      {
         platformTransactionManager.getValue().commit(currentTransaction);
      }
      catch (HeuristicCompletionException e)
      {

         switch (e.getOutcomeState())
         {
            case HeuristicCompletionException.STATE_ROLLED_BACK:
               log.error("Exception cause:", e);
               throw new HeuristicRollbackException(e.getMessage());
            default:
               log.error("Exception cause:", e);
               throw new HeuristicMixedException(e.getMessage());
         }
      }
      catch (TransactionSystemException e)
      {
         log.error("Exception cause:", e);
         throw new SystemException(e.getMessage());
      }
      catch (UnexpectedRollbackException e)
      {
         log.error("Exception cause:", e);
         throw new RollbackException(e.getMessage());
      }
      finally
      {
         currentTransaction = null;
      }
   }

   public int getStatus() throws SystemException
   {
      if (TransactionSynchronizationManager.isActualTransactionActive())
      {
         TransactionStatus transaction = null;
         try
         {
            if (currentTransaction == null)
            {
               transaction = platformTransactionManager.getValue().getTransaction(definition);
               if (transaction.isNewTransaction())
               {
                  return Status.STATUS_COMMITTED;
               }
            }
            else
            {
               transaction = currentTransaction;
            }
            // If SynchronizationManager things it has an active transaction but
            // our transaction is a new one
            // then we must be in the middle of committing
            if (transaction.isCompleted())
            {
               if (transaction.isRollbackOnly())
               {
                  return Status.STATUS_ROLLEDBACK;
               }
               return Status.STATUS_COMMITTED;
            }
            else
            {
               if (transaction.isRollbackOnly())
               {
                  return Status.STATUS_MARKED_ROLLBACK;
               }
               return Status.STATUS_ACTIVE;
            }
         }
         finally
         {
            if (currentTransaction == null)
            {
               platformTransactionManager.getValue().commit(transaction);
            }
         }
      }
      return Status.STATUS_NO_TRANSACTION;
   }

   public void rollback() throws IllegalStateException, SecurityException, SystemException
   {
      assertActive();
      try
      {
         platformTransactionManager.getValue().rollback(currentTransaction);
      }
      catch (TransactionException e)
      {
         log.error("Exception cause:", e);
         throw new SystemException(e.getMessage());
      }
      finally
      {
         currentTransaction = null;
      }
   }

   /**
    *
    */
   private void assertActive()
   {
      if (!TransactionSynchronizationManager.isActualTransactionActive()
               || currentTransaction == null)
      {
         throw new IllegalStateException("No transaction currently active that Seam started."
                  + "Seam should only be able to committ or rollback transactions it started.");
      }
   }

   public void setRollbackOnly() throws IllegalStateException, SystemException
   {
      if (!TransactionSynchronizationManager.isActualTransactionActive())
      {
         throw new IllegalStateException("No Spring Transaction is currently available.");
      }
      TransactionStatus transaction = null;
      try
      {
         if (currentTransaction == null)
         {
            transaction = platformTransactionManager.getValue().getTransaction(definition);
         }
         else
         {
            transaction = currentTransaction;
         }
         transaction.setRollbackOnly();
      }
      catch (TransactionException e)
      {
         log.error("Exception cause:", e);
         throw new SystemException(e.getMessage());
      }
      finally
      {
         if (currentTransaction == null)
         {
            platformTransactionManager.getValue().commit(transaction);
         }
      }
   }

   public void setTransactionTimeout(int timeout) throws SystemException
   {
      if (TransactionSynchronizationManager.isActualTransactionActive())
      {
         // cannot set timeout on already running transaction
         return;
      }
      definition.setTimeout(timeout);
   }

   @Override
   public void enlist(EntityManager entityManager) throws SystemException
   {
      if (joinTransaction == null)
      {
         // If not set attempt to detect if we should join or not
         if (!(platformTransactionManager.getValue() instanceof JpaTransactionManager))
         {
            super.enlist(entityManager);
         }
      }
      else if (joinTransaction)
      {
         super.enlist(entityManager);
      }
   }
  
   @Destroy
   public void cleanupCurrentTransaction() {
      if(currentTransaction != null) {
         try {
            log.debug("Attempting to rollback left over transaction.  Should never be called.");
            platformTransactionManager.getValue().rollback(currentTransaction);
         } catch(Throwable e) {
            //ignore
         }
      }
   }

   public void setPlatformTransactionManager(
            ValueExpression<PlatformTransactionManager> platformTransactionManager)
   {
      this.platformTransactionManager = platformTransactionManager;
   }

   @Override
   public boolean isConversationContextRequired()
   {
      return conversationContextRequired;
   }

   public void setConversationContextRequired(boolean conversationContextRequired)
   {
      this.conversationContextRequired = conversationContextRequired;
   }

   public void setJoinTransaction(Boolean joinTransaction)
   {
      this.joinTransaction = joinTransaction;
   }
  
   public class JtaSpringSynchronizationAdapter extends TransactionSynchronizationAdapter
   {
      @Override
      public int getOrder()
      {
         return SeamLifecycleUtils.SEAM_LIFECYCLE_SYNCHRONIZATION_ORDER - 1;
      }

      private final Synchronization sync;

      public JtaSpringSynchronizationAdapter(Synchronization sync)
      {
         this.sync = sync;
      }

      @Override
      public void afterCompletion(int status)
      {
         sync.afterCompletion(status);
      }

      @Override
      public void beforeCompletion()
      {
         sync.beforeCompletion();
      }
   }
}
TOP

Related Classes of org.jboss.seam.ioc.spring.SpringTransaction$JtaSpringSynchronizationAdapter

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.