org.jboss.soa.esb.actions.soap
Class SOAPClient

java.lang.Object
  extended by org.jboss.soa.esb.actions.AbstractActionLifecycle
      extended by org.jboss.soa.esb.actions.AbstractActionPipelineProcessor
          extended by org.jboss.soa.esb.actions.soap.SOAPClient
All Implemented Interfaces:
ActionLifecycle, ActionPipelineProcessor

@Publish(value=WsdlContractPublisher.class)
public class SOAPClient
extends AbstractActionPipelineProcessor

SOAP Client action processor.

Uses the soapUI Client Service to construct and populate a message for the target service. This action then routes that message to that service.

Endpoint Operation Specification

Specifying the endpoint operation is a straightforward task. Simply specify the "wsdl" and "SOAPAction" properties on the SOAPClient action as follows:
        <action name="soapui-client-action" class="org.jboss.soa.esb.actions.soap.SOAPClient">
            <property name="wsdl" value="http://localhost:18080/acme/services/OrderManagement?wsdl"/>
            <property name="SOAPAction" value="http://www.acme.com/OrderManagement/SendSalesOrderNotification"/>
        </action>
The SOAP operation is derived from the SOAPAction.

SOAP Request Message Construction

The SOAP operation parameters are supplied in one of 2 ways:
  1. As a Map instance set on the default body location (Message.getBody().add(Map))
  2. As a Map instance set on in a named body location (Message.getBody().add(String, Map)), where the name of that body location is specified as the value of the "paramsLocation" action property.
The parameter Map itself can also be populated in one of 2 ways:
  1. Option 1: With a set of Objects that are accessed (for SOAP message parameters) using the OGNL
  2. framework. More on the use of OGNL below.
  3. Option 2: With a set of String based key-value pairs(<String, Object>), where the key is an OGNL expression identifying the SOAP parameter to be populated with the key's value. More on the use of OGNL below.
As stated above, OGNL is the mechanism we use for selecting the SOAP parameter values to be injected into the SOAP message from the supplied parameter Map. The OGNL expression for a specific parameter within the SOAP message depends on that the position of that parameter within the SOAP body. In the following message:
        <soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/"
                             xmlns:cus="http://schemas.acme.com">
           <soapenv:Header/>
           <soapenv:Body>
                <cus:customerOrder>
                        <cus:header>
                                <cus:customerNumber>123456</cus:customerNumber>
                        </cus:header>
                </cus:customerOrder>
           </soapenv:Body>
        </soapenv:Envelope>
 
The OGNL expression representing the customerNumber parameter is "customerOrder.header.customerNumber".

Once the OGNL expression has been calculated for a parameter, this class will check the supplied parameter map for an Object keyed off the full OGNL expression (Option 1 above). If no such parameter Object is present on the map, this class will then attempt to load the parameter by supplying the map and OGNL expression instances to the OGNL toolkit (Option 2 above). If this doesn't yield a value, this parameter location within the SOAP message will remain blank.

Taking the sample message above and using the "Option 1" approach to populating the "customerNumber" requires an object instance (e.g. an "Order" object instance) to be set on the parameters map under the key "customerOrder". The "customerOrder" object instance needs to contain a "header" property (e.g. a "Header" object instance). The object instance behind the "header" property (e.g. a "Header" object instance) should have a "customerNumber" property.

OGNL expressions associated with Collections are constructed in a slightly different way. This is easiest explained through an example:

 <soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/"
               xmlns:cus="http://schemas.active-endpoints.com/sample/customerorder/2006/04/CustomerOrder.xsd"
               xmlns:stan="http://schemas.active-endpoints.com/sample/standardtypes/2006/04/StandardTypes.xsd">
        <soapenv:Header/>
        <soapenv:Body>
                <cus:customerOrder>
                        <cus:items>
                                <cus:item>
                                        <cus:partNumber>FLT16100</cus:partNumber>
                                        <cus:description>Flat 16 feet 100 count</cus:description>
                                        <cus:quantity>50</cus:quantity>
                                        <cus:price>490.00</cus:price>
                                        <cus:extensionAmount>24500.00</cus:extensionAmount>
                                </cus:item>
                                <cus:item>
                                        <cus:partNumber>RND08065</cus:partNumber>
                                        <cus:description>Round 8 feet 65 count</cus:description>
                                        <cus:quantity>9</cus:quantity>
                                        <cus:price>178.00</cus:price>
                                        <cus:extensionAmount>7852.00</cus:extensionAmount>
                                </cus:item>
                        </cus:items>
                </cus:customerOrder>
        </soapenv:Body>
 </soapenv:Envelope>
 
