Skip to content

Events reference

Every run event uses the same envelope:

FieldTypeNotes
runIdstringRun identifier
sequenceintegerPer-run monotonic ordering key
typestringEvent type from the fixed taxonomy
timestampISO 8601 stringInformational only
payloadobjectType-specific data
callIdstringA field of the payload on tool events (tool.call, tool.result, tool.error); pairs a tool outcome with its call

Clients should order and deduplicate events by sequence.

Event taxonomy

TypeWhen it firesPayload fields
agent.turn.startWhen the model begins a turnturnId
agent.message.deltaWhen the model streams a chunk of visible text — emitted by both the GitHub Copilot and Microsoft Foundry runnersdelta, messageId
agent.messageFallback: emitted only when a turn produced no token deltas (Foundry runner only; tool-call-only turns or empty streams)content
agent.turn.endWhen the model finishes a turn (emitted by both runners; closes the turn bubble in the frontend)turnId
agent.intentWhen the agent calls report_intent before a major stepintent
agent.system_promptAt run start, after the system prompt is setprovider, prompt (full text), note (optional)
agent.toolsAt run start, listing the tools registered for this runtools (string array of tool names)
tool.callBefore the runtime evaluates a tool invocation against the sandbox policycallId, toolName, arguments
tool.resultAfter an approved tool runs successfullycallId, content
tool.errorAfter a tool is denied by the sandbox policy, or fails for any other reason such as a missing file or I/O failurecallId, errorMessage
tool.approval_requiredWhen a tool call is paused awaiting human approvalrequest_id, tool_name, url (optional), intention (optional)
tool.auto_approvedWhen the per-run auto-approve-tools option is ON and an allow-with-approval tool request is auto-granted at the gate instead of waiting for a human; audit-only (the tool then runs)requestId, toolName, url (optional)
agent.question_askedWhen an agent calls ask_question to bubble a clarifying question or permission request; the run suspends inside the tool call until answered or timed outrequestId, question
agent.question_answeredWhen a pending ask_question request is answered (or resolved by timeout) and the agent resumesrequestId, answer, timedOut
run.completedWhen the watch loop determines the run is terminal with no file changes (watch-loop only; never emitted by the runner)result
run.outcomeAgent self-assessment of task completion, emitted just before run.completedachieved (bool), reason
run.failedWhen the runtime, provider, or content-safety flow ends the run in failurereason
run.boundedWhen the run hits a step-count or wall-clock boundlimit_type, step_count
run.cancelledWhen an in-progress run is cancelled because its project was deleted(none)
run.errorWhen an operation fails but the run is reverted to a retryable state (e.g. back to AwaitingReview after a merge internal error); non-terminal — the stream stays openreason
run.degradedWhen the sandbox blocks at least one tool call during a run; non-terminal — the run continues with a degraded outcometoolName, reason
rai.verdictThe RAI reviewer's verdict for a run; written to the {runId}-rai sub-streamverdict (green / yellow / red / revise), runId
workflow.stepWhen each workflow executor stage transitions (start/complete/fail/skip), for every node in both the full and child pipelinesstep, status, label, timestamp_utc, agent_name (agent step only), reviewer (review step only), message (optional)
run.workflow_graphOnce at run start, carrying the full workflow graph descriptor for rendering the run topologyGraphDescriptor (see below)
review.requestedAfter the worktree is committed and the review tree hash is storedtree_hash, request_id
review.approvedWhen the owner approves the run and the merge proceeds(none)
review.declinedWhen the owner declines the run(none)
review.changes_requestedWhen the human reviewer requests a revisioncomment
revision.startedWhen a request-changes revision cycle beginsrevision, message
merge.startedWhen an approved merge begins executiontree_hash
merge.completedAfter an approved run merges cleanly into the originating branchmerged_commit_hash; previous_head_sha (direct path only)
merge.failedAfter an approved run cannot merge back cleanlyreason
coordinator.startedWhen a coordinator run begins drafting an OutcomeSpec from the user's goalgoal
coordinator.recoveredWhen an interrupted coordinator run is resumed after a process restart and its dispatch / collective-assembly engine is re-armed from the persisted work planstatus (the work-plan status it resumed from)
coordinator.outcome_specWhen the coordinator has drafted an OutcomeSpec and suspended at the await-confirmation gatespecId, status, desiredOutcome, scope, assumptions, clarifyingQuestions
coordinator.outcome_spec.confirmedWhen a human confirms the drafted OutcomeSpec and the coordinator run proceedsspecId, confirmedBy
coordinator.work_planWhen the coordinator has decomposed the confirmed spec into a persisted work planworkPlanId, status, subtasks, dependencies
coordinator.workflow_selectedWhen the coordinator selects which workflow to run from a project's multi-workflow set (skipped silently when the project carries only one workflow)selectedId, selectedName, rationale, wasAutoSelected, overrideHint, available ([{ id, name }])
coordinator.child_stall_detectedWhen the proactive stuck-child sweep finds an in-progress child subtask that has made no progress past the stall timeout; the child is force-completed and re-dispatchedchildRunId, subtaskId, staleSinceUtc, stallTimeoutMinutes
coordinator.topologyWhen the orchestration graph is first dispatched (snapshot) and on every subsequent subtask lifecycle transition (delta)version, kind, seq, nodes (snapshot) / changed (delta), edges (snapshot)
coordinator.graphWhen the unified coordinator graph shape changes (a subtask child run is dispatched, or the plan reaches its terminal snapshot)a shape-only GraphDescriptor (variant coordinator)
subtask.dispatched / subtask.running / subtask.assemble_ready / subtask.rai_flagged / subtask.completed / subtask.failedAs a subtask's child run advances through its lifecyclesubtaskId, childRunId, assignedAgent, selectedModelId, status
run.assemble_readyOn a coordinator CHILD run's own stream when the child finishes its trimmed (agent + RAI) pipeline and is ready to be collected/assembledrunId, subtaskId, parentRunId, worktreeBranch, treeHash, hasChanges, stepCount, raiSafetyFlagged
run.no_changes_producedOn a coordinator CHILD run when it reaches assemble-ready with no committed changes (the worker wrote no files)runId, subtaskId, parentRunId, message
coordinator.steeringWhen a steering directive is created or changes statedirectiveId, kind, targetChildRunId, status, instruction
coordinator.children_completeWhen every child subtask has reached a terminal status and the work plan moves to awaiting_assemblyworkPlanId
coordinator.assembly_startedWhen the collective-assembly pipeline claims the plan (awaiting_assembly → assembling, exactly-once)workPlanId, integrationBranch, subtaskCount
coordinator.assembly_blockedWhen assembly stops with NO partial work — an ineligible subtask, or a conflict building the integration branchworkPlanId, reason, and (conflict only) conflictingBranch, conflictingFiles
coordinator.assembly_rai_started / coordinator.assembly_rai_completedThe ONE collective RAI pass over the aggregate diff (advisory; never hard-blocks)workPlanId, integrationBranch / raiSafetyFlagged
coordinator.assembly_review_requestedWhen the pipeline suspends at the ONE collective human-review gateworkPlanId, integrationBranch, treeHash, includedSubtaskIds, raiSafetyFlagged, hasChanges
coordinator.assembly_review_approvedWhen the reviewer approves the combined outputworkPlanId
coordinator.assembly_changes_requestedWhen the reviewer requests changes; the coordinator re-dispatches the inferred childrenworkPlanId, redispatchSubtaskIds, inferredFiles, fellBackToAll, feedback
coordinator.assembly_merge_started / coordinator.assembly_merge_completed / coordinator.assembly_merge_failedThe ONE collective merge of the integration branch into the originating branchworkPlanId, integrationBranch / commitHash / reason, conflictingFiles
coordinator.assembly_scribe_started / coordinator.assembly_scribe_completedThe ONE collective scribe pass after a successful merge (best-effort)workPlanId
coordinator.assembly_completedWhen collective assembly finishes and the work plan reaches completeworkPlanId, integrationBranch, commitHash
coordinator.assembly_declinedWhen the reviewer declines the combined output (terminal); the coordinator run ends declinedworkPlanId, reason, reviewer
coordinator.assembly_failedWhen the assembly background task hits an UNEXPECTED fault; the work plan moves to assembly_failed and the coordinator run ends with a human-readable assembly_error: <message> resultworkPlanId, reason, phase
coordinator.child_questionWhen a coordinator child run bubbles a question via ask_question; re-projected onto the coordinator stream so the operator can answer the child runchildRunId, subtaskId, requestId, question
coordinator.child_approval_requiredWhen a coordinator child run pauses on a tool-approval gate; re-projected onto the coordinator stream so the operator can grant/deny on the child runchildRunId, subtaskId, requestId, toolName, url (optional), message (optional)
coordinator.autopilot_answeredWhen the coordinator's Autopilot option is ON and the coordinator model auto-answers a clarifying question (its own or one bubbled from a child); the answer is also resolved on the child's question gate, so the normal agent.question_answered still surfacesrunId, childRunId (optional), requestId, question, answer

