Package org.infinispan.transaction

Source Code of org.infinispan.transaction.TransactionTable

/*
* JBoss, Home of Professional Open Source.
* Copyright 2000 - 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.infinispan.transaction;

import org.infinispan.CacheException;
import org.infinispan.context.InvocationContext;
import org.infinispan.context.TransactionContext;
import org.infinispan.factories.annotations.Inject;
import org.infinispan.factories.annotations.NonVolatile;
import org.infinispan.factories.annotations.Start;
import org.infinispan.factories.context.ContextFactory;
import org.infinispan.logging.Log;
import org.infinispan.logging.LogFactory;
import org.infinispan.remoting.RpcManager;
import org.infinispan.remoting.transport.Address;
import org.infinispan.remoting.transport.Transport;

import javax.transaction.Status;
import javax.transaction.SystemException;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
* Maintains the mapping between a local {@link Transaction} and a {@link GlobalTransaction}. Also stores {@link
* TransactionContext} instances under a given transaction.
*
* @author <a href="mailto:bela@jboss.org">Bela Ban</a> Apr 14, 2003
* @since 4.0
*/
@NonVolatile
public class TransactionTable {
   private static final Log log = LogFactory.getLog(TransactionTable.class);
   private static final boolean trace = log.isTraceEnabled();

   /**
    * Mapping between local (javax.transaction.Transaction) and a TransactionContext
    */
   protected final Map<Transaction, TransactionContext> txMapping = new ConcurrentHashMap<Transaction, TransactionContext>();

   /**
    * Mappong between GlobalTransaction and a TransactionContext
    */
   protected final Map<GlobalTransaction, TransactionContext> gtxMapping = new ConcurrentHashMap<GlobalTransaction, TransactionContext>();

   private TransactionManager transactionManager = null;

   private RpcManager rpcManager;
   private Transport transport;

   private ContextFactory contextFactory;

   @Inject
   public void initialize(TransactionManager transactionManager, RpcManager rpcManager, ContextFactory contextFactory) {
      this.transactionManager = transactionManager;
      this.rpcManager = rpcManager;
      this.contextFactory = contextFactory;
   }

   @Start(priority = 12)
   // needs to happen after RpcManager
   public void start() {
      transport = rpcManager == null ? null : rpcManager.getTransport();
   }


   /**
    * Returns the number of local transactions.
    */
   public int getNumLocalTransactions() {
      return txMapping.size();
   }

   /**
    * Returns the number of global transactions.
    */
   public int getNumGlobalTransactions() {
      return txMapping.size();
   }

   /**
    * Returns the global transaction associated with the local transaction. Returns null if tx is null or it was not
    * found.
    */
   public GlobalTransaction get(Transaction tx) {
      if (tx == null) return null;
      TransactionContext ctx = txMapping.get(tx);
      return ctx == null ? null : ctx.getGobalTransaction();
   }

   /**
    * If assers exists is true and the coresponding local transaction is null an IllegalStateExcetpion is being thrown.
    */
   public Transaction getLocalTransaction(GlobalTransaction gtx, boolean assertExists) {
      Transaction ltx = getLocalTransaction(gtx);
      if (!assertExists) {
         return ltx;
      }
      if (ltx != null) {
         if (log.isDebugEnabled()) log.debug("Found local TX=" + ltx + ", global TX=" + gtx);
         return ltx;
      } else {
         throw new IllegalStateException(" found no local TX for global TX " + gtx);
      }
   }

   /**
    * Associates 3 elements of a transaction - a local Transaction, a GlobalTransaction and a TransactionContext - with
    * each other.
    *
    * @param tx  transaction to associate
    * @param gtx global transaction to associate
    * @param ctx transaction context to associate
    */
   public void associateTransaction(Transaction tx, GlobalTransaction gtx, TransactionContext ctx) {
      if (ctx.getTransaction() == null) ctx.setTransaction(tx);
      if (ctx.getGobalTransaction() == null) ctx.setGlobalTransaction(gtx);

      txMapping.put(tx, ctx);
      gtxMapping.put(gtx, ctx);
   }

   public Transaction getLocalTransaction(GlobalTransaction gtx) {
      TransactionContext ctx = gtxMapping.get(gtx);
      return ctx == null ? null : ctx.getTransaction();
   }

   /**
    * Returns summary debug information.
    */
   @Override
   public String toString() {
      StringBuilder sb = new StringBuilder();
      sb.append(txMapping.size()).append(" transactions");
      return sb.toString();
   }

   /**
    * Returns detailed debug information.
    */
   public String toString(boolean printDetails) {
      if (!printDetails)
         return toString();
      StringBuilder sb = new StringBuilder();
      sb.append("Transactions: ").append(txMapping.size()).append("\n");
      sb.append("mappings:\n");
      for (Map.Entry<Transaction, TransactionContext> entry : txMapping.entrySet()) {
         sb.append(entry.getKey()).append(": ").append(entry.getValue()).append("\n");
      }
      return sb.toString();
   }

