DB Adapter returning more than just results: status and messages

When working with the DB Adapter within the Oracle SOA Suite (or Oracle Service Bus) you may feel the need to return more than just the result of your call to the database. When calling you web service normally you get a normal SOAP response or a SOAP fault. But even when you get an normal SOAP response, it is possible that this is not the expected behavior and you need to know if, for instance, you supplied enough input arguments in your web service request.

In that case you need more than just the results, you need status (OK/NOT OK) and one or more messages indicating what went wrong if the result was not ok.

In this tutorial I will show you an example of how how can implement this within the Oracle SOA Suite (and because the DB Adapter can also be used within the Oracle Service Bus, the principles also apply to implementing it within the OSB).

First of all we will use object types within the database for communicating with the SOA Suite. Using object types is quite easy in PL/SQL and is natively supported by the DB Adapter (converting it to XML automatically).

Prerequisite

the tutorial is done using the HR demo schema and I assume that the DB Adapter Connection Pools and JDBC Data Source within the Weblogic Server are already in place for connecting to the HR database schema in your database.

The service we will create will provide a list of employees within a selected department.

PL/SQL

We start by creating the object types for this service:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
create or replace type hrm_emp_obj_t as object
( empno number(6,0)
, name  varchar2(50)
)
/
 
create or replace type hrm_emp_tab_t as table of hrm_emp_obj_t
/
 
create or replace type hrm_dept_emp_obj_t as object
( deptno     number(4,0)
, name       varchar2(30)
, location   varchar2(30)
, employees  hrm_emp_tab_t
)
/
 
create or replace type hrm_dept_emp_tab_t as table of hrm_dept_emp_obj_t
/

Because we like to return some status information, we create some more object types which will hold the information we need:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
create or replace type hrm_service_msg_obj_t as object
 (code        varchar2(15)
 ,description varchar2(4000)
 ,type        varchar2(1)
 ,constructor function hrm_service_msg_obj_t
  return self as result
 )
/
 
create or replace type body hrm_service_msg_obj_t
is
 
  constructor hrm_service_msg_obj_t
    return self is result
  is
  begin
    null;
  end;
end;
/
 
create or replace type hrm_service_messages_tab_t as table of hrm_service_msg_obj_t
/

Next we create the package with the function which will be called by the DB Adapter (HRM_EMP_PCK):

Package Specification:

Package Body:

And the supporting package HRM_SERVICE_MSG_PCK:

Package Specification:

Package Body:

It may seem at lot at first, but the service object types and hrm_services_msg_pck package will be reused for every new service function you write in PL/SQL.

Let look at the hrm_emp_pck.get_employee_list function.

You’ll see two argument: the first is the department id to look for and the second is the result of the search (if all goes well the list of employees within the selected department). And as return type: hrm_service_status_obj_t. This object type will hold the additional information you want, being the status (OK or NOT OK) and any messages describing a functional error or technical database error (the latter being ORA-messages).

This status object will, when called using the DB Adapter (and after transformation to the right format), results in a XML structure like this:

1
2
3
4
5
6
7
8
9
10
11
<status>
  <result>NOT OK</result>
  <programUnit>hrm_emp_pck.get_employee_list</programUnit>
  <messages>
    <message>
      <code>SRV-00001</code>
      <description>One or more mandatory arguments missing</description>
      <type>F</type>
    </message>
  </messages>
</status>

The hrm_service_msg_pck package now contains only some basic functions, but you can extend it with, for instance, support for returning Headstart messages (if you use this in the application the services are for) or for more default error handling. Within this tutorial the service objects and package are created within the application schema, but if you plan to use it for real, then I suggest you implement this within a dedicated schema for handling (common) service functionality.

SOA Suite

To finish the tutorial and make it actually work, we start JDeveloper 11g and create a new SOA Application and project.

JDeveloper_new_SOA_Application

Enter a name for your application, e.g. HRServices

Enter a name for your project, e.g. GetEmployeeList

Select as Composite Template: Empty Composite.

JDeveloper_empty_composite

Press Finish.

Before we continue, we first need to add (create) some files to the project: the WSDL file which defines the service we want to create and the accompanying XML Schema (XSD) files.