Tool event pairing

Each tool.call carries a callId in its payload. The matching tool.result or tool.error repeats that same callId, so clients can pair tool outcomes with calls without relying on adjacent events. A policy denial is reported as a tool.error, not a separate event type.

Provider parity

Both the GitHub Copilot and Microsoft Foundry runners stream text as agent.message.delta events. Each delta carries a delta chunk and the messageId it belongs to. Both providers emit agent.turn.end to close the final turn, giving the frontend a consistent signal to close the turn bubble regardless of which provider is active.

Both providers surface the same tool event vocabulary. For each tool the agent runs, the stream carries a tool.call, followed by a tool.result for an approved tool (with its real content) or a tool.error for a denial or failure. The Copilot provider reads these from the tool-execution lifecycle that flows inline through the streaming response, so an observer sees individual tool activity on Copilot runs at parity with Foundry.

SDK-internal tools (report_outcome, glob) are suppressed from the event stream. report_intent is translated into an agent.intent event rather than suppressed — the raw tool call is hidden, but the intent text surfaces as a first-class event. agent.tools is a synthetic event emitted by the runtime, not an SDK tool.

Event details

agent.message

This event is emitted only by the Foundry runner, as a fallback when a turn produced no token deltas — for example, a tool-call-only turn or an empty stream. content carries the full text for that turn. It is never emitted alongside agent.message.delta events for the same turn.

