Skip to content

temporal-sa/temporal-order-management-demo

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

61 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Temporal Order Management Demo

Demos various aspects of Temporal using the Go SDK for backend and Python SDK for frontend.

Prerequisites __ Features __ Patterns
Network Connection __ Schedule __ Entity
Golang 1.18+ __ Local Activity __ Long-Running
Python 3.11 __ Timer __ Fanout
Poetry 1.3.2 __ Signal __ Continue As New
Node 20 __ Query __ Manual Intervention
__ Update __ Saga
__ Heartbeat __ Long-polling
__ Retry __
__ Data Converter __
__ Polyglot __
__ API Keys __

This demo walks through several scenarios using an order management process. The scenarios are:

  • HappyPath
  • AdvancedVisibility
  • HumanInLoopSignal
  • HumanInLoopUpdate
  • ChildWorkflow
  • NexusOperation
  • APIFailure
  • RecoverableFailure
  • NonRecoverableFailure

(Optional) Using Temporal CLI

Execute the following command:

$ temporal server start-dev --http-port 7243 --dynamic-config-value frontend.enableUpdateWorkflowExecution=true --dynamic-config-value system.forceSearchAttributesCacheRefreshOnRead=false --dynamic-config-value system.enableNexus=true --search-attribute OrderStatus="Keyword"

(Optional) Create Nexus Endpoint Temporal CLI

$ temporal operator nexus --address 127.0.0.1:7233 --namespace default endpoint create --name shipping-endpoint --target-namespace default --target-task-queue shipping

Run Worker

Ensure the following environment variables are set:

$ export TEMPORAL_NAMESPACE=<namespace>.<accountId>
$ export TEMPORAL_WORKER_METRICS_PORT=9090
$ export TEMPORAL_TASK_QUEUE=orders
$ export TEMPORAL_NEXUS_TASK_QUEUE=shipping
$ export NEXUS_SHIPPING_ENDPOINT=shipping-endpoint

If using mTLS authentication, set the following environment variables:

$ export TEMPORAL_ADDRESS=<namespace>.<accountId>.tmprl.cloud:7233
$ export TEMPORAL_CERT_PATH=/home/ktenzer/temporal/certs/ca.pem
$ export TEMPORAL_KEY_PATH=/home/ktenzer/temporal/certs/ca.key

If using API key authentication, set the following environment variables:

$ export TEMPORAL_ADDRESS=<region>.<cloudProvider>.api.temporal.io:7233
$ export TEMPORAL_APIKEY=eyABCD1234.eyABCD1234.ABCD-EFGH
$ cd go
$ go run worker/main.go

2024/02/20 14:05:20 prometheus metrics scope created
2024/02/20 14:05:20 INFO  No logger configured for temporal client. Created default one.
2024/02/20 14:05:21 INFO  Started Worker Namespace helloworld.sdvdw TaskQueue orders WorkerID 485217@fedora@

(Optional) Create Nexus Endpoint Cloud

This is only required for NexusOperation scenario.

$ tcld nexus endpoint create \
  --name shipping-endpoint \
  --target-task-queue shipping \
  --target-namespace helloworld.sdvdw\
  --allow-namespace helloworld.sdvdw \
  --description-file description.md

(Optional) Run Nexus Worker

This is only required for NexusOperation scenario.

$ cd go
$ go run nexus/worker/worker.go

time=2024-10-17T12:03:34.341-07:00 level=INFO msg="Started Worker" Namespace=helloworld.sdvdw TaskQueue=shipping [email protected]@

Run UI

Ensure the following environment variables:

$ export TEMPORAL_NAMESPACE=<namespace>.<accountId>
$ export TEMPORAL_TASK_QUEUE=orders

If using mTLS authentication, set the following environment variables:

$ export TEMPORAL_ADDRESS=<namespace>.<accountId>.tmprl.cloud:7233
$ export TEMPORAL_CERT_PATH=/home/ktenzer/temporal/certs/ca.pem
$ export TEMPORAL_KEY_PATH=/home/ktenzer/temporal/certs/ca.key

If using API key authentication, set the following environment variables:

$ export TEMPORAL_ADDRESS=<region>.<cloudProvider>.api.temporal.io:7233
$ export TEMPORAL_APIKEY=eyABCD1234.eyABCD1234.ABCD-EFGH

Install Poetry

Change to the UI directory

$ cd ui

Install Poetry Dependencies

$ poetry install --no-root

Run Flask App

$ poetry run python app.py

 * Serving Flask app 'app'
 * Debug mode: on
WARNING: This is a development server. Do not use it in a production deployment. Use a production WSGI server instead.
 * Running on http://127.0.0.1:5000
Press CTRL+C to quit
 * Restarting with stat
 * Debugger is active!
 * Debugger PIN: 685-645-739

Happy Path

