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

org.jibx.ws.wsdl.tools.custom.OperationCustom Maven / Gradle / Ivy

/*
 * Copyright (c) 2007-2008, Dennis M. Sosnoski. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 * 
 * Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 * disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 * following disclaimer in the documentation and/or other materials provided with the distribution. Neither the name of
 * JiBX nor the names of its contributors may be used to endorse or promote products derived from this software without
 * specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package org.jibx.ws.wsdl.tools.custom;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.log4j.Logger;
import org.jibx.custom.CustomUtils;
import org.jibx.custom.classes.CustomBase;
import org.jibx.custom.classes.IDocumentFormatter;
import org.jibx.custom.classes.SharedNestingBase;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.JiBXException;
import org.jibx.runtime.impl.UnmarshallingContext;
import org.jibx.util.IClass;
import org.jibx.util.IClassItem;
import org.jibx.util.IClassLocator;
import org.jibx.util.StringArray;
import org.jibx.ws.wsdl.tools.SignatureParser;

/**
 * Operation customization information. This supports direct operation customizations (such as the corresponding request
 * and/or response element name) and also acts as a container for parameter and/or return customizations.
 * 
 * @author Dennis M. Sosnoski
 */
public class OperationCustom extends NestingBase
{
    /** Logger for class. */
    private static final Logger s_logger = Logger.getLogger(OperationCustom.class.getName());
    
    /** Enumeration of allowed attribute names */
    public static final StringArray s_allowedAttributes =
        new StringArray(new String[] { "method-name", "operation-name", "optionals", "request-message",
            "request-wrapper", "requireds", "response-message", "response-wrapper", "soap-action" },
            SharedNestingBase.s_allowedAttributes);
    
    // values specific to class level
    private String m_methodName;
    
    private String m_operationName;
    
    private String m_requestMessageName;
    
    private String m_requestWrapperName;
    
    private String m_responseMessageName;
    
    private String m_responseWrapperName;
    
    private String m_soapAction;
    
    private List m_documentation;
    
    private String[] m_requireds;
    
    private String[] m_optionals;
    
    // list of contained parameter customizations
    private final ArrayList m_parameters;
    
    // contained result customization (null if none)
    private ValueCustom m_return;
    
    // list of contained throws customizations
    private final ArrayList m_throws;
    
    /**
     * Constructor.
     * 
     * @param parent
     * @param name method name
     */
    OperationCustom(NestingBase parent, String name) {
        super(parent);
        m_methodName = name;
        m_parameters = new ArrayList();
        m_throws = new ArrayList();
    }
    
    /**
     * Get the namespace for WSDL definitions of this service.
     * 
     * @return WSDL namespace
     */
    public String getWsdlNamespace() {
        return ((NestingBase)getParent()).getWsdlNamespace();
    }
    
    /**
     * Get method name.
     * 
     * @return name
     */
    public String getMethodName() {
        return m_methodName;
    }
    
    /**
     * Get the operation name.
     * 
     * @return operation name
     */
    public String getOperationName() {
        return m_operationName;
    }
    
    /**
     * Get request message name.
     * 
     * @return name
     */
    public String getRequestMessageName() {
        return m_requestMessageName;
    }
    
    /**
     * Get request wrapper element name.
     * 
     * @return name
     */
    public String getRequestWrapperName() {
        return m_requestWrapperName;
    }
    
    /**
     * Get response message name.
     * 
     * @return name
     */
    public String getResponseMessageName() {
        return m_responseMessageName;
    }
    
    /**
     * Get response wrapper name.
     * 
     * @return name
     */
    public String getResponseWrapperName() {
        return m_responseWrapperName;
    }
    
    /**
     * Get return value.
     * 
     * @return return
     */
    public ValueCustom getReturn() {
        return m_return;
    }
    
    /**
     * Get SOAPAction.
     * 
     * @return soapAction
     */
    public String getSoapAction() {
        return m_soapAction;
    }
    
