Package com.caucho.ejb.protocol

Source Code of com.caucho.ejb.protocol.EjbProtocolManager

/*
* Copyright (c) 1998-2011 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source 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, or any warranty
* of NON-INFRINGEMENT.  See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
*   Free Software Foundation, Inc.
*   59 Temple Place, Suite 330
*   Boston, MA 02111-1307  USA
*
* @author Scott Ferguson
*/

package com.caucho.ejb.protocol;

import com.caucho.config.ConfigException;
import com.caucho.ejb.manager.EjbManager;
import com.caucho.ejb.server.AbstractEjbBeanManager;
import com.caucho.server.e_app.EnterpriseApplication;
import com.caucho.naming.Jndi;
import com.caucho.util.L10N;

import javax.enterprise.inject.spi.AnnotatedType;
import javax.naming.NamingException;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Iterator;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
* Server containing all the EJBs for a given configuration.
*
* <p>
* Each protocol will extend the container to override Handle creation.
*/
public class EjbProtocolManager {
  @SuppressWarnings("unused")
  private static final L10N L = new L10N(EjbProtocolManager.class);
  protected static final Logger log = Logger.getLogger(EjbProtocolManager.class
      .getName());

  private static ThreadLocal<String> _protocolLocal = new ThreadLocal<String>();

  private static Hashtable<String, WeakReference<AbstractEjbBeanManager<?>>> _staticServerMap
    = new Hashtable<String, WeakReference<AbstractEjbBeanManager<?>>>();

  private final EjbManager _ejbContainer;
  private final ClassLoader _loader;

  private String _localJndiPrefix; // = "java:comp/env/cmp";
  private String _remoteJndiPrefix; // = "java:comp/env/ejb";

  private String _jndiPrefix; // java:comp/env/ejb/FooBean/local

  private HashMap<String, AbstractEjbBeanManager<?>> _serverMap
    = new HashMap<String, AbstractEjbBeanManager<?>>();

  // handles remote stuff
  protected ProtocolContainer _protocolContainer;
  protected HashMap<String, ProtocolContainer> _protocolMap = new HashMap<String, ProtocolContainer>();

  /**
   * Create a server with the given prefix name.
   */
  public EjbProtocolManager(EjbManager ejbContainer) throws ConfigException
  {
    _ejbContainer = ejbContainer;
    _loader = _ejbContainer.getClassLoader();

    EjbManager parent = ejbContainer.getParent();
    if (parent != null) {
      _localJndiPrefix = parent.getProtocolManager().getLocalJndiPrefix();
      _remoteJndiPrefix = parent.getProtocolManager().getRemoteJndiPrefix();
      _jndiPrefix = parent.getProtocolManager().getJndiPrefix();
    }
  }

  public void setJndiPrefix(String name)
  {
    _jndiPrefix = name;
  }

  public String getJndiPrefix()
  {
    return _jndiPrefix;
  }

  public void setLocalJndiPrefix(String name)
  {
    _localJndiPrefix = name;
  }

  public String getLocalJndiPrefix()
  {
    return _localJndiPrefix;
  }

  public void setRemoteJndiPrefix(String name)
  {
    _remoteJndiPrefix = name;
  }

  public String getRemoteJndiPrefix()
  {
    return _remoteJndiPrefix;
  }

  /**
   * Returns the EJB server.
   */
  public EjbManager getEjbContainer()
  {
    return _ejbContainer;
  }

  /**
   * Initialize the protocol manager.
   */
  public void init() throws NamingException
  {
  }

  /**
   * Gets the current protocol.
   */
  public static String getThreadProtocol()
  {
    return _protocolLocal.get();
  }

  /**
   * Gets the current protocol.
   */
  public static String setThreadProtocol(String protocol)
  {
    String oldProtocol = _protocolLocal.get();

    _protocolLocal.set(protocol);

    return oldProtocol;
  }

  public void setProtocolContainer(ProtocolContainer protocol)
  {
    _protocolContainer = protocol;

    synchronized (_protocolMap) {
      _protocolMap.put(protocol.getName(), protocol);
    }

    addProtocolServers(protocol);
  }

  public void addProtocolContainer(ProtocolContainer protocol)
  {
    if (_protocolContainer == null)
      _protocolContainer = protocol;

    addProtocolContainer(protocol.getName(), protocol);
  }

  public void removeProtocolContainer(ProtocolContainer protocol)
  {
    if (_protocolContainer == protocol)
      _protocolContainer = null;

    synchronized (_protocolMap) {
      _protocolMap.remove(protocol.getName());
    }
  }

