Reference

Human in the Loop

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.

Human in the Loop
Display Data-
Notification-
Resume Form-
error

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: true on the original execute call) — The resume response streams SSE events with selectedOutputs chunks, just like the initial execution.

  • Async mode (X-Execution-Mode: async on the original execute call) — The resume dispatches execution to a background worker and returns immediately with 202, including a jobId and statusUrl for 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-key

Returns 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-key

Returns 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

OutputWhat it is
urlThe approval portal URL
resumeEndpointThe resume API endpoint
responseThe display data shown to the approver
submissionThe approver's form submission
submittedAtISO 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> === true

The 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>.

  • Condition - Branch based on approval decisions
  • Variables - Store approval history and metadata
  • Response - Return workflow results to API callers

Common Questions

The workflow pauses indefinitely until a human provides input through the approval portal, the REST API, or a webhook. There is no automatic timeout — it will wait until someone responds.
You can configure notifications through Slack, Gmail, Microsoft Teams, SMS (via Twilio), or custom webhooks. Include the approval URL in your notification message so approvers can access the portal directly.
Use the syntax <blockId.fieldName> to reference specific fields from the resume form. For example, if your block name is 'approval1' and the form has an 'approved' field, use <approval1.approved>.
Yes. You can place multiple Human in the Loop blocks in sequence to create multi-stage approval workflows. Each block pauses independently and can have its own notification configuration and resume form fields.
Yes. Each block exposes a resume API endpoint that you can call with a POST request containing the form data as JSON. This lets you build custom approval UIs or integrate with existing systems like Jira or ServiceNow.
The block outputs include the approval portal URL, the resume API endpoint URL, the display data shown to the approver, the form submission data, the raw resume input, and an ISO timestamp of when the workflow was resumed.

On this page