    /**
     * Get operation documentation.
     * 
     * @return list of documentation nodes (null if none)
     */
    public List getDocumentation() {
        return m_documentation;
    }
    
    /**
     * Get list of children.
     * 
     * @return list
     */
    public ArrayList getParameters() {
        return m_parameters;
    }
    
    /**
     * Get list of throws customizations.
     * 
     * @return list
     */
    public ArrayList getThrows() {
        return m_throws;
    }
    
    /**
     * Add child.
     * 
     * @param child
     */
    protected void addChild(CustomBase child) {
        if (child.getParent() == this) {
            m_parameters.add(child);
        } else {
            throw new IllegalStateException("Internal error: child not linked");
        }
    }
    
    /**
     * Unmarshalling factory. This gets the containing element and the name so that the standard constructor can be
     * used.
     * 
     * @param ictx
     * @return created instance
     * @throws JiBXException
     */
    private static OperationCustom factory(IUnmarshallingContext ictx) throws JiBXException {
        String mname = ((UnmarshallingContext)ictx).attributeText(null, "method-name");
        s_logger.debug("Creating operation for operation name " + mname);
        return new OperationCustom((NestingBase)getContainingObject(ictx), mname);
    }
    
    /**
     * Check if type is a collection type (specifically collection, not array).
     * 
     * @param type
     * @return item type, null if not a collection type
     */
    private boolean isCollection(String type, IClassLocator icl) {
        IClass info = icl.getRequiredClassInfo(type);
        return info.isImplements("Ljava/util/Collection;");
    }
    
    /**
     * Parse parameter type.
     * 
     * @param parse
     * @return parameter type
     */
    private String parameterType(SignatureParser parse) {
        String itype = null;
        while (parse.next() != SignatureParser.TYPE_PARAMETERS_END_EVENT) {
            if (itype == null && parse.getEvent() == SignatureParser.TYPE_EVENT) {
                itype = parse.getType();
            }
        }
        return itype;
    }
    
    /**
     * Check if a particular value is required or optional.
     * 
     * @param name
     * @param reqset
     * @param optset
     * @return TRUE if required, FALSE if optional, null if unknown
     */
    private static Boolean checkRequired(String name, Set reqset, Set optset) {
        if (reqset != null && reqset.contains(name)) {
            return Boolean.TRUE;
        } else if (optset != null && optset.contains(name)) {
            return Boolean.FALSE;
        } else {
            return null;
        }
    }
    
