A task should have the following structure:
name
: The name of the task. This is required if it's defined in an array. We recommend using a unique name for each task and task name should contain only letters, numbers, and underscores.title
: A brief description of the task. This is optional.handler
: The name of the task handler aka task type. Please refer to the Task Handler document for more details. This is required.parameters
: An object that provides input and configuration for the task. This field supports interpolation, which means you can reference the output of other tasks or the workflow input. This field is optional.ignoreError
: Whether to ignore errors during task execution and let other tasks continue running. This is optional.tasks
: An array of tasks to be executed (will be executed orderly) or an object of tasks (will be executed concurrently). This is optional.retryCount
: The number of times to retry the task if it fails (defaults to 3). This is optional.retryStrategy
: The strategy to use for retrying the task (defaults to 'fixed'). This is optional.retryDelaySeconds
: The delay between retries in seconds (defaults to 3 seconds). This is optional.
For retrying configuration, they are inherited from the parent container (workflow or task) if not defined.
There are some special fields that depend on the handler:
then
,else
: Tasks to execute when the handler is If.decisionCases
,defaultCase
: Tasks to execute when the handler is Switch.loopOver
: Tasks to execute when the handler is For, While, or Iterate.catch
,finally
: Tasks to execute when the handler is Catch.
This is an example of a task definition:
{
"name": "log_message",
"handler": "log",
"parameters": {
"message": "Hello, world!"
}
}
A task handler is a string that represents the task type.
It can be a built-in task or a custom task.
The task handler should be defined in the handler
field.
The handler should follow the format: type:identify[:task-name]
.
type
: The type of the task, it can be:package
: A task handler from a node package which is published to npm.external
: A task handler from an external node package which is not published.script
: A task handler from a standalone script file.
identify
: The identifier of the task handler, it can be:- The package name if the type is
package
. The version can be specified by using the@
symbol, e.g.@my-scope/[email protected]
. - The path to the node module if the type is
external
. - The path to the script file if the type is
script
.
- The package name if the type is
task-name
: The name of the task handler, this is optional if the identify isn't a group task package.
If the handler is a path, it will treat as a script task handler.
This is some valid task handler examples:
package:@letrun-task/[email protected]:read
package:example-task
external:./tasks/example-task
script:./tasks/example-task.js
You can nest tasks inside other tasks by defining the nested tasks in the tasks
field. This is useful when you want to group tasks together.
Almost tasks support the
tasks
field, except for the If, Switch tasks and loop tasks (For, While and Iterate). These tasks support nested tasks by using other fields likethen
,else
,decisionCases
,defaultCase
,loopOver
.
Here is an example:
{
"name": "my_daily_life",
"tasks": {
"play_game": {
"handler": "game.js"
},
"listen_to_music": {
"handler": "music_player.js",
"parameters": {
"songName": "Baby Shark"
},
"tasks": [
{
"name": "download_song",
"handler": "download.js",
"parameters": {
"url": "https://example.com/baby-shark.mp3"
}
},
{
"name": "copy_song",
"handler": "copy.js",
"parameters": {
"source": "baby-shark.mp3",
"destination": "music/baby-shark.mp3"
}
}
]
}
}
}
The nested tasks will be executed before the parent task, thus the execution order will be:
download_song
andplay_game
copy_song
andplay_game
(may still run)listen_to_music
andplay_game
(may still run)
Tasks in the
tasks
field will be executed concurrently or orderly based on the parent task's definition. See more in the Run Tasks Orderly and Run Tasks Concurrently sections.