Skip to main content

Oracle SOA Transaction Management and Understanding compensation

What is transaction management???

Two words with complex level of understanding, Many of us are still not able to understand what is transaction management and how it work, or how can some one control this transaction. Their are many implementation of transaction management based on technologies in which you are implementing it but TRANSACTION MANAGEMENT term remain same for all.

Today we will talk about transaction management in respect of oracle soa...

As per oracle documents, "transaction is a logical unit of work that contains one or more SQL statements. A transaction is an atomic unit. The effects of all the SQL statements in a transaction can be either all committed (applied to the database) or all rolled back (undone from the database)."


Points to keep in mind when designing transaction management:

  1. CommitCommitting a transaction means making permanent the changes performed by the SQL statements within the transaction., You need to analysis when you want to commit.
  1. RollbackRolling back means undoing any changes to data that have been performed by SQL statements within an uncommitted transaction. Oracle uses undo tablespaces (or rollback segments) to store old values. , You need to analysis when you want to rollback.
Types of transaction: Now when we design transactions, its required to analysis what type of transaction you need to design, Type of transaction always differ and it will always depends on what type of implementation you need for your system.
  1. Local: The transaction involved only the single database and all updates made to the database were committed at the end of these changes. This is referred to as a local transaction.
  2. Global: A global transaction involves a complicated set of management objects--objects that track all of the objects and databases involved in the transaction. These global transaction objects--TransactionManager and Transaction--track all objects and resources involved in the global transaction. At the end of the transaction, the TransactionManager and Transaction objects ensure that all database changes are atomically committed at the same time.
How to define transactions:Transactions manage changes to multiple databases within a single application as a unit of work. That is, if you have an application that manages data within one or more databases, you can ensure that all changes in all databases are committed at the same time if they are managed within a transaction.

Transactions are described in terms of ACID properties, which are as follows:
  • Atomic: all changes to the database made in a transaction are rolled back if any change fails.
  • Consistent: the effects of a transaction take the database from one consistent state to another consistent state.
  • Isolated: the intermediate steps in a transaction are not visible to other users of the database.
  • Durable: when a transaction is completed (committed or rolled back), its effects persist in the database.
Transaction Management in Oracle SOA:
  • Multiple DB-Transactions invoked by Oracle BPEL process, where all these DB invocations are related to XA Datasources .
  • Multiple DB and Non-DB Transactions invoked by Oracle BPEL Process, where all these transactions are related to heterogeneous applications.
How Oracle SOA BPEL handle transaction: Oracle BPEL Process Manager by default creates a new transaction on a request basis. That is, if a transaction exists, it is suspended, and a new transaction is created. Upon completion of the child (new) transaction, the master (suspended) transaction resumes.
However, if the request is asynchronous (that is, one-way), the transaction is either:
  • Inherited for insertion into the dehydration store (table dlv_message).
  • Enlisted transparently into the transaction (if one exists).
