This is article is the second part of a five article series on a working example of SOA at work. In the last article I introduced the challenge, coming up with a fairly robust WSDL which could be implemented in multiple different programming languages consistently.
The service we are creating has one operation. The input for this sample operation takes an xml message with one element as well as an array of a nested object. The output returns an array of the same input objects (I had to redact the more meaningful input and output messages as they were work related). It initially seems complicated because I'm trying to capture all the complexities of xml structures and how the map to data structures that programmers in modern programming languages often take for granted. Ok, enough introduction, let's get onto the WSDL.
WSDL
Click the expand source link and voila! That's the WSDL. Now to run you through the xml elements that constitute this WSDL.- <?xml version="1.0" encoding="utf-8"?>
- <wsdl:definitions xmlns="http://schemas.xmlsoap.org/wsdl/" xmlns:tns="http://kimenye.com/soap/" xmlns:impl="http://kimenye.com/soap/" xmlns:xsd1="http://kimenye.com/soap/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:soapenc="http://schemas.xmlsoap.org/soap/encoding/" name="SampleService" targetNamespace="http://kimenye.com/soap/">
- <wsdl:types>
- <xsd:schema elementFormDefault="qualified" targetNamespace="http://kimenye.com/soap/">
- <!-- Exception message, contains a string with the field message -->
- <xsd:complexType name="ServiceException">
- <xsd:sequence>
- <xsd:element name="message" nillable="true" type="xsd:string"/>
- </xsd:sequence>
- </xsd:complexType>
- <!-- A sample field with two elements, FieldName and FieldValue -->
- <xsd:complexType name="SampleField">
- <xsd:sequence>
- <xsd:element minOccurs="1" maxOccurs="1" name="FieldName" type="xsd:string"/>
- <xsd:element minOccurs="1" maxOccurs="1" name="FieldValue" type="xsd:string"/>
- </xsd:sequence>
- </xsd:complexType>
- <!-- An array of sample fields -->
- <xsd:complexType name="ArrayOfSampleField">
- <xsd:sequence>
- <xsd:element minOccurs="0" maxOccurs="unbounded" name="SampleField" nillable="true" type="tns:SampleField"/>
- </xsd:sequence>
- </xsd:complexType>
- <xsd:complexType name="AnObject">
- <xsd:sequence>
- <xsd:element minOccurs="0" maxOccurs="1" name="DemoField" type="xsd:string" nillable="true"/>
- <xsd:element minOccurs="0" maxOccurs="1" name="SampleFields" type="tns:ArrayOfSampleField"/>
- </xsd:sequence>
- </xsd:complexType>
- <xsd:complexType name="ArrayOfReturnObjects">
- <xsd:sequence>
- <xsd:element minOccurs="0" maxOccurs="unbounded" name="AnObject" nillable="true" type="tns:AnObject"/>
- </xsd:sequence>
- </xsd:complexType>
- <xsd:element name="ServiceException" type="tns:ServiceException"/>
- <xsd:element name="in" type="tns:AnObject" />
- <xsd:element name="out" type="tns:ArrayOfReturnObjects" />
- </xsd:schema>
- </wsdl:types>
- <message name="ServiceException">
- <part name="ServiceException" element="tns:ServiceException"/>
- </message>
- <message name="InputRequest">
- <part name="in" element="tns:in"/>
- </message>
- <message name="OutputResponse">
- <part name="out" element="tns:out"/>
- </message>
- <portType name="SampleServicePortType">
- <operation name="SampleOperation">
- <input message="tns:InputRequest"/>
- <output message="tns:OutputRequest"/>
- <fault name="ServiceException" message="tns:ServiceException"/>
- </operation>
- </portType>
- <binding name="SampleServiceBinding" type="tns:SampleServicePortType">
- <soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"/>
- <operation name="SampleOperation">
- <soap:operation soapAction="urn://SampleService#SampleOperation"/>
- <input>
- <soap:body use="literal"/>
- </input>
- <output>
- <soap:body use="literal"/>
- </output>
- <fault name="ServiceException">
- <soap:fault name="ServiceException" use="literal"/>
- </fault>
- </operation>
- </binding>
- <service name="SampleService">
- <port name="SampleServicePort" binding="tns:SampleServiceBinding">
- <soap:address location="http://127.0.0.1:8000/index.php"/>
- </port>
- </service>
- </wsdl:definitions>
Exceptions
Exceptions in SOAP are xml messages that can be defined in the schema. For this example, the exception is aptly named ServiceException, and it has an element named "message" which happens to be a string.
- <!-- Exception message, contains a string with the field message -->
- <xsd:complexType name="ServiceException">
- <xsd:sequence>
- <xsd:element name="message" nillable="true" type="xsd:string"/>
- </xsd:sequence>
- </xsd:complexType>
Arrays & Collections
There are several ways to represent arrays in XML. I've chosen for this sample, the simplest way which is to use the xml attribute "maxOccurs" and set it to "unbounded".
- <!-- A sample field with two elements, FieldName and FieldValue -->
- <xsd:complexType name="SampleField">
- <xsd:sequence>
- <xsd:element minOccurs="1" maxOccurs="1" name="FieldName" type="xsd:string"/>
- <xsd:element minOccurs="1" maxOccurs="1" name="FieldValue" type="xsd:string"/>
- </xsd:sequence>
- </xsd:complexType>
- <!-- An array of sample fields -->
- <xsd:complexType name="ArrayOfSampleField">
- <xsd:sequence>
- <xsd:element minOccurs="0" maxOccurs="unbounded" name="SampleField" nillable="true" type="tns:SampleField"/>
- </xsd:sequence>
- </xsd:complexType>
Declarative SOAP Exceptions
For this sample, soap exceptions are declared in the WSDL, this is in order to detect multiple exceptions. In this sample, we only have ServiceException, but could have included others, to programmatically detect which exception was thrown. This declaration is achieved by using the "fault" element. Note that the exception needs to be declared in both the binding and port type.
- <binding name="SampleServiceBinding" type="tns:SampleServicePortType">
- <soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"/>
- <operation name="SampleOperation">
- <soap:operation soapAction="urn://SampleService#SampleOperation"/>
- <input>
- <soap:body use="literal"/>
- </input>
- <output>
- <soap:body use="literal"/>
- </output>
- <fault name="ServiceException">
- <soap:fault name="ServiceException" use="literal"/>
- </fault>
- </operation>
- </binding>
That's it for the most part. In the next article, I'll be looking at the PHP implementation of this WSDL.