Package org.jboss.remoting.transport.rmi

Source Code of org.jboss.remoting.transport.rmi.RMIServerInvoker

/*
* JBoss, Home of Professional Open Source
* Copyright 2005, JBoss Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt 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.remoting.transport.rmi;

import org.jboss.remoting.Home;
import org.jboss.remoting.InvocationRequest;
import org.jboss.remoting.InvokerLocator;
import org.jboss.remoting.Remoting;
import org.jboss.remoting.ServerInvoker;
import org.jboss.remoting.marshal.MarshalFactory;
import org.jboss.remoting.marshal.Marshaller;
import org.jboss.remoting.marshal.MarshallerDecorator;
import org.jboss.remoting.marshal.UnMarshaller;
import org.jboss.remoting.marshal.UnMarshallerDecorator;
import org.jboss.remoting.marshal.VersionedMarshaller;
import org.jboss.remoting.marshal.VersionedUnMarshaller;
import org.jboss.remoting.marshal.rmi.RMIMarshaller;
import org.jboss.remoting.marshal.rmi.RMIUnMarshaller;
import org.jboss.remoting.marshal.serializable.SerializableMarshaller;
import org.jboss.remoting.serialization.SerializationManager;
import org.jboss.remoting.serialization.SerializationStreamFactory;
import org.jboss.util.propertyeditor.PropertyEditors;
import org.jboss.logging.Logger;

import javax.net.SocketFactory;

import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.net.InetAddress;
import java.rmi.Remote;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;
import java.rmi.server.ExportException;
import java.rmi.server.RMIClientSocketFactory;
import java.rmi.server.RMIServerSocketFactory;
import java.rmi.server.RemoteServer;
import java.rmi.server.ServerNotActiveException;
import java.rmi.server.UnicastRemoteObject;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;

/**
* RMIServerInvoker
*
* @author <a href="mailto:jhaynie@vocalocity.net">Jeff Haynie</a>
* @author <a href="mailto:tom.elrod@jboss.com">Tom Elrod</a>
* @version $Revision: 3663 $
*/
public class RMIServerInvoker extends ServerInvoker implements RMIServerInvokerInf, Cloneable
{
   public static String RMI_ONEWAY_MARSHALLING = "rmiOnewayMarshalling";
  
   private static final Logger log = Logger.getLogger(RMIServerInvoker.class);

   protected boolean isPrimaryServer = true;
   protected Set secondaryServers = new HashSet();
   protected boolean rmiOnewayMarshalling;
  
   private Remote stub;
   private RemotingRMIClientSocketFactory csf;
  

   /**
    * Default for how many connections are queued.  Value is 200.
    */
   public static final int BACKLOG_DEFAULT = 200;

   /**
    * Default port on which rmi registry will be started.  Value is 3455.
    */
   public static final int DEFAULT_REGISTRY_PORT = 3455;

   /**
    * Key for port on which rmi registry should be started on.
    */
   public static final String REGISTRY_PORT_KEY = "registryPort";

   private Marshaller marshaller = null;
   private UnMarshaller unmarshaller = null;

   public RMIServerInvoker(InvokerLocator locator)
   {
      super(locator);
   }

   public void start() throws IOException
   {
      super.start();

      if (isPrimaryServer)
      {
         List connectHomes = getConnectHomes();
         List homes = getHomes();
        
         if (connectHomes.size() != homes.size())
            throw new IOException("number of connect homes and bind homes must match in RMI transport");
        
         Home bindHome = (Home) homes.get(0);
         Home connectHome = (Home) connectHomes.get(0);
         initRMI(bindHome, connectHome);

         for (int i = 1; i < homes.size(); i++)
         {
            bindHome = (Home) homes.get(i);
            connectHome = (Home) connectHomes.get(i);
            RMIServerInvoker copy = copy();
            secondaryServers.add(copy);
            copy.initRMI(bindHome, connectHome);
         }
      }
   }
  
   protected void setup() throws Exception
   {
      Properties props = new Properties();
      props.putAll(getConfiguration());
      PropertyEditors.mapJavaBeanProperties(this, props, false);
      super.setup();
   }
  
   protected RMIServerInvoker copy() throws IOException
   {
      Object o = null;
      try
      {
         o = clone();
      }
      catch (CloneNotSupportedException e)
      {
         log.error("This should not happen", e);
         return this;
      }
      RMIServerInvoker server = (RMIServerInvoker) o;
      server.locator = locator;
      server.locator = new InvokerLocator(locator.getLocatorURI());
      server.locator.setHomeInUse(locator.getHomeInUse());
      server.isPrimaryServer = false;
      server.start();
      return server;
   }
  
