Package org.jboss.ha.jndi

Source Code of org.jboss.ha.jndi.HAJNDI

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

import java.util.Collection;
import java.util.List;

import javax.naming.Binding;
import javax.naming.InitialContext;
import javax.naming.Name;
import javax.naming.NameClassPair;
import javax.naming.NameNotFoundException;
import javax.naming.NamingException;

import org.jboss.ha.framework.interfaces.HAPartition;
import org.jboss.ha.jndi.spi.DistributedTreeManager;
import org.jboss.logging.Logger;
import org.jnp.interfaces.Naming;

/**
*  Provides the Naming implementation. Lookups will look for Names in
*  the injected DistributedTreeManager and if not found will delegate to the local
*  InitialContext. If still not found, a group RPC will be sent to the cluster
*  using the provided partition.  All other Naming operations delegate to the
*  DistributedTreeManager.
*
@author <a href="mailto:bill@burkecentral.com">Bill Burke</a>
@author Scott.Stark@jboss.org
@author Brian Stansberry
@author Galder Zamarreño
@version $Revision: 105825 $
*/
public class HAJNDI
   implements org.jnp.interfaces.Naming
{
   private static final Logger log = Logger.getLogger(HAJNDI.class);
  
   /** @since 1.12.2.4, jboss-3.2.2 */
   static final long serialVersionUID = -6277328603304171620L;
  
   // Attributes --------------------------------------------------------
  
   private final HAPartition partition;
   private final DistributedTreeManager distributedTreeManager;
   private final Naming localNamingInstance;
   private boolean missingLocalNamingLogged;
   private final RemoteLookupHandler rpcHandler;

   // Constructor --------------------------------------------------------
 
   public HAJNDI(HAPartition partition, DistributedTreeManager distributedTreeManager, Naming localNamingInstance)
   {
      if (partition == null)
      {
         throw new IllegalArgumentException("Null partition");
      }
     
      if (distributedTreeManager == null)
      {
         throw new IllegalArgumentException("Null distributedTreeManager");
      }
     
      if (localNamingInstance == null)
      {
         log.debug("No localNamingInstance provided; injecting a local naming instance is recommended");
      }
     
      this.partition = partition;
      this.distributedTreeManager = distributedTreeManager;
      this.localNamingInstance = localNamingInstance;
      this.rpcHandler = new RemoteLookupHandler();
   }
  
   // Public --------------------------------------------------------

   public void init()
   {
      log.debug("HAJNDI registering RPC Handler with HAPartition");
      this.partition.registerRPCHandler("HAJNDI", this.rpcHandler);
      this.distributedTreeManager.init();
   }

   public void shutdown()
   {
      log.debug("HAJNDI unregistering RPCHandler with HAPartition");
      this.partition.unregisterRPCHandler("HAJNDI", this.rpcHandler);
      this.distributedTreeManager.shutdown();
   }

   // Naming implementation -----------------------------------------
  

   public synchronized void bind(Name name, Object obj, String className) throws NamingException
   {
      this.distributedTreeManager.bind(name, obj, className);
   }

   public synchronized void rebind(Name name, Object obj, String className) throws NamingException
   {
      this.distributedTreeManager.rebind(name, obj, className);
   }

   public synchronized void unbind(Name name) throws NamingException
   {
      this.distributedTreeManager.unbind(name);
   }

   public Object lookup(Name name) throws NamingException
   {
      Object binding = this.distributedTreeManager.lookup(name);
      if (binding == null)
      {
         try
         {
            binding = lookupLocally(name);
         }
         catch (NameNotFoundException nne)
         {
            binding = lookupRemotely(name);
            if (binding == null)
            {
               throw nne;
            }
         }
      }
      return binding;
   }

   public Collection<NameClassPair> list(Name name) throws NamingException
   {
      return this.distributedTreeManager.list(name) ;
   }
   
   public Collection<Binding> listBindings(Name name) throws NamingException
   {
      return this.distributedTreeManager.listBindings(name);
   }
  
   public javax.naming.Context createSubcontext(Name name) throws NamingException
   {
      return this.distributedTreeManager.createSubcontext(name);
   }
  
   // ----------------------------------------------------------------  Private

   /**
    * Performs a lookup against the local Naming service.
    *
    * @param name the name
    * @return     the object bound locally under name
    * @throws NamingException
    */
   private Object lookupLocally(Name name) throws NamingException
   {
      boolean trace = log.isTraceEnabled();
      if (trace)
      {
         log.trace("lookupLocally, name="+name);
      }

      // We cannot do InitialContext().lookup(name) because
      // we get ClassNotFound errors and ClassLinkage errors.
      // So, what we prefer to use an injected local Naming instance
      try
      {
         if (localNamingInstance != null)
         {
            return localNamingInstance.lookup(name);
         }

         return new InitialContext().lookup(name);
      }
      catch (NameNotFoundException e)
      {
         if (trace)
         {
            log.trace("lookupLocally failed, name=" + name, e);
         }
         throw e;
      }
      catch (NamingException e)
      {
         if (!logMissingLocalNamingInstance(e) && trace)
         {
            log.trace("lookupLocally failed, name=" + name, e);
         }
         throw e;
      }
      catch (java.rmi.RemoteException e)
      {
         NamingException ne = new NamingException("unknown remote exception");
         ne.setRootCause(e);
         if( trace )
         {
            log.trace("lookupLocally failed, name=" + name, e);
         }
         throw ne;
      }
      catch (RuntimeException e)
      {
         if (!logMissingLocalNamingInstance(e) && trace)
         {
            log.trace("lookupLocally failed, name=" + name, e);
         }
         throw e;
      }
      catch (Error e)
      {
         logMissingLocalNamingInstance(e);
         throw e;
      }
   }
  
   private Object lookupRemotely(Name name) throws NameNotFoundException
   {
      boolean trace = log.isTraceEnabled();
     
      // if we get here, this means we need to try on every node.
      Object[] args = new Object[1];
      args[0] = name;
      List<?> rsp = null;
      Exception cause = null;
      try
      {
         if (trace)
         {
            log.trace("calling lookupLocally(" + name + ") on HAJNDI cluster");
         }
         rsp = this.partition.callMethodOnCluster("HAJNDI", "remoteLookup", args, new Class[] { Name.class }, Object.class, true, new LookupSucceededFilter(), this.partition.getMethodCallTimeout(), false);
      }
      catch (Exception ignored)
      {
         if (trace)
         {
            log.trace("Clustered lookupLocally("+name+") failed", ignored);
         }
         cause = ignored;
      }

      if (trace)
      {
         log.trace("Returned results size: "+ (rsp != null ? rsp.size() : 0));
      }
      if (rsp == null || rsp.size() == 0)
      {
         NameNotFoundException nnfe2 = new NameNotFoundException(name.toString());
         nnfe2.setRootCause(cause);
         throw nnfe2;
      }

      for (int i = 0; i < rsp.size(); i++)
      {
         Object result = rsp.get(i);
         if (trace)
         {
            String type = (result != null ? result.getClass().getName() : "null");
            log.trace("lookupLocally, i="+i+", value="+result+", type="+type);
         }
         // Ignore null and Exception return values
         if ((result != null) && !(result instanceof Exception))
         {
            return result;
         }
      }
     
      return null;
   }
  
   /**
    * One time and one time only logs a WARN if localNamingInstance was
    * not passed to our constructor and an error condition occurred while
    * attempting to do a local lookup via new InitialContext().
    *
    * @param t the error that occurred
    * @return <code>true</code> if this method logged a WARN
    */
   private boolean logMissingLocalNamingInstance(Throwable t)
   {
      if (localNamingInstance == null && !missingLocalNamingLogged)
      {
         log.warn("No localNamingInstance configured and lookup via new InitialContext() failed; " +
               "injecting a local naming instance is recommended", t);
         missingLocalNamingLogged = true;
         return true;
      }
      return false;
   }
  
   /**
    * Exposes the group RPC interface of the parent class.
    */
   public class RemoteLookupHandler
   {
      /**
       * Performs a lookup against the local Naming service. Will not throw a
       * NamingException, instead returning null. See JBAS-7947.
       *
       * @param name the name
       * @return     the object bound locally under name, or null if a
       *             NamingException is caught.
       */
      public Object remoteLookup(Name name)
      {
         Object result = null;
         try
         {
            result = HAJNDI.this.lookupLocally(name);
         }
         catch (NamingException ne)
         {
            // JBAS-7947 -- don't propagate this across cluster as the
            // remote caller is just going to ignore it
            if (log.isTraceEnabled())
            {
               log.trace("Caught NamingException doing a lookup of " + name, ne);
            }
         }
         return result;
      }
   }
}
TOP

Related Classes of org.jboss.ha.jndi.HAJNDI

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.