All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.enterprise.deployment.MethodDescriptor Maven / Gradle / Ivy

There is a newer version: 10.0-b28
Show newest version
/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 * 
 * Copyright 1997-2007 Sun Microsystems, Inc. All rights reserved.
 * 
 * The contents of this file are subject to the terms of either the GNU
 * General Public License Version 2 only ("GPL") or the Common Development
 * and Distribution License("CDDL") (collectively, the "License").  You
 * may not use this file except in compliance with the License. You can obtain
 * a copy of the License at https://glassfish.dev.java.net/public/CDDL+GPL.html
 * or glassfish/bootstrap/legal/LICENSE.txt.  See the License for the specific
 * language governing permissions and limitations under the License.
 * 
 * When distributing the software, include this License Header Notice in each
 * file and include the License file at glassfish/bootstrap/legal/LICENSE.txt.
 * Sun designates this particular file as subject to the "Classpath" exception
 * as provided by Sun in the GPL Version 2 section of the License file that
 * accompanied this code.  If applicable, add the following below the License
 * Header, with the fields enclosed by brackets [] replaced by your own
 * identifying information: "Portions Copyrighted [year]
 * [name of copyright owner]"
 * 
 * Contributor(s):
 * 
 * If you wish your version of this file to be governed by only the CDDL or
 * only the GPL Version 2, indicate your decision by adding "[Contributor]
 * elects to include this software in this distribution under the [CDDL or GPL
 * Version 2] license."  If you don't indicate a single choice of license, a
 * recipient has the option to distribute your version of this file under
 * either the CDDL, the GPL Version 2 or to extend the choice of license to
 * its licensees as provided above.  However, if you add GPL Version 2 code
 * and therefore, elected the GPL Version 2 license, then the option applies
 * only if the new code is made subject to such option by the copyright
 * holder.
 */
package com.sun.enterprise.deployment;

import java.lang.reflect.Method;
import java.util.*;
import com.sun.enterprise.util.LocalStringManagerImpl;
import java.util.logging.*;
import com.sun.enterprise.deployment.util.LogDomains;
import com.sun.enterprise.deployment.util.TypeUtil;
import org.glassfish.deployment.common.DeploymentUtils;

    /** I am a deployment object representing a single method or a collection
    * of methods on Enterprise Bean classes.
    * @author Danny Coward
    */

public final class MethodDescriptor extends Descriptor {
    /** Represents the bean home interface ejbClassSymbol.*/
    public static final String EJB_HOME = "Home";
    /** Represents the bean local home interface ejbClassSymbol.*/
    public static final String EJB_LOCALHOME = "LocalHome";
    /** Represents the bean local home interface ejbClassSymbol.*/
    public static final String EJB_OPTIONAL_LOCALHOME = "OptionalLocalHome";
    /** Represents the bean remote interface ejbClassSymbol.*/
    public static final String EJB_REMOTE = "Remote";
    /** Represents the bean local interface ejbClassSymbol.*/
    public static final String EJB_LOCAL = "Local";
    /** Represents the optional local interface */
    public static final String EJB_OPTIONAL_LOCAL = "LocalBean";
    /** Represents the web service interface ejbClassSymbol.*/
    public static final String EJB_WEB_SERVICE = "ServiceEndpoint";
    /** Represents the bean class ejbClassSymbol.*/
    public static final String EJB_BEAN = "Bean";
    /** Unused.*/
    public static final String ALL_OF_NAME = "AllOfName";
    /** The method descriptor name representing all methods.*/
    public static final String ALL_EJB_METHODS = "*";
    public static final String ALL_METHODS = "*";
    
    private String[] parameterClassNames = null;
    private String[] javaParameterClassNames = null;
    private String className = ""; // cache this
    private String ejbClassSymbol;
    private String ejbName;
    private static LocalStringManagerImpl localStrings =
	    new LocalStringManagerImpl(MethodDescriptor.class);    
	    
     final static Logger _logger = LogDomains.getLogger(DeploymentUtils.class, LogDomains.DPL_LOGGER);
		
    private final int JAVA_FORMAT = 1;
    private final int XML_FORMAT = -1;
    private final int XML_JAVA_FORMAT = 0;
    private boolean isExact = false;
    
    /** 
    * Constructs a method descriptor corresponding to methods on the ejb class defined by the ejbClassSymbol (or home
    * and remote if null) with the same name (or all if ALL_EJB_METHODS) and paramater list (or just all by name of this is null).
    * (Styles 1 2 and 3 in the ejb specification)
    */
    public MethodDescriptor(String name, String description, String[] parameterClassNames, String ejbClassSymbol) {
	super(name, description);
	if (name == null) {
	    super.setName("");
	}
        if (parameterClassNames != null)
            convertToAppropriateFormat (parameterClassNames);
	this.setEjbClassSymbol(ejbClassSymbol);
    }
           