  public void addProtocolContainer(String name, ProtocolContainer protocol)
  {
    synchronized (_protocolMap) {
      if (_protocolMap.get(name) == null)
        _protocolMap.put(name, protocol);
    }

    addProtocolServers(protocol);
  }

  public ProtocolContainer getProtocol(String name)
  {
    synchronized (_protocolMap) {
      return _protocolMap.get(name);
    }
  }

  private void addProtocolServers(ProtocolContainer protocol)
  {
    for (AbstractEjbBeanManager<?> server : _serverMap.values()) {
      protocol.addServer(server);
    }
  }

  /**
   * Returns the named server if it's in the same JVM.
   */
  public static AbstractEjbBeanManager<?> getJVMServer(String serverId)
  {
    WeakReference<AbstractEjbBeanManager<?>> serverRef
      = _staticServerMap.get(serverId);

    return serverRef != null ? serverRef.get() : null;
  }

  /**
   * Adds a server.
   */
  public void addServer(AbstractEjbBeanManager<?> server)
  {
    _serverMap.put(server.getProtocolId(), server);

    for (ProtocolContainer protocol : _protocolMap.values()) {
      protocol.addServer(server);
    }

    Thread thread = Thread.currentThread();
    ClassLoader loader = thread.getContextClassLoader();

    try {
      Thread.currentThread().setContextClassLoader(_loader);

      // ejb/0g11
      // remote without a local interface should not get bound
      // with the local prefix

      bindDefaultJndi(_jndiPrefix, server);
     
      bindPortableJndiApis(server);
    } catch (RuntimeException e) {
      throw e;
    } catch (Exception e) {
      throw ConfigException.create(e);
    } finally {
      Thread.currentThread().setContextClassLoader(loader);
    }
  }

  @SuppressWarnings("unchecked")
  private void bindDefaultJndi(String prefix, AbstractEjbBeanManager server)
  {
    try {
      EnterpriseApplication eApp = EnterpriseApplication.getCurrent();

      if (prefix == null)
        prefix = "";
      else if (!prefix.endsWith("/"))
        prefix = prefix + "/";

      if (eApp != null && eApp.getName() != null)
        prefix = prefix + eApp.getName() + "/";

      prefix = prefix + server.getEJBName();

      ClassLoader loader = Thread.currentThread().getContextClassLoader();

      ArrayList<AnnotatedType<?>> apiList = server.getLocalApi();
      if (apiList != null && apiList.size() > 0) {
        String jndiName = prefix + "/local";

        Class<?> localApi = apiList.get(0).getJavaClass();
       
        Jndi.bindDeep(jndiName, new ServerLocalProxy(server, localApi));

        log.finest(server + " local binding to '" + jndiName + "' " + loader);
      }
     

      Object localHome = null;

      if (localHome != null) {
        String jndiName = prefix + "/local-home";

        Jndi.bindDeep(jndiName, localHome);

        log.finest(server + " local-home binding to '" + jndiName + "' "
                   + loader);
      }
    } catch (Exception e) {
      throw ConfigException.create(e);
    }
  }

  private void bindPortableJndi(String appName,
                                String moduleName,
                                String suffix,
                                ServerLocalProxy proxy)
  {
    try {
      Thread thread = Thread.currentThread();
      ClassLoader oldLoader = thread.getContextClassLoader();
     
      try {
        thread.setContextClassLoader(_loader);

        String jndiName = null;
       
        // global _may_ mean across applications, but is not required
        // (EJB 3.1 spec. sec. 3.2.2).
        if (appName != null)
          jndiName = "java:global/" + appName + '/' + moduleName + '/' + suffix;
        else
          jndiName = "java:global/" + moduleName + '/' + suffix;

        // XXX: for embed client must be true global 
        thread.setContextClassLoader(_ejbContainer.getGlobalClassLoader());

        Jndi.bindDeep(jndiName, proxy);
        log.finer(proxy + " global binding to '" + jndiName + "' "
                  + _loader);
       
        thread.setContextClassLoader(_loader);
       
        // application means across modules within an application
        jndiName = "java:app/" + moduleName + '/' + suffix;
        Jndi.bindDeep(jndiName, proxy);
        log.finest(proxy + " application binding to '" + jndiName + "' "
                   + _loader);

        // XXX module binding - this is problematic because this will
        // expose a module-level binding to the application context
        // used by the EJBContainer
       
        // module means local to a single module within an application
        jndiName = "java:module/" + suffix;
        Jndi.bindDeep(jndiName, proxy);
       
        log.finest(proxy + " module binding to '" + jndiName + "' "
                   + _loader);
      }
      finally {
        Thread.currentThread().setContextClassLoader(oldLoader);
      }
    } catch (Exception e) {
      throw ConfigException.create(e);
    }
  }

