Simple SOAP Service Simple SOAP Service
Home > Books > Tutorials and Training Guides > Web > Simple SOAP Service

Rate this page:
Really useful
Satisfactory
Not helpful
Confusing
Incorrect
Unsure
Extra comments:


HowTo Build a SOAP Service

This howto explains the steps to follow to build a SOAP web-service on NetKernel.

Source Code

A working implementation of this tutorial is provided for download here.

Specification

The service will be exposed at a URL endpoint of:

http:/localhost:8080/howto/soap/service

The service will have one port that echos back the SOAP message it recieves. The port will have the SOAP action URI:

urn:org:ten60:howto:soap:service:echo

Recipe

  1. Create a module and configure the mapping section like this:

    <mapping>
      <this>
        <match>ffcpl:/etc.*</match>
        <match>ffcpl:/resources/.*</match>
      </this>
      <import>
        <uri>urn:org:ten60:netkernel:mod:ws</uri>
      </import>
      <import>
        <uri>urn:org:ten60:netkernel:ext:dpml</uri>
      </import>
      <super />
    </mapping>

    The <this> section makes sure that resources in the /etc and /resources directories may be found in the module's address space. The imports hook-in the folloing libraries:

    urn:org:ten60:netkernel:mod:ws- provides the soap mapper and message processing tools
    urn:org:ten60:netkernel:ext:dpml- provides the DPML runtime for scripting the echo service.
  2. Import your module into the Front-End fulcrum so that it is exposed to the HTTP transport running on port 8080.
  3. Edit your module definition to export a public interface:

    <export>
      <match>service:(wsSOAPServer|wsdlGenerator|wsDocumentation).*?/howto/soap/service.*</match>
    </export>

    This rule will capture all pre-processed SOAP service requests coming from the HTTPBridge in the fulcrum module that match our service endpoint /howto/soap/service.

    The HTTPBridge will automatically treat any URL request with a path containing /soap/ as being a SOAP service request using the HTTP protocol binding. In this 'soap' mode the HTTPBridge automatically extracts the soap message from the body of the HTTP request together with the endpoint and action HTTP headers. The HTTPBridge collects all this information and presents it as a uniform active URI. For our service the Bridge will pre-process the HTTP request and issue an internal active URI request:

    service:SOAPServer
    +endpoint@http://localhost:8080/howto/soap/service
    +action@urn:org:ten60:howto:soap:service:echo
    +message@[somemessage]

    [Don't worry about the other service request types, wsdlGenerator and wsDocumentation - we'll show how these can be wired up to provide WSDL and human readable service descriptions below.]

  4. Now add this rewrite rule to your module definition.

    <rewrite>
      <rule>
        <match>service:((wsSOAPServer|wsdlGenerator|wsDocumentation).*)</match>
        <to>active:$1+config@ffcpl:/etc/SOAPServiceMap.xml</to>
      </rule>
    </rewrite>

    This rule will turn the service:wsSOAPServer request into an active:wsSOAPServer request and in the process attaches a config argument that tells active:wsSOAPServer where the SOAP service map is located. active:sSOAPServer is a specialist mapper service, its job is to route the external SOAPy requests to an internal active URI request for a service implementation.

  5. Create the SOAP service map at /etc/SOAPServiceMap.xml

    <SOAPServiceMap>
      <service>
        <name>HowTo SOAP Service</name>
        <endpoint>/howto/soap/service</endpoint>
      </service>
      <port>
        <name>Echo</name>
        <type>doc</type>
        <endpoint>/howto/soap/service</endpoint>
        <action>urn:org:ten60:howto:soap:service:echo</action>
        <int>active:dpml+operand@ffcpl:/resources/echo.idoc</int>
      </port>
    </SOAPServiceMap>

    Here the service tag contains a named endpoint matching our public endpoint URL. The port tag contains all the information needed by the wsSOAPServer to locate the internal service implementation of the echo service. So the port name is Echo, its type is 'doc' - meaning we expect document/literal soap messages (not RPC), the endpoint and action tags match our service endpoint and action specification. Finally the int tag provides the URI call to the internal implementation for the service - in this case it is a call to the DPML runtime to run /resources/echo.idoc. The wsSOAPServer will issue a request for the int URI with the addition of an argument called message containing the incoming SOAP message. Within the implementing script the message can be processed with the usual "this:param:message" syntax.

    The service map has one service tag per endpoint. Each endpoint may have any number of ports and each port must have a unique action.

  6. Implement the echo service at /resources/echo.idoc by copying and pasting this script:

    <idoc>
      <seq>
        <instr>
          <type>copy</type>
          <operand>this:param:message</operand>
          <target>this:response</target>
        </instr>
      </seq>
    </idoc>

    This script echos the message it receives back as its response. You could of course use any of NetKernels supported languages to implement this service. Just change the int service implementation URI appropriately.

  7. The service is finished. Now write a test soap client. Copy the following script into the workbench module at a location /howto/testClient.idoc

    <idoc>
      <seq>
        <instr>
          <type>wsSOAPClient</type>
          <endpoint>http://localhost:8080/howto/soap/service</endpoint>
          <action>urn:org:ten60:howto:soap:service:echo</action>
          <message>
            <soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/">
              <soapenv:Body>
                <p:message xmlns:p="urn:org:ten60:howto">Hello World</p:message>
              </soapenv:Body>
            </soapenv:Envelope>
          </message>
          <target>this:response</target>
        </instr>
      </seq>
    </idoc>

    This script uses the wsSOAPClient accessor to call the soap echo service. You can run this script at http://localhost:8080/workbench/howto/testClient.idoc