    // converts an XML style parameter class name to java style and vice versa
    private void convertToAppropriateFormat (String[] parameterClassNames){
	int format = isJavaFormat (parameterClassNames);
	// not java format so fix the java string
	if (format == JAVA_FORMAT) {
	    this.javaParameterClassNames = parameterClassNames;
	    this.parameterClassNames =
		fixParamClassNames (parameterClassNames);
	} else if (format == XML_FORMAT) { // fix the non java string
	    this.javaParameterClassNames = 
		xmlFormat2JavaClassNames (parameterClassNames);
	    this.parameterClassNames = parameterClassNames;
	} else if (format == XML_JAVA_FORMAT){
	    // let them be as it is makes no difference
	    this.javaParameterClassNames = parameterClassNames;
	    this.parameterClassNames = parameterClassNames;
	}
    }
    /** Constructor for styles 2 and 1. 
    ** Style 1 iff ALL_METHODS is used
    */
    
    public MethodDescriptor(String name, String description, String ejbClassSymbol) {
	super(name, description);
	this.parameterClassNames = null;
	this.setEjbClassSymbol(ejbClassSymbol);
    }
    
    /** Construct an exact method descriptor from the given method object, classloader and ejb descriptor.
    */
    public MethodDescriptor(Method method, String methodIntf) {
        this(method);
        isExact=true;
	this.setEjbClassSymbol(methodIntf);
    }

    /** Construct an method descriptor from the given method object.
     */
    public MethodDescriptor(Method method) {
        super(method.getName(), "");
        Class methodClass = method.getClass();
        Method[] methods = methodClass.getMethods();
        this.parameterClassNames = getParameterClassNamesFor(method);
        this.javaParameterClassNames = getJavaFormatClassNamesFor(method);
        this.className = method.getDeclaringClass().getName();
    }
    
    public MethodDescriptor() {
    }
    
    public void setEmptyParameterClassNames() {
        parameterClassNames = new String[0];
    }

