Asynchronous ProcessingAsynchronous Processing
Creating a simple asynchronous process
Home > Books > Tutorials and Training Guides > Asynchronous Processing

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


Introduction

NetKernel has a fully asynchronous architecture which allows many concurrent processes to execute simultaneously. In general it is only necessary to write synchronous processes which defer to the kernel to optimize and manage scheduling and thread allocation. However occasionally, such as with high-latency requests to remote systems or requests for which your current process doesn't require a response, it can be valuable to explicitly indicate to the kernel that a service should be invoked asynchronously (ie in parallel with a parent process).

Service Specification

We will create a service which asynchronously calls a time-delayed echo service. The calling process will record the time at which the asynchronous request is issued. Whilst the asynchronous request is being processed the parent process will create a result document, it will then join with the asynchronous process to retrieve its result. Finally the duration of the sub-process and it's echoed response will be added to the result document.

We will show two solutions to this specification. The first is a Beanshell scripted solution using the NetKernel Foundation API. The second is a DPML process using the async and join accessors.

The delayed-echo service (shown below) is implemented in DPML and simply returns the parameter argument it receives after a delay of 3-seconds.

You can try the NKF solution here. The DPML solution can be tried here.

NKF Solution

Below is a solution implemented in beanshell and using the NKF API - the Java comments describe in detail each part of the process.


import org.ten60.netkernel.xml.xda.*;
import org.ten60.netkernel.xml.representation.*;
import org.ten60.netkernel.xml.util.*;
import org.ten60.netkernel.layer1.representation.*;
import org.ten60.netkernel.layer1.nkf.*;
import org.ten60.netkernel.layer1.nkf.impl.*;
import java.io.*;

void main()
    {   //Create a simple message XML doc
        message=new StringAspect("<message>Are your receiving? Over.</message>");
                
        //Create a request to invoke the delayed-echo service
        req=context.createSubRequest();
        req.setURI("active:dpml");
        req.addArgument("operand", "delayed-echo.idoc");
        req.addArgument("param", message);
        req.setAspectClass(IXAspect.class);

        //Get current time
        t1=System.currentTimeMillis();
        
        //Issue asynchronous request - non-blocking returns immediately
        handle=context.issueAsyncSubRequest(req);
        
        //In the mean time prepare the partially completed response document
        sr=new StringReader("<results><issued><start>"+ t1+
         "</start>"+message.getString()+ "</issued><returned></returned></results>");
        dxda=new DOMXDA( XMLUtils.getInstance().parse(sr), false );
        
        //Join with the asynch process
        echoresult=handle.join();

        //Get current time
        t2=System.currentTimeMillis();

        //Finally process the results document
        dxda.appendPath("/results/returned", "duration", ""+(t2-t1) );
        erxa=(IXAspect)echoresult.getAspect(IXAspect.class);
        dxda.append(erxa.getXDA(), "/", "/results/returned");

        //Create response
        dxaa=new DOMXDAAspect(dxda);
        resp=context.createResponseFrom(dxaa);
        resp.setMimeType("text/xml");
        resp.setExpired();

        //Issue response
        context.setResponse(resp);
    }

DPML Solution

Below is a solution implemented in DPML using the async and join accessors - the comments describe in detail each part of the process.

<idoc>
  <seq>
    <comment> ************ Prepare a message ************ </comment>
    <instr>
      <type>copy</type>
      <operand>
        <message>Are your receiving? Over.</message>
      </operand>
      <target>var:message</target>
    </instr>
    <comment> ************ Start the stopwatch ************ </comment>
    <instr>
      <type>org.ten60.ura.util.stopWatch</type>
      <operator>
        <stopWatch>
          <start />
        </stopWatch>
      </operator>
      <target>var:start</target>
    </instr>
    <comment> ************ Issue asynchronous request to execute DPML script "delayed-echo.idoc" with the message as parameter. This will take 3 seconds to complete. ************ </comment>
    <instr>
      <type>async</type>
      <uri>active:dpml</uri>
      <operand>delayed-echo.idoc</operand>
      <param>var:message</param>
      <target>var:syncHandle</target>
    </instr>
    <comment> ************ In the mean time do some work preparing response ************ </comment>
    <instr>
      <type>copy</type>
      <operand>
        <results>
          <issued>
            <start />
            <message1 />
          </issued>
          <returned>
            <stop />
            <message2 />
          </returned>
        </results>
      </operand>
      <target>this:response</target>
    </instr>
    <instr>
      <type>copy</type>
      <operand>var:message</operand>
      <target>this:response#xpointer(/results/issued/message1)</target>
    </instr>
    <instr>
      <type>copy</type>
      <operand>var:start</operand>
      <target>this:response#xpointer(/results/issued/start)</target>
    </instr>
    <comment> ************ Join with the async request putting it's response in a variable ************ </comment>
    <instr>
      <type>join</type>
      <operand>var:syncHandle</operand>
      <target>var:echoedmessage</target>
    </instr>
    <comment> ************ Stop the stopwatch ************ </comment>
    <instr>
      <type>org.ten60.ura.util.stopWatch</type>
      <operator>
        <stopWatch>
          <stop />
        </stopWatch>
      </operator>
      <operand>var:start</operand>
      <target>var:stop</target>
    </instr>
    <comment> ************ Finally fill out the response with the duration and echoed message ************ </comment>
    <instr>
      <type>copy</type>
      <operand>var:echoedmessage</operand>
      <target>this:response#xpointer(/results/returned/message2)</target>
    </instr>
    <instr>
      <type>copy</type>
      <operand>var:stop</operand>
      <target>this:response#xpointer(/results/returned/stop)</target>
    </instr>
    <comment> ************ Force to be expired so the demo runs every time ************ </comment>
    <instr>
      <type>expire</type>
      <operand>this:response</operand>
      <target>this:response</target>
    </instr>
  </seq>
</idoc>

Delayed Echo Service

Below is a simple delayed echo service implemented in DPML.

<idoc>
  <seq>
    <comment> ************ Delayed Echo: After 3 seconds it sends back the parameter it receives. ************ </comment>
    <instr>
      <type>copy</type>
      <operand>this:param:param</operand>
      <target>this:response</target>
    </instr>
    <instr>
      <type>sleep</type>
      <operator>
        <time>3000</time>
      </operator>
    </instr>
  </seq>
</idoc>

Deployment

You can experiment further with asynchronous processes by copying these examples to a new module created with the wizard (remember to choose DPML and scripting support).

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