Long Running Workflows

Having a graphical notation to orchestrate services is nice, but you might wonder if it is worth it!

The real power of a workflow engine gets into action when the workflow is long running: in other words, when it reaches a pause status, because it has to wait an asynchronous event, a manual intervention or a timer.

In this section, you will evolve the workflow to add this long running nature!

From a use case perspective, the workflow has to wait for an asynchronous event which confirms the completion shipment procedures.

The following picture shows an high level design:

high level flow 2

Wait Event Sprint

For sake of simplicity, let’s imagine that an external integration service is able to homogenize events arriving from the internal department and those arriving from the external supplier.

The event will bring an important information the correlation key, in our use case is the orderId. As the name implies, the correlation key is important to address the correct workflow instance.

The correlation key MUST BE initialized at creation time with an incoming CloudEvent.

This implies two catching event nodes are required:

  • one at the beginning of the flow to start the workflow

  • another after the Prepare for Shipping and Forward to External Supplier to wait the shipping event

Open the workflow editor and follow these steps:

  1. Before the start declaration, add the following event declaration:

    events:
    - name: orderEvent
      kind: consumed
      type: OrderEventType
      source: Client
      correlation:
      - contextAttributeName: orderid
    - name: shippingEvent
      kind: consumed
      type: ShippingEventType
      source: Shipper
      correlation:
      - contextAttributeName: orderid

    The definition is quite self explanatory: the workflow can consume two different types of events: OrderEventType and ShippingEventType, from two different sources. One important aspect is the correlation definition which says that CloudEvent has to carry a special attribute in the header named orderid (CloudEvents attributes must be lowercase).

  2. Add an event state at the beginning of the workflow:

    1. Insert the following immediately after the declaration of states:

      - name: Order Received
        type: event
        onEvents:
          - eventRefs:
              - orderEvent
        transition: Check Inventory
    2. Update the start attribute to match the event state Order Received, this is the graphical outcome:

      04 order received
  3. Add another event state at the end of the workflow:

    1. Insert another event state declaration:

      • name: Order Shipped

      • eventRef: shippingEvent

        It’s worth noting that the order of the declarations is irrelevant, but to make the file easier to read, it’s a good practice to follow the same order in the textual representation as in the graphical one.
    2. Update the Prepare for Shipping and Forward to External Supplier states to transition in the Order Shipped event node; on both declarations:

      1. remove end: true

      2. add transition: Order Shipped

      The resulting diagram is:

    order shipped
  4. To complete the use case implementation, you have to add another operation that is responsible of notifying the customer about the shipping status with usual tracking information:

    • copy the previous operation state Prepare for Shipping

    • rename it to Notify Customer

    • update the message to match the operation name

    • remove the existing transition and add the attribute end: true

  5. Update the previous Order Shipped event state adding the transition to Notify Customer

You can compare your resulting workflow with the expected solution at this stage in solution/03-long-wait

Eventing

Events in Serverless Workflow comply to the CloudEvents standard which is under the Cloud Native Computing Foundation.

Moreover, Serverless Workflow, and in general OpenShift Serverless Logic, aims to fit seamlessly in Knative eventing.

For such a reason, Serverless Workflow exposes two channels to ingest CloudEvents:

  • HTTP POST

  • Kafka Broker

The developer runtime listens for CloudEvents on the root path of the standard HTTP channel (e.g. http:/localhost:8080/). In general, the Serverless Workflow runtime can expose multiple endpoints to ingest different types of events.

Test the events

To start the workflow and subsequently advance it, you have to inject two types of CloudEvent:

  • OrderEventType

  • ShippingEventType

Make sure that the workflow latest updates are deployed:

  1. Select Try on OpenShift button and then Upload "order" to Dev Mode.

    upload
  2. Wait for the confirmation message and select Go to Serverless Workflow Dev UI ↗ to open the Dev UI in a new browser tab.

    Dev UI

The Dev UI offers a convenient web form to inject CloudEvents in the Serverless Workflow runtime:

  1. Open the Workflow Definitions page.

  2. In workflow list, click on the event icon:

    cloud event button
  3. Fill in the form with the event information:

    • In Event Type type: OrderEventType

    • Click Add Header to the right of Event Custom Headers and type ce-orderid and 1000 for Header Name and Value

      custom event header
    • In Event Data copy the following json: {"orderId": "1000", "item": "1111"}

  4. Click Trigger

The CloudEvent is sent via the HTTP protocol, and by convention the CloudEvent attributes are sent as an HTTP header with a ce- prefix. The CloudEvent attribute orderid matches the contextAttributeName used in the event definition for correlation. In other words, orderid is the hook to the workflow instance, you can run multiple workflow instances in parallel, but they must wait for a distinct orderid.

Open the Workflow Details page to inspect the instance status: you should see the the workflow instance still active, waiting for the shipping event:

Shipping Event

To advance the workflow execution, you have to send another event:

  1. Open the Workflow Definitions page.

  2. In workflow list, click on the event icon:

    cloud event button
  3. Fill in the form with the following values:

    • Event Type : ShippingEventType

    • Event Custom Headers : ce-orderid = 1000

  4. Click Trigger

Use the Workflow Details page to check that the workflow instance is correctly completed.

CONGRATULATION!!! Your Serverless Workflow resumes when asynchronous events arrives!