Skip to main content

Handling Failures

Orkes Conductor automatically handles transient workflow and task failures without the need to write custom code. Various failure-handling configurations can be set ahead of time, which will take effect during execution.

For tasks, you can configure the following resilience parameters in its task definition:

  • Retries
  • Timeouts
  • Rate limits

For workflows, you can configure the following resilience parameters in its workflow definition:

  • Compensation flows (known as failure workflow in Conductor)
note

To deal with workflow failures post-execution, refer to Debugging Workflows.

Message delivery guarantees

Conductor guarantees at least once message delivery, meaning all messages are persistent and will be delivered to task workers one or more times. In the event of failure, the message will be delivered more than once. This semantic ensures that:

  1. If a workflow has started, it will run to completion as long as all its tasks are completed.
  2. If a task worker fails due to restarts, crashes, or other issues, the message will be redelivered to another worker node that is alive and responding.

Task retries

Automatic retries are a key strategy for handling transient task failures. If a task fails to complete, the Conductor server will make the task available for polling again after a given duration.

Retry configuration

You can configure retry behavior for tasks in its task definition. The parameters for defining a task’s retry behavior are:

  • Retry count
  • Retry logic
  • Retry delay seconds
  • Backoff scale factor
ParameterDescriptionRequired/ Optional
retryCountThe number of retry attempts if the task fails. Default value is 3.Optional.
retryLogicThe policy that determines the retry mechanism for the tasks. Supported values:
  • FIXED—Retries after a fixed interval defined by retryDelaySeconds.
  • LINEAR_BACKOFF—Retries occur with a delay that increases linearly based on retryDelaySeconds x backoffScaleFactor x attemptNumber.
  • EXPONENTIAL_BACKOFF—Retries occur with a delay that increases exponentially based on retryDelaySeconds x (backoffScaleFactor ^ attemptNumber).
Optional.
retryDelaySecondsThe time (in seconds) to wait before each retry attempt. This provides time for the task service to recover from any transient failure before it is retried. Default value is 60.

Note: The actual duration depends on the retry policy set in retryLogic.
Optional.
backoffScaleFactorThe value multiplied with retryDelaySecondsto determine the interval for retry. Default value is 1.Optional.

Example

// task definition
{
"name": "someTaskDefName",
...
"retryCount": 3,
"retryLogic": "FIXED|EXPONENTIAL_BACKOFF|LINEAR_BACKOFF",
"retryDelaySeconds": 1,
"backoffScaleFactor": 1
}

Example retry behavior

Diagram showing how the Conductor server and worker interact in the event of a retry.

Based on the retry configuration in the above figure, the following sequence of events will occur in the event of a retry:

  1. Worker (W1) polls the Conductor server for task T1 and receives the task.
  2. After processing the task, the worker determines that the task execution is a failure and reports to the server with a FAILED status after 10 seconds.
  3. The server will persist this failed execution of T1.
  4. A new task T1 execution is created and scheduled for polling. Based on the retry configuration, the task will be available for polling after 5 seconds

Task timeouts

A task timeout can occur if:

  • There are no workers available for a given task type. This could be due to longer-than-expected system downtime or a system misconfiguration.
  • The worker receives the message but dies before completely processing the task, so the task never reaches completion.
  • The worker has completed the task but could not communicate with the Conductor server due to network failures, the server being down, or other issues.

Timeout configuration

You can configure timeout behavior for tasks in its task definition to handle the various abovementioned cases. The parameters for a task’s timeout behavior are:

  • Poll timeout seconds
  • Response timeout seconds
  • Timeout seconds
  • Timeout policy
ParameterDescriptionRequired/Optional
pollTimeoutSecondsThe maximum duration in seconds that a worker has to poll a task before it gets marked as TIMED_OUT. When configured with a value > 0, Conductor will wait for the task to be picked up by a worker.

Useful for detecting a backlogged task queue with insufficient workers.