  private <X> void bindPortableJndiApis(AbstractEjbBeanManager<X> manager)
  {
    String appName = null;

    EnterpriseApplication app = EnterpriseApplication.getCurrent();
    if (app != null)
      appName = app.getName();
   
    String moduleName = manager.getModuleName();

    if (moduleName != null) {
      ServerLocalProxy proxy = null;
     
      if (manager.getLocalBean() != null) {
        String suffix = manager.getEJBName();
        Class<?> api = manager.getEjbClass();

        proxy = new ServerLocalProxy(manager, api);

        bindPortableJndi(appName, moduleName, suffix, proxy);

        suffix = suffix + '!' + manager.getEjbClass().getName();
        bindPortableJndi(appName, moduleName, suffix, proxy);
      }

      ArrayList<AnnotatedType<? super X>> apiList = manager.getLocalApi();

      if (apiList.size() == 1) {
        String suffix = manager.getEJBName();
        Class<?> api = apiList.get(0).getJavaClass();

        proxy = new ServerLocalProxy(manager, api);

        if (manager.getLocalBean() == null)
          bindPortableJndi(appName, moduleName, suffix, proxy);

        suffix = suffix + '!' + api.getName();
        bindPortableJndi(appName, moduleName, suffix, proxy);
      }
      else {
        for (AnnotatedType<?> api : apiList) {
          String suffix = manager.getEJBName() + '!' + api.getJavaClass().getName();

          proxy = new ServerLocalProxy(manager, api.getJavaClass());

          bindPortableJndi(appName, moduleName, suffix, proxy);
        }
      }
    }
  }

  /**
   * Adds a server.
   */
  public void removeServer(AbstractEjbBeanManager<?> server)
    throws NamingException
  {
    for (ProtocolContainer protocol : _protocolMap.values()) {
      protocol.removeServer(server);
    }
  }

  /**
   * Returns the server specified by the serverId.
   */
  public AbstractEjbBeanManager<?> getServerByEJBName(String ejbName)
  {
    if (!ejbName.startsWith("/"))
      ejbName = "/" + ejbName;

    return _serverMap.get(ejbName);
  }

  /**
   * Returns the server specified by the serverId.
   */
  public AbstractEjbBeanManager<?> getServerByServerId(String protocolId)
  {
    for (AbstractEjbBeanManager<?> server : _serverMap.values()) {
      if (protocolId.equals(server.getProtocolId()))
        return server;
    }

    return null;
  }

  @SuppressWarnings("unchecked")
  public Iterator getLocalNames()
  {
    return _serverMap.keySet().iterator();
  }

  /**
   * Returns a list of child EJB names.
   *
   * @param ejbName
   *          the name which might be a prefix.
   */
  public ArrayList<String> getLocalChildren(String ejbName)
  {
    if (!ejbName.startsWith("/"))
      ejbName = "/" + ejbName;

    if (!ejbName.endsWith("/"))
      ejbName = ejbName + "/";

    ArrayList<String> children = new ArrayList<String>();

    Iterator<String> iter = _serverMap.keySet().iterator();
    while (iter.hasNext()) {
      String name = iter.next();

      AbstractEjbBeanManager<?> server = _serverMap.get(name);

      if (server.getLocalProxy(null) == null)
        continue;

      if (name.startsWith(ejbName)) {
        int prefixLength = ejbName.length();
        int p = name.indexOf('/', prefixLength);

        if (p > 0)
          name = name.substring(prefixLength, p);
        else
          name = name.substring(prefixLength);

        if (!children.contains(name))
          children.add(name);
      }
    }

    return children;
  }

  /**
   * Returns a list of child EJB names.
   *
   * @param ejbName
   *          the name which might be a prefix.
   */
  public ArrayList<String> getRemoteChildren(String ejbName)
  {
    if (!ejbName.startsWith("/"))
      ejbName = "/" + ejbName;

    ArrayList<String> children = new ArrayList<String>();

    Iterator<String> iter = _serverMap.keySet().iterator();
    while (iter.hasNext()) {
      String name = iter.next();

      AbstractEjbBeanManager<?> server = _serverMap.get(name);
    }

    if (children.size() == 0)
      return null;
    else
      return children;
  }

  /**
   * Destroys the manager.
   */
  public void destroy()
  {
  }
}
TOP

Related Classes of com.caucho.ejb.protocol.EjbProtocolManager

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.