    /**
     * Apply customizations to method to fill out parameter and return information.
     * 
     * @param method
     * @param icl
     * @param fmt
     */
    public void apply(IClassItem method, IClassLocator icl, IDocumentFormatter fmt) {
        
        // fill in missing details
        s_logger.debug("Applying operation from method " + m_methodName);
        if (m_operationName == null) {
            String name = convertName(m_methodName, CAMEL_CASE_NAMES);
            m_operationName = registerName(name, this);
            s_logger.debug("Set operation name " + m_operationName);
        } else if (!m_operationName.equals(registerName(m_operationName, this))) {
            throw new IllegalStateException("Operation name conflict for '" + m_operationName + '\'');
        }
        if (m_requestMessageName == null) {
            m_requestMessageName = m_operationName + "Message";
        }
        if (m_requestWrapperName == null) {
            m_requestWrapperName = m_operationName;
        }
        if (m_responseMessageName == null) {
            m_responseMessageName = m_operationName + "ResponseMessage";
        }
        if (m_responseWrapperName == null) {
            m_responseWrapperName = m_operationName + "Response";
        }
        if (m_soapAction == null && isSoapAction()) {
            m_soapAction = "urn:" + m_operationName;
        }
        if (m_documentation == null) {
            m_documentation = fmt.docToNodes(method.getJavaDoc());
        }
        
        // find parameter types, and item types for collections
        int count = method.getArgumentCount();
        String[] ptypes = new String[count];
        String[] pitypes = new String[count];
        String rtype = null;
        String ritype = null;
        String sig = method.getGenericsSignature();
        if (sig == null) {
            
            // no signature, just use basic type information
            for (int i = 0; i < count; i++) {
                String type = method.getArgumentType(i);
                ptypes[i] = type;
                if (isCollection(type, icl)) {
                    pitypes[i] = "java.lang.Object";
                }
            }
            rtype = method.getTypeName();
            if (isCollection(rtype, icl)) {
                ritype = "java.lang.Object";
            }
            
        } else {
            
            // parse the signature to check collection item types
            SignatureParser parse = new SignatureParser(sig);
            int index = 0;
            boolean inparms = false;
            while (parse.next() != SignatureParser.END_EVENT) {
                switch (parse.getEvent()) {
                    
                    case SignatureParser.METHOD_PARAMETERS_START_EVENT:
                        inparms = true;
                        index = 0;
                        break;
                    
                    case SignatureParser.METHOD_PARAMETERS_END_EVENT:
                        inparms = false;
                        break;
                    
                    case SignatureParser.TYPE_EVENT:
                        String type = parse.getType();
                        String itype = null;
                        if (parse.isParameterized()) {
                            String ptype = parameterType(parse);
                            IClass info = icl.getRequiredClassInfo(type);
                            if (info.isImplements("Ljava/util/Collection;")) {
                                itype = ptype;
                            }
                        }
                        if (inparms) {
                            ptypes[index] = type;
                            pitypes[index++] = itype;
                        } else {
                            rtype = type;
                            ritype = itype;
                        }
                        break;
                    
                }
            }
        }
        
        // create map of parameters using separate customizations
        Map namevalue = new HashMap();
        for (int i = 0; i < m_parameters.size(); i++) {
            ValueCustom value = (ValueCustom)m_parameters.get(i);
            namevalue.put(value.getBaseName(), value);
        }
        m_parameters.clear();
        
        // fill in the parameters and return customizations
        Set reqset = CustomUtils.noCaseNameSet(m_requireds);
        Set optset = CustomUtils.noCaseNameSet(m_optionals);
        for (int i = 0; i < count; i++) {
            String name = method.getParameterName(i);
            if (name == null) {
                name = "arg" + (i + 1);
            }
            ValueCustom value = (ValueCustom)namevalue.get(name);
            if (value == null) {
                value = new ValueCustom(this, name);
                s_logger.debug("Added customization for parameter " + name + " with type " + pitypes[i]);
            } else {
                s_logger.debug("Found customization for parameter " + name + " with type " + pitypes[i]);
            }
            m_parameters.add(value);
            Boolean req = checkRequired(name, reqset, optset);
            List docs = fmt.docToNodes(method.getParameterJavaDoc(i));
            value.complete(icl.getRequiredClassInfo(ptypes[i]), docs, req, pitypes[i]);
        }
        Boolean req = checkRequired("return", reqset, optset);
        String text = method.getReturnJavaDoc();
        boolean isname = false;
        if (text != null && Character.isJavaIdentifierStart(text.charAt(0))) {
            isname = true;
            for (int i = 1; i < text.length(); i++) {
                if (!Character.isJavaIdentifierPart(text.charAt(i))) {
                    isname = false;
                    break;
                }
            }
        }
        String name = "return";
        if (isname) {
            name = text;
            text = null;
        }
        List docs = fmt.docToNodes(text);
        if (m_return == null) {
            m_return = new ValueCustom(this, name);
        }
        m_return.complete(icl.getRequiredClassInfo(rtype), docs, req, ritype);
        
        // add throws information
        count = method.getExceptions().length;
        if (m_throws.size() == 0) {
            for (int i = 0; i < count; i++) {
                name = method.getExceptions()[i];
                ThrowsCustom thrw = new ThrowsCustom(this, name);
                thrw.complete(fmt.docToNodes(method.getExceptionJavaDoc(i)));
                m_throws.add(thrw);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy