Package org.infinispan.interceptors

Source Code of org.infinispan.interceptors.InvocationContextInterceptor

/*
* 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.interceptors;


import org.infinispan.commands.VisitableCommand;
import org.infinispan.commands.tx.CommitCommand;
import org.infinispan.commands.tx.PrepareCommand;
import org.infinispan.commands.tx.RollbackCommand;
import org.infinispan.commands.write.ClearCommand;
import org.infinispan.commands.write.PutKeyValueCommand;
import org.infinispan.commands.write.PutMapCommand;
import org.infinispan.commands.write.RemoveCommand;
import org.infinispan.commands.write.ReplaceCommand;
import org.infinispan.context.InvocationContext;
import org.infinispan.context.TransactionContext;
import org.infinispan.factories.annotations.Inject;
import org.infinispan.invocation.Flag;
import org.infinispan.remoting.RpcManager;
import org.infinispan.transaction.GlobalTransaction;
import org.infinispan.transaction.TransactionTable;

import javax.transaction.SystemException;
import javax.transaction.Transaction;

public class InvocationContextInterceptor extends BaseTransactionalContextInterceptor {
   private RpcManager rpcManager;

   @Inject
   public void setDependencies(RpcManager rpcManager) {
      this.rpcManager = rpcManager;
   }

   @Override
   public Object visitPutMapCommand(InvocationContext ctx, PutMapCommand command) throws Throwable {
      return handleAll(ctx, command, ctx.getGlobalTransaction(), false);
   }

   @Override
   public Object visitPutKeyValueCommand(InvocationContext ctx, PutKeyValueCommand command) throws Throwable {
      return handleAll(ctx, command, ctx.getGlobalTransaction(), true);
   }

   @Override
   public Object visitClearCommand(InvocationContext ctx, ClearCommand command) throws Throwable {
      return handleAll(ctx, command, ctx.getGlobalTransaction(), false);
   }

   @Override
   public Object visitRemoveCommand(InvocationContext ctx, RemoveCommand command) throws Throwable {
      return handleAll(ctx, command, ctx.getGlobalTransaction(), false);
   }

   @Override
   public Object visitReplaceCommand(InvocationContext ctx, ReplaceCommand command) throws Throwable {
      return handleAll(ctx, command, ctx.getGlobalTransaction(), false);
   }

   @Override
   public Object visitPrepareCommand(InvocationContext ctx, PrepareCommand command) throws Throwable {
      return handleAll(ctx, command, command.getGlobalTransaction(), true);
   }

   @Override
   public Object visitRollbackCommand(InvocationContext ctx, RollbackCommand command) throws Throwable {
      return handleAll(ctx, command, command.getGlobalTransaction(), true);
   }

   @Override
   public Object visitCommitCommand(InvocationContext ctx, CommitCommand command) throws Throwable {
      return handleAll(ctx, command, command.getGlobalTransaction(), true);
   }

   @Override
   public Object handleDefault(InvocationContext ctx, VisitableCommand command) throws Throwable {
      return handleAll(ctx, command, null, true);
   }

   private Object handleAll(InvocationContext ctx, VisitableCommand command, GlobalTransaction gtx, boolean scrubContextOnCompletion) throws Throwable {
      boolean suppressExceptions = false;
      Transaction suspendedTransaction = null;
      boolean resumeSuspended = false;

      if (trace) log.trace("Invoked with command " + command + " and InvocationContext [" + ctx + "]");

      try {
         if (txManager != null) {
            Transaction tx = getTransaction();
            GlobalTransaction realGtx = getGlobalTransaction(tx, gtx);
            if (tx == null && realGtx != null && realGtx.isRemote()) tx = txTable.getLocalTransaction(gtx);
            setTransactionalContext(tx, realGtx, null, ctx);
         } else {
            setTransactionalContext(null, null, null, ctx);
         }

         if (ctx.hasFlag(Flag.FAIL_SILENTLY)) {
            log.debug("FAIL_SILENTLY flag is present - suspending any ongoing transaction.");
            suppressExceptions = true;
            if (ctx.getTransaction() != null) {
               suspendedTransaction = txManager.suspend();
               setTransactionalContext(null, null, null, ctx);
               if (trace) log.trace("Suspending transaction " + suspendedTransaction);
               resumeSuspended = true;
            } else {
               if (trace) log.trace("No ongoing transaction to suspend");
            }
         }

         Object retval;
         try {
            return invokeNextInterceptor(ctx, command);
         }
         catch (Throwable th) {
            retval = th;
            // if fail silently return a null
            if (suppressExceptions) return null;
            Throwable t = (Throwable) retval;
            if (t instanceof RuntimeException && t.getCause() != null) {
               throw t.getCause();
            } else {
               throw t;
            }
         }
         // assume we're the first interceptor in the chain.  Handle the exception-throwing.
      }
      finally {
         // TODO: scope upgrading should happen transparently
         /*
          * we should scrub txs after every call to prevent race conditions
          * basically any other call coming in on the same thread and hijacking any running tx's
          * was highlighted in JBCACHE-606
          */
         if (scrubContextOnCompletion) setTransactionalContext(null, null, null, ctx);

         // clean up any invocation-scope flags set up
         if (trace) log.trace("Resetting invocation-scope flags");
         ctx.resetFlags();

         // if this is a prepare, opt prepare or

         if (resumeSuspended) {
            txManager.resume(suspendedTransaction);
         } else {
            if (ctx.getTransaction() != null && (TransactionTable.isValid(ctx.getTransaction()))) {
               copyInvocationScopeFlagsToTxScope(ctx);
            }
         }

         // TODO: Calling ctx.reset() here breaks stuff.  Check whether this is just becuse UTs expect stuff in the ctx or whether this really breaks functionality.
//         ctx.reset();
         // instead, for now, just wipe contents of the looked up entries
         ctx.clearLookedUpEntries();
      }
   }

   private GlobalTransaction getGlobalTransaction(Transaction tx, GlobalTransaction gtx) {
      if (gtx == null) gtx = txTable.getCurrentTransaction(tx, false);
      if (gtx != null) gtx.setRemote(isRemoteGlobalTx(gtx));
      return gtx;
   }

   private Transaction getTransaction() throws SystemException {
      // this creates a context if one did not exist.
      if (txManager == null) {
         if (trace) log.trace("no transaction manager configured, setting tx as null.");
         return null;
      } else {
         return txManager.getTransaction();
      }
   }

   /**
    * Tests if a global transaction originated from a different cache in the cluster
    *
    * @param gtx
    * @return true if the gtx is remote, false if it originated locally.
    */
   private boolean isRemoteGlobalTx(GlobalTransaction gtx) {
      return gtx != null && (gtx.getAddress() != null) && (!gtx.getAddress().equals(rpcManager.getTransport().getAddress()));
   }

   private void copyInvocationScopeFlagsToTxScope(InvocationContext ctx) {
      // notify the transaction tCtx that this override is in place.
      TransactionContext tCtx = ctx.getTransactionContext();
      if (tCtx != null) {
         if (ctx.hasFlag(Flag.CACHE_MODE_LOCAL)) tCtx.setFlags(Flag.CACHE_MODE_LOCAL);
         if (ctx.hasFlag(Flag.SKIP_CACHE_STATUS_CHECK)) tCtx.setFlags(Flag.SKIP_CACHE_STATUS_CHECK);
      }
   }
}
TOP

Related Classes of org.infinispan.interceptors.InvocationContextInterceptor

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.