The above order message contains a collection of order "items". Each entry in the collection is represented by an "item" element. The OGNL expressions for the order item "partNumber" is constructed as "customerOrder.items[0].partnumber" and "customerOrder.items[1].partnumber". As you can see from this, the collection entry element (the "item" element) makes no explicit appearance in the OGNL expression. It is represented implicitly by the indexing notation. In terms of an Object Graph (Option 1 above), this could be represented by an Order object instance (keyed on the map as "customerOrder") containing an "items" list (List or array), with the list entries being "OrderItem" instances, which in turn contains "partNumber" etc properties.

Option 2 (above) provides a quick-and-dirty way to populate a SOAP message without having to create an Object model ala Option 1. The OGNL expressions that correspond with the SOAP operation parameters are exactly the same as for Option 1, except that there's not Object Graph Navigation involved. The OGNL expression is simply used as the key into the Map, with the corresponding key-value being the parameter.

SOAP Response Message Consumption

The SOAP response object instance can be is attached to the ESB Message instance in one of the following ways:
  1. On the default body location (Message.getBody().add(Map))
  2. On in a named body location (Message.getBody().add(String, Map)), where the name of that body location is specified as the value of the "responseLocation" action property.
The response object instance can also be populated (from the SOAP response) in one of 3 ways:
  1. Option 1: As an Object Graph created and populated by the XStream toolkit.
  2. Option 2: As a set of String based key-value pairs(<String, String>), where the key is an OGNL expression identifying the SOAP response element and the value is a String representing the value from the SOAP message. This option should only be used for simple responses. If your response contains collections etc, this option is not viable. You need to use option 1.
  3. Option 3: If Options 1 or 2 are not specified in the action configuration, the raw SOAP response message (String) is attached to the message.
Using XStream as a mechanism for populating an Object Graph (Option 1 above) is straightforward and works well, as long as the XML and Java object models are in line with each other.

The XStream approach (Option 1) is configured on the action as follows:

  <action name="soapui-client-action" class="org.jboss.soa.esb.actions.soap.SOAPClient">
      <property name="wsdl" value="http://localhost:18080/acme/services/OrderManagement?wsdl"/>
      <property name="SOAPAction" value="http://www.acme.com/OrderManagement/GetOrder"/>
      <property name="paramsLocation" value="get-order-params" />
      <property name="responseLocation" value="get-order-response" />
      <property name="responseXStreamConfig">
          <alias name="customerOrder" class="com.acme.order.Order" namespace="http://schemas.acme.com/services/CustomerOrder.xsd" />
          <alias name="orderheader" class="com.acme.order.Header" namespace="http://schemas.acme.com/services/CustomerOrder.xsd" />
          <alias name="item" class="com.acme.order.OrderItem" namespace="http://schemas.acme.com/services/CustomerOrder.xsd" />
      </property>
  </action>
 
In the above example, we also include an example of how to specify non-default named locations for the request parameters Map and response object instance.

We also provide, in addition to the above XStream configuration options, the ability to specify field name mappings and XStream annotated classes.

      <property name="responseXStreamConfig">
          <fieldAlias name="header" class="com.acme.order.Order" fieldName="headerFieldName" />
          <annotation class="com.acme.order.Order" />
      </property>
 
Field mappings can be used to map XML elements onto Java fields on those occasions when the local name of the element does not correspond to the field name in the Java class.

To have the SOAP response data extracted into an OGNL keyed map (Option 2 above) and attached to the ESB Message, simply replace the "responseXStreamConfig" property with the "responseAsOgnlMap" property having a value of "true" as follows:

  <action name="soapui-client-action" class="org.jboss.soa.esb.actions.soap.SOAPClient">
      <property name="wsdl" value="http://localhost:18080/acme/services/OrderManagement?wsdl"/>
      <property name="SOAPAction" value="http://www.acme.com/OrderManagement/GetOrder"/>
      <property name="paramsLocation" value="get-order-params" />
      <property name="responseLocation" value="get-order-response" />
      <property name="responseAsOgnlMap" value="true" />
  </action>
 