During normal streaming, both the GitHub Copilot and Microsoft Foundry runners produce agent.message.delta events, each carrying a delta chunk and the messageId it belongs to.

tool.call

This event fires before the runtime evaluates the request against the sandbox policy. toolName is the tool the model invoked, and arguments is the argument object it passed (for file tools, this includes the requested path).

tool.result

This event records a successful tool execution. content carries the result the tool returned, such as the text of an approved file read.

tool.error

This event records every tool outcome that is not a success. It covers sandbox policy denials — an absolute path, .. traversal, or a symlink escape — as well as non-policy failures such as a missing file or an I/O error. errorMessage explains what went wrong. It never carries the contents of a file outside the sandbox, because a denied tool never runs.

run.completed

This event is emitted exclusively by the watch loop (RunWatchLoopService) when the workflow reaches a terminal state with no file changes. The result field is "no_changes". Neither the GitHub Copilot runner nor the Foundry runner emits this event; they emit agent.turn.end to close their final turn and let the watch loop determine terminal state. When the agent produces changes, run.completed is not emitted; the run transitions to review.requested instead.

run.failed

This event marks a terminal failure. The reason field identifies the cause. When the agent's output is blocked by content safety policy, reason is "content_safety" and the run never reaches the review gate. Other values reflect infrastructure or watch-loop errors (for example, "watch_loop_error").

run.bounded