Useful patterns and tools

This howto shows how to connect an external SOAP service to an internal NetKernel implementation. The typical pattern for the internal implementation is to extract a message body part. Process it. Construct a new SOAP message. Set some body value and issue the message as the service's response.

The mod-ws module provides many utility tools for slicing and dicing SOAP messages without the pain of dealing with the XML directly. See for example wsSOAPGetBody for an 'extract by example' tool to get at a SOAP body part.

Documenting the Service

If you make a browser request for http://localhost:8080/howto/soap/service?DOC - you will see an auto-generated service description. When you call the endpoint with a ?DOC query parameter the HTTPBridge knows that this is a request for documentation and so it issues a request for service:wsDocumentation+[....]. Earlier we created a rewrite rule that mappted service:wsDocumentation to the active:wsDocumentation engine. With no arguments, this accessor simply styles the SOAP service map to a human readable HTML form.

You can serve human readable documentation for each port by adding a doc tag to the port specification. Like int, this should provide the internal URI request to a script or service that provides an HTML document describing the port. Note this *must* be an active URI since, just like the wsSOAPServer, the wsDocumentation server issues the internal request for the URI but adds the action and endpoint arguments.

If you have a strong stomach you can add a WSDL description by adding a 'wsdl' tag to the main service description tag. This should point to the internal URI of the WSDL document. A request for http://localhost:8080/howto/soap/service?WSDL will serve this WSDL service description for the endpoint. WSDL has the unique distinction amongst standards as being a service metadata format that is invariably neither human nor machine readable - well done. Call me old fashioned but I think that a well written human readable HTML engineering specification can't be beaten.

Manualy configuring the HTTPBridge to use SOAP mode

You might prefer to control the public URL endpoint interface more specifically than relying on the default /soap/ match we used above. Here's how:

  • On the public interface of your module export ffcpl:/etc/HTTPBridgeConfig.xml
  • In your module create a file /etc/HTTPBridgeConfig.xml with the following contents

    <HTTPBridgeConfig>
      <!--SOAP Service Zone-->
      <zone>
        <match>.*/mymodule/path1/path2/myservice.*</match>
        <soap />
      </zone>
    </HTTPBridgeConfig>

    Any URL request that matches this zone will be processed using the SOAP HTTP protocol binding rules - you'll have to take care of the endpoint matching and mapping in your module to work with the new soap service path.

© 2003-2007, 1060 Research Limited. 1060 registered trademark, NetKernel trademark of 1060 Research Limited.