Introduction
In this guide, we’ll look at best practices and caveats for using Upstash Workflow.Core Principles
Execute business logic in context.run
Your workflow endpoint will be called multiple times during a workflow run. Therefore:
- Place your business logic code inside the
context.runfunction for each step - Code outside
context.runonly serves to connect steps
Return Results from context.run for Later Use
Always return step results if needed in subsequent steps.result will be unitialized when the endpoint is called again to run step-2.
If you are curious about why an endpoint is called multiple times, see how Workflow works.
Avoiding Common Pitfalls
Avoid Non-deterministic Code Outside context.run
A workflow endpoint should always produce the same results, even if it’s called multiple times. Avoid:
- Non-idempotent functions
- Time-dependent code
- Randomness
Failed to authenticate Workflow request. errors. This can happen if you return based on the result of the non-idempotent code before any workflow step.
To prevent this, ensure that the non-idempotent code (such as getResultFromDb in the example) runs within context.run.
Ensure Idempotency in context.run
Business logic should be idempotent due to potential retries in distributed systems. In other words, when a workflow runs twice with the same input, the end result should be the same as if the workflow only ran once.
In the example below, the someWork function must be idempotent:
someWork executes once and makes a change to a database. However, before the database had a chance to respond with the successful change, the connection is lost. Your Workflow cannot know if the database change was successful or not. The caller has no choice but to retry, which will cause someWork to run twice.
If someWork is not idempotent, this could lead to unintended consequences. For example duplicated records or corrupted data. Idempotency is crucial to maintaining the integrity and reliability of your workflow.
Don’t Nest Context Methods
Avoid callingcontext.call, context.sleep, context.sleepFor, or context.run within another context.run.
Include At Least One Step in Workflow
Every workflow must include at least one step execution withcontext.run. If no steps are defined, the workflow will throw a Failed to authenticate Workflow request. error.
Avoid Promise.any
In workflow-js, you can usePromise.all to run steps in parallel. However, a similar method, Promise.any, is not supported for workflow steps.
While Promise.all works seamlessly, Promise.any does not currently function with workflow steps. We are exploring the possibility of adding support for Promise.any in the future.
If you have a specific use case that requires Promise.any, don’t hesitate to reach out to Upstash support.