This event marks a run that exceeded enforced limits. limit_type is step-count or wall-clock.

run.cancelled

This event marks a run that was cancelled because its project was deleted. The run transitions to a terminal state immediately and no further events are emitted. It carries no payload fields. The originating branch and any worktree state are cleaned up as part of the project deletion.

review.requested

This event anchors the review gate. tree_hash identifies the committed worktree state that the human reviews and that the merge step later verifies. request_id is an informational correlation id for the underlying workflow review request; it is not required for the review decision.

review.approved

This event records that an approve decision was accepted. It carries no payload fields. It is followed immediately by either merge.completed or merge.failed. A blocked (retriable) approve does not emit this event — the run stays at the review gate and review.requested remains the last event on the stream.

review.declined

This event records a decline decision. It carries no payload fields. The originating branch remains unchanged.

merge.started

This event fires immediately before the merge operation begins, bridging the gap between the approve action and the terminal merge.completed or merge.failed event. tree_hash identifies the committed worktree state being merged — the same value recorded by review.requested.

merge.completed

This event records a successful merge. merged_commit_hash is always present. In the primary workflow path, the field contains the full result string in the format merged:{commitHash} (e.g., "merged:34c09ee..."). In the direct fallback path (post-restart recovery with no checkpoint), the field contains just the commit hash, and previous_head_sha is also present — the SHA the originating branch pointed to before the merge, useful for auditing and rollback.

merge.failed

This event records why an approved run could not merge. Terminal reasons are branch divergence with unresolvable conflicts, and a tree-hash mismatch (the worktree branch changed after the run was reviewed). A checked-out originating branch is not a terminal reason — if the branch is checked out but the working tree is dirty, the approve is blocked retriably and no event is emitted until the condition is resolved.

agent.intent

Emitted when the agent calls the report_intent internal tool before a major step. Not shown as a tool call card in the frontend — rendered as a lightweight lifecycle card with the intent text. The intent field is always a non-empty string.

agent.system_prompt

Emitted once at run start when the system prompt is set. provider identifies which model provider is active. prompt carries the full system prompt text. In the frontend, this renders as a collapsible card showing a 120-character preview with character count; clicking expands the full text.

agent.tools

Emitted once at run start listing the tool names registered for this run. Varies by sandbox policy (run_command is absent when shell execution is disabled). Rendered in the frontend as a row of <Badge> components.

revision.started

Emitted when a request-changes revision cycle begins. revision is a 1-based counter. message is the reviewer's comment passed to the agent.

run.outcome

The agent's self-assessment of task completion. achieved: true when the agent reports the task was completed; false when a critical step failed or was blocked (for example, a required tool call was denied by the sandbox). Emitted by the report_outcome internal tool, which is suppressed from ToolCallCard rendering. The frontend uses this to style run.completed: green for achieved, amber warning for not achieved. If the agent never calls report_outcome (older prompts or prompts that don't include the instruction), run.completed renders as success by default.

review.changes_requested

Emitted when the human reviewer calls POST /api/runs/{id}/request-changes. comment is the reviewer's feedback passed to the agent for the revision cycle.

tool.approval_required

Emitted when a tool call is paused waiting for human approval. request_id identifies the request and is used by POST /api/runs/{id}/tool-approvals and POST /api/runs/{id}/tool-denials. tool_name is the tool being called. url is the resource being accessed (for web_fetch and similar tools). intention is an optional human-readable description of what the agent intends to do. The run is paused until the human approves or denies; tool.result or tool.error follows once settled.

workflow.step

Emitted by each workflow executor stage when it starts, completes, fails, or is skipped — for every executor node in BOTH the full and the trimmed coordinator-child pipelines, so the graph updates live (no node stays "Pending" while it is actually running). The step field identifies the stage:

StepExecutorEmitter
agentAI agent turn (task execution)executor self-emit
raiRAI safety reviewexecutor self-emit
reviewHuman review gateHITL (review.requested / decision endpoints)
mergeBranch mergeexecutor self-emit
scribeSession loggerexecutor self-emit
assemble-readyCoordinator child assemble-ready terminalwatch loop (MAF executor-lifecycle translation)