   /**
    * Returns the transaction associated with the current thread. If a local transaction exists, but doesn't yet have a
    * mapping to a GlobalTransaction, a new GlobalTransaction will be created and mapped to the local transaction.  Note
    * that if a local transaction exists, but is not ACTIVE or PREPARING, null is returned.
    *
    * @return A GlobalTransaction, or null if no (local) transaction was associated with the current thread
    */
   public GlobalTransaction getCurrentTransaction() {
      return getCurrentTransaction(true);
   }


   /**
    * Returns the transaction associated with the thread; optionally creating it if is does not exist.
    */
   public GlobalTransaction getCurrentTransaction(boolean createIfNotExists) {
      Transaction tx;

      if ((tx = getLocalTransaction()) == null) {// no transaction is associated with the current thread
         return null;
      }

      if (!isValid(tx)) {// we got a non-null transaction, but it is not active anymore
         int status = -1;
         try {
            status = tx.getStatus();
         }
         catch (SystemException e) {
         }

         // JBCACHE-982 -- don't complain if COMMITTED
         if (status != Status.STATUS_COMMITTED) {
            log.warn("status is " + status + " (not ACTIVE or PREPARING); returning null)");
         } else {
            log.trace("status is COMMITTED; returning null");
         }

         return null;
      }

      return getCurrentTransaction(tx, createIfNotExists);
   }

   /**
    * Returns the transaction associated with the current thread. We get the initial context and a reference to the
    * TransactionManager to get the transaction. This method is used by {@link #getCurrentTransaction()}
    */
   protected Transaction getLocalTransaction() {
      if (transactionManager == null) {
         return null;
      }
      try {
         return transactionManager.getTransaction();
      }
      catch (Throwable t) {
         return null;
      }
   }

   /**
    * Returns true if transaction is ACTIVE, false otherwise
    */
   public static boolean isActive(Transaction tx) {
      if (tx == null) return false;
      int status;
      try {
         status = tx.getStatus();
         return status == Status.STATUS_ACTIVE;
      }
      catch (SystemException e) {
         return false;
      }
   }

   /**
    * Returns true if transaction is PREPARING, false otherwise
    */
   public static boolean isPreparing(Transaction tx) {
      if (tx == null) return false;
      int status;
      try {
         status = tx.getStatus();
         return status == Status.STATUS_PREPARING;
      }
      catch (SystemException e) {
         return false;
      }
   }

   /**
    * Return s true of tx's status is ACTIVE or PREPARING
    *
    * @param tx
    * @return true if the tx is active or preparing
    */
   public static boolean isValid(Transaction tx) {
      return isActive(tx) || isPreparing(tx);
   }

   /**
    * Tests whether the caller is in a valid transaction.  If not, will throw a CacheException.
    */
   public static void assertTransactionValid(InvocationContext ctx) {
      Transaction tx = ctx.getTransaction();
      if (!isValid(tx)) try {
         throw new CacheException("Invalid transaction " + tx + ", status = " + (tx == null ? null : tx.getStatus()));
      }
      catch (SystemException e) {
         throw new CacheException("Exception trying to analyse status of transaction " + tx, e);
      }
   }


   /**
    * Returns the global transaction for this local transaction.
    */
   public GlobalTransaction getCurrentTransaction(Transaction tx) {
      return getCurrentTransaction(tx, true);
   }

   /**
    * Returns the global transaction for this local transaction.
    *
    * @param createIfNotExists if true, if a global transaction is not found; one is created
    */
   public GlobalTransaction getCurrentTransaction(Transaction tx, boolean createIfNotExists) {
      // removed synchronization on txTable because underlying implementation is thread safe
      // and JTA spec (section 3.4.3 Thread of Control, par 2) says that only one thread may
      // operate on the transaction at one time so no concern about 2 threads trying to call
      // this method for the same Transaction instance at the same time
      //
      GlobalTransaction gtx = get(tx);
      if (gtx == null && createIfNotExists) {
         Address addr = getAddress();
         gtx = GlobalTransaction.create(addr);
         if (trace) log.trace("Creating new GlobalTransaction " + gtx);
         TransactionContext transactionContext;
         try {
            transactionContext = contextFactory.createTransactionContext(tx);
         }
         catch (Exception e) {
            throw new CacheException("Unable to create a transaction entry!", e);
         }
         associateTransaction(tx, gtx, transactionContext);
         if (trace) {
            log.trace("created new GTX: " + gtx + ", local TX=" + tx);
         }
      }
      return gtx;
   }

   private Address getAddress() {
      return transport == null ? null : transport.getAddress();
   }

   public TransactionContext getTransactionContext(GlobalTransaction gtx) {
      return gtxMapping.get(gtx);
   }

   public void cleanup(GlobalTransaction gtx) {
      TransactionContext ctx = gtxMapping.remove(gtx);
      if (ctx != null) txMapping.remove(ctx.getTransaction());
   }
}
TOP

Related Classes of org.infinispan.transaction.TransactionTable

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.