Default value is 3600.
Optional.
responseTimeoutSecondsThe maximum duration in seconds that a worker has to respond to the server with a status update before it gets marked as TIMED_OUT. When configured with a value > 0, Conductor will wait for the worker to return a status update, starting from when the task was picked up.

If a task requires more time to complete, the worker can respond with the IN_PROGRESS status.

Default value is 600.
Optional.
timeoutSecondsThe maximum duration in seconds for the task to reach a terminal state before it gets marked as TIMED_OUT. When configured with a value > 0, Conductor will wait for the task to complete, starting from when the task was picked up.

Useful for governing the overall SLA for completion.

Default value is 3600.
Required.
timeoutPolicyThe policy for handling timeout. Supported values:
  • RETRY—Retries the task based on the retry configuration.
  • TIME_OUT_WF—The task is marked as TIMED_OUT and is terminated, which also sets the workflow status as TIMED_OUT.
  • ALERT_ONLY—An alert message is logged when the timeout occurs.
Note: The ALERT_ONLY option should be used only when you have your own metrics monitoring system to send alerts.
Optional.
note

To configure tasks that never timeout, set timeOutSeconds and pollTimeoutSeconds to 0.

Example

// task definition
{
"name": "someTaskDefName",
...
"retryCount": 3,
"retryLogic": "FIXED|EXPONENTIAL_BACKOFF|LINEAR_BACKOFF",
"retryDelaySeconds": 1,
"backoffScaleFactor": 1
}

Example timeout behavior

Poll timeout
In the figure below, task T1 isn’t polled by the worker within 60 seconds, so Conductor marks it as `TIMED_OUT`.

Diagram showing how the Conductor server and worker interact in the event of a poll timeout.

Response timeout

Diagram showing how the Conductor server and worker interact in the event of a response timeout.

Based on the timeout configuration in the above figure, the following sequence of events will occur in the event of a delayed worker response:

  1. At 0 seconds, the worker polls the Conductor server for task T1 and receives it. T1 is marked as IN_PROGRESS by the server.
  2. The worker starts processing the task, but the worker instance dies during the execution.
  3. At 20 seconds (T1’s responseTimeoutSeconds), the server marks T1 as TIMED_OUT since the worker has not updated the task within the configured duration.
  4. A new instance of task T1 is scheduled based on the retry configuration.
  5. At 25 seconds, the retried instance of T1 is available for polling after the retryDelaySeconds (5) has elapsed.
Poll timeout

Diagram showing how the Conductor server and worker interact in the event of a timeout.

Based on the timeout configuration in the above figure, the following sequence of events will occur when a task cannot be completed within the given duration:

  1. At 0 seconds, a worker polls the Conductor server for task T1 and receives the task. T1 is marked as IN_PROGRESS by the server.
  2. The worker starts processing the task but is unable to complete it within the response timeout. The worker updates the server with T1 set to an IN_PROGRESS status and a callback of 9 seconds.
  3. The server puts T1 back in the queue but makes it invisible and the worker continues to poll for the task but does not receive T1 for 9 seconds.
  4. After 9 seconds, the worker receives T1 from the server but is still unable to finish processing the task. As such, the worker updates the server again with a callback of 9 seconds.
  5. The same cycle repeats for the next few seconds.
  6. At 30 seconds (T1 timeout), the server marks T1 as TIMED_OUT because it is not in a terminal state after first being moved to IN_PROGRESS status. The server schedules a new task based on the retry count.
  7. At 32 seconds, the worker finishes processing T1 and updates the server with COMPLETED status. The server ignores this update since T1 has already been moved to a terminal status (TIMED_OUT).

Task rate limits

A task’s rate limit controls the number of task executions in a given period and serves as a critical strategy for managing task load and worker capacity.

When the number of tasks scheduled within a given duration exceeds the defined rate limit, the Conductor server will place the additional tasks in a PENDING status. Once an IN_PROGRESS task is completed, the rate limit is freed up and the server will make the next PENDING task available for polling.

Rate limit configuration