Lifecycle (started / completed / failed) is translated from the MAF ExecutorInvoked / ExecutorCompleted / ExecutorFailed events by the run watch loop for nodes that do not already self-emit (currently the child assemble-ready terminal); the agent/rai/merge/scribe executors self-emit their own lifecycle plus the richer branch statuses.

Possible status values: started, completed, failed, skipped, revise (RAI step only).

The step value always equals the descriptor node id the frontend keys on (run.workflow_graph nodes[].id). The label field is a short human-readable description (e.g. "Agent turn", "RAI review", "Assemble-ready"). timestamp_utc is an ISO 8601 ("O") timestamp; the timestamp_utc on the started event drives the per-node live elapsed timer. The agent step includes agent_name (the team member running the turn). The review step includes reviewer (GitHub username) when a human review decision is recorded. An optional message field carries a short human-readable status note when available (e.g. on the assemble-ready node); consumers must tolerate its absence.

The web UI uses workflow.step events to drive the workflow diagram — each card in the Agent → Rai → Review → Merge → Scribe pipeline (or Agent → Rai → Assemble-ready for a child) updates live as these events arrive.

run.workflow_graph

Emitted once at run start, carrying a full snapshot of the run's workflow topology as a GraphDescriptor. It is built from the same code that wires the MAF workflow (no runtime reflection), so the rendered graph never drifts from the executors that actually run. The descriptor is persisted with the run's other events, so it is also available for terminal runs via replay and through GET /api/runs/{id}/graph. Child runs (parent_run_id != null) carry the child variant; all others carry the full variant.

GraphDescriptor shape (snake_case JSON):

json
{
  "graph_id": "agentweaver-workflow-full",
  "variant": "full",
  "start_node_id": "agent",
  "nodes": [
    { "id": "agent", "label": "Agent", "role": "agent", "kind": "live", "node_type": "agent", "child_graph_ref": null }
  ],
  "edges": [
    { "from": "agent", "to": "rai", "cardinality": "direct", "loopback": false }
  ]
}
  • variant: "full" | "child" | "coordinator".
  • nodes[].id: the logical node id (equals the step key in workflow.step events for live business nodes). kind: "live" | "planned". child_graph_ref: optional reference to a nested graph.
  • nodes[].node_type: self-declared category that drives the frontend's rendered shape — one of "agent" (an AI agent turn), "action" (a deterministic system op), "gate" (a human-in-the-loop decision/approval), "terminal" (a workflow endpoint/checkpoint), or "subtask" (a coordinator fan-out child reference). Required on every node. In the full variant: agent/rai/scribe are agent, review is gate, merge is action; in the child variant assemble-ready is terminal.
  • edges[].cardinality: "direct" | "fanout" | "fanin". loopback: true for revision-cycle back-edges (the target is an ancestor of the source).

Plumbing executors (input storers, adapters, terminals) are collapsed or hidden: hidden nodes are dropped and their edges are transitively re-stitched, and the scribe-path executors collapse into the single scribe node. The full variant nodes are agent, rai, review, merge, scribe; the child variant nodes are agent, rai, assemble-ready.

coordinator.started

Emitted once when a coordinator run begins. A coordinator run is a Run with ParentRunId == null driven by the built-in Coordinator agent. goal carries the user's submitted goal text. The run reads the project's Feature 006 memories and decision-inbox entries as grounding context, then drafts an OutcomeSpec.

coordinator.recovered

Emitted when an interrupted coordinator run is resumed after an API process restart. A coordinator run stays InProgress across the (non-MAF) dispatch + collective-assembly window, so a restart would otherwise strand it. On startup — after the generic restart sweep has failed any stranded child runs — CoordinatorRunService.RecoverInterruptedRunsAsync reconstructs the orchestration entirely from the persisted work plan and re-arms the correct engine: the dispatch loop re-launches in-flight subtasks (reset to pending), or the collective-assembly pipeline rebuilds the integration branch and re-arms the human-review gate. (The pre-confirm spec phase is a checkpointed MAF workflow and is resumed from its checkpoint instead, with no coordinator.recovered event.) status is the work-plan status the run resumed from (planned, dispatching, awaiting_assembly, assembling, or in_review). The re-armed engine then re-emits its topology / assembly snapshots so the live view renders without client-side computation.

