The Human in the Loop block pauses a run and waits for a person before it continues. Use it for approval gates, to collect feedback, or to gather input at a decision point. The run stays paused — with no timeout — until someone responds through the approval portal, the API, or a webhook.
Configuration
Display Data
What the approver sees — the context shown in the portal to help them decide. Build it field by field or as JSON, referencing earlier outputs with <blockName.output>.
{
"customerName": "<agent1.content.name>",
"proposedAction": "<router1.selectedPath>",
"confidenceScore": "<evaluator1.score>",
"generatedEmail": "<agent2.content>"
}Notification
How approvers are alerted that a decision is waiting. Include the approval URL (<blockId.url>) in the message so they can open the portal. Available channels:
- Slack — a message to a channel or DM
- Gmail — an email with the approval link
- Microsoft Teams — a channel notification
- SMS — a text alert via Twilio
- Webhook — a request to your own notification system
Resume Form
The fields the approver fills in when responding. Each becomes available to downstream blocks once the run resumes.
{
"approved": {
"type": "boolean",
"description": "Approve or reject this request"
},
"comments": {
"type": "string",
"description": "Optional feedback or explanation"
}
}Access resume data in downstream blocks using <blockId.fieldName>.
Approval Methods
Approval Portal
Every block generates a unique portal URL (<blockId.url>) with a visual interface showing all paused output data and form fields for resume input. Mobile-responsive and secure.
Share this URL in notifications for approvers to review and respond.
REST API
Programmatically resume workflows using the resume endpoint. The contextId is available from the block's resumeEndpoint output or from the _resume object in the paused execution response.
POST /api/resume/{workflowId}/{executionId}/{contextId}
Content-Type: application/json
X-API-Key: your-api-key
{
"input": {
"approved": true,
"comments": "Looks good to proceed"
}
}The resume endpoint automatically respects the execution mode used in the original execute call:
- Sync mode (default) — The response waits for the remaining workflow to complete and returns the full result:
{
"success": true,
"status": "completed",
"executionId": "<resumeExecutionId>",
"output": { ... },
"metadata": { "duration": 1234, "startTime": "...", "endTime": "..." }
}If the resumed workflow hits another HITL block, the response returns "status": "paused" with new _resume URLs in the output.
-
Stream mode (
stream: trueon the original execute call) — The resume response streams SSE events withselectedOutputschunks, just like the initial execution. -
Async mode (
X-Execution-Mode: asyncon the original execute call) — The resume dispatches execution to a background worker and returns immediately with202, including ajobIdandstatusUrlfor polling:
{
"success": true,
"async": true,
"jobId": "<jobId>",
"executionId": "<resumeExecutionId>",
"message": "Resume execution queued",
"statusUrl": "/api/jobs/<jobId>"
}Polling execution status
Poll the statusUrl from the async response to check when the resume completes:
GET /api/jobs/{jobId}
X-API-Key: your-api-keyReturns job status and, when completed, the full workflow output.
To check on a paused execution's pause points and resume links:
GET /api/resume/{workflowId}/{executionId}
X-API-Key: your-api-keyReturns the paused execution detail with all pause points, their statuses, and resume links. Returns 404 when the execution has completed and is no longer paused.
Webhook
Add a webhook tool to the Notification section to send approval requests to external systems. Integrate with ticketing systems like Jira or ServiceNow.
API Execute Behavior
When triggering a workflow via the execute API (POST /api/workflows/{id}/execute), HITL blocks cause the execution to pause and return the _resume data in the response:
The response includes the full pause data with resume URLs:
{
"success": true,
"executionId": "<executionId>",
"output": {
"data": {
"operation": "human",
"_resume": {
"apiUrl": "/api/resume/{workflowId}/{executionId}/{contextId}",
"uiUrl": "/resume/{workflowId}/{executionId}",
"contextId": "<contextId>",
"executionId": "<executionId>",
"workflowId": "<workflowId>"
}
}
}
}Blocks before the HITL stream their selectedOutputs normally. When execution pauses, the final SSE event includes status: "paused" and the _resume data:
data: {"blockId":"agent1","chunk":"streamed content..."}
data: {"event":"final","data":{"success":true,"output":{...,"_resume":{...}},"status":"paused"}}
data: "[DONE]"On resume, blocks after the HITL stream their selectedOutputs the same way.
HITL blocks are automatically excluded from the selectedOutputs dropdown since their data is always included in the pause response.
Returns 202 immediately. Use the polling endpoint to check when the execution pauses.
Examples
Approve content before it ships
The run pauses at the Human in the Loop block until someone approves; on resume, the API publishes. The same gate works before any action, like sending a customer email.
Chain multiple approvals
For a high-stakes change, chain two approval steps — a manager, then a director — before the workflow executes.
Verify extracted data
A reviewer checks the data an Agent extracted before a Function processes it.
Outputs
| Output | What it is |
|---|---|
url | The approval portal URL |
resumeEndpoint | The resume API endpoint |
response | The display data shown to the approver |
submission | The approver's form submission |
submittedAt | ISO timestamp of when the run resumed |
<fieldName> | Each Resume Form field, by name, after the run resumes |
Read them downstream as <blockName.output> — for a block named approval, that's <approval.approved>.
The approval portal
Paused Output:
{
"title": "<agent1.content.title>",
"body": "<agent1.content.body>",
"qualityScore": "<evaluator1.score>"
}Resume Input:
{
"approved": { "type": "boolean" },
"feedback": { "type": "string" }
}Downstream Usage:
// Condition block
<approval1.approved> === trueThe example below shows an approval portal as seen by an approver after the workflow is paused. Approvers can review the data and provide inputs as a part of the workflow resumption. The approval portal can be accessed directly via the unique URL, <blockId.url>.
Related Blocks
- Condition - Branch based on approval decisions
- Variables - Store approval history and metadata
- Response - Return workflow results to API callers