Package org.springframework.ws.wsdl.wsdl11.provider

Source Code of org.springframework.ws.wsdl.wsdl11.provider.AbstractPortTypesProvider

/*
* Copyright 2005-2014 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
*      http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package org.springframework.ws.wsdl.wsdl11.provider;

import java.util.Iterator;
import java.util.List;
import javax.wsdl.Definition;
import javax.wsdl.Fault;
import javax.wsdl.Input;
import javax.wsdl.Message;
import javax.wsdl.Operation;
import javax.wsdl.OperationType;
import javax.wsdl.Output;
import javax.wsdl.PortType;
import javax.wsdl.WSDLException;
import javax.xml.namespace.QName;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.springframework.util.Assert;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.MultiValueMap;
import org.springframework.util.StringUtils;

/**
* Abstract base class for {@link PortTypesProvider} implementations.
*
* @author Arjen Poutsma
* @since 1.5.0
*/
public abstract class AbstractPortTypesProvider implements PortTypesProvider {

    /** Logger available to subclasses. */
    protected final Log logger = LogFactory.getLog(getClass());

    private String portTypeName;

    /** Returns the port type name used for this definition. */
    public String getPortTypeName() {
        return portTypeName;
    }

    /** Sets the port type name used for this definition. Required. */
    public void setPortTypeName(String portTypeName) {
        this.portTypeName = portTypeName;
    }

    /**
     * Creates a single {@link PortType}, and calls {@link #populatePortType(Definition, PortType)} with it.
     *
     * @param definition the WSDL4J {@code Definition}
     * @throws WSDLException in case of errors
     */
    @Override
    public void addPortTypes(Definition definition) throws WSDLException {
        Assert.notNull(getPortTypeName(), "'portTypeName' is required");
        PortType portType = definition.createPortType();
        populatePortType(definition, portType);
        createOperations(definition, portType);
        portType.setUndefined(false);
        definition.addPortType(portType);
    }

    /**
     * Called after the {@link PortType} has been created.
     *
     * <p>Default implementation sets the name of the port type to the defined value.
     *
     * @param portType the WSDL4J {@code PortType}
     * @throws WSDLException in case of errors
     * @see #setPortTypeName(String)
     */
    protected void populatePortType(Definition definition, PortType portType) throws WSDLException {
        QName portTypeName = new QName(definition.getTargetNamespace(), getPortTypeName());
        if (logger.isDebugEnabled()) {
            logger.debug("Creating port type [" + portTypeName + "]");
        }
        portType.setQName(portTypeName);
    }

    private void createOperations(Definition definition, PortType portType) throws WSDLException {
        MultiValueMap<String, Message> operations = new LinkedMultiValueMap<String, Message>();
        for (Iterator<?> iterator = definition.getMessages().values().iterator(); iterator.hasNext();) {
            Message message = (Message) iterator.next();
            String operationName = getOperationName(message);
            if (StringUtils.hasText(operationName)) {
                operations.add(operationName,message);
            }
        }
        if (operations.isEmpty() && logger.isWarnEnabled()) {
            logger.warn("No operations were created, make sure the WSDL contains messages");
        }
        for (String operationName : operations.keySet()) {
            Operation operation = definition.createOperation();
            operation.setName(operationName);
            List<Message> messages = operations.get(operationName);
            for (Message message : messages) {
                if (isInputMessage(message)) {
                    Input input = definition.createInput();
                    input.setMessage(message);
                    populateInput(definition, input);
                    operation.setInput(input);
                }
                else if (isOutputMessage(message)) {
                    Output output = definition.createOutput();
                    output.setMessage(message);
                    populateOutput(definition, output);
                    operation.setOutput(output);
                }
                else if (isFaultMessage(message)) {
                    Fault fault = definition.createFault();
                    fault.setMessage(message);
                    populateFault(definition, fault);
                    operation.addFault(fault);
                }
            }
            operation.setStyle(getOperationType(operation));
            operation.setUndefined(false);
            if (logger.isDebugEnabled()) {
                logger.debug(
                        "Adding operation [" + operation.getName() + "] to port type [" + portType.getQName() + "]");
            }
            portType.addOperation(operation);
        }
    }

    /**
     * Template method that returns the name of the operation coupled to the given {@link Message}. Subclasses can
     * return {@code null} to indicate that a message should not be coupled to an operation.
     *
     * @param message the WSDL4J {@code Message}
     * @return the operation name; or {@code null}
     */
    protected abstract String getOperationName(Message message);

    /**
     * Indicates whether the given name name should be included as {@link Input} message in the definition.
     *
     * @param message the message
     * @return {@code true} if to be included as input; {@code false} otherwise
     */
    protected abstract boolean isInputMessage(Message message);

    /**
     * Called after the {@link javax.wsdl.Input} has been created, but it's added to the operation. Subclasses can
     * override this method to define the input name.
     *
     * <p>Default implementation sets the input name to the message name.
     *
     * @param definition the WSDL4J {@code Definition}
     * @param input      the WSDL4J {@code Input}
     */
    protected void populateInput(Definition definition, Input input) {
        input.setName(input.getMessage().getQName().getLocalPart());
    }

    /**
     * Indicates whether the given name name should be included as {@link Output} message in the definition.
     *
     * @param message the message
     * @return {@code true} if to be included as output; {@code false} otherwise
     */
    protected abstract boolean isOutputMessage(Message message);

    /**
     * Called after the {@link javax.wsdl.Output} has been created, but it's added to the operation. Subclasses can
     * override this method to define the output name.
     *
     * <p>Default implementation sets the output name to the message name.
     *
     * @param definition the WSDL4J {@code Definition}
     * @param output     the WSDL4J {@code Output}
     */
    protected void populateOutput(Definition definition, Output output) {
        output.setName(output.getMessage().getQName().getLocalPart());
    }

    /**
     * Indicates whether the given name name should be included as {@link Fault} message in the definition.
     *
     * @param message the message
     * @return {@code true} if to be included as fault; {@code false} otherwise
     */
    protected abstract boolean isFaultMessage(Message message);

    /**
     * Called after the {@link javax.wsdl.Fault} has been created, but it's added to the operation. Subclasses can
     * override this method to define the fault name.
     *
     * <p>Default implementation sets the fault name to the message name.
     *
     * @param definition the WSDL4J {@code Definition}
     * @param fault      the WSDL4J {@code Fault}
     */
    protected void populateFault(Definition definition, Fault fault) {
        fault.setName(fault.getMessage().getQName().getLocalPart());
    }

    /**
     * Returns the {@link OperationType} for the given operation.
     *
     * <p>Default implementation returns {@link OperationType#REQUEST_RESPONSE} if both input and output are set; {@link
     * OperationType#ONE_WAY} if only input is set, or {@link OperationType#NOTIFICATION} if only output is set.
     *
     * @param operation the WSDL4J {@code Operation}
     * @return the operation type for the operation
     */
    protected OperationType getOperationType(Operation operation) {
        if (operation.getInput() != null && operation.getOutput() != null) {
            return OperationType.REQUEST_RESPONSE;
        }
        else if (operation.getInput() != null && operation.getOutput() == null) {
            return OperationType.ONE_WAY;
        }
        else if (operation.getInput() == null && operation.getOutput() != null) {
            return OperationType.NOTIFICATION;
        }
        else {
            return null;
        }
    }


}


TOP

Related Classes of org.springframework.ws.wsdl.wsdl11.provider.AbstractPortTypesProvider

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.