coordinator.outcome_spec

Emitted when the coordinator has drafted an OutcomeSpec and suspended at the await-confirmation gate (coordinator-confirmation-gate). The OutcomeSpec is persisted to the memory store with status awaiting_confirmation before this event fires. specId is the persisted row id; status is awaiting_confirmation. desiredOutcome, scope, and assumptions are the drafted strings; clarifyingQuestions is an optional array. No decomposition or child dispatch occurs before a human confirms — the run blocks here until the confirm or revise seam is called.

When autopilot is off on a pickup run (or on any interactive run), the SSE stream closes with a done frame after this event. This done is not a permanent terminal — the run remains in_progress at the confirmation gate. After the user confirms the spec, the frontend reopens the stream from the last received sequence and the run continues. When autopilot is on for a pickup run, ScheduleUnattendedConfirm fires automatically so the stream typically stays live without a manual reconnect.

coordinator.outcome_spec.confirmed

Emitted when a human confirms the drafted OutcomeSpec through the confirm seam. The persisted OutcomeSpec advances to status confirmed. specId is the confirmed row id; confirmedBy is the confirming user. In Phase 1 the coordinator run terminates after confirmation (decomposition and child dispatch are Phase 2); a run.completed event follows.

coordinator.work_plan

Emitted when the coordinator decomposes a confirmed OutcomeSpec into a work plan and persists it. workPlanId is the persisted plan id; status begins at planned. subtasks is the array of decomposed units of work, each carrying its subtaskId, title, scope, assignedAgent, selectedModelId, phase, isolation, and status. dependencies is the array of { subtaskId, dependsOnSubtaskId } edges that constrain dispatch order. The work plan is the durable artifact behind the live coordinator.topology graph; subagents read from it once dispatched.

coordinator.topology

Emitted to describe the orchestration graph as it executes. The payload is versioned (version: 1) and comes in two kinds, ordered by a per-coordinator seq counter:

  • A snapshot (kind: "snapshot", seq: 0) fires once when dispatch begins. It carries the complete nodes array and the edges array. There is one coordinator node (id: "coordinator") plus one node per subtask (id: "subtask-{id}"). Each node carries id, kind (coordinator or subtask), subtaskId, status, label, agent, model, childRunId, phase, and isolation. Each edge is { from, to }, meaning the from node (a dependency) must reach assemble_ready/completed before the to node (its dependent) is dispatched.
  • A delta (kind: "delta", seq > 0) fires on every subtask lifecycle transition. It carries a changed array of the node(s) whose state moved (replace by id); edges never change after the snapshot, so deltas omit them. A delta may carry the coordinator node when the work plan's own status transitions.

Clients render directly from these events and never compute topology themselves. Edge direction is always dependency to dependent.

coordinator.graph

The unified coordinator view in the shared GraphDescriptor contract (the same shape returned by GET /api/runs/{id}/graph and emitted per-run as run.workflow_graph), so the frontend's generic renderer draws the coordinator, its fan-out subtask children, and the PLANNED Phase 3 collective-assembly stage with one code path. Emitted on the coordinator stream as a FULL, shape-only snapshot whenever the topology shape changes (a subtask child run is dispatched, or the plan reaches its terminal snapshot). It is built from the work plan (no reflection).