There is no message loss. Either the invocation message is inserted into the dehydration store for processing or the consumer is notified through a fault.

  1. If Service (bpel-process or composite) "A" invokes Service "B", and  Service B's transaction participant type = "NEVER" then,  Service "B" always runs in its own transaction and never comes under Service "A" transaction.
  2. If Service B Transaction participant type = "MANDATORY" then, Service B uses the Service A transaction scope, if Service A transaction is not available then, Service B creates its own transaction.
  3. In Service A, if "Idempotent" property at the ServiceB-partner link is set to false, then Service A and Service B runs in different transaction units.
  4. Even though Service A and Service B are in agreement to use same transaction, but Service B is not a part of XA Type transaction source then, BPEL-Engineer needs to write the code in Compensation handler to rollback the transactions based on business requirement and fault-handlers.

    How to configure transaction in oracle BPEL:

    <component name="InternalWarehouseService">
        <implementation.bpel src="InternalWarehouseService.bpel"/>
        <property name="bpel.config.transaction" 
             many="false" type="xs:string">required | requiresNew</property>
      </component>
    
    
    
    
    
    
    As earlier we was talking about GLOBAL & LOCAL TRANSACTION, Now their will be a case where 
    you need to communicate with two systems which are not in XA transaction enable or atomic transaction.
    Use Case:
    1.System "A" is a type of SOAP service but non-transactional.
    2.System "B" is a type of SOAP service but non-transactional.
    3.BPEL will orchestrate both A and B, Now when A is done with it's calling, B makes a fault but A already done with it's commit as 
       they are not in global transaction and creating there own local transaction.
    4.How System "A" transaction will rollback.
    
    
    
    
    For such cases, Oracle SOA provide an activity "compensation", Which is used to control/rollback transaction for such cases. 
    Such implementation are called "Heterogeneous transactions in a single transaction unit".
    • The only catch here is, Heterogeneous transactions may involve with flatfiles, XA-Based databases, non XA-Based databases, third party applications, webservices and many more.  
    • Now the point here is, how we can bring all these applications, transactions, databases bring together as a single transaction unit?  
    • What is the best way to rollback or commit these XA-based, non-XA Based transactions? 
    
    
    As mentioned in the previous section, bpel-pm commit or rollback only the transactions those are part of that specific "component-trx-context" and part of "XA based datasources".  
    That means, if any of the application datasource is not an XA Datasource then, it does not comes under the rollback or commit.
    
    
    In BPEL, "compensation handlers" provides a flexibility to rollback or reverse the transactions for business or/and system exceptions. 
    Compensation handler functionality is a kind of "catch" block in Java programming language. 
    Using this compensation handlers, pro-grammatically we can rollback the transactions.
    
    
    • Usually, bpel engineers, writes the programatic rollback code in "compensation handler" of bpel-scope to reverse the transactions executed of that specific scope.  
    • "Compensation handlers" are not triggered automatically  rather they need to explicitly invoked using "compensate" activity., which can only be invoked from within a fault-handler or another compensation handler.
    • When the "compensate" activity (which is invoked from fault-handler of that scope) is executed, it will only invoke the "compensation handlers" of that particular scope associated to that fault-handler. 
    • If "compensate" activity at process level is invoked then, it will only execute the compensation handlers for the top-level scopes. The compensation handlers will only be invoked for those scopes which have completed successfully and will be invoked in the reverse order of completion. That means, the most recently completed scope's compensation handler is invoked first, and then the next most recent and so on.
    • which will automatically calls the code (or activities) included in "compensation handler" associated to this scope.
    How to design compensation for your BPEL process:
    Compensations consist of 2 parts, the compensation handler, and the compensate activity.
    Defination: 
    1. Each scope can have one compensation handler
    2. A compensation handler defines an undo sequence for the scope it is bound to, for example "cancel flight"
    3. A compensation handler can be called directly (by specifying a scope) or indirectly by a compensate activity
    The use case: Consider a flow, where a user can pass a data, and the flow will search for / book a flight - and afterwards try to book a hotel room. Given the hotel is not available, the flight should be cancelled.
    The solution: As per above definitions the ReserveAirline scope will have a compensation handler attached, that contains the logic for canceling the previously booked flight.
    <scope name="ReserveAirline">
     <!-- define a compensation handler --> 
     <compensationHandler> <!-- call cancel ticked --> 
     <invoke name="Invoke_CancelTicket" partnerLink="AirlineBookingService" portType="airline:airlineBooking" operation="cancelSeat"/>
     </compensationHandler> 
     <sequence name="Sequence_1"> 
     <invoke name="Invoke_BookTicket" partnerLink="AirlineBookingService" portType="airline:airlineBooking" operation="reserveSeat"/> 
     </sequence> 
     </scope>
    The next step in this little sample happens within the scope, responsible for booking a hotel room ().
    The operation to reserve a hotel room (reserveHotel) throws a fault if no room is available for the given dates hotel:NoRoomAvailfault.
    <scope name="ReserveHotel"> 
     <faultHandlers> 
     <catch faultName="hotel:NoRoomAvailfault" >
     <throw name="RollBack" faultName="client:rollBackFault"/> 
     </catch> 
     </faultHandlers> 
     <sequence name="Sequence_2"> 
     <invoke name="Invoke_ReserveHotel" partnerLink="hotel" portType="ReservationSystem" operation="bookRoom"/> 
     </sequence> 
     </scope>
    In the above code a catch block is defined to catch the specific fault we expect and throws itself a rollBackFault to the parent scope, that will trigger the compensation, we have defined earlier.
    The last part is is to define a catch block on the parent scope, that triggers in case of a rollbackFault and fires the compensation defined on the ReserveAirline scope. 
      <faultHandlers> 
     <!-- catch the rollback fault and call --> 
     <catch faultName="client:rollbackFault"> 
     <compensate name="CompensateBoookings" scope="ReserveAirline"/> 
     </catch> 
     </faultHandlers>
    Having questions? send your feedback......





    Comments

    1. Hi Sandeep,

      I have a scenario :-

      Master Bpel has transaction set to requiresNew
      Child Bpel has transaction set to required.

      Master Bpel calls DB Adapter(non XA) to insert and then calls Child Bpel which also tries to insert the same record in same DBTable using same non XA DB Adapter. Child Bpel has a catchAll block and throws a RollBack Exception. Master Bpel has a CatchAll block to catches the child exception.

      Will it write in Database or not?

      I followed this blog
      (http://bhupisoa.blogspot.com/2015/01/soa-11g-bpel-transaction-handling-part-2.html)
      and our result does not match. For me its writing in the DB. For the blog author its not.

      Your blog explains my issue though. Please help.
      Thanks!

      ReplyDelete
      Replies
      1. Hi, it will write into dB in all cases and reason is quite simple which is usages of non-xa data source

        Delete

    Post a Comment

    Popular posts from this blog

    Solution for BPM standard dashboard & activity guide not working in Oracle SOA 12.2.1.0 C

    As earlier i publish a post about different issue of Oracle BPM, After some oracle support i got to fixed them. Issue Blog Here... Issue 3:BPM 12.2.1 process workspace activity guide not working. if you have a normal BPM Process, then this is the issue. In order to have data populated in "Activity Guide" firstly you should create a BPMN Guided Business Process. Below you can find the documentation about activity guide and how to create a Guided Business Process: https://docs.oracle.com/middleware/1221/bpm/bpm-develop/GUID-F765955D-90A5-48D4-8D2A-2F01FBB539E3.htm#BPMPD901 And here is specified: " A Guided Business Process is modeled as an activity guide that is based on a business process. The Activity Guide includes a set of Milestones. A milestone is a contained set of tasks that the end user has to complete. A milestone is complete when the user successfully runs a specific set of tasks in the milestone. " And in the below chapters you will find the ...

    Oracle SOA 12C rest adapter with Custom HTTP headers

    Most existing web applications are connected through web services, which are commonly known as SOAP services. More and more users are relying on mobile devices for communication, and they’re looking for lighter ways to access enterprise information on the go. REST services are the answer for the mobile device platform, because they get rapid responses and fast access to data. Oracle SOA Suite 12 c  provides a complete set of service infrastructure components for designing, deploying, and managing composite applications. Oracle SOA Suite 12 c  enables services to be created, managed, and orchestrated into composite applications and business processes. Some time we have need to send HTTP headers in REST service, In OSB we use header component and add what ever is needed but in oracle SOA 12C it's little bit different. Let see how we can do it. Create one SOA Application. Create one SOA Sample project inside SOA Application. Go to composite and drag drop REST ...

    Dynamic Routing in OSB 12C

    Dynamic Routing in OSB cab be used when the BusinessService endpoint required to be determine at runtime in message flow. Consider a scenario where OSB has to route the incoming requests to 2 different services based on the CustomerType element value sent in the payload. So create a XQuery resource with the following contents. Observe that we are using the absolute path of business service in configuration as required by dynamic routing. Following is the XML schema that we use:>>  Customer.xsd <?xml version="1.0" encoding="windows-1252" ?> <xsd:schema targetNamespace="http://xmlns.oracle.com/schema/Customer" xmlns:xsd="http://www.w3.org/2001/XMLSchema"             xmlns:ns1="http://xmlns.oracle.com/schema/Customer">   <xsd:complexType name="Customer">     <xsd:sequence>       <xsd:element name="CustomerId" type="xsd:string"/>       <xsd:elemen...