Skip to content

Expressions

SuperPlane uses Expr for expressions. Expressions let you reference data from upstream nodes, transform values, and control workflow logic.

As a run executes, each node’s output is added to the message chain. Access it through $, referencing nodes by display name:

{{$['Node Name'].data.field}}
{{$['Node Name'].data.nested.field}}
{{$['Node Name'].data.array[0].value}}

Every entry also includes a .config property — the node’s resolved configuration at run time:

{{$['HTTP Request'].config.url}}
{{$['HTTP Request'].config.method}}
FunctionReturns
root()The payload that started the run (the trigger event).
previous()The immediate upstream node’s payload.
previous(n)Walk n levels upstream.
{{root().data.ref}}
{{previous().data.status}}

previous() is not available when a node has multiple inputs (e.g. after a Merge). Use $['Node Name'] instead.


Expressions appear in two contexts with slightly different syntax:

Text fields (URLs, message bodies, labels) — wrap each expression in {{ }}:

Deployment of {{$['Release'].data.name}} failed. See: {{$['Deploy'].data.workflow_run.html_url}}

Condition fields (If, Filter) — the entire field is one bare expression that must return true or false:

$['Get cat fact'].data.body.length <= 160 && $['Health Check'].data.body.healthy

Type $ in a condition field or {{ in a text field to trigger autocomplete.


Beyond standard arithmetic (+, -, *, /, %, **) and comparison (==, !=, <, >, <=, >=):

OperatorWhat it doesExample
&& || !Logical (aliases: and, or, not){{$['a'].data.ok && !$['b'].data.failed}}
containsString contains substring{{$['node'].data.body contains "error"}}
startsWithString prefix check{{$['node'].data.ref startsWith "refs/heads/"}}
endsWithString suffix check{{$['node'].data.branch endsWith "-hotfix"}}
matchesRegex match (RE2){{$['node'].data.msg matches "^fix\\(.*\\)"}}
in / not inMembership test{{$['node'].data.env in ["staging", "prod"]}}
??Nil coalescing (fallback){{$['node'].data.label ?? "default"}}
? :Ternary{{$['node'].data.ok ? "pass" : "fail"}}
?.Optional chaining{{$['node'].data?.user?.name}}

Array functions accept a closure where # is the current element:

{{filter($['node'].data.items, # > 10)}}
{{map($['node'].data.users, #.name)}}
{{any($['node'].data.tags, # == "critical")}}
{{sortBy($['node'].data.alerts, #.severity, "desc")}}

reduce() adds #acc for the accumulator:

{{reduce($['node'].data.items, #acc + #.price, 0)}}

Fallback for missing fields:

{{$['Webhook'].data.user.name ?? "unknown"}}

Ternary in a text field:

Status: {{$['Deploy'].data.success ? "Deployed" : "Failed"}}

Check array membership:

{{"production" in $['node'].data.environments}}

Filter and join:

{{join(filter($['node'].data.tags, # startsWith "env:"), ", ")}}

Date comparison (event in the last hour):

{{now().Sub(date($['node'].data.timestamp)).Hours() < 1}}

Build a JSON string:

{{toJSON({status: $['Deploy'].data.result, ref: root().data.ref})}}

SuperPlane expressions have access to the full set of Expr built-in functions for strings, arrays, dates, math, and type conversion. See the Expression Functions Reference for the complete list with signatures and examples.

For the Expr language specification, see the Expr documentation.