Unlike coordinator.topology, runtime status is NOT baked into the descriptor — it is shape only (consistent with run.workflow_graph); project status separately from the subtask.* / coordinator.topology streams. The payload is a GraphDescriptor with variant: "coordinator", graph_id: "coordinator:{coordinatorRunId}", start_node_id: "coordinator":

  • Node coordinator (node_type: "agent", role: "coordinator", kind: "live").
  • One plan:subtask-{id} node per subtask (node_type: "subtask", kind: "live") carrying optional agent, model, phase, isolation, child_run_id fields (omitted when null) and a child_graph_ref of run:{childRunId} once dispatched (null until then) so the child's own graph can be expanded via GET /api/runs/{childRunId}/graph.
  • PLANNED collective-assembly chain (kind: "planned"): planned:assembly-rai (agent) → planned:assembly-review (gate) → planned:assembly-merge (action) → planned:assembly-scribe (agent).
  • Edges: coordinator → each root subtask; dependency edges between subtasks; each terminal (leaf) subtask → planned:assembly-rai; then the assembly chain. Two loopback back-edges (loopback: true) close the cycle — planned:assembly-raicoordinator and planned:assembly-reviewcoordinator — reflecting that an RAI flag or a review request-changes re-dispatches affected subtasks through the coordinator; all forward edges are loopback: false, and loopback edges are always direct and excluded from the fan-out/fan-in degree counts. coordinator.topology remains emitted alongside for existing consumers; coordinator.graph is the unified-contract event.

subtask.*

The subtask.dispatched, subtask.running, subtask.assemble_ready, subtask.rai_flagged, subtask.completed, and subtask.failed events track a single subtask's child run through its lifecycle on the coordinator stream. Each carries subtaskId, childRunId, assignedAgent, selectedModelId, and status. The subtask status advances pending -> dispatched -> running -> {assemble_ready | rai_flagged | completed | failed}. Each subtask.* event is paired with a coordinator.topology delta for the changed node, so observers can choose either the granular per-subtask signal or the graph view.

coordinator.steering

Emitted when a steering directive is created through POST /api/runs/{id}/steer and as it changes state. directiveId identifies the directive; kind is stop, redirect, or amend; targetChildRunId is the targeted child run, or null for a broadcast to every active child; instruction is the direction relayed to the subagent(s); status advances pending -> queued -> relayed -> applied. A stop collapses to applied essentially immediately because it cancels the in-flight turn's token. A redirect or amend reaches applied only at the targeted subagent's next turn boundary, so observers should surface it as queued until then. Pause is not supported in Phase 2.

coordinator.children_complete and coordinator.assembly_*

Phase 3 collective assembly runs ONE pipeline over the COMBINED output of all child runs, then flows back to the coordinator. Child output is git state, not in-memory text: each child commits to its own worktree branch. When every child subtask reaches a terminal status, the coordinator emits coordinator.children_complete and moves the work plan to awaiting_assembly.

A single background pipeline then drives the collective stages, each emitting a paired coordinator.graph so its planned assembly node flips to kind: "live":

  1. Exactly-once claim — a DB compare-and-swap transitions awaiting_assembly → assembling; only the winner proceeds. coordinator.assembly_started carries the integrationBranch name (agentweaver/integration/{coordinatorRunId}) and subtaskCount.
  2. Eligibility gate (no partial assembly) — every subtask must be assembly-eligible (assemble_ready, or completed with no changes). If any is failed / rai_flagged / pending / blocked, or merging the eligible child branches into the integration branch conflicts, the pipeline emits coordinator.assembly_blocked (with reason, and conflictingBranch/conflictingFiles on a conflict) and STOPS — no RAI, no merge.
  3. Integration branch — the eligible child branches are merged in dependency (topological) order off the coordinator's originating branch, producing one aggregate diff + tree hash.
  4. Collective RAI (coordinator.assembly_rai_startedcoordinator.assembly_rai_completed) — one RAI pass over the aggregate diff. It is advisory: it never hard-blocks, but raiSafetyFlagged is surfaced to the human reviewer.
  5. One human review gate (coordinator.assembly_review_requested, carrying treeHash and includedSubtaskIds so the UI can render the assembled tree and which subtasks it covers) — the pipeline suspends until a decision arrives via POST /api/runs/{coordinatorRunId}/assembly/review. Approve → coordinator.assembly_review_approved. Request changes → coordinator.assembly_changes_requested (the coordinator infers the affected children from the reviewer's target_files ∪ path tokens in feedback, intersected with each child's touched-files and expanded to dependents — redispatchSubtaskIds, inferredFiles, fellBackToAll), resets those subtasks to pending, returns the plan to dispatching, and re-dispatches. A pure decline is the terminal assembly_declined status.
  6. One merge (coordinator.assembly_merge_startedcoordinator.assembly_merge_completed with commitHash, or coordinator.assembly_merge_failed with reason/conflictingFiles).
  7. One scribe (coordinator.assembly_scribe_startedcoordinator.assembly_scribe_completed) — best-effort; a scribe failure does not fail the already-merged assembly.
  8. Completioncoordinator.assembly_completed with the integrationBranch and commitHash; the work plan reaches complete.