   protected void initRMI(Home bindHome, Home connectHome) throws IOException
   {
      Registry registry = null;
      try
      {
         registry = getRegistry();
      }
      catch (Exception e)
      {
         throw new IOException(e.getMessage());
      }

      String bindHost = bindHome.host;
      int bindPort = bindHome.port;
      String clientConnectHost = connectHome.host;

      if(clientConnectHost == null)
      {
         clientConnectHost = bindHost;
      }

      locator.setHomeInUse(bindHome);
      RMIServerSocketFactory ssf = new RemotingRMIServerSocketFactory(getServerSocketFactory(), BACKLOG_DEFAULT, bindHost, getTimeout());
      csf = getRMIClientSocketFactory(clientConnectHost);
      stub = UnicastRemoteObject.exportObject(this, bindPort, csf, ssf);

      log.debug("Binding server to \"remoting/RMIServerInvoker/" + bindPort + "\" in registry");
      registry.rebind("remoting/RMIServerInvoker/" + bindPort, this);

      unmarshaller = MarshalFactory.getUnMarshaller(getLocator(), this.getClass().getClassLoader());
      marshaller = MarshalFactory.getMarshaller(getLocator(), this.getClass().getClassLoader());
   }


   protected RemotingRMIClientSocketFactory getRMIClientSocketFactory(String clientConnectHost)
   {
      // Remove server side socket creation listeners.
      HashMap remoteConfig = new HashMap(configuration);
      remoteConfig.remove(Remoting.CUSTOM_SERVER_SOCKET_FACTORY);
      remoteConfig.remove(Remoting.SOCKET_CREATION_CLIENT_LISTENER);
      remoteConfig.remove(Remoting.SOCKET_CREATION_SERVER_LISTENER);
      return new RemotingRMIClientSocketFactory(locator, clientConnectHost, getTimeout(), remoteConfig);
   }
  
   protected SocketFactory getDefaultSocketFactory()
   {
//      return SocketFactory.getDefault();
      /**
       * Returning null because by default, this socket factory
       * will be need to be serialized when exported.  Since the
       * default factory implementation returned from SocketFactory.getDefault()
       * is not serializable, it will not work.  Therefore, if return null,
       * will delay the creation of the socket factory until the RMIClientSocketFactory is
       * exported.
       */
      return null;
   }


   public RMIServerInvoker(InvokerLocator locator, Map configuration)
   {
      super(locator, configuration);
   }

   private Registry getRegistry() throws Exception
   {
      Registry registry = null;

      int port = DEFAULT_REGISTRY_PORT;

      // See if locator contains a specific registry port
      Map params = getConfiguration();
      if(params != null)
      {
         String value = (String) params.get(REGISTRY_PORT_KEY);
         if(value != null)
         {
            try
            {
               port = Integer.parseInt(value);
               log.debug("Using port " + port + " for rmi registry.");
            }
            catch(NumberFormatException e)
            {
               throw new Exception("Can not set the RMIServerInvoker RMI registry to port " + value + ".  This is not a valid port number.");
            }
         }
      }

      try
      {
         log.debug("Creating registry for " + port);

         registry = LocateRegistry.createRegistry(port);
      }
      catch(ExportException exportEx)
      {
         log.debug("Locating registry for " + port);

         // Probably means that the registry already exists, so just get it.
         registry = LocateRegistry.getRegistry(port);
      }
      if(log.isTraceEnabled())
      {
         log.trace("Got registry: " + registry);
      }
      return registry;
   }

   protected String getDefaultDataType()
   {
      return SerializableMarshaller.DATATYPE;
   }