Happy Path

Order Process

Order Confirmation

The happy path demonstrates the base functionality that is consistent across all other scenarios. In this demo you will place an order for various items that make a table. The workflow will generate a trackingId using a Temporal Side Effect.

The workflow will execute a side effect:

  • getTrackingId

Shows how to handle simple non-deterministic code that cannot fail.

The workflow will expose two queries:

  • getItems
  • progress

The getItems query shows the list of items that are part of order.

The progress query is used by the status bar in the UI to show progress. As workflow progress, progress is updated accordingly.

The workflow will execute one local activity:

  • GetItems The GetItems activity is used to get the items that need to be shipped as part of order in deterministic way. Lists in Go are not deterministic and as such should be properly sorted.

The workflow will execute several activities:

  • CheckFraud
  • PrepareShipment
  • ChargeCustomer
  • ShipOrder

The items just simulate reaching out to a service. They simply print that they are doing what they are supposed to do. The ChargeCustomer activity is used for failure scenarios. The ShipOrder activity will batch and execute parallel activities for every item in the order that needs to ship. The workflow will await until all are completed.

Advanced Visibility

Advanced Visibility

This scenario follows Happy Path and in addition shows search attributes to increase workflow visibility. When executing workflow a search attribute is upserted which keeps track of the state of the workflow. In order to use this scenario ensure the following:

  • A search attributed called OrderStatus with type Keyword is defined for your namespace Add Search Attribute
  • The OrderStatus custom search attribute field is added to your workflows UI panel Update UI Menu

Human in the Loop Signal

Human in the Loop SIgnal

This scenario follows Happy Path and in addition adds human in the loop. During the order processing you are asked to update the order with a new address. The address you provide is sent and updated in the workflow using a signal. The signal is fire and forget with no validation. You have 30 seconds to submit an updated address otherwise the workflow will timeout. The timeout is done using an asynchronous timer. This scenario also shows how to use Go coroutines with Temporal as both the timer and the signal are coroutines.

Signal

Human in the Loop Update

Human in the Loop Update

This scenario follows Happy Path and in addition adds human in the loop. During the order processing you are asked to update the order with a new address. The address you provide is sent and updated in the workflow using a update. The update allows for validation and if you send an address that doesn't start with a number validation will fail and an error is shown. You have 30 seconds to submit an updated address otherwise the workflow will timeout. The timeout is done using an asynchronous timer. This scenario also shows how to use Go coroutines with Temporal as both the timer and the update are coroutines.

Update Failed

Update Success

Child Workflow

Child Workflow

This scenario follows Happy Path but instead of shipping the items using parallel activities it does so using child workflows instead.

Shipping Workflows

NexusOperation

Nexus Operation

This scenario follows Happy Path but instead of shipping items using parallel activities or Child Workflows, it does so using Nexus Operations to trigger the Shipping Workflow.

Note: This scenario is currently only supported using Go.

Shipping Workflows

API Failure

API Failure

This scenario follows Happy Path, however after instead of executing ChargeCustomer this workflow will execute ChargeCustomerAPIFailure which will fail and on 5th attempt succeed. This scenario shows how Temporal workflows handle failures from activities.

Activity Attempts

Recoverable Failure

Recoverable Failure

This scenario follows Happy Path, however after ChargeCustomer activity executes a divide by zero bug is introduced. Simply comment-out the divide by zero code in OrderWorkflowREcoverableFailure.go and restart worker. Temporal will perform a replay, recover state of the workflow and proceed exactly where it left of as if nothing happened.

Before starting scenario uncomment following code in OrderWorkflowRecoverableFailure.go:

Divide(1, 0)

Non Recoverable Failure

Non Recoverable Failure

This scenario follows Happy Path, however after instead of executing ChargeCustomer this workflow will execute ChargeCustomerNonREcoverableFailure activity. This activity throws a non-retryable application error which causes the workflow to fail.

Workflow Failed

Live API Key Rotation

Live API Key Rotation

This scenario demonstrates updating the API key for the worker without the need to restart the worker. Steps;

  1. Invalidate the worker API key with Remove Worker API Key
    • This doesn't disable/revoke the key with the Temporal server, it just sets a blank key on the worker.
    • Alternatively, you could diable/revoke the key on the Temporal server.
  2. Start the HappyPath scenario. The workflow will fail to make progress.
  3. In a new browser tab/window, open the Order Management UI (http://127.0.0.1:5000) and update the worker API key.
    • The predefined value in the Update Key text field is the value of the TEMPORAL_APIKEY environment variable. If you used Remove Worker API Key earlier, then this key will still work fine.
    • Alternatively, if you have disabled/revoked the key on the Temporal server then you will need to make sure you provide a vaild key here.
  4. Return to the in-flight HappyPath scenario. The workflow will now progress/complete.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published