Workflow Engine evaluates a series of actions and control instructions to perform requests against our internal API (using the BXDK package), tag data API, external HTTP APIs etc. using a JSON based Workflow Language.
result_state =
WorkflowEngine.evaluate(workflow,
params: %{
context: %{}
},
json_logic: MyService.JsonLogic,
actions: %{
"foo" => MyApp.FooAction
}
)This package can be installed by adding workflow_engine to your list of dependencies in mix.exs:
def deps do
[
{:workflow_engine, github: "box-id/workflow_engine", tag: "0.1.0"}
]
endTo extend Workflow Engine with custom actions, implement the WorkflowEngine.Action behaviour.
This is a minimal example of a custom action that multiplies a value by a given factor and stores the result in the workflow state:
defmodule MyApp.FooAction do
@behaviour WorkflowEngine.Action
@impl true
def execute(workflow_state, %{"type" => "multiply"} = step) do
# Implement your action logic here
multiply_by = get_required(step, "multiply_by")
source_key = get_required(step, "source_key")
value = Map.get(workflow_state, source_key, 1)
new_state =
Map.put(workflow_state, "multiply_result", value * multiply_by)
{:ok, {new_state, nil}}
rescue
# Wrap all error messages & add current state
e in WorkflowEngine.Error ->
reraise WorkflowEngine.Error,
[message: "FooAction: " <> e.message, state: state],
__STACKTRACE__
end
defp get_required(step, key) do
case Map.fetch(step, key) do
{:ok, value} when not is_nil(value) ->
value
_ ->
raise WorkflowEngine.Error,
message: "Missing required step parameter \"#{key}\"."
end
end
endThen, setup the action in a customized module that implements the WorkflowEngine:
defmodule MyAppNamespace.WorkflowEngine do
def evaluate(workflow, opts \\ []) do
state = %WorkflowEngine.State{
vars: Keyword.fetch!(opts, :vars),
actions: %{
"multiply" => MyApp.FooAction,
}
}
WorkflowEngine.evaluate(state, workflow)
end
end
### WorkflowEngine.State Attributes
- `vars`: A map of variables that can be used in the workflow.
- `json_logic_mod`: The module implementing the JSON Logic evaluation logic.
- `actions`: A map of action types to their respective modules. This allows you to define custom actions that can be used in workflows.
## Error Handling
Since workflows are dynamic (and potentially user-provided), Workflow Engine and its actions need
to take great care of handling errors in a transparent way.
Workflow Engine uses the `{:error, %WorkflowEngine.Error{}}` return type if something unexpected
ocurred. Workflow Engine should never return an error tuple with another data type at the second
position.
Also, no exceptions should be raised during workflow evaluation. However, due to the manifold ways
invalid configuration or data could be supplied, it can't be fully ruled out. These cases are
considered a bug, though, and are worth fixing by expanding input validation and error handling.
##### `message`.
Errors contain a `message` binary, which is a human-readable description of the error, including
the formatted instruction pointer (to hint to the position in the workflow that likely caused the
error) and the configuration of the workflow step that failed.
Depending on the actions and configuration of a workflow, `message` could contain
sensitive information and might thus not be suitable for displaying to users.
##### `state`
The `state` field allows introspecting the state of the Workflow Engine at the time the error
occurred.
##### `recoverable`
The boolean `recoverable` field gives a best-effort estimation about whether retrying the workflow
could lead to a positive/error-free outcome or not. The decision is made based on the reason for
the error.
For example, retrying workflows with invalid configuration such as an invalid step descriptions,
unknown action names, malformed JsonLogic etc., will never lead to a successful result. In such
cases, errors are tagged with `recoverable: false`.
Softer errors, that typically originate from an external system such as a failed HTTP request or a
syntax error in a CSV file, are marked with `recoverable: true`. For these errors, the caller can
decide whether it wants to automatically or manually retry running the workflow.
## Tests
Run tests using `mix test` or, during development, `mix test.watch`.
Tests touching external systems should be tagged with `@tag :external_service` s.t.
they are (by default) skipped, which keeps the execution of the test suite fast and reliable.
To include those tests as well, use `mix test.including_external` or the `--include
external_service` flag.