Package org.jboss.test.tm.resource

Source Code of org.jboss.test.tm.resource.MTOperation

/*
* JBoss, Home of Professional Open Source.
* Copyright 2008, 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.test.tm.resource;

import java.io.Serializable;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

import javax.naming.InitialContext;
import javax.transaction.Synchronization;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;

import org.jboss.logging.Logger;
import org.jboss.tm.TxUtils;

/**
* MultiThreaded Operations that can be executed concurrently.
*
* Based on Operation class.
*
* @author <a href="dimitris@jboss.org">Dimitris Andreadis</a>
* @version $Revision: 89025 $
*/
public class MTOperation implements Serializable
{
   // Static Data ---------------------------------------------------

   /** The serialVersionUID */
   private static final long serialVersionUID = 2924873545494045020L;
   /** Available Operations */
   public static final int TM_GET_STATUS = 0;
   public static final int TM_BEGIN = 1;
   public static final int TM_RESUME = 2;
   public static final int TM_COMMIT = 3;
   public static final int TX_COMMIT = 4;
   public static final int TX_REGISTER_SYNC = 5;
   public static final int XX_SLEEP_200 = 6;
   public static final int XX_WAIT_FOR = 7;

   /** The Logger */
   protected static Logger log;

   /** TM instance */
   protected static TransactionManager tm = null;

   /** Shared resources */
   protected static Map resources = Collections.synchronizedMap(new HashMap());

   /** Active Transactions */
   protected static Map transactions = Collections.synchronizedMap(new HashMap());

   // Protected Data ------------------------------------------------

   /** An id for this transaction */
   protected Integer id;

   /** The operation to execute */
   protected int op;

   /** Set when an exception is expected */
   protected Throwable throwable;

   /** If a throwable is required, or only *may* occur */
   private boolean require;

   // Static Methods ------------------------------------------------

   /**
    * Setup static objects for the test
    */
   public static void init(Logger log) throws Exception
   {
      MTOperation.log = log;

      if (getTM().getTransaction() != null)
      {
         throw new IllegalStateException("Invalid thread association " + getTM().getTransaction());
      }
      resources.clear();
      transactions.clear();
   }

   /**
    * Lazy TransactionManager lookup
    */
   public static TransactionManager getTM() throws Exception
   {
      if (tm == null)
      {
         tm = (TransactionManager) new InitialContext().lookup("java:/TransactionManager");
      }
      return tm;
   }

   /**
    * Cleanup
    */
   public static void destroy()
   {
      resources.clear();
      transactions.clear();
   }

   // Constructors --------------------------------------------------

   public MTOperation(int op)
   {
      this(op, 0);
   }

   public MTOperation(int op, int id)
   {
      this.id = new Integer(id);
      this.op = op;
   }

   public MTOperation(int op, int id, Throwable throwable)
   {
      this(op, id, throwable, true);
   }

   public MTOperation(int op, int id, Throwable throwable, boolean require)
   {
      this.id = new Integer(id);
      this.op = op;
      this.throwable = throwable;
      this.require = require;
   }

   // Public Methods ------------------------------------------------

   public void perform() throws Exception
   {
      Throwable caught = null;
      try
      {
         switch (op)
         {
            case TM_GET_STATUS:
               tmGetStatus();
               break;

            case TM_BEGIN:
               tmBegin();
               break;

            case TM_RESUME:
               tmResume();
               break;

            case TM_COMMIT:
               tmCommit();
               break;

            case TX_COMMIT:
               txCommit();
               break;

            case TX_REGISTER_SYNC:
               txRegisterSync();
               break;

            case XX_SLEEP_200:
               xxSleep200();
               break;

            case XX_WAIT_FOR:
               xxWaitForTx();
               break;

            default:
               throw new IllegalArgumentException("Invalid operation " + op);
         }
      }
      catch (Throwable t)
      {
         caught = t;
      }

      // required an exception but caught none
      if (require && throwable != null && caught == null)
      {
         throw new Exception("Expected throwable " + throwable);
      }

      // got an exception, but it was the wrong one
      if (throwable != null && caught != null && !throwable.getClass().isAssignableFrom(caught.getClass()))
      {
         log.warn("Caught wrong throwable", caught);
         throw new Exception("Expected throwable " + throwable + " caught " + caught);
      }

      // did not expect an exception but caught one
      if (throwable == null && caught != null)
      {
         log.warn("Caught unexpected throwable", caught);
         throw new Exception("Unexpected throwable " + caught);
      }
   }

   public void tmGetStatus() throws Exception
   {
      log.info(tid() + " " + TxUtils.getStatusAsString(getTM().getStatus()));
   }

   public void tmBegin() throws Exception
   {
      log.info(tid() + " TM_BEGIN (" + id + ")");
      getTM().begin();
      Transaction tx = getTM().getTransaction();
      synchronized (transactions)
      {
         transactions.put(id, tx);
         transactions.notifyAll();
      }
   }

   public void tmResume() throws Exception
   {
      log.info(tid() + " TM_RESUME (" + id + ")");
      Transaction tx = (Transaction)transactions.get(id);
      if (tx == null)
      {
         throw new IllegalStateException("Tx not found:" + id);
      }
      else
      {
         getTM().resume(tx);
      }
   }

   public void tmCommit() throws Exception
   {
      log.info(tid() + " TM_COMMIT");
      getTM().commit();
   }

   public void txCommit() throws Exception
   {
      log.info(tid() + " TX_COMMIT (" + id + ")");
      Transaction tx = (Transaction)transactions.get(id);
      if (tx == null)
      {
         throw new IllegalStateException("Tx not found: " + id);
      }
      else
      {
         tx.commit();
      }
   }

   public void txRegisterSync() throws Exception
   {
      log.info(tid() + " TX_REGISTER_SYNC (" + id + ")");
      Transaction tx = (Transaction)transactions.get(id);
      if (tx == null)
      {
         throw new IllegalStateException("Tx not found: " + id);
      }
      Synchronization sync = new Synchronization()
      {
         public void beforeCompletion()
         {
            log.info(tid() + " beforeCompletion() called");
         }

         public void afterCompletion(int status)
         {
            log.info (tid() + " afterCompletion(" + TxUtils.getStatusAsString(status) + ") called");
         }
      };
      tx.registerSynchronization(sync);
   }

   public void xxWaitForTx() throws Exception
   {
      log.info(tid() + " XX_WAIT_FOR (" + id + ")");

      Transaction tx = (Transaction)transactions.get(id);
      while (tx == null)
      {
         log.info(tid() + " Sleeping for 100 msecs");
         synchronized (transactions)
         {
            try
            {
               transactions.wait(100);
            }
            catch (InterruptedException ignore) {}
         }
         tx = (Transaction)transactions.get(id);
      }
      log.info(tid() + " Got it");
   }

   public void xxSleep200() throws Exception
   {
      log.info(tid() + " XX_SLEEP_200");
      Thread.sleep(200);
   }

   private String tid()
   {
      return Thread.currentThread().getName();
   }
}
TOP

Related Classes of org.jboss.test.tm.resource.MTOperation

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.