You can configure rate limit behavior for tasks in its task definition. The parameters for defining a task’s rate limit behavior are:

  • Rate limit
  • Rate limit frequency
  • Concurrent executions
ParameterDescriptionRequired/ Optional
rateLimitPerFrequencyThe maximum number of task executions that can be scheduled in a given duration. Default value is 0.Optional.
rateLimitFrequencyInSecondsThe frequency window (in seconds) for the rate limit.Optional.
concurrentExecLimitThe number of task executions that can be executed concurrently. Default value is 0.Optional.
note

To configure tasks with no rate limits, set rateLimitPerFrequency and concurrentExecLimit to 0.

Example

// task definition
{
"name": "someTaskDefName",
"pollTimeoutSeconds": 3600,
"responseTimeoutSeconds": 600,
"timeoutSeconds": 3600,
"timeoutPolicy": "TIME_OUT_WF",
}

Workflow compensation flows

A compensation flow can be configured to take place when a workflow execution fails (FAILED status). Known as a failure workflow in Conductor, this failure workflow must be created in Conductor and added to the main workflow definition.

When triggered, the failure workflow receives the failed workflow details, such as its workflow ID and tasks, as input. This enables you to implement compensating logic to handle the failure.

Setting a failure workflow

You can set a failure workflow for a workflow in its workflow definition. Before setting the failure workflow, ensure that you have created it first.

To set a failure workflow:

  1. Go to Definitions > Workflow.
  2. Select the workflow that you want to add a failure workflow to.
  3. In the Workflow tab on the right, scroll down to Failure workflow name and select the failure workflow from the dropdown box.
  4. Select Save > Confirm save.

Configuring failure workflow in UI.

Example

// workflow definition
{
...
"failureWorkflow": "<name of the workflow that will run upon failure>"
}

Workflow rate limits

A workflow’s rate limit controls the number of concurrent executions that can be active. Beyond this limit, workflows will be queued for execution based on their start time.

When the number of scheduled workflows exceeds the defined rate limit, the Conductor server will place these workflows in a RUNNING state with the first task set to a PENDING status. Once a workflow completes, the rate limit is freed up and the server will schedule the next PENDING task for polling.

Rate limit configuration

You can configure the limit on concurrent workflow executions in its workflow definition.

ParameterDescriptionRequired/ Optional
rateLimitConfigA map of the workflow rate limit configuration.Optional.
rateLimitConfig. rateLimitKeyA unique identifier to group workflow executions for rate limits.

Can be a fixed value (for example, “max”) or a dynamic variable from the workflow input (for example, ${workflow.input.correlationId}).
Optional.
rateLimitConfig. concurrentExecLimitThe number of workflow executions that can run concurrently for each rate limit key. Cannot be passed as a dynamic variable.Optional.

Routing rate limits with dynamic key

Using a dynamic rateLimitKey, you can apply separate rate limit queues based on workflow inputs like correlationId or version. The rate limit for each group of workflows will be the same, based on the concurrentExecLimit.

For example, if workflow executions are grouped according to their correlation ID with concurrentExecLimit set to 100, workflows triggered with correlation IDs 1 and 2 will have their rate limit queues capped at 100 each.

Example

// Workflow definition
{
...
"rateLimitConfig": {
"rateLimitKey": "${workflow.input.correlationId}",
"concurrentExecLimit": 100
}
}

Client SDK methods

Using a fixed rateLimitKey

        RateLimitConfig rateLimitConfig = new RateLimitConfig();
rateLimitConfig.setRateLimitKey("http");
rateLimitConfig.setConcurrentExecLimit(3);
workflowDef.setRateLimitConfig(rateLimitConfig);

Using a dynamic rateLimitKey

In this case, a separate rate limit is applied for each correlation ID.

        RateLimitConfig rateLimitConfig = new RateLimitConfig();
rateLimitConfig.setRateLimitKey("${workflow.correlationId}");
rateLimitConfig.setConcurrentExecLimit(3);
workflowDef.setRateLimitConfig(rateLimitConfig);