The creating of the WSDL and XML Schema files (mostly done in the reverse order) is something you will do for every new service you built. You start with either the declaration of the objecttypes and PL/SQL package functions or with the XML Schema and WSDL. By the way, you do not need to have actual PL/SQL code when creating the DB Adapter (you only need the correct objecttypes and function specification). But for making it work you need the code, of course.

Add the WSDL file to the project root folder and the XML Schema (XSD) files to the xsd subfolder.

The WSDL (web service description) (employeelistbydepartment.wsdl) for our service:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?xml version= '1.0' encoding= 'UTF-8' ?>
<wsdl:definitions
     name="employeelistbydepartment"
     targetNamespace="www.amis.nl/services/hrm"
     xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/"
     xmlns:hrm="www.amis.nl/services/hrm"
     xmlns:tns="www.amis.nl/services/hrm"
    >
    <wsdl:types>
      <schema xmlns="http://www.w3.org/2001/XMLSchema" >
        <import namespace="www.amis.nl/services/hrm" schemaLocation="xsd/employeelistbydepartment.xsd" />
      </schema>
    </wsdl:types>
    <wsdl:message name="getEmployeeListByDepartmentRequest">
        <wsdl:part name="request" element="hrm:employeeListRequest"/>
    </wsdl:message>
    <wsdl:message name="getEmployeeListByDepartmentResponse">
        <wsdl:part name="reply" element="hrm:employeeListResponse"/>
    </wsdl:message>
    <wsdl:portType name="employeelistbydepartment_ptt">
        <wsdl:operation name="getEmployeeListByDepartment">
            <wsdl:input message="tns:getEmployeeListByDepartmentRequest"/>
            <wsdl:output message="tns:getEmployeeListByDepartmentResponse"/>
        </wsdl:operation>
    </wsdl:portType>
</wsdl:definitions>

And the XML Schema files:

XML Schema for service request and response types (employeelistbydepartment.xsd)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<?xml version= '1.0' encoding= 'UTF-8' ?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
            xmlns="www.amis.nl/services/hrm"
            targetNamespace="www.amis.nl/services/hrm"
            elementFormDefault="qualified" xmlns:hrm="www.amis.nl/hrm">
    <xsd:import namespace="www.amis.nl/hrm" schemaLocation="hrm.xsd"/>
    <xsd:element name="employeeListRequest">
        <xsd:complexType>
            <xsd:sequence>
                <xsd:element name="departmentIdentifier" type="xsd:int"/>
            </xsd:sequence>
        </xsd:complexType>
    </xsd:element>
    <xsd:element name="employeeListResponse">
        <xsd:complexType>
            <xsd:sequence>
                <xsd:element name="status" type="hrm:statusType" minOccurs="1" maxOccurs="1"/>
                <xsd:element name="departments" type="hrm:departmentsType" minOccurs="0" maxOccurs="1"/>
            </xsd:sequence>
        </xsd:complexType>
    </xsd:element>
</xsd:schema>

XML Schema for the result (list of employees within a department) (hrm.xsd)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
<?xml version= '1.0' encoding= 'UTF-8' ?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
            xmlns="www.amis.nl/services/hrm"
            targetNamespace="www.amis.nl/hrm"
            elementFormDefault="qualified"
            xmlns:hrm="www.amis.nl/hrm">
    <xsd:complexType name="departmentsType">
        <xsd:sequence minOccurs="1" maxOccurs="unbounded">
            <xsd:element name="department" type="hrm:departmentType"/>
        </xsd:sequence>
    </xsd:complexType>
    <xsd:complexType name="departmentType">
        <xsd:sequence>
            <xsd:element name="id" type="xsd:integer"/>
            <xsd:element name="name" type="xsd:string"/>
            <xsd:element name="location" type="xsd:string"/>
            <xsd:element name="employees">
                <xsd:complexType>
                    <xsd:sequence>
                        <xsd:element name="employee" type="hrm:employeeType"
                                     minOccurs="0" maxOccurs="unbounded"/>
                    </xsd:sequence>
                </xsd:complexType>
            </xsd:element>
        </xsd:sequence>
    </xsd:complexType>
    <xsd:complexType name="employeeType">
        <xsd:sequence>
            <xsd:element name="personelNo" type="xsd:integer"/>
            <xsd:element name="name" type="xsd:string"/>
        </xsd:sequence>
    </xsd:complexType>
