Guides

Invoking workflows

There are multiple ways to execute workflow:

  • Use a trigger.
  • Use the REST API.
  • Use C# code.

Use a trigger

When a workflow is configured to start with a trigger, that workflow will start whenever the trigger is activated. How the trigger is activated depends on the trigger type.

For example, a workflow that starts with an HTTP endpoint trigger can be started by calling the path configured in the trigger.

Http endpoint configuration

The trigger ain't triggering

Triggers become active only when their Can start workflow is enabled.

Can start workflow

Without this, the trigger will not start the workflow (but it will still be used as a blocking activity).

This workflow can be triggered with the following request:

curl --location --request POST 'https://localhost:5001/workflows/test' \
--header 'Content-Type: application/json'

Other examples of triggers include:

  • Timer triggers, which start a workflow at a specific time.
  • Event triggers, which start a workflow when an event is received.
  • Message triggers, which start a workflow when a message is received.

Use the REST API

The Elsa.Workflows.Api package provides two endpoints for executing a workflow:

  • /workflow-definitions/{workflow_definition_id}/execute: Executes a workflow definition synchronously.
  • /workflow-definitions/{workflow_definition_id}/dispatch: Executes a workflow definition asynchronously.

Synchronous execution

Synchronously in this context doesn't mean that the workflow will be executed in the same thread as the caller. It still uses task-based asynchronous programming. The difference is that the caller will wait for the workflow to finish before getting a response.

To execute a workflow definition synchronously, you can use the following request:

curl --location --request POST 'https://localhost:5001/elsa/api/workflow-definitions/<workflow-definition-id>/execute' \
--header 'Content-Type: application/json' \
--header 'Authorization: ApiKey {api_key}'
--data-raw '{
    "input": {
        "InputData": {
            "dummyProp": "hello"
        }
    }
}'

You can define inputs and outputs for each workflow from the Input/Output tab. Workflow input and output

The response should look something like this:

{
    "workflowState": {
        "definitionId": "2f6ba5802e254082b00bd4dab00e650a",
        "definitionVersion": 1,
        "status": "Finished",
        "subStatus": "Finished",
        "bookmarks": [],
        "completionCallbacks": [],
        "activityExecutionContexts": [],
        "output": {
            "OutputData": "hello there!"
        },
        "properties": {
            "PersistentVariablesDictionary": {
                "f765d2c1ef7843ed862488ae6f191884:Workflow1:subvar": ""
            }
        },
        "id": "f765d2c1ef7843ed862488ae6f191884"
    }
}

The id field contains the workflow instance ID. The output fields contains the outputs of the workflow under.

For more information about input and output, see the Input / Output chapter.

Asynchronous execution

The following endpoint executes a workflow asynchronously:

/workflow-definitions/{workflow_definition_id}/dispatch`

The request is pretty much the same as the /execute endpoint, except that it doesn't wait for the workflow to finish.

Instead, the endpoint will enqueue the workflow for execution and return the workflow instance ID.

Here is an example of a response:

{
    "workflowInstanceId": "b235209459294b97904eaee89fe7f930"
}

Workflow instance ID

The workflow instance ID is generated before the workflow instance is actually created. This means that the workflow instance ID can be used to track the workflow instance, even if it doesn't exist yet.

Use C# code

There are two ways to execute a workflow from C# code:

  • Using the IWorkflowRunner service.
  • Using the IWorkflowRuntime service.

Using the IWorkflowRunner service

The IWorkflowRunner is a low-level service that can execute any IActivity object.

For example, the following code executes a workflow called HelloWorld:

var workflow = new Workflow
{
   Root = new WriteLine("Hello world!")
};

var workflowRunner = serviceProvider.GetRequiredService<IWorkflowRunner>();

await workflowRunner.RunWorkflowAsync(workflow);

Workflow instance ID

The Workflow class itself is an activity, too!

Using the IWorkflowRuntime service

Using IWorkflowRunner is easy, but it requires you to know what workflow to execute and process any bookmarks it creates, yourself. Furthermore, you will have to manage persisting the workflow instance yourself.

The IWorkflowRuntime service, on the other hand, is a high-level service that lets you trigger one or more workflows, depending on what input you provide. The workflow runtime works together with the IWorkflowDefinitionService to find workflows from the IWorkflowDefinitionStore.

This means that before you can trigger workflows using the IWorkflowRuntime service, you will have to register your workflows.

When using the workflow designer (or the REST API directly) to create workflows, the workflows will be registered automatically. And if you have a custom IWorkflowDefinitionProvider, its workflows will be registered automatically, too. But, if you need to, you can also register workflows manually. For example:

public class HelloWorldWorkflow : WorkflowBase
{
   protected override void Build(IWorkflowBuilder builder)
   {
        builder.Root = new WriteLine("Hello world!");
   }
}

services.AddElsa(elsa => elsa.AddWorkflow<MyWorkflow>());

The following examples demonstrates invoking an existing workflow using the IWorkflowRuntime service:


var workflowRuntime = serviceProvider.GetRequiredService<IWorkflowRuntime>();
var result = await workflowRuntime.StartWorkflowAsync("HelloWorldWorkflow");

Workflow instance ID

The IWorkflowRuntime service is often used when you want to trigger a workflow from an event handler. For example, if you implement a custom trigger activity that monitors a directory for file changes, you can use the IWorkflowRuntime service to trigger a workflow when a file is created. Another example is when you want to trigger a workflow from a message queue. In that case, you can use the IWorkflowRuntime service to trigger a workflow when a message is received.

Previous
Loading workflows from JSON