Problem statement
Your enterprise might have invested a lot in exposing software assets as SOAP Web Services to integrate with internal and external applications. If a new "modern" consumer who can (or wants to) consume only REST API, would you rewrite the entire thing to support REST clients?WSO2 ESB (for example, version 4.8.1 which I have used) supports RESTful integration using API. We can leverage this feature to expose SOAP web services through REST API without changes to SOAP services. So, same time you would be able to support existing SOAP clients and new REST clients.
There are many blogs / web sites available on detailed explanation on REST style. You can follow this https://www.ics.uci.edu/~fielding/pubs/dissertation/top.htm from Roy Fielding. Another blog on REST can be found here in http://rest.elkstein.org/2008/02/what-is-rest.html
Steps involved
The following steps are required to achieve what we said.- Define REST API
- Transform request REST data into SOAP data
- Call the back end application / URL
- Transform response SOAP data into REST data (for example JSON)
- Respond with JSON response
Sample use case - Student Exam Result Service
Usecase overview
We will implement a simple use case - student exam results application. The organisation has the SOAP service already running in the premise that will take student id as input parameter and responds with marks for four subjects.
Sample request SOAP message:
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:stud="http://www.example.org/student/">
<soapenv:Header/>
<soapenv:Body>
<stud:getMarks>
<student>John Smith</student>
</stud:getMarks>
</soapenv:Body>
</soapenv:Envelope>
Sample response message:
<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/" xmlns:stud="http://www.example.org/student/">
<soapenv:Header/>
<soapenv:Body>
<stud:getMarksResponse>
<marks>
<Mathematics>95</Mathematics>
<Science>97</Science>
<Lang-I>92</Lang-I>
<Lang-II>94</Lang-II>
</marks>
</stud:getMarksResponse>
</soapenv:Body>
</soapenv:Envelope>
WSDL:
<wsdl:definitions name="Student" targetNamespace="http://www.example.org/student/" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:tns="http://www.example.org/student/" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<wsdl:types>
<xsd:schema targetNamespace="http://www.example.org/student/">
<xsd:element name="getMarks">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="student" type="xsd:string"/>
</xsd:sequence>
</xsd:complexType>
</xsd:element>
<xsd:element name="getMarksResponse">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="marks" type="tns:Marks"/>
</xsd:sequence>
</xsd:complexType>
</xsd:element>
<xsd:complexType name="Marks">
<xsd:sequence>
<xsd:element name="Mathematics" type="xsd:int"/>
<xsd:element name="Science" type="xsd:int"/>
<xsd:element name="Lang-I" type="xsd:int"/>
<xsd:element name="Lang-II" type="xsd:int"/>
</xsd:sequence>
</xsd:complexType>
</xsd:schema>
</wsdl:types>
<wsdl:message name="getMarksRequest">
<wsdl:part element="tns:getMarks" name="parameters"/>
</wsdl:message>
<wsdl:message name="getMarksResponse">
<wsdl:part element="tns:getMarksResponse" name="parameters"/>
</wsdl:message>
<wsdl:portType name="Student">
<wsdl:operation name="getMarks">
<wsdl:input message="tns:getMarksRequest"/>
<wsdl:output message="tns:getMarksResponse"/>
</wsdl:operation>
</wsdl:portType>
<wsdl:binding name="StudentSOAP" type="tns:Student">
<soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"/>
<wsdl:operation name="getMarks">
<soap:operation soapAction="http://www.example.org/student/getMarks"/>
<wsdl:input>
<soap:body use="literal"/>
</wsdl:input>
<wsdl:output>
<soap:body use="literal"/>
</wsdl:output>
</wsdl:operation>
</wsdl:binding>
<wsdl:service name="Student">
<wsdl:port binding="tns:StudentSOAP" name="StudentSOAP">
<soap:address location="http://www.example.org/"/>
</wsdl:port>
</wsdl:service>
</wsdl:definitions>
Now, lets do some hands on in WSO2 ESB.
1. Define REST API
In WSO2 ESB Management Console, goto Main --> Service Bus --> API and create a new API. The steps are very clearly mentioned in the WSO2 documentation site https://docs.wso2.com/display/ESB480/Getting+Started+with+REST+APIs how to create API.The API definition will look like this.
<api xmlns="http://ws.apache.org/ns/synapse"
name="StudentAPI"
context="/exam-results">
<resource methods="GET" uri-template="/student/{student}" >
<inSequence> </inSequence>
<outSequence> </outSequence>
</resources>
</api>
This will provide a REST GET url like this http://localhost:8280/exam-results/student/1002 where 1002 is the student id for which we need the marks scored in four subjects. The URL may directly be accessed from a browser for instance.
2. Transform request REST data into SOAP data
Use Payload Mediator to transform REST data into SOAP data<payloadFactory media-type="xml">
<format>
<stud:getMarks xmlns:stud="http://www.example.org/student/">
<student>$1</student>
</stud:getMarks>
</format>
<args>
<arg evaluator="xml" expression="get-property('uri.var.student')"/>
</args>
</payloadFactory>
3. Call the back end application / URL
Set Action property and configure the SOAP end point URL<header name="Action" value="http://www.example.org/student/getMarks"/>
<send>
<endpoint>
<address uri="http://localhost:8088/services/Student" format="soap11"/>
</endpoint>
</send>
4. Transform response SOAP data into REST data (for example JSON)
Now, you will get the response in SOAP format. Use Payload Mediator to transform into REST data i.e., JSON<payloadFactory media-type="json">
<format>{"Language_1": "$1", "Language_2": "$2", "Maths": "$3", "Science": "$4"}</format>
<args>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Lang-I"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Lang-II"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Mathematics"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Science"/>
</args>
</payloadFactory>
5. Respond with JSON response
Change the response type to JSON and send the response to REST client
<property name="messageType" value="application/json" scope="axis2"/>
<send/>
When you execute, the output will look like
{
"Language_1": "92",
"Language_2": "94",
"Maths": "95",
"Science": "97"
}
See below the complete API definition
<?xml version="1.0" encoding="UTF-8"?>
<api xmlns="http://ws.apache.org/ns/synapse"
name="StudentAPI"
context="/exam-results">
<resource methods="GET" uri-template="/student/{student}">
<inSequence>
<log level="custom">
<property name="student_name_##########"
expression="get-property('uri.var.student')"/>
</log>
<payloadFactory media-type="xml">
<format>
<stud:getMarks xmlns:stud="http://www.example.org/student/">
<student>$1</student>
</stud:getMarks>
</format>
<args>
<arg evaluator="xml" expression="get-property('uri.var.student')"/>
</args>
</payloadFactory>
<header name="Action" value="http://www.example.org/student/getMarks"/>
<property name="Accept-Encoding" scope="transport" action="remove"/>
<log level="custom">
<property name="#body" expression="$body/*"/>
</log>
<send>
<endpoint>
<address uri="http://localhost:8088/services/Student" format="soap11"/>
</endpoint>
</send>
</inSequence>
<outSequence>
<log level="custom">
<property name="#response-body-from-Backend" expression="$body/*"/>
</log>
<payloadFactory media-type="json">
<format>{"Language_1": "$1", "Language_2": "$2", "Maths": "$3", "Science": "$4"}</format>
<args>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Lang-I"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Lang-II"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Mathematics"/>
<arg xmlns:stud="http://www.example.org/student/"
evaluator="xml"
expression="$body/stud:getMarksResponse/marks/Science"/>
</args>
</payloadFactory>
<property name="messageType" value="application/json" scope="axis2"/>
<log level="custom">
<property name="#body" expression="$body/*"/>
</log>
<send/>
</outSequence>
</resource>
</api>
Hope this is useful to you. Please let me know if any further help.
No comments:
Post a Comment