</xsd:schema>

XML Schema for the service message (service.xsd)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
<?xml version= '1.0' encoding= 'UTF-8' ?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema"
            xmlns="www.amis.nl/services/hrm" targetNamespace="www.amis.nl/hrm"
            elementFormDefault="qualified" xmlns:hrm="www.amis.nl/hrm">
    <xsd:complexType name="statusType">
        <xsd:sequence>
            <xsd:element name="result" type="xsd:string" minOccurs="1" maxOccurs="1"/>
            <xsd:element name="programUnit" type="xsd:string" minOccurs="1" maxOccurs="1"/>
            <xsd:element name="messages" type="tns:messagesType" minOccurs="1" maxOccurs="1"/>
        </xsd:sequence>
    </xsd:complexType>
    <xsd:complexType name="messagesType">
        <xsd:sequence minOccurs="1" maxOccurs="unbounded">
            <xsd:element name="message" type="tns:messageType"/>
        </xsd:sequence>
    </xsd:complexType>
    <xsd:complexType name="messageType">
        <xsd:sequence>
            <xsd:element name="code" type="xsd:string"/>
            <xsd:element name="description" type="xsd:string"/>
            <xsd:element name="type" type="xsd:string"/>
        </xsd:sequence>
    </xsd:complexType>
</xsd:schema>

Now drag and drop a DB Adapter service to the External References lanes inside the composite.xml.

Enter a name for the DB Adapter Service, e.g. hrmGetEmployeeList

In step 3 select or add a database connection to the HR Schema. This database connection is only used at design time and provide the means to select the appropriate database package function. Enter at JNDI Name the name of the JNDI Data Source as recorded in the Weblogic server, e.g. eis/DB/HR

DBAdapter_step3

Press Next.

Because we want to call a database package function, we select as operation type: Call a Stored Procedure or Function.

Press Next.

In step 5 press the Browse button to select the HRM_EMP_PCK.GET_EMPLOYEE_LIST function.

Press OK.

DBAdapter_function

Press Next.

Press Next.

Press Finish.

You will see that 3 new files are added to the project: a WSDL file and two XML Schema files.

Next add a Mediator component by drag and drop it on the Components lane.

Enter a name for the Mediator, e.g. GetEmployeeListMediator

Select as Template: Interface Definition from WSDL.

And select the WSDL file we created earlier: employeelistbydepartment.wsdl

Check the check box ‘Create Composite Service with Bindings’. This will ensure that the service is exposed to the outside world by the Mediator.

Mediator

Press OK.

Connect the Mediator with the DB Adapter:

composite

Double click on the Mediator component and click on the transformation icon in the first routing rule (to).

Select Create New Mapper File, accept the default name and press OK.

Create the mapping and save the file.

Mediator_transform_1

Return to the Mediator editor and click on the transformation icon in the second routing rule (from).

Select Create New Mapper File, accept the default name and press OK.

Create the mapping and save the file.

Mediator_transform_2

Save all files.

Now you are ready (assuming you have configured Weblogic).

Deploy and test

Deploy the service to the application server and test if it works.

Call the web service with:

  • No department id
  • An existing department id (e.g. 10 or 100)
  • A not existing department id (e.g. 77)

You can use the SOA console for testing or soapui.

Note that the first test can not be performed within the SOA console (only in soapUI) because it marks the request parameter as mandatory

The results in the SOA console:

Department id = 100:

response_deptid100

Department id = 77:

response_deptid77

With no department id (in soapUI):

soapui_no_deptid

I hope this tutorial has given you some insight how you can return status and messages back from the database to the web service. And if needed you can add functionality in your web service when a certain status and/or messages are returned from the DB Adapter, e.g. for raising a predefined SOAP Fault or call other services depending on the result. For this you need to use BPEL or, when in the Oracle Service Bus, within a proxy service.

One Response

  1. Lucas Jellema January 4, 2011