    // XXX JD fix this
    public void addParameterClass(String parameter) {
        if (parameterClassNames==null) {
            parameterClassNames = new String[1];
        } else {
            String [] newParameterClassNames = new String[parameterClassNames.length + 1];
            for (int i=0;i
     * @return the style level of this method descriptors. According to the J2EE spec, methods
     * can be described byt using style 1, style 2 or style 3 xml tags. 
     * 

*/ public int getStyle() { if (getName().equals(ALL_EJB_METHODS)) return 1; if (getParameterClassNames()==null) return 2; return 3; } public Method getMethod(EjbDescriptor ejbDescriptor) { Method method = null; try { ClassLoader classloader = ejbDescriptor.getEjbBundleDescriptor().getClassLoader(); String[] javaParamClassNames = getJavaParameterClassNames(); if ( ejbClassSymbol == null || ejbClassSymbol.equals("") || ejbClassSymbol.equals(EJB_BEAN) ) { try { if ( !(className.equals("")) ) { // If declaring class is known, use it. Since method // can have any access type and there is no need // to search super-classes, use // Class.getDeclaredMethod() lookup behavior. Class declaringClass =classloader.loadClass(className); return TypeUtil.getDeclaredMethod (declaringClass, classloader, getName(), javaParamClassNames); } else { // Method is public but can be anywhere in class // hierarchy. Class ejbClass = classloader.loadClass (ejbDescriptor.getEjbClassName()); return TypeUtil.getMethod(ejbClass, classloader, getName(), javaParamClassNames); } } catch(NoSuchMethodException nsme) {} try { if( ejbDescriptor.isRemoteInterfacesSupported() ) { Class homeClass = classloader.loadClass (ejbDescriptor.getHomeClassName()); return TypeUtil.getMethod(homeClass, classloader, getName(), javaParamClassNames); } } catch(NoSuchMethodException nsme) {} try { if( ejbDescriptor.isLocalInterfacesSupported() ) { Class cl = classloader.loadClass (ejbDescriptor.getLocalHomeClassName()); return TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } } catch(NoSuchMethodException nsme) {} try { if( ejbDescriptor.hasWebServiceEndpointInterface() ) { Class cl = classloader.loadClass (ejbDescriptor.getWebServiceEndpointInterfaceName()); return TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } } catch(NoSuchMethodException nsme) {} } else if ( ejbClassSymbol.equals(EJB_HOME) ) { try { Class homeClass = classloader.loadClass(ejbDescriptor.getHomeClassName()); method = TypeUtil.getMethod(homeClass, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} } else if ( ejbClassSymbol.equals(EJB_LOCALHOME) ) { try { Class cl = classloader.loadClass( ejbDescriptor.getLocalHomeClassName()); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} } else if ( ejbClassSymbol.equals(EJB_REMOTE) ) { if( ejbDescriptor.isRemoteInterfacesSupported() ) { try { Class cl = classloader.loadClass( ejbDescriptor.getRemoteClassName()); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} } if( (method == null) && ejbDescriptor.isRemoteBusinessInterfacesSupported() ) { for(String intf : ejbDescriptor.getRemoteBusinessClassNames() ) { try { Class cl = classloader.loadClass(intf); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} if( method != null ) { break; } } } } else if ( ejbClassSymbol.equals(EJB_OPTIONAL_LOCAL) ) { if (ejbDescriptor.isOptionalLocalBusinessViewSupported()) { try { Class cl = classloader.loadClass( ejbDescriptor.getEjbClassName()); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch (NoSuchMethodException nsme) { } } } else if ( ejbClassSymbol.equals(EJB_LOCAL) ) { if( ejbDescriptor.isLocalInterfacesSupported() ) { try { Class cl = classloader.loadClass( ejbDescriptor.getLocalClassName()); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} } if( (method == null) && ejbDescriptor.isLocalBusinessInterfacesSupported() ) { for(String intf : ejbDescriptor.getLocalBusinessClassNames() ) { try { Class cl = classloader.loadClass(intf); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} if( method != null ) { break; } } } } else if ( ejbClassSymbol.equals(EJB_WEB_SERVICE) ) { try { Class cl = classloader.loadClass (ejbDescriptor.getWebServiceEndpointInterfaceName()); method = TypeUtil.getMethod(cl, classloader, getName(), javaParamClassNames); } catch(NoSuchMethodException nsme) {} } } catch(Exception e) { _logger.log(Level.SEVERE,"enterprise.deployment.backend.methodClassLoadFailure",new Object[]{ejbDescriptor}); } return method; } public Method getMethod(Class declaringClass) { try { return TypeUtil.getMethod(declaringClass, declaringClass.getClassLoader(), getName(), getJavaParameterClassNames()); } catch(Exception e) { _logger.log(Level.SEVERE,"enterprise.deployment.backend.methodClassLoadFailure",new Object[]{declaringClass}); return null; } } public Method getDeclaredMethod(Class declaringClass) { try { return TypeUtil.getDeclaredMethod(declaringClass, declaringClass.getClassLoader(), getName(), getJavaParameterClassNames()); } catch(Exception e) { _logger.log(Level.SEVERE,"enterprise.deployment.backend.methodClassLoadFailure",new Object[]{declaringClass}); return null; } } /** * Performs a conversion from the style1 style2 and style3 (no interface symbol) to * method descriptors of style3 with an interface symbol. */ public Vector doStyleConversion(EjbDescriptor ejbDescriptor, Collection allMethods) { // must be exact methods Vector v = new Vector(); if (this.getName().equals(ALL_EJB_METHODS)) { // STYLE 1 for (Iterator itr = allMethods.iterator(); itr.hasNext();) { MethodDescriptor next = (MethodDescriptor) itr.next(); // when ejb-name is present // since it is an optional element in some case if (this.getEjbName() != null && this.getEjbName().length() > 0) { next.setEjbName(ejbDescriptor.getName()); } if (!next.isExact()) { //throw new RuntimeException("Conversion failed: " + next); } if (this.getDescription() != null && this.getDescription().length() > 0) { next.setDescription(this.getDescription()); } if (getEjbClassSymbol()==null) { v.addElement(next); } else if (this.getEjbClassSymbol().equals(next.getEjbClassSymbol())) { v.addElement(next); } } } else if (this.getParameterClassNames() == null) { // STYLE 2 v.addAll(this.getMethodDescriptorsOfName(this.getName(), allMethods)); } else { // STYLE 3, but maybe not exact if (getEjbClassSymbol()==null) { v.addAll(this.getMethodDescriptorsOfNameAndParameters(this.getName(), this.getParameterClassNames(), allMethods)); } else { v.addElement(this); // this must be exact } } return v; } private Set getMethodDescriptorsOfNameAndParameters(String name, String[] parameterArray, Collection methodDescriptors) { Set methods = new HashSet(); for (Iterator itr = getMethodDescriptorsOfName(name, methodDescriptors).iterator(); itr.hasNext();) { MethodDescriptor next = (MethodDescriptor) itr.next(); next.setEjbName(getEjbName()); if (stringArrayEquals(parameterArray, next.getParameterClassNames())) { methods.add(next); } } return methods; } private Set getMethodDescriptorsOfName(String name, Collection methodDescriptors) { Set set = new HashSet(); for (Iterator itr = methodDescriptors.iterator(); itr.hasNext();) { MethodDescriptor next = (MethodDescriptor) itr.next(); next.setEjbName(getEjbName()); if (name.equals(next.getName())) { if (getEjbClassSymbol()==null) { set.add(next); } else if (getEjbClassSymbol().equals(next.getEjbClassSymbol())) { set.add(next); } } } return set; } /** Returns the ejb class sybol for this method descriptor. */ public String getEjbClassSymbol() { return this.ejbClassSymbol; } /** Sets the ejb class sybol for this method descriptor. */ public void setEjbClassSymbol(String ejbClassSymbol) { this.ejbClassSymbol = ejbClassSymbol; } public String getFormattedString() { return this.getName() + this.getPrettyParameterString(); } public String getPrettyParameterString() { String prettyParameterString = "("; if (this.parameterClassNames != null) { for (int i = 0; i < this.parameterClassNames.length; i++) { int j = i + 1; if (i > 0) { prettyParameterString = prettyParameterString + ", " + this.parameterClassNames[i] + " p" + j; } else { prettyParameterString = prettyParameterString + this.parameterClassNames[i] + " p" + j; } } } prettyParameterString = prettyParameterString + ")"; return prettyParameterString; } public String[] getParameterClassNames() { return parameterClassNames; } public String[] getJavaParameterClassNames (){ return javaParameterClassNames; } private boolean stringArrayEquals(String[] s1, String[] s2) { if (s1 == null && s2 == null) { return true; } if (s1 == null && s2 != null) { return false; } if (s2 == null && s1 != null) { return false; } if (s1.length == s2.length) { for (int i = 0; i < s1.length; i++) { if (!s1[i].equals(s2[i])) { return false; } } return true; } else { return false; } } /** Equlity iff the parameter names match and the name matches.*/ public boolean equals(Object other) { if (other != null && other instanceof MethodDescriptor) { MethodDescriptor otherMethodDescriptor = (MethodDescriptor) other; if (otherMethodDescriptor.getName().equals(getName()) && stringArrayEquals(otherMethodDescriptor.getParameterClassNames(), getParameterClassNames())) { if (getEjbClassSymbol()!=null && otherMethodDescriptor.getEjbClassSymbol()!=null) { return getEjbClassSymbol().equals(otherMethodDescriptor.getEjbClassSymbol()); } // if the ejb class symbol is not defined in both descriptor, we consider // the method described being the same. return true; } } return false; } /** Indicates if a method descriptor implies the other one*/ public boolean implies(Object other) { if (other != null && other instanceof MethodDescriptor) { MethodDescriptor otherMethodDescriptor = (MethodDescriptor) other; if (getName().equals(ALL_METHODS) || getName().equals(otherMethodDescriptor.getName())) { if (getParameterClassNames() == null || stringArrayEquals(getParameterClassNames(), otherMethodDescriptor.getParameterClassNames())) { return true; } } } return false; } /** My Hashcode. */ public int hashCode() { return this.getPrettyParameterString().hashCode() + this.getName().hashCode(); } /** My pretty format. */ public void print(StringBuffer toStringBuffer) { toStringBuffer.append("Method Descriptor").append((ejbName==null?"":" for ejb " + ejbName)).append( " name: ").append(this.getName()).append(" params: ").append(this.getPrettyParameterString()).append( " intf: ").append(this.ejbClassSymbol); } public String prettyPrint() { return "Name : " + this.getName() + " Params: " + this.getPrettyParameterString() + " Intf: " + this.ejbClassSymbol; } public String[] getParameterClassNamesFor(Method method) { Class[] classes = method.getParameterTypes(); String[] classNames = new String[classes.length]; for (int i = 0; i < classes.length; i++) { Class compType = classes[i].getComponentType(); if ( compType == null ) { // not an array classNames[i] = classes[i].getName(); } else { // name of array types should be like int[][][] // Class.getName() returns something like [[[I int dimensions = 1; while(compType.getComponentType()!=null) { dimensions++; compType=compType.getComponentType(); } classNames[i] = compType.getName(); // add "[]" depending on array dimension for (int j=0;j




© 2015 - 2025 Weber Informatics LLC | Privacy Policy