Journey management endpoints
Journeys API requests are authenticated by passing your workspace API key as a Bearer token in theAuthorization header: Authorization: Bearer <your-api-key>. Requests without this header are rejected.
| Endpoint | Purpose |
|---|---|
GET /journeys | List journeys (paginated). |
POST /journeys | Create a journey. |
GET /journeys/{templateId} | Fetch a journey by ID. |
PUT /journeys/{templateId} | Replace a journey draft. |
DELETE /journeys/{templateId} | Archive a journey. |
GET /journeys/{templateId}/versions | List published versions. |
POST /journeys/{templateId}/publish | Publish the current draft. |
POST /journeys/{templateId}/invoke | Invoke a journey (start a run). |
Journey-scoped notification template endpoints
Each journey manages its own notification templates, created and managed exclusively within that journey’s scope. These endpoints mirror the workspace-level/notifications endpoints but are nested under a specific journey.
| Endpoint | Purpose |
|---|---|
GET /journeys/{templateId}/templates | List templates scoped to a journey. |
POST /journeys/{templateId}/templates | Create a scoped template. |
GET /journeys/{templateId}/templates/{notificationId} | Fetch a template by ID. |
PUT /journeys/{templateId}/templates/{notificationId} | Replace a template draft. |
DELETE /journeys/{templateId}/templates/{notificationId} | Archive a template. |
POST /journeys/{templateId}/templates/{notificationId}/publish | Publish a template. |
GET /journeys/{templateId}/templates/{notificationId}/versions | List template versions. |
Journey structure: triggers, nodes, and the DAG
A journey is a directed acyclic graph (DAG) of nodes. Each node performs one action (send a notification, wait, branch, fetch data, etc.), and the sequence defines execution order.Triggers
Every journey starts with a trigger node that determines how runs begin. You’ve got two options:- API invoke (
trigger_type: "api-invoke"): Starts a run when you callPOST /journeys/{templateId}/invoke. You can optionally attach a JSON Schema to the trigger via theschemafield. Courier validates thedatapayload against it when the trigger node executes. - Segment (
trigger_type: "segment"): Starts a run when a matching Segment event arrives. Requiresrequest_type(identify,group, ortrack) and optionally anevent_id.
Node types
Nodes are the building blocks of your journey. Each node has atype field that determines its behavior, and some node types have a secondary discriminator. Here’s the full set:
| Node type | Description |
|---|---|
trigger | Entry point. Discriminated by trigger_type: api-invoke or segment. |
send | Send a notification using a journey-scoped template. References the template via message.template. |
delay | Pause the run. Discriminated by mode: duration (e.g., "PT1H" for one hour) or until (a specific timestamp). |
fetch | Make an HTTP request and merge the response into run state. Discriminated by method: get, delete, post, or put. |
branch | Conditional routing. Evaluates paths[] in order and routes to the first match, with a default fallback. |
throttle | Rate-limit runs. Discriminated by scope: user, global, or dynamic. |
ai | Run an LLM prompt with optional web search. Returns structured output per output_schema. |
exit | End the run immediately. |
Conditions
Several node types support aconditions field for conditional execution. Conditions are expressed as tuples:
- Binary (3 elements):
[path, operator, value], for example["data.plan", "is equal", "pro"] - Unary (2 elements):
[path, operator], for example["data.email", "exists"]
is equal, is not equal, contains, does not contain, starts with, ends with, greater than, greater than or equal, less than, less than or equal.
Available operators for unary conditions: exists, does not exist.
Conditions can be grouped with AND / OR logic, and groups can be nested for complex expressions.
Standard workflow: create → template → wire → publish → invoke
Here’s the typical process for building a journey programmatically. Each step builds on the previous one, so follow them in order:- Create the journey shell with
POST /journeys. Pass anameand at least one node (typically a trigger). This gives you a journey ID and a draft you can build on. - Create journey-scoped templates with
POST /journeys/{id}/templates. These are the notification templates your send nodes will reference. Define the content using Elemental format and specify thechannel(email, sms, push, etc.). - Wire templates into the journey with
PUT /journeys/{id}. Replace the draft with your full node graph, including send nodes that reference the template IDs you created in step 2. - Publish with
POST /journeys/{id}/publish. This locks in the current draft as a versioned snapshot. All new runs execute against the published version. - Invoke with
POST /journeys/{id}/invoke. Pass auser_id(orprofilewith contact info) and optionaldatapayload. Courier starts a run that walks through the DAG.
Example: building a welcome journey
Let’s walk through creating a “Welcome Journey” that sends a welcome email when invoked via the API.Step 1: Create the journey shell
id from the response. You’ll use it in every subsequent request.
Step 2: Create a scoped email template
id from the response as your template ID.
Step 3: Wire the template into the journey
Step 4: Publish the journey
Step 5: Invoke the journey
runId immediately (HTTP 202). Courier processes the run asynchronously, validating data against the trigger’s JSON Schema and walking through the DAG.
Journey-scoped templates vs. workspace templates
These two template types serve different purposes and are managed differently. Journey-scoped templates are created underPOST /journeys/{id}/templates and live exclusively within that journey. They use Elemental format for content, can’t be shared across journeys, and can’t be referenced from the Send API. They’re published either alongside the journey (when you publish the journey itself) or independently via their own publish endpoint.
Workspace templates are the ones you create at POST /notifications or in Design Studio. They live at the workspace level, can be used from the Send API, and aren’t tied to any specific journey.
If you need a template that’s reusable across multiple journeys or callable from the Send API, use a workspace template. If the template is specific to one journey’s logic, keep it scoped to the journey.
Node types reference
Here’s a detailed reference for each node type with their discriminator fields and required properties.| Type | Discriminator | Required fields |
|---|---|---|
| Trigger (API) | type: "trigger", trigger_type: "api-invoke" | None beyond discriminators. Optional: id, schema, conditions. |
| Trigger (Segment) | type: "trigger", trigger_type: "segment" | request_type (identify, group, or track). Optional: event_id, conditions. |
| Send | type: "send" | message.template (journey-scoped template ID). Optional: message.to (overrides), message.delay, message.data, conditions. |
| Delay (duration) | type: "delay", mode: "duration" | duration (ISO 8601 duration string, e.g. "PT30M"). Optional: conditions. |
| Delay (until) | type: "delay", mode: "until" | until (ISO 8601 timestamp or context reference). Optional: conditions. |
| Fetch (GET/DELETE) | type: "fetch", method: "get" or "delete" | url, merge_strategy. Optional: headers, query_params, response_schema, conditions. |
| Fetch (POST/PUT) | type: "fetch", method: "post" or "put" | url, merge_strategy. Optional: body, headers, query_params, response_schema, conditions. |
| Branch | type: "branch" | paths[] (each with conditions and nodes[]), default (with nodes[]). Optional: paths[].label, default.label. |
| Throttle (static) | type: "throttle", scope: "user" or "global" | max_allowed, period. Optional: conditions. |
| Throttle (dynamic) | type: "throttle", scope: "dynamic" | max_allowed, period, throttle_key. Optional: conditions. |
| AI | type: "ai" | output_schema. Optional: model, user_prompt, web_search, conditions. |
| Exit | type: "exit" | None. Optional: id. |
Merge strategies for fetch nodes
When a fetch node receives a response, themerge_strategy field determines how it’s incorporated into the journey run state:
| Strategy | Behavior |
|---|---|
overwrite | Deep-merges response into state. Response values overwrite existing fields. |
soft-merge | Adds new fields from the response without overwriting existing values. |
replace | Replaces the entire run state with the response. |
none | Discards the response body. Useful for fire-and-forget requests. |
What’s next
Building Journeys via UI
Compose journeys visually with the canvas editor
Run Inspection
Debug journey runs node by node