Thursday, August 31, 2006

Constructing a document literal-wrapped WSDL document

caveat: This document was originally written for an intranet wiki page to aid our team to construct WSDL for ActiveBPEL that was suitable for use by XFire Web Service clients and by .Net Web Service method calls. It was my first attempt at the problem and as such is probably wrong in several ways [one such example is the stuff I'm doing with namespaces, I now know that I should have two namespaces, one for my schemas and one for my definitions (WSDL documents), I didn't know /hadn't discovered this at the time, hence the uri: approach. (sorry!) ]

Anyway, hopefully this will help someone else going down this route :)

Steps to construct a document literal wrapped WSDL document:
  • Decide on your service name, and the parameters it takes .
    • Parameters seem to be defined in0,in1,in2 and out etc.
    • For this example we're going to define a service called sprinkleDust which takes a Pixie entity and returns a tired Pixie.
  • Construct your WSDL Document's basic structure
    • We need to bring in the standard namespaces/schema definitions:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema">
</definitions>
  • Next we want to define the PortType that defines our service method:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema">
<portType name="SprinkleDustServiceInterface">
<operation name="sprinkleDust">
</operation>
</portType>
</definitions>
  • Web methods aren't that useful without inputs and outputs, for this we need to define two WSDL message elements, and tell our operation to use them, note the naming scheme used here ServiceNameRequest and ServiceNameResponse [this is a convention but I don't think it actually matters]. Also note carefully where the prefix has been used.:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema">

<message name="sprinkleDustRequest">
</message>

<message name="sprinkleDustResponse">
</message>

<portType name="SprinkleDustServiceInterface">
<operation name="sprinkleDust">
<input message="smf:sprinkleDustRequest" name="sprinkleDustRequest" />
<output message="smf:sprinkleDustResponse" name="sprinkleDustResponse" />
</operation>
</portType>
</definitions>

  • Now in order for the WSDL document to be truly document-literal/wrapped, we need to define our WSDL message elements carefully, they must conform to the following requirements:
    • The input message has a single part
    • The part is an element (not a type/reference)
    • The element has the same name as the operation
    • The element's complex type has no attributes
  • Because this is a fairly complex set of requirements, first of all we'll flesh out the message elements, then we'll deal with the element it actually uses:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:smfprocess='urn:SprinkleDustServiceProcess'>

<message name="sprinkleDustRequest">
<part name="parameters" element="smfprocess:sprinkleDust" />
</message>

<message name="sprinkleDustResponse">
<part name="parameters" element="smfprocess:sprinkleDustResponse"/>
</message>

<portType name="SprinkleDustServiceInterface">
<operation name="sprinkleDust">
<input message="smf:sprinkleDustRequest" name="sprinkleDustRequest" />
<output message="smf:sprinkleDustResponse" name="sprinkleDustResponse" />
</operation>
</portType>
</definitions>
  • Looking at the above WSDL you should be able to see we satisfy the previously defined constraints:
    • The input message has a single part - in this case named parameters
    • The part is an element (not a type/reference) - the element referenced is smfprocess:sprinkleDust or smfprocess:sprinkleDustResponse
    • The element has the same name as the operation - smfprocess:sprinkleDust is the same name as the operation's name ( sprinkleDust )
    • The element's complex type has no attributes - Can't see this just yet, but it doesn't, honest.
  • Now we need to pull in the element's we're referring to that make up the document that is sent and received to/from the web service method:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:smfprocess='urn:SprinkleDustServiceProcess'>
<types>
<xs:schema targetNamespace="urn:SprinkleDustServiceProcess">
<xs:import schemaLocation="SAAll.xsd" namespace="http://yournamespace"/>
<xs:element name="sprinkleDust" >
<xs:complexType>
<xs:sequence>
<xs:element name="in0" type="smf:Pixie" nillable="true" minOccurs="1" maxOccurs="1"/>
</xs:sequence>
</xs:complexType>
</xs:element>

<xs:element name="sprinkleDustResponse">
<xs:complexType>
<xs:sequence>
<xs:element name="out" type="smf:Pixie" nillable="true" minOccurs="1" maxOccurs="1"/>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
</types>
<message name="sprinkleDustRequest">
<part name="parameters" element="smfprocess:sprinkleDust" />
</message>

<message name="sprinkleDustResponse">
<part name="parameters" element="smfprocess:sprinkleDustResponse"/>
</message>

<portType name="SprinkleDustServiceInterface">
<operation name="sprinkleDust">
<input message="smf:sprinkleDustRequest" name="sprinkleDustRequest" />
<output message="smf:sprinkleDustResponse" name="sprinkleDustResponse" />
</operation>
</portType>
</definitions>
  • This step was quite complex, we've added a types section, that defines our document-literal/wrapped elements, however it belongs to its own namespace. This is because in this WSDL file we're importing an external Schema that contains the definitions of our Pixie complex types and there are some special semantics around namespaces an imports + includes.
Last but not least, If we want our service to be used by other BPEL processes we need to add in a PartnerLink element, so our structure now looks like:

<?xml version="1.0" encoding="UTF-8"?>
<definitions name="SprinkleDust"
targetNamespace="http://yournamespace"
xmlns="http://schemas.xmlsoap.org/wsdl/"
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
xmlns:smf="http://yournamespace"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:smfprocess='urn:SprinkleDustServiceProcess'
xmlns:plnk="http://schemas.xmlsoap.org/ws/2003/05/partner-link/">
<types>
<xs:schema targetNamespace="urn:SprinkleDustServiceProcess">
<xs:import schemaLocation="SAAll.xsd" namespace="http://yournamespace"/>
<xs:element name="sprinkleDust" >
<xs:complexType>
<xs:sequence>
<xs:element name="in0" type="smf:Pixie" nillable="true" minOccurs="1" maxOccurs="1"/>
</xs:sequence>
</xs:complexType>
</xs:element>

<xs:element name="sprinkleDustResponse">
<xs:complexType>
<xs:sequence>
<xs:element name="out" type="smf:Pixie" nillable="true" minOccurs="1" maxOccurs="1"/>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
</types>
<message name="sprinkleDustRequest">
<part name="parameters" element="smfprocess:sprinkleDust" />
</message>

<message name="sprinkleDustResponse">
<part name="parameters" element="smfprocess:sprinkleDustResponse"/>
</message>

<portType name="SprinkleDustServiceInterface">
<operation name="sprinkleDust">
<input message="smf:sprinkleDustRequest" name="sprinkleDustRequest" />
<output message="smf:sprinkleDustResponse" name="sprinkleDustResponse" />
</operation>
</portType>

<plnk:partnerLinkType name="sprinkleDustPartnerLink">
<plnk:role name="User">
<plnk:portType name="smf:SprinkleDustServiceInterface" />
</plnk:role>
</plnk:partnerLinkType>
</definitions>