   /**
    * destroy the RMI Server Invoker, which will unexport the RMI server
    */
   public void destroy()
   {
      super.destroy();
      try
      {
         try
         {
            log.debug("locator: " + locator + ", home: " + locator.getHomeInUse());
            log.debug(this + " primary: " + isPrimaryServer + " unbinding " + "remoting/RMIServerInvoker/" + locator.getPort() + " from registry");
            Registry registry = getRegistry();
            registry.unbind("remoting/RMIServerInvoker/" + locator.getPort());
            log.debug("unbound " + "remoting/RMIServerInvoker/" + locator.getPort() + " from registry");
         }
         catch(Exception e)
         {
            if ("Finalizer".equalsIgnoreCase(Thread.currentThread().getName()))
               log.debug("thread: " + Thread.currentThread().getName() + " Error unbinding RMIServerInvoker from RMI registry.", e);
            else
               log.error("thread: " + Thread.currentThread().getName() + " Error unbinding RMIServerInvoker from RMI registry.", e);
         }

         UnicastRemoteObject.unexportObject(this, true);

      }
      catch(java.rmi.NoSuchObjectException e)
      {

      }
     
      if (csf != null)
         csf.clear();
     
      if (isPrimaryServer)
      {
         Iterator it = secondaryServers.iterator();
         while (it.hasNext())
         {
            RMIServerInvoker server = (RMIServerInvoker) it.next();
            it.remove();
            server.destroy();
         }
      }
   }

   protected void finalize() throws Throwable
   {
      destroy();
      super.finalize();
   }

   /**
    * returns true if the transport is bi-directional in nature, for example,
    * SOAP in unidirectional and SOCKETs are bi-directional (unless behind a firewall
    * for example).
    */
   public boolean isTransportBiDirectional()
   {
      return true;
   }

   public final Remote getStub()
   {
      return stub;
   }

   public Object transport(Object invocation)
         throws RemoteException, IOException
   {

      Object payload = invocation;
      if(unmarshaller != null && !(unmarshaller instanceof RMIUnMarshaller))
      {
         if(unmarshaller instanceof UnMarshallerDecorator)
         {
            payload = ((UnMarshallerDecorator) unmarshaller).removeDecoration(payload);
         }
         else
         {
            ByteArrayInputStream is = null;
            if (rmiOnewayMarshalling)
            {
               // Legacy treatment, pre 2.4.0
               ByteArrayOutputStream baos = new ByteArrayOutputStream();
               SerializationManager manager = SerializationStreamFactory.getManagerInstance(getSerializationType());
               ObjectOutputStream oos = manager.createOutput(baos);
               oos.writeObject(payload);
               oos.flush();
               oos.close();
               is = new ByteArrayInputStream(baos.toByteArray());
            }
            else
            {
               is = new ByteArrayInputStream((byte[]) payload);
            }

            try
            {
               if (unmarshaller instanceof VersionedUnMarshaller)
               {
                  payload = ((VersionedUnMarshaller) unmarshaller).read(is, null, getVersion());
                  is.close();
               }
               else
               {
                  payload = unmarshaller.read(is, null);
                  is.close();
               }
            }
            catch(ClassNotFoundException e)
            {
               log.debug("Could not unmarshall invocation request" + payload, e);
               throw new IOException(e.getMessage());
            }
         }
      }

      if (payload instanceof InvocationRequest)
      {
         InvocationRequest ir = (InvocationRequest) payload;
         Map metadata = ir.getRequestPayload();
         if (metadata == null)
         {
            metadata = new HashMap();
            ir.setRequestPayload(metadata);
         }
         try
         {
            String clientHost = RemoteServer.getClientHost();
            metadata.put(Remoting.CLIENT_ADDRESS, InetAddress.getByName(clientHost));
         }
         catch (ServerNotActiveException e)
         {
            throw new RemoteException(e.getMessage());
         }
      }
      Object response = invoke(payload);

      if(marshaller != null && !(marshaller instanceof RMIMarshaller) && !rmiOnewayMarshalling)
      {
         if(marshaller instanceof MarshallerDecorator)
         {
            response = ((MarshallerDecorator) marshaller).addDecoration(response);
         }
         else
         {
            ByteArrayOutputStream byteOut = new ByteArrayOutputStream();
            if (marshaller instanceof VersionedMarshaller)
               ((VersionedMarshaller) marshaller).write(response, byteOut, getVersion());
            else
               marshaller.write(response, byteOut);
           
            byteOut.close();
            response = byteOut.toByteArray();
         }
      }
      return response;
   }

   public boolean isRmiOnewayMarshalling()
   {
      return rmiOnewayMarshalling;
   }

   public void setRmiOnewayMarshalling(boolean rmiOnewayMarshalling)
   {
      this.rmiOnewayMarshalling = rmiOnewayMarshalling;
   }
}
TOP

Related Classes of org.jboss.remoting.transport.rmi.RMIServerInvoker

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.