Work-plan status flows dispatching → awaiting_assembly → assembling → in_review → assembling (during merge/scribe after approval) → complete, plus the parked/terminal states assembly_blocked, assembly_failed, and assembly_declined. On every terminal assembly path the coordinator run itself is moved to a terminal RunStatus that carries a human-readable result (the reason): assembly_blocked: <reason> (Failed), assembly_merge_failed: <reason> (MergeFailed), assembly_declined (Declined), assembly_error: <message> (Failed, unexpected fault), or assembly_complete (Completed). This result is surfaced as the statusReason on GET /api/runs/{coordinatorRunId}/work-plan and as the result on the run summary/detail, so the UI never shows a bare "Failed" with no explanation. The per-child subtask.rai_flagged events and the collective coordinator.assembly_rai_* events are DISTINCT: the former is each agent stream's own RAI, the latter is the single RAI over the combined output. All events carry a monotonic seq on the coordinator stream.

Model-assisted casting

Creating a casting proposal in free_text or analysis mode starts a MAF run on GitHub Copilot. That run emits events under the same event model as a regular agentweaver run — the same envelope, the same event types, and the same SSE endpoint.

The run_id returned in the POST /api/projects/{id}/casting/proposals response identifies that run. Stream its events from GET /api/runs/{run_id}/stream to observe the model's reasoning as it builds the proposal.

The casting wizard in the web UI streams these events during the review step using the same timeline rendering as the watch screen. MCP clients receive them as progress notifications during a team_cast tool call.

No .squad/ files are written during this run. The run produces a proposal, not a commit. Files are only written after the user confirms the proposal through POST /api/projects/{id}/casting/proposals/{pid}/confirm.

ask_question bubbling

Any agent (a worker, or the coordinator during decomposition) can call the ask_question tool to surface a clarifying question or permission request instead of silently guessing. On invoke the tool emits agent.question_asked (carrying a generated requestId and the question) on the run's own stream, then suspends inside the tool call on the per-run question gate.

The wait is resolved by POST /api/runs/{id}/questions/{requestId}/answer with body { "answer": "..." }. When an answer arrives (or the wait times out after 30 minutes), the tool emits agent.question_answered (requestId, answer, timedOut) and returns the answer text to the model so it can continue. On timeout the model receives a proceed-with-best-judgement instruction rather than hanging. The question gate is cleared on run completion alongside the tool-approval gate; any still-pending wait resolves to a proceed instruction.

For a coordinator CHILD run, the coordinator's child watcher (CoordinatorDispatchService.ObserveChildAsync) re-projects the child's agent.question_asked onto the COORDINATOR stream as coordinator.child_question, and the child's tool.approval_required as coordinator.child_approval_required, each carrying childRunId + subtaskId + requestId. The answer/approval flows back to the CHILD run: answer via POST /api/runs/{childRunId}/questions/{requestId}/answer, approval via the existing POST /api/runs/{childRunId}/tool-approvals / tool-denials. Re-projection does not affect terminal-event mapping.

Scenario-mode proposals resolve without a model run. The run_id field in their proposal response is null.