To return the raw SOAP message as a String (Option 3), simply omit both the "responseXStreamConfig" and "responseAsOgnlMap" properties.

Transforming the SOAP Request Message

It's often necessary to be able to transform the SOAP request before sending it. This may be to simply add some SOAP headers.

Transformation of the SOAP request (before sending) is supported by configuring the SOAPClient action with a Smooks transformation configuration property as follows:

     <property name="smooksTransform" value="/transforms/order-transform.xml" />
The value of the "smooksTransform" property is resolved by first checking it as a filesystem based resource. Failing that, it's checked as a classpath resource and failing that, as a URI based resource.

Specifying a different SOAP schema

    <property name="SOAPNS" value="http://www.w3.org/2009/09/soap-envelope"/>
 
This is an optional property and can be used to specify the SOAP schema that should be used by OGNL.

Author:
tom.fennelly@jboss.com

Nested Class Summary
protected static class SOAPClient.Response
           
 
Field Summary
 
Fields inherited from interface org.jboss.soa.esb.actions.ActionPipelineProcessor
PROCESS_EXCEPTION_METHOD, PROCESS_METHOD, PROCESS_SUCCESS_METHOD
 
Constructor Summary
SOAPClient(ConfigTree config)
           
 
Method Summary
 void destroy()
          Destroy the action instance.
protected  java.lang.String getEndpointOperation()
           
 java.lang.String getSoapNS()
           
 SoapUIInvoker getSoapUIInvoker()
           
 void initialise()
          Initialise the action instance.
 Message process(Message message)
          Perform the action processing on the specified message.
protected  void processResponse(Message message, SOAPClient.Response response)
          Process the SOAP response into the ESB Message instance.
 void setSoapUIInvoker(SoapUIInvoker soapUIInvoker)
           
 
Methods inherited from class org.jboss.soa.esb.actions.AbstractActionPipelineProcessor
processException, processSuccess
 
Methods inherited from class java.lang.Object
clone, equals, finalize, getClass, hashCode, notify, notifyAll, toString, wait, wait, wait
 

Constructor Detail

SOAPClient

public SOAPClient(ConfigTree config)
           throws ConfigurationException
Throws:
ConfigurationException
Method Detail

initialise

public void initialise()
                throws ActionLifecycleException
Description copied from class: AbstractActionLifecycle
Initialise the action instance.

This method is called after the action instance has been instantiated so that configuration options can be validated.

Specified by:
initialise in interface ActionLifecycle
Overrides:
initialise in class AbstractActionLifecycle
Throws:
ActionLifecycleException - for errors during initialisation.

getSoapUIInvoker

public SoapUIInvoker getSoapUIInvoker()

setSoapUIInvoker

public void setSoapUIInvoker(SoapUIInvoker soapUIInvoker)

destroy

public void destroy()
             throws ActionLifecycleException
Description copied from class: AbstractActionLifecycle
Destroy the action instance.

This method is called prior to the release of the action instance. All resources associated with this action instance should be released as the instance will no longer be used.

Specified by:
destroy in interface ActionLifecycle
Overrides:
destroy in class AbstractActionLifecycle
Throws:
ActionLifecycleException

process

public Message process(Message message)
                throws ActionProcessingException
Description copied from interface: ActionPipelineProcessor
Perform the action processing on the specified message. This method is invoked for each message passing throught the pipeline.

Parameters:
message - The current message being processed.
Returns:
The message to be passed to the next stage of the pipeline.
Throws:
ActionProcessingException - for errors during processing.

getSoapNS

public java.lang.String getSoapNS()

getEndpointOperation

protected java.lang.String getEndpointOperation()

processResponse

protected void processResponse(Message message,
                               SOAPClient.Response response)
                        throws ActionProcessingException
Process the SOAP response into the ESB Message instance.

Parameters:
message - The ESB Message.
response - The SOAP response.
Throws:
ActionProcessingException - Error processing the response.