rename: open_threads → loose_ends + cancel_thread → cancel_loose_end across wire / tools / web ui

This commit is contained in:
damocles 2026-05-18 18:22:49 +02:00
parent b1d0a62cb9
commit 6e23d087d2
16 changed files with 152 additions and 139 deletions

View file

@ -53,11 +53,12 @@ hive-c0re/ host daemon + CLI (one binary, subcommand-dispatched)
transient/sockets) + tombstone enumeration + transient/sockets) + tombstone enumeration +
kick_agent + notify_agent (helper-event push) + kick_agent + notify_agent (helper-event push) +
last_containers cache + rescan_and_emit diff helper last_containers cache + rescan_and_emit diff helper
src/open_threads.rs loose-ends aggregator (pending approvals + src/loose_ends.rs loose-ends aggregator (pending approvals +
unanswered questions) — for_agent (filtered) and unanswered questions + pending reminders) —
hive_wide (manager surface). Backs for_agent (filtered) and hive_wide (manager
AgentRequest::GetOpenThreads + ManagerRequest:: surface). Backs AgentRequest::GetLooseEnds +
GetOpenThreads (the get_open_threads MCP tool). ManagerRequest::GetLooseEnds (the
get_loose_ends MCP tool).
src/actions.rs approve/deny/destroy (transient-aware) src/actions.rs approve/deny/destroy (transient-aware)
src/auto_update.rs startup rebuild scan + ensure_manager + src/auto_update.rs startup rebuild scan + ensure_manager +
meta::lock_update_hyperhive bump meta::lock_update_hyperhive bump

View file

@ -40,7 +40,7 @@ how often the friction bites in normal use.
into the prompt builder in `hive-ag3nt::turn.rs`. Even better: add a into the prompt builder in `hive-ag3nt::turn.rs`. Even better: add a
one-shot `recv_batch(max: u32)` MCP tool that returns up to `max` one-shot `recv_batch(max: u32)` MCP tool that returns up to `max`
pending messages in a single round-trip. pending messages in a single round-trip.
- ~~**Self-management of own asks + reminders**~~ ✓ landed — unified with `get_open_threads` rather than a separate listing surface. `OpenThread` gained a `Reminder { id, owner, message, due_at, age_seconds }` variant (sub-agent flavour filters by `owner == self`; manager unfiltered). New `mcp__hyperhive__cancel_thread(kind, id)` on both surfaces — `kind` is `"question"` (asker gets `[cancelled by <self>]` answer, unblocks) or `"reminder"` (hard-deleted before fire). Auth: sub-agent must own the row; manager bypasses for hive-wide cleanup. New helpers `OperatorQuestions::cancel` + `Broker::cancel_reminder_as` push the auth check down so both flavours stay aligned. Shared dispatch in `hive-c0re/src/questions.rs::handle_cancel_thread`. - ~~**Self-management of own asks + reminders**~~ ✓ landed — unified with `get_loose_ends` (renamed from `get_open_threads` per the naming pass). `LooseEnd` enum (renamed from `OpenThread`) gained a `Reminder { id, owner, message, due_at, age_seconds }` variant (sub-agent flavour filters by `owner == self`; manager unfiltered). New `mcp__hyperhive__cancel_loose_end(kind, id)` on both surfaces — `kind` is `"question"` (asker gets `[cancelled by <self>]` answer, unblocks) or `"reminder"` (hard-deleted before fire). Auth: sub-agent must own the row; manager bypasses for hive-wide cleanup. New helpers `OperatorQuestions::cancel` + `Broker::cancel_reminder_as` push the auth check down so both flavours stay aligned. Shared dispatch in `hive-c0re/src/questions.rs::handle_cancel_loose_end`. Per-agent web UI's `/api/open-threads``/api/loose-ends` too, with reminder-row rendering added.
- **Optional `in_reply_to: <msg_id>` on send** — pure wire addition; no - **Optional `in_reply_to: <msg_id>` on send** — pure wire addition; no
behavioural change. The dashboard could render conversation threads behavioural change. The dashboard could render conversation threads
(already wants this for the agent-to-agent question UI in the (already wants this for the agent-to-agent question UI in the

View file

@ -393,43 +393,43 @@
} }
renderStateBadge(); renderStateBadge();
} }
// Open-threads section: same data the get_open_threads MCP tool // Loose-ends section: same data the get_loose_ends MCP tool
// returns. Best-effort fetch on cold load + after every turn_end // returns. Best-effort fetch on cold load + after every turn_end
// (a turn likely answered or asked something). Silent failure // (a turn likely answered or asked something). Silent failure
// keeps the section hidden rather than surfacing an empty banner. // keeps the section hidden rather than surfacing an empty banner.
let lastOpenThreadsCount = 0; let lastLooseEndsCount = 0;
async function refreshOpenThreads() { async function refreshLooseEnds() {
try { try {
const resp = await fetch('/api/open-threads'); const resp = await fetch('/api/loose-ends');
if (!resp.ok) { if (!resp.ok) {
renderOpenThreads([]); renderLooseEnds([]);
return; return;
} }
const data = await resp.json(); const data = await resp.json();
renderOpenThreads(data.threads || []); renderLooseEnds(data.loose_ends || []);
} catch (err) { } catch (err) {
console.warn('open-threads fetch failed', err); console.warn('loose-ends fetch failed', err);
renderOpenThreads([]); renderLooseEnds([]);
} }
} }
function renderOpenThreads(threads) { function renderLooseEnds(threads) {
const root = $('open-threads-section'); const root = $('loose-ends-section');
const list = $('open-threads-list'); const list = $('loose-ends-list');
const summary = $('open-threads-summary'); const summary = $('loose-ends-summary');
if (!root || !list || !summary) return; if (!root || !list || !summary) return;
if (!threads.length) { if (!threads.length) {
root.hidden = true; root.hidden = true;
lastOpenThreadsCount = 0; lastLooseEndsCount = 0;
return; return;
} }
root.hidden = false; root.hidden = false;
summary.textContent = 'open threads · ' + threads.length; summary.textContent = 'loose ends · ' + threads.length;
list.innerHTML = ''; list.innerHTML = '';
// Auto-expand on first appearance of any open thread so the // Auto-expand on first appearance of any open thread so the
// operator notices new loose ends; collapse only on operator // operator notices new loose ends; collapse only on operator
// click (sticky after that). // click (sticky after that).
if (lastOpenThreadsCount === 0) root.open = true; if (lastLooseEndsCount === 0) root.open = true;
lastOpenThreadsCount = threads.length; lastLooseEndsCount = threads.length;
const fmtAge = (s) => { const fmtAge = (s) => {
if (s < 60) return s + 's'; if (s < 60) return s + 's';
if (s < 3600) return Math.floor(s / 60) + 'm'; if (s < 3600) return Math.floor(s / 60) + 'm';
@ -455,6 +455,18 @@
el('span', { class: 'inbox-ts' }, fmtAge(t.age_seconds || 0) + ' ago'), el('span', { class: 'inbox-ts' }, fmtAge(t.age_seconds || 0) + ' ago'),
el('div', { class: 'inbox-body' }, t.question || ''), el('div', { class: 'inbox-body' }, t.question || ''),
); );
} else if (t.kind === 'reminder') {
// due_at is an absolute unix-seconds value; show time-until-fire
// (negative when overdue, fmtAge handles 0/positive case here).
const now = Math.floor(Date.now() / 1000);
const dueIn = (t.due_at || 0) - now;
const dueLabel = dueIn >= 0 ? 'in ' + fmtAge(dueIn) : fmtAge(-dueIn) + ' overdue';
li.append(
el('span', { class: 'inbox-from' }, '⏰ reminder #' + t.id), ' ',
el('span', { class: 'inbox-sep' }, t.owner + ' · due ' + dueLabel), ' ',
el('span', { class: 'inbox-ts' }, 'scheduled ' + fmtAge(t.age_seconds || 0) + ' ago'),
el('div', { class: 'inbox-body' }, t.message || ''),
);
} else { } else {
li.append(el('span', { class: 'inbox-body' }, JSON.stringify(t))); li.append(el('span', { class: 'inbox-body' }, JSON.stringify(t)));
} }
@ -614,7 +626,7 @@
// Open-threads aren't part of /api/state (kept on the broker // Open-threads aren't part of /api/state (kept on the broker
// db, fetched via the per-agent socket). Cold-load fetches // db, fetched via the per-agent socket). Cold-load fetches
// it here; turn_end refreshes it via the renderer below. // it here; turn_end refreshes it via the renderer below.
refreshOpenThreads(); refreshLooseEnds();
// Skip the re-render if nothing structurally changed. The most // Skip the re-render if nothing structurally changed. The most
// common case is `online` polling itself — without this guard, the // common case is `online` polling itself — without this guard, the
// operator's <input value> gets clobbered every cycle. // operator's <input value> gets clobbered every cycle.
@ -956,7 +968,7 @@
} else { } else {
setBannerActive(false); setState('idle'); setBannerActive(false); setState('idle');
// Likely answered/asked/scheduled something — refresh. // Likely answered/asked/scheduled something — refresh.
refreshOpenThreads(); refreshLooseEnds();
} }
const cls = ev.ok ? 'turn-end-ok' : 'turn-end-fail'; const cls = ev.ok ? 'turn-end-ok' : 'turn-end-fail';
api.row(cls, api.row(cls,

View file

@ -29,9 +29,9 @@
<ul id="inbox-list"></ul> <ul id="inbox-list"></ul>
</details> </details>
<details id="open-threads-section" class="agent-inbox" hidden> <details id="loose-ends-section" class="agent-inbox" hidden>
<summary><span id="open-threads-summary">open threads</span></summary> <summary><span id="loose-ends-summary">loose ends</span></summary>
<ul id="open-threads-list"></ul> <ul id="loose-ends-list"></ul>
</details> </details>
<div class="terminal-wrap"> <div class="terminal-wrap">

View file

@ -7,8 +7,8 @@ Tools (hyperhive surface):
- (some agents only) **extra MCP tools** surfaced as `mcp__<server>__<tool>` — these are agent-specific (matrix client, scraper, db connector, etc.) declared in your `agent.nix` under `hyperhive.extraMcpServers`. Treat them as first-class tools alongside the hyperhive surface; the operator already auto-approved them at deploy time. - (some agents only) **extra MCP tools** surfaced as `mcp__<server>__<tool>` — these are agent-specific (matrix client, scraper, db connector, etc.) declared in your `agent.nix` under `hyperhive.extraMcpServers`. Treat them as first-class tools alongside the hyperhive surface; the operator already auto-approved them at deploy time.
- `mcp__hyperhive__ask(question, options?, multi?, ttl_seconds?, to?)` — surface a structured question to the human operator (default, or `to: "operator"`) OR a peer agent (`to: "<agent-name>"`). Returns immediately with a question id — do NOT wait inline. When the recipient answers, a system message with event `question_answered { id, question, answer, answerer }` lands in your inbox; handle it on a future turn. Use this for clarifications, permission for risky actions, choice between options, or peer Q&A without burning regular inbox slots. `options` is advisory: a short fixed-choice list when applicable, otherwise leave empty for free text. `multi: true` lets the answerer pick multiple (checkboxes), answer comes back comma-joined. `ttl_seconds` auto-cancels with answer `[expired]` (and `answerer: "ttl-watchdog"`) when the decision becomes moot. - `mcp__hyperhive__ask(question, options?, multi?, ttl_seconds?, to?)` — surface a structured question to the human operator (default, or `to: "operator"`) OR a peer agent (`to: "<agent-name>"`). Returns immediately with a question id — do NOT wait inline. When the recipient answers, a system message with event `question_answered { id, question, answer, answerer }` lands in your inbox; handle it on a future turn. Use this for clarifications, permission for risky actions, choice between options, or peer Q&A without burning regular inbox slots. `options` is advisory: a short fixed-choice list when applicable, otherwise leave empty for free text. `multi: true` lets the answerer pick multiple (checkboxes), answer comes back comma-joined. `ttl_seconds` auto-cancels with answer `[expired]` (and `answerer: "ttl-watchdog"`) when the decision becomes moot.
- `mcp__hyperhive__answer(id, answer)` — answer a question that was routed to YOU. You'll see one in your inbox as a `question_asked { id, asker, question, options, multi }` system event when a peer or the manager calls `ask(to: "<your-name>", ...)`. The answer surfaces in the asker's inbox as a `question_answered` event. Strict authorisation: you can only answer questions where you are the declared target. - `mcp__hyperhive__answer(id, answer)` — answer a question that was routed to YOU. You'll see one in your inbox as a `question_asked { id, asker, question, options, multi }` system event when a peer or the manager calls `ask(to: "<your-name>", ...)`. The answer surfaces in the asker's inbox as a `question_answered` event. Strict authorisation: you can only answer questions where you are the declared target.
- `mcp__hyperhive__get_open_threads()` — list your loose ends: unanswered questions where you're asker (waiting on someone) or target (owing a reply), plus reminders you've scheduled that haven't fired. No args, cheap server-side sweep. Useful at turn start to remember what's outstanding without scanning inbox archaeology. - `mcp__hyperhive__get_loose_ends()` — list your loose ends: unanswered questions where you're asker (waiting on someone) or target (owing a reply), plus reminders you've scheduled that haven't fired. No args, cheap server-side sweep. Useful at turn start to remember what's outstanding without scanning inbox archaeology.
- `mcp__hyperhive__cancel_thread(kind, id)` — cancel one of your own open threads. `kind` is `"question"` (the asker — you, in this case — gets a `[cancelled by <you>]` answer so the waiter unblocks) or `"reminder"` (hard-deleted before it fires). `id` from the matching `get_open_threads` row or the original submission reply. - `mcp__hyperhive__cancel_loose_end(kind, id)` — cancel one of your own open threads. `kind` is `"question"` (the asker — you, in this case — gets a `[cancelled by <you>]` answer so the waiter unblocks) or `"reminder"` (hard-deleted before it fires). `id` from the matching `get_loose_ends` row or the original submission reply.
- `mcp__hyperhive__whoami()` — self-introspection: returns your canonical agent name (from socket identity, not the prompt-substituted label), role, and current hyperhive rev. No args. Use it when you want a trustworthy identity stamp for state files, commit messages, or cross-agent attribution that won't drift across renames. - `mcp__hyperhive__whoami()` — self-introspection: returns your canonical agent name (from socket identity, not the prompt-substituted label), role, and current hyperhive rev. No args. Use it when you want a trustworthy identity stamp for state files, commit messages, or cross-agent attribution that won't drift across renames.
Need new packages, env vars, or other NixOS config for yourself? You can't edit your own config directly — message the manager (recipient `manager`) describing what you need + why. The manager evaluates the request (it doesn't rubber-stamp), edits `/agents/{label}/config/agent.nix` on your behalf, commits, and submits an approval that the operator can accept on the dashboard; on approve hive-c0re rebuilds your container with the new config. Need new packages, env vars, or other NixOS config for yourself? You can't edit your own config directly — message the manager (recipient `manager`) describing what you need + why. The manager evaluates the request (it doesn't rubber-stamp), edits `/agents/{label}/config/agent.nix` on your behalf, commits, and submits an approval that the operator can accept on the dashboard; on approve hive-c0re rebuilds your container with the new config.

View file

@ -12,8 +12,8 @@ Tools (hyperhive surface):
- `mcp__hyperhive__request_apply_commit(agent, commit_ref, description?)` — submit a config change for any agent (`hm1nd` for self) for operator approval. Pass an optional `description` and it appears on the dashboard approval card so the operator knows what changed without opening the diff. At submit time hive-c0re fetches your commit into the agent's applied repo and pins it as `proposal/<id>`; from that moment your proposed-side commit can be amended or force-pushed freely without changing what the operator will build. - `mcp__hyperhive__request_apply_commit(agent, commit_ref, description?)` — submit a config change for any agent (`hm1nd` for self) for operator approval. Pass an optional `description` and it appears on the dashboard approval card so the operator knows what changed without opening the diff. At submit time hive-c0re fetches your commit into the agent's applied repo and pins it as `proposal/<id>`; from that moment your proposed-side commit can be amended or force-pushed freely without changing what the operator will build.
- `mcp__hyperhive__ask(question, options?, multi?, ttl_seconds?, to?)` — surface a structured question to the operator (default, or `to: "operator"`) OR a sub-agent (`to: "<agent-name>"`). Returns immediately with a question id; the answer arrives later as a system `question_answered { id, question, answer, answerer }` event in your inbox. Options are advisory: the dashboard always lets the operator type a free-text answer in addition. Set `multi: true` to render options as checkboxes (operator can pick multiple); the answer comes back as `, `-separated. Set `ttl_seconds` to auto-cancel after a deadline (capped at 6h server-side) — on expiry the answer is `[expired]` and `answerer` is `"ttl-watchdog"`. Do not poll inside the same turn — finish the current work and react when the event lands. - `mcp__hyperhive__ask(question, options?, multi?, ttl_seconds?, to?)` — surface a structured question to the operator (default, or `to: "operator"`) OR a sub-agent (`to: "<agent-name>"`). Returns immediately with a question id; the answer arrives later as a system `question_answered { id, question, answer, answerer }` event in your inbox. Options are advisory: the dashboard always lets the operator type a free-text answer in addition. Set `multi: true` to render options as checkboxes (operator can pick multiple); the answer comes back as `, `-separated. Set `ttl_seconds` to auto-cancel after a deadline (capped at 6h server-side) — on expiry the answer is `[expired]` and `answerer` is `"ttl-watchdog"`. Do not poll inside the same turn — finish the current work and react when the event lands.
- `mcp__hyperhive__answer(id, answer)` — answer a question that was routed to YOU (a sub-agent did `ask(to: "manager", ...)`). The triggering event in your inbox is `question_asked { id, asker, question, options, multi }`. The answer surfaces in the asker's inbox as a `question_answered` event. - `mcp__hyperhive__answer(id, answer)` — answer a question that was routed to YOU (a sub-agent did `ask(to: "manager", ...)`). The triggering event in your inbox is `question_asked { id, asker, question, options, multi }`. The answer surfaces in the asker's inbox as a `question_answered` event.
- `mcp__hyperhive__get_open_threads()` — hive-wide loose ends: every pending approval + every unanswered question + every pending reminder across the swarm. Cheap server-side sweep, no args. Use to find stalled threads (sub-agent A asked B something three days ago and B never answered) before they rot. - `mcp__hyperhive__get_loose_ends()` — hive-wide loose ends: every pending approval + every unanswered question + every pending reminder across the swarm. Cheap server-side sweep, no args. Use to find stalled threads (sub-agent A asked B something three days ago and B never answered) before they rot.
- `mcp__hyperhive__cancel_thread(kind, id)` — cancel any question or reminder in the swarm (manager bypasses the owner check used on sub-agents). Use for hive-wide cleanup when a sub-agent is offline / can't withdraw its own ask / reminder. - `mcp__hyperhive__cancel_loose_end(kind, id)` — cancel any question or reminder in the swarm (manager bypasses the owner check used on sub-agents). Use for hive-wide cleanup when a sub-agent is offline / can't withdraw its own ask / reminder.
- `mcp__hyperhive__whoami()` — self-introspection: canonical name (`manager`), role, current hyperhive rev. No args. Useful for boot announcements and cross-agent attribution that won't drift across config reloads. - `mcp__hyperhive__whoami()` — self-introspection: canonical name (`manager`), role, current hyperhive rev. No args. Useful for boot announcements and cross-agent attribution that won't drift across config reloads.
Approval boundary: lifecycle ops on *existing* sub-agents (`kill`, `start`, `restart`) are at your discretion — no operator approval. *Creating* a new agent (`request_spawn`) and *changing* any agent's config (`request_apply_commit`) still go through the approval queue. The operator only signs off on changes; you run the day-to-day. Approval boundary: lifecycle ops on *existing* sub-agents (`kill`, `start`, `restart`) are at your discretion — no operator approval. *Creating* a new agent (`request_spawn`) and *changing* any agent's config (`request_apply_commit`) still go through the approval queue. The operator only signs off on changes; you run the day-to-day.

View file

@ -240,7 +240,7 @@ async fn serve(
| AgentResponse::Status { .. } | AgentResponse::Status { .. }
| AgentResponse::Recent { .. } | AgentResponse::Recent { .. }
| AgentResponse::QuestionQueued { .. } | AgentResponse::QuestionQueued { .. }
| AgentResponse::OpenThreads { .. } | AgentResponse::LooseEnds { .. }
| AgentResponse::PendingRemindersCount { .. } | AgentResponse::PendingRemindersCount { .. }
| AgentResponse::Whoami { .. }, | AgentResponse::Whoami { .. },
) => { ) => {
@ -320,11 +320,11 @@ fn now_unix() -> i64 {
async fn fetch_agent_post_turn_counts(socket: &Path) -> (Option<u64>, Option<u64>) { async fn fetch_agent_post_turn_counts(socket: &Path) -> (Option<u64>, Option<u64>) {
let threads = match client::request::<_, AgentResponse>( let threads = match client::request::<_, AgentResponse>(
socket, socket,
&AgentRequest::GetOpenThreads, &AgentRequest::GetLooseEnds,
) )
.await .await
{ {
Ok(AgentResponse::OpenThreads { threads }) => u64::try_from(threads.len()).ok(), Ok(AgentResponse::LooseEnds { loose_ends }) => u64::try_from(loose_ends.len()).ok(),
_ => None, _ => None,
}; };
let reminders = match client::request::<_, AgentResponse>( let reminders = match client::request::<_, AgentResponse>(

View file

@ -208,7 +208,7 @@ async fn serve(
| ManagerResponse::QuestionQueued { .. } | ManagerResponse::QuestionQueued { .. }
| ManagerResponse::Recent { .. } | ManagerResponse::Recent { .. }
| ManagerResponse::Logs { .. } | ManagerResponse::Logs { .. }
| ManagerResponse::OpenThreads { .. } | ManagerResponse::LooseEnds { .. }
| ManagerResponse::PendingRemindersCount { .. } | ManagerResponse::PendingRemindersCount { .. }
| ManagerResponse::Whoami { .. }, | ManagerResponse::Whoami { .. },
) => { ) => {
@ -259,11 +259,11 @@ fn now_unix() -> i64 {
async fn fetch_manager_post_turn_counts(socket: &Path) -> (Option<u64>, Option<u64>) { async fn fetch_manager_post_turn_counts(socket: &Path) -> (Option<u64>, Option<u64>) {
let threads = match client::request::<_, ManagerResponse>( let threads = match client::request::<_, ManagerResponse>(
socket, socket,
&ManagerRequest::GetOpenThreads, &ManagerRequest::GetLooseEnds,
) )
.await .await
{ {
Ok(ManagerResponse::OpenThreads { threads }) => u64::try_from(threads.len()).ok(), Ok(ManagerResponse::LooseEnds { loose_ends }) => u64::try_from(loose_ends.len()).ok(),
_ => None, _ => None,
}; };
let reminders = match client::request::<_, ManagerResponse>( let reminders = match client::request::<_, ManagerResponse>(

View file

@ -40,7 +40,7 @@ pub enum SocketReply {
QuestionQueued(i64), QuestionQueued(i64),
Recent(Vec<hive_sh4re::InboxRow>), Recent(Vec<hive_sh4re::InboxRow>),
Logs(String), Logs(String),
OpenThreads(Vec<hive_sh4re::OpenThread>), LooseEnds(Vec<hive_sh4re::LooseEnd>),
PendingRemindersCount(u64), PendingRemindersCount(u64),
Whoami { Whoami {
name: String, name: String,
@ -59,7 +59,7 @@ impl From<hive_sh4re::AgentResponse> for SocketReply {
hive_sh4re::AgentResponse::Status { unread } => Self::Status(unread), hive_sh4re::AgentResponse::Status { unread } => Self::Status(unread),
hive_sh4re::AgentResponse::Recent { rows } => Self::Recent(rows), hive_sh4re::AgentResponse::Recent { rows } => Self::Recent(rows),
hive_sh4re::AgentResponse::QuestionQueued { id } => Self::QuestionQueued(id), hive_sh4re::AgentResponse::QuestionQueued { id } => Self::QuestionQueued(id),
hive_sh4re::AgentResponse::OpenThreads { threads } => Self::OpenThreads(threads), hive_sh4re::AgentResponse::LooseEnds { loose_ends } => Self::LooseEnds(loose_ends),
hive_sh4re::AgentResponse::PendingRemindersCount { count } => { hive_sh4re::AgentResponse::PendingRemindersCount { count } => {
Self::PendingRemindersCount(count) Self::PendingRemindersCount(count)
} }
@ -87,7 +87,7 @@ impl From<hive_sh4re::ManagerResponse> for SocketReply {
hive_sh4re::ManagerResponse::QuestionQueued { id } => Self::QuestionQueued(id), hive_sh4re::ManagerResponse::QuestionQueued { id } => Self::QuestionQueued(id),
hive_sh4re::ManagerResponse::Recent { rows } => Self::Recent(rows), hive_sh4re::ManagerResponse::Recent { rows } => Self::Recent(rows),
hive_sh4re::ManagerResponse::Logs { content } => Self::Logs(content), hive_sh4re::ManagerResponse::Logs { content } => Self::Logs(content),
hive_sh4re::ManagerResponse::OpenThreads { threads } => Self::OpenThreads(threads), hive_sh4re::ManagerResponse::LooseEnds { loose_ends } => Self::LooseEnds(loose_ends),
hive_sh4re::ManagerResponse::PendingRemindersCount { count } => { hive_sh4re::ManagerResponse::PendingRemindersCount { count } => {
Self::PendingRemindersCount(count) Self::PendingRemindersCount(count)
} }
@ -128,16 +128,16 @@ pub fn format_recv(resp: Result<SocketReply, anyhow::Error>) -> String {
} }
} }
/// Format helper for `get_open_threads`: renders a short bulleted list /// Format helper for `get_loose_ends`: renders a short bulleted list
/// of pending approvals + questions. Empty list collapses to a clear /// of pending approvals + questions. Empty list collapses to a clear
/// marker so claude doesn't go hunting for a payload that isn't there. /// marker so claude doesn't go hunting for a payload that isn't there.
pub fn format_open_threads(resp: Result<SocketReply, anyhow::Error>) -> String { pub fn format_loose_ends(resp: Result<SocketReply, anyhow::Error>) -> String {
use std::fmt::Write as _; use std::fmt::Write as _;
let threads = match resp { let threads = match resp {
Ok(SocketReply::OpenThreads(t)) => t, Ok(SocketReply::LooseEnds(t)) => t,
Ok(SocketReply::Err(m)) => return format!("get_open_threads failed: {m}"), Ok(SocketReply::Err(m)) => return format!("get_loose_ends failed: {m}"),
Ok(other) => return format!("get_open_threads unexpected response: {other:?}"), Ok(other) => return format!("get_loose_ends unexpected response: {other:?}"),
Err(e) => return format!("get_open_threads transport error: {e:#}"), Err(e) => return format!("get_loose_ends transport error: {e:#}"),
}; };
if threads.is_empty() { if threads.is_empty() {
return "(no open threads)".to_owned(); return "(no open threads)".to_owned();
@ -145,7 +145,7 @@ pub fn format_open_threads(resp: Result<SocketReply, anyhow::Error>) -> String {
let mut out = format!("{} open thread(s):\n", threads.len()); let mut out = format!("{} open thread(s):\n", threads.len());
for t in &threads { for t in &threads {
match t { match t {
hive_sh4re::OpenThread::Approval { hive_sh4re::LooseEnd::Approval {
id, id,
agent, agent,
commit_ref, commit_ref,
@ -161,7 +161,7 @@ pub fn format_open_threads(resp: Result<SocketReply, anyhow::Error>) -> String {
"- approval #{id} ({agent} @ {commit_ref}, {age_seconds}s old){desc}" "- approval #{id} ({agent} @ {commit_ref}, {age_seconds}s old){desc}"
); );
} }
hive_sh4re::OpenThread::Question { hive_sh4re::LooseEnd::Question {
id, id,
asker, asker,
target, target,
@ -174,7 +174,7 @@ pub fn format_open_threads(resp: Result<SocketReply, anyhow::Error>) -> String {
"- question #{id} ({asker} → {to}, {age_seconds}s old): {question}" "- question #{id} ({asker} → {to}, {age_seconds}s old): {question}"
); );
} }
hive_sh4re::OpenThread::Reminder { hive_sh4re::LooseEnd::Reminder {
id, id,
owner, owner,
message, message,
@ -191,16 +191,16 @@ pub fn format_open_threads(resp: Result<SocketReply, anyhow::Error>) -> String {
out out
} }
/// Parse the user-facing `kind` string for `cancel_thread` into the /// Parse the user-facing `kind` string for `cancel_loose_end` into the
/// wire enum. Accepts a small alias set so claude doesn't have to /// wire enum. Accepts a small alias set so claude doesn't have to
/// remember the exact spelling (`"q"` / `"r"` shorthand falls out /// remember the exact spelling (`"q"` / `"r"` shorthand falls out
/// for free). /// for free).
fn parse_cancel_kind(raw: &str) -> Result<hive_sh4re::CancelThreadKind, String> { fn parse_loose_end_kind(raw: &str) -> Result<hive_sh4re::CancelLooseEndKind, String> {
match raw.trim().to_ascii_lowercase().as_str() { match raw.trim().to_ascii_lowercase().as_str() {
"question" | "q" => Ok(hive_sh4re::CancelThreadKind::Question), "question" | "q" => Ok(hive_sh4re::CancelLooseEndKind::Question),
"reminder" | "r" => Ok(hive_sh4re::CancelThreadKind::Reminder), "reminder" | "r" => Ok(hive_sh4re::CancelLooseEndKind::Reminder),
other => Err(format!( other => Err(format!(
"cancel_thread: unknown kind '{other}' (expected \"question\" or \"reminder\")" "cancel_loose_end: unknown kind '{other}' (expected \"question\" or \"reminder\")"
)), )),
} }
} }
@ -454,13 +454,13 @@ impl AgentServer {
sweep, no args. Useful at turn start to remember what you owe / what's owed to \ sweep, no args. Useful at turn start to remember what you owe / what's owed to \
you without scrolling inbox history. Output is a short bulleted list with ids, \ you without scrolling inbox history. Output is a short bulleted list with ids, \
ages in seconds, and the relevant context. Each `question` or `reminder` row \ ages in seconds, and the relevant context. Each `question` or `reminder` row \
can be cancelled by passing its id + kind to `cancel_thread`. Empty result \ can be cancelled by passing its id + kind to `cancel_loose_end`. Empty result \
is reported clearly." is reported clearly."
)] )]
async fn get_open_threads(&self) -> String { async fn get_loose_ends(&self) -> String {
run_tool_envelope("get_open_threads", String::new(), async move { run_tool_envelope("get_loose_ends", String::new(), async move {
let (resp, retries) = self.dispatch(hive_sh4re::AgentRequest::GetOpenThreads).await; let (resp, retries) = self.dispatch(hive_sh4re::AgentRequest::GetLooseEnds).await;
annotate_retries(format_open_threads(resp), retries) annotate_retries(format_loose_ends(resp), retries)
}) })
.await .await
} }
@ -485,24 +485,24 @@ impl AgentServer {
description = "Cancel an open thread you own — a `question` you asked (the \ description = "Cancel an open thread you own — a `question` you asked (the \
asker gets `[cancelled by <you>]` as the answer and unblocks) or a `reminder` \ asker gets `[cancelled by <you>]` as the answer and unblocks) or a `reminder` \
you scheduled (hard-deleted before it fires). `kind` is `\"question\"` or \ you scheduled (hard-deleted before it fires). `kind` is `\"question\"` or \
`\"reminder\"`; `id` is the row id from the matching `get_open_threads` entry \ `\"reminder\"`; `id` is the row id from the matching `get_loose_ends` entry \
or the `question_queued` reply you got when you submitted. Auth: you can only \ or the `question_queued` reply you got when you submitted. Auth: you can only \
cancel rows where you're the asker / owner. Returns `ok` or an error string." cancel rows where you're the asker / owner. Returns `ok` or an error string."
)] )]
async fn cancel_thread(&self, Parameters(args): Parameters<CancelThreadArgs>) -> String { async fn cancel_loose_end(&self, Parameters(args): Parameters<CancelLooseEndArgs>) -> String {
let log = format!("{args:?}"); let log = format!("{args:?}");
let kind_label = args.kind.clone(); let kind_label = args.kind.clone();
let id = args.id; let id = args.id;
run_tool_envelope("cancel_thread", log, async move { run_tool_envelope("cancel_loose_end", log, async move {
let kind = match parse_cancel_kind(&args.kind) { let kind = match parse_loose_end_kind(&args.kind) {
Ok(k) => k, Ok(k) => k,
Err(e) => return e, Err(e) => return e,
}; };
let (resp, retries) = self let (resp, retries) = self
.dispatch(hive_sh4re::AgentRequest::CancelThread { kind, id }) .dispatch(hive_sh4re::AgentRequest::CancelLooseEnd { kind, id })
.await; .await;
annotate_retries( annotate_retries(
format_ack(resp, "cancel_thread", format!("cancelled {kind_label} {id}")), format_ack(resp, "cancel_loose_end", format!("cancelled {kind_label} {id}")),
retries, retries,
) )
}) })
@ -654,13 +654,13 @@ pub struct AnswerArgs {
} }
#[derive(Debug, serde::Deserialize, schemars::JsonSchema)] #[derive(Debug, serde::Deserialize, schemars::JsonSchema)]
pub struct CancelThreadArgs { pub struct CancelLooseEndArgs {
/// Which kind of thread to cancel — `"question"` for an open /// Which kind of thread to cancel — `"question"` for an open
/// `ask` that's still waiting on an answer, `"reminder"` for a /// `ask` that's still waiting on an answer, `"reminder"` for a
/// scheduled `remind` that hasn't fired yet. Use the `kind` /// scheduled `remind` that hasn't fired yet. Use the `kind`
/// field straight off the `get_open_threads` row. /// field straight off the `get_loose_ends` row.
pub kind: String, pub kind: String,
/// Row id from the matching `get_open_threads` entry (or the /// Row id from the matching `get_loose_ends` entry (or the
/// `question_queued` reply when you submitted it). /// `question_queued` reply when you submitted it).
pub id: i64, pub id: i64,
} }
@ -995,14 +995,14 @@ impl ManagerServer {
approvals stuck waiting on the operator, reminders piling up on an offline \ approvals stuck waiting on the operator, reminders piling up on an offline \
agent, etc. No args. The sub-agent flavour only returns the agent's own \ agent, etc. No args. The sub-agent flavour only returns the agent's own \
threads; the manager flavour is unfiltered. Cancel any question or reminder \ threads; the manager flavour is unfiltered. Cancel any question or reminder \
row via `cancel_thread` (manager bypasses the owner check)." row via `cancel_loose_end` (manager bypasses the owner check)."
)] )]
async fn get_open_threads(&self) -> String { async fn get_loose_ends(&self) -> String {
run_tool_envelope("get_open_threads", String::new(), async move { run_tool_envelope("get_loose_ends", String::new(), async move {
let (resp, retries) = self let (resp, retries) = self
.dispatch(hive_sh4re::ManagerRequest::GetOpenThreads) .dispatch(hive_sh4re::ManagerRequest::GetLooseEnds)
.await; .await;
annotate_retries(format_open_threads(resp), retries) annotate_retries(format_loose_ends(resp), retries)
}) })
.await .await
} }
@ -1025,24 +1025,24 @@ impl ManagerServer {
description = "Cancel any open thread in the swarm — a `question` (cancels \ description = "Cancel any open thread in the swarm — a `question` (cancels \
with the operator-override sentinel so the asker unblocks) or a `reminder` \ with the operator-override sentinel so the asker unblocks) or a `reminder` \
(hard-deleted before fire). `kind` is `\"question\"` or `\"reminder\"`; `id` \ (hard-deleted before fire). `kind` is `\"question\"` or `\"reminder\"`; `id` \
is the row id from `get_open_threads` or the original submission reply. \ is the row id from `get_loose_ends` or the original submission reply. \
Manager surface bypasses the owner check on the sub-agent flavour use for \ Manager surface bypasses the owner check on the sub-agent flavour use for \
hive-wide cleanup of stuck or stale threads." hive-wide cleanup of stuck or stale threads."
)] )]
async fn cancel_thread(&self, Parameters(args): Parameters<CancelThreadArgs>) -> String { async fn cancel_loose_end(&self, Parameters(args): Parameters<CancelLooseEndArgs>) -> String {
let log = format!("{args:?}"); let log = format!("{args:?}");
let kind_label = args.kind.clone(); let kind_label = args.kind.clone();
let id = args.id; let id = args.id;
run_tool_envelope("cancel_thread", log, async move { run_tool_envelope("cancel_loose_end", log, async move {
let kind = match parse_cancel_kind(&args.kind) { let kind = match parse_loose_end_kind(&args.kind) {
Ok(k) => k, Ok(k) => k,
Err(e) => return e, Err(e) => return e,
}; };
let (resp, retries) = self let (resp, retries) = self
.dispatch(hive_sh4re::ManagerRequest::CancelThread { kind, id }) .dispatch(hive_sh4re::ManagerRequest::CancelLooseEnd { kind, id })
.await; .await;
annotate_retries( annotate_retries(
format_ack(resp, "cancel_thread", format!("cancelled {kind_label} {id}")), format_ack(resp, "cancel_loose_end", format!("cancelled {kind_label} {id}")),
retries, retries,
) )
}) })
@ -1092,8 +1092,8 @@ impl ManagerServer {
any agent including yourself), `ask` (structured question to the operator or a \ any agent including yourself), `ask` (structured question to the operator or a \
sub-agent non-blocking, answer arrives later as a `question_answered` event), \ sub-agent non-blocking, answer arrives later as a `question_answered` event), \
`answer` (respond to a `question_asked` event directed at you), \ `answer` (respond to a `question_asked` event directed at you), \
`get_open_threads` (hive-wide loose ends pending approvals + unanswered \ `get_loose_ends` (hive-wide loose ends pending approvals + unanswered \
questions + pending reminders across the swarm), `cancel_thread` (cancel any \ questions + pending reminders across the swarm), `cancel_loose_end` (cancel any \
question or reminder row by id), `whoami` (self-introspection canonical \ question or reminder row by id), `whoami` (self-introspection canonical \
name, role, current hyperhive rev). The manager's own config lives at \ name, role, current hyperhive rev). The manager's own config lives at \
`/agents/hm1nd/config/agent.nix`." `/agents/hm1nd/config/agent.nix`."
@ -1136,9 +1136,9 @@ pub fn allowed_mcp_tools(flavor: Flavor) -> Vec<String> {
"ask", "ask",
"answer", "answer",
"remind", "remind",
"get_open_threads", "get_loose_ends",
"whoami", "whoami",
"cancel_thread", "cancel_loose_end",
], ],
Flavor::Manager => &[ Flavor::Manager => &[
"send", "send",
@ -1152,10 +1152,10 @@ pub fn allowed_mcp_tools(flavor: Flavor) -> Vec<String> {
"ask", "ask",
"answer", "answer",
"get_logs", "get_logs",
"get_open_threads", "get_loose_ends",
"remind", "remind",
"whoami", "whoami",
"cancel_thread", "cancel_loose_end",
], ],
}; };
let mut out: Vec<String> = names let mut out: Vec<String> = names

View file

@ -106,7 +106,7 @@ pub async fn serve(
.route("/api/compact", post(post_compact)) .route("/api/compact", post(post_compact))
.route("/api/model", post(post_set_model)) .route("/api/model", post(post_set_model))
.route("/api/new-session", post(post_new_session)) .route("/api/new-session", post(post_new_session))
.route("/api/open-threads", get(api_open_threads)) .route("/api/loose-ends", get(api_loose_ends))
.with_state(state); .with_state(state);
let addr = SocketAddr::from(([0, 0, 0, 0], port)); let addr = SocketAddr::from(([0, 0, 0, 0], port));
let listener = bind_with_retry(addr, "web UI").await?; let listener = bind_with_retry(addr, "web UI").await?;
@ -240,25 +240,25 @@ struct SessionView {
exit_note: Option<String>, exit_note: Option<String>,
} }
/// Proxy this agent's open-threads via the per-agent socket. The /// Proxy this agent's loose-ends list via the per-agent socket. The
/// web UI surfaces the result as a collapsible section in the page /// web UI surfaces the result as a collapsible section in the page
/// so the operator can see at a glance what's pending against the /// so the operator can see at a glance what's pending against the
/// agent (questions asked by it, peer questions targeting it, /// agent (questions asked by it, peer questions targeting it,
/// approvals for the manager). Same data the /// reminders it scheduled, approvals for the manager). Same data
/// `mcp__hyperhive__get_open_threads` tool sees from inside the /// the `mcp__hyperhive__get_loose_ends` tool sees from inside the
/// container. /// container.
async fn api_open_threads(State(state): State<AppState>) -> Response { async fn api_loose_ends(State(state): State<AppState>) -> Response {
let threads: Vec<hive_sh4re::OpenThread> = match state.flavor() { let loose_ends: Vec<hive_sh4re::LooseEnd> = match state.flavor() {
Flavor::Agent => { Flavor::Agent => {
match client::request::<_, hive_sh4re::AgentResponse>( match client::request::<_, hive_sh4re::AgentResponse>(
&state.socket, &state.socket,
&hive_sh4re::AgentRequest::GetOpenThreads, &hive_sh4re::AgentRequest::GetLooseEnds,
) )
.await .await
{ {
Ok(hive_sh4re::AgentResponse::OpenThreads { threads }) => threads, Ok(hive_sh4re::AgentResponse::LooseEnds { loose_ends }) => loose_ends,
Ok(hive_sh4re::AgentResponse::Err { message }) => { Ok(hive_sh4re::AgentResponse::Err { message }) => {
return error_response(&format!("get_open_threads: {message}")); return error_response(&format!("get_loose_ends: {message}"));
} }
Ok(other) => return error_response(&format!("unexpected response: {other:?}")), Ok(other) => return error_response(&format!("unexpected response: {other:?}")),
Err(e) => return error_response(&format!("transport: {e:#}")), Err(e) => return error_response(&format!("transport: {e:#}")),
@ -267,20 +267,20 @@ async fn api_open_threads(State(state): State<AppState>) -> Response {
Flavor::Manager => { Flavor::Manager => {
match client::request::<_, hive_sh4re::ManagerResponse>( match client::request::<_, hive_sh4re::ManagerResponse>(
&state.socket, &state.socket,
&hive_sh4re::ManagerRequest::GetOpenThreads, &hive_sh4re::ManagerRequest::GetLooseEnds,
) )
.await .await
{ {
Ok(hive_sh4re::ManagerResponse::OpenThreads { threads }) => threads, Ok(hive_sh4re::ManagerResponse::LooseEnds { loose_ends }) => loose_ends,
Ok(hive_sh4re::ManagerResponse::Err { message }) => { Ok(hive_sh4re::ManagerResponse::Err { message }) => {
return error_response(&format!("get_open_threads: {message}")); return error_response(&format!("get_loose_ends: {message}"));
} }
Ok(other) => return error_response(&format!("unexpected response: {other:?}")), Ok(other) => return error_response(&format!("unexpected response: {other:?}")),
Err(e) => return error_response(&format!("transport: {e:#}")), Err(e) => return error_response(&format!("transport: {e:#}")),
} }
} }
}; };
axum::Json(serde_json::json!({ "threads": threads })).into_response() axum::Json(serde_json::json!({ "loose_ends": loose_ends })).into_response()
} }
async fn api_state(State(state): State<AppState>) -> axum::Json<StateSnapshot> { async fn api_state(State(state): State<AppState>) -> axum::Json<StateSnapshot> {

View file

@ -174,8 +174,8 @@ async fn dispatch(req: &AgentRequest, agent: &str, coord: &Arc<Coordinator>) ->
timing, timing,
file_path, file_path,
} => handle_remind(coord, agent, message, timing, file_path.as_deref()), } => handle_remind(coord, agent, message, timing, file_path.as_deref()),
AgentRequest::GetOpenThreads => match crate::open_threads::for_agent(coord, agent) { AgentRequest::GetLooseEnds => match crate::loose_ends::for_agent(coord, agent) {
Ok(threads) => AgentResponse::OpenThreads { threads }, Ok(loose_ends) => AgentResponse::LooseEnds { loose_ends },
Err(e) => AgentResponse::Err { Err(e) => AgentResponse::Err {
message: format!("{e:#}"), message: format!("{e:#}"),
}, },
@ -193,7 +193,7 @@ async fn dispatch(req: &AgentRequest, agent: &str, coord: &Arc<Coordinator>) ->
role: "agent".to_owned(), role: "agent".to_owned(),
hyperhive_rev: crate::auto_update::current_flake_rev(&coord.hyperhive_flake), hyperhive_rev: crate::auto_update::current_flake_rev(&coord.hyperhive_flake),
}, },
AgentRequest::CancelThread { kind, id } => crate::questions::handle_cancel_thread( AgentRequest::CancelLooseEnd { kind, id } => crate::questions::handle_cancel_loose_end(
coord, agent, *kind, *id, coord, agent, *kind, *id,
) )
.map_or_else( .map_or_else(

View file

@ -1,7 +1,7 @@
//! Loose-ends aggregator. Walks the `approvals` + `operator_questions` //! Loose-ends aggregator. Walks the `approvals` + `operator_questions`
//! tables once per call and assembles a `Vec<OpenThread>` for either //! tables once per call and assembles a `Vec<LooseEnd>` for either
//! a single agent (`for_agent`) or the whole hive (`hive_wide`). Both //! a single agent (`for_agent`) or the whole hive (`hive_wide`). Both
//! `AgentRequest::GetOpenThreads` and `ManagerRequest::GetOpenThreads` //! `AgentRequest::GetLooseEnds` and `ManagerRequest::GetLooseEnds`
//! land here so the routing logic + age-seconds derivation stay in //! land here so the routing logic + age-seconds derivation stay in
//! one place. //! one place.
//! //!
@ -16,7 +16,7 @@
use std::time::{SystemTime, UNIX_EPOCH}; use std::time::{SystemTime, UNIX_EPOCH};
use anyhow::Result; use anyhow::Result;
use hive_sh4re::{MANAGER_AGENT, OpenThread}; use hive_sh4re::{MANAGER_AGENT, LooseEnd};
use crate::coordinator::Coordinator; use crate::coordinator::Coordinator;
@ -32,7 +32,7 @@ use crate::coordinator::Coordinator;
/// Ordered approvals → questions → reminders within the returned /// Ordered approvals → questions → reminders within the returned
/// vector. Within each kind, source-of-truth ordering (sqlite's /// vector. Within each kind, source-of-truth ordering (sqlite's
/// `pending()` queries return newest-first within their indexes). /// `pending()` queries return newest-first within their indexes).
pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<OpenThread>> { pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<LooseEnd>> {
let now = now_unix(); let now = now_unix();
let mut out = Vec::new(); let mut out = Vec::new();
// Approvals are only submitted by the manager today. When that // Approvals are only submitted by the manager today. When that
@ -41,7 +41,7 @@ pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<OpenThread>> {
// here on that column — for now MANAGER_AGENT == sole submitter. // here on that column — for now MANAGER_AGENT == sole submitter.
if agent == MANAGER_AGENT { if agent == MANAGER_AGENT {
for a in coord.approvals.pending()? { for a in coord.approvals.pending()? {
out.push(OpenThread::Approval { out.push(LooseEnd::Approval {
id: a.id, id: a.id,
agent: a.agent, agent: a.agent,
commit_ref: a.commit_ref, commit_ref: a.commit_ref,
@ -55,7 +55,7 @@ pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<OpenThread>> {
if !role_match { if !role_match {
continue; continue;
} }
out.push(OpenThread::Question { out.push(LooseEnd::Question {
id: q.id, id: q.id,
asker: q.asker, asker: q.asker,
target: q.target, target: q.target,
@ -67,7 +67,7 @@ pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<OpenThread>> {
if r.agent != agent { if r.agent != agent {
continue; continue;
} }
out.push(OpenThread::Reminder { out.push(LooseEnd::Reminder {
id: r.id, id: r.id,
owner: r.agent, owner: r.agent,
message: r.message, message: r.message,
@ -82,11 +82,11 @@ pub fn for_agent(coord: &Coordinator, agent: &str) -> Result<Vec<OpenThread>> {
/// unanswered question + EVERY pending reminder. Manager surface /// unanswered question + EVERY pending reminder. Manager surface
/// only; sub-agents can't see each other's threads via the agent /// only; sub-agents can't see each other's threads via the agent
/// surface (`for_agent` filters by name). /// surface (`for_agent` filters by name).
pub fn hive_wide(coord: &Coordinator) -> Result<Vec<OpenThread>> { pub fn hive_wide(coord: &Coordinator) -> Result<Vec<LooseEnd>> {
let now = now_unix(); let now = now_unix();
let mut out = Vec::new(); let mut out = Vec::new();
for a in coord.approvals.pending()? { for a in coord.approvals.pending()? {
out.push(OpenThread::Approval { out.push(LooseEnd::Approval {
id: a.id, id: a.id,
agent: a.agent, agent: a.agent,
commit_ref: a.commit_ref, commit_ref: a.commit_ref,
@ -95,7 +95,7 @@ pub fn hive_wide(coord: &Coordinator) -> Result<Vec<OpenThread>> {
}); });
} }
for q in coord.questions.pending_all()? { for q in coord.questions.pending_all()? {
out.push(OpenThread::Question { out.push(LooseEnd::Question {
id: q.id, id: q.id,
asker: q.asker, asker: q.asker,
target: q.target, target: q.target,
@ -104,7 +104,7 @@ pub fn hive_wide(coord: &Coordinator) -> Result<Vec<OpenThread>> {
}); });
} }
for r in coord.broker.list_pending_reminders()? { for r in coord.broker.list_pending_reminders()? {
out.push(OpenThread::Reminder { out.push(LooseEnd::Reminder {
id: r.id, id: r.id,
owner: r.agent, owner: r.agent,
message: r.message, message: r.message,

View file

@ -20,10 +20,10 @@ mod events_vacuum;
mod forge; mod forge;
mod lifecycle; mod lifecycle;
mod limits; mod limits;
mod loose_ends;
mod manager_server; mod manager_server;
mod meta; mod meta;
mod migrate; mod migrate;
mod open_threads;
mod operator_questions; mod operator_questions;
mod questions; mod questions;
mod reminder_scheduler; mod reminder_scheduler;

View file

@ -329,8 +329,8 @@ async fn dispatch(req: &ManagerRequest, coord: &Arc<Coordinator>) -> ManagerResp
}, },
} }
} }
ManagerRequest::GetOpenThreads => match crate::open_threads::hive_wide(coord) { ManagerRequest::GetLooseEnds => match crate::loose_ends::hive_wide(coord) {
Ok(threads) => ManagerResponse::OpenThreads { threads }, Ok(loose_ends) => ManagerResponse::LooseEnds { loose_ends },
Err(e) => ManagerResponse::Err { Err(e) => ManagerResponse::Err {
message: format!("{e:#}"), message: format!("{e:#}"),
}, },
@ -348,7 +348,7 @@ async fn dispatch(req: &ManagerRequest, coord: &Arc<Coordinator>) -> ManagerResp
role: "manager".to_owned(), role: "manager".to_owned(),
hyperhive_rev: crate::auto_update::current_flake_rev(&coord.hyperhive_flake), hyperhive_rev: crate::auto_update::current_flake_rev(&coord.hyperhive_flake),
}, },
ManagerRequest::CancelThread { kind, id } => crate::questions::handle_cancel_thread( ManagerRequest::CancelLooseEnd { kind, id } => crate::questions::handle_cancel_loose_end(
coord, coord,
MANAGER_AGENT, MANAGER_AGENT,
*kind, *kind,

View file

@ -129,20 +129,20 @@ pub fn handle_answer(
Ok(()) Ok(())
} }
/// Handle `CancelThread` from either surface. Dispatches by kind to /// Handle `CancelLooseEnd` from either surface. Dispatches by kind to
/// either `OperatorQuestions::cancel` or `Broker::cancel_reminder_as`, /// either `OperatorQuestions::cancel` or `Broker::cancel_reminder_as`,
/// both of which do their own auth check (canceller == owner / /// both of which do their own auth check (canceller == owner /
/// asker, or `operator`, or `manager`). On question cancel, fires /// asker, or `operator`, or `manager`). On question cancel, fires
/// the `QuestionAnswered` event back to the asker so the harness /// the `QuestionAnswered` event back to the asker so the harness
/// loop can react (mirrors the operator-cancel dashboard path). /// loop can react (mirrors the operator-cancel dashboard path).
pub fn handle_cancel_thread( pub fn handle_cancel_loose_end(
coord: &Arc<Coordinator>, coord: &Arc<Coordinator>,
canceller: &str, canceller: &str,
kind: hive_sh4re::CancelThreadKind, kind: hive_sh4re::CancelLooseEndKind,
id: i64, id: i64,
) -> Result<(), String> { ) -> Result<(), String> {
match kind { match kind {
hive_sh4re::CancelThreadKind::Question => { hive_sh4re::CancelLooseEndKind::Question => {
let (question, asker, target) = coord let (question, asker, target) = coord
.questions .questions
.cancel(id, canceller) .cancel(id, canceller)
@ -161,7 +161,7 @@ pub fn handle_cancel_thread(
coord.emit_question_resolved(id, &sentinel, canceller, true, target.as_deref()); coord.emit_question_resolved(id, &sentinel, canceller, true, target.as_deref());
Ok(()) Ok(())
} }
hive_sh4re::CancelThreadKind::Reminder => { hive_sh4re::CancelLooseEndKind::Reminder => {
let owner = coord let owner = coord
.broker .broker
.cancel_reminder_as(id, canceller) .cancel_reminder_as(id, canceller)

View file

@ -185,7 +185,7 @@ pub enum ReminderTiming {
At { unix_timestamp: i64 }, At { unix_timestamp: i64 },
} }
/// One row in the response to `GetOpenThreads`. Tagged enum so new /// One row in the response to `GetLooseEnds`. Tagged enum so new
/// thread kinds (forge PRs, long-running approvals from a privileged /// thread kinds (forge PRs, long-running approvals from a privileged
/// bot, etc) can land later without breaking existing handlers. The /// bot, etc) can land later without breaking existing handlers. The
/// caller (claude in the agent harness) is expected to render these /// caller (claude in the agent harness) is expected to render these
@ -193,13 +193,13 @@ pub enum ReminderTiming {
/// needed without a follow-up fetch. /// needed without a follow-up fetch.
/// ///
/// `Question` and `Reminder` rows are cancellable via the /// `Question` and `Reminder` rows are cancellable via the
/// `CancelThread` request (and the `cancel_thread` MCP tool); /// `CancelLooseEnd` request (and the `cancel_loose_end` MCP tool);
/// `Approval` rows are not (operator approves/denies via the /// `Approval` rows are not (operator approves/denies via the
/// dashboard, manager has no withdraw path today). /// dashboard, manager has no withdraw path today).
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(tag = "kind", rename_all = "snake_case")] #[serde(tag = "kind", rename_all = "snake_case")]
pub enum OpenThread { pub enum LooseEnd {
/// A pending approval. For agent-flavour `GetOpenThreads` calls /// A pending approval. For agent-flavour `GetLooseEnds` calls
/// this only surfaces when the agent itself is the manager /// this only surfaces when the agent itself is the manager
/// (sub-agents don't submit approvals). For manager-flavour calls /// (sub-agents don't submit approvals). For manager-flavour calls
/// it lists every pending approval in the swarm. `agent` is the /// it lists every pending approval in the swarm. `agent` is the
@ -245,14 +245,14 @@ pub enum OpenThread {
}, },
} }
/// Kind discriminator for `CancelThread`. Maps to which underlying /// Kind discriminator for `CancelLooseEnd`. Maps to which underlying
/// store the dispatcher reaches into (`OperatorQuestions` vs /// store the dispatcher reaches into (`OperatorQuestions` vs
/// `Broker::reminders`). Approvals are deliberately not cancellable /// `Broker::reminders`). Approvals are deliberately not cancellable
/// — the operator approves/denies via the dashboard, manager has no /// — the operator approves/denies via the dashboard, manager has no
/// withdraw path today. /// withdraw path today.
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)] #[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)]
#[serde(rename_all = "snake_case")] #[serde(rename_all = "snake_case")]
pub enum CancelThreadKind { pub enum CancelLooseEndKind {
Question, Question,
Reminder, Reminder,
} }
@ -340,7 +340,7 @@ pub enum AgentRequest {
/// agent submitted them (which only ever happens for the /// agent submitted them (which only ever happens for the
/// manager); questions surface where the agent is `asker` or /// manager); questions surface where the agent is `asker` or
/// `target`. Cheap O(n) sweep server-side — no caching. /// `target`. Cheap O(n) sweep server-side — no caching.
GetOpenThreads, GetLooseEnds,
/// Count of this agent's pending (un-delivered) reminders. Used /// Count of this agent's pending (un-delivered) reminders. Used
/// by the harness's per-turn stats sink to snapshot "what was /// by the harness's per-turn stats sink to snapshot "what was
/// queued at turn-end time" without paying for a full list. /// queued at turn-end time" without paying for a full list.
@ -357,7 +357,7 @@ pub enum AgentRequest {
/// Authorisation on the sub-agent surface: caller must own the /// Authorisation on the sub-agent surface: caller must own the
/// row. The manager surface uses the same wire variant but /// row. The manager surface uses the same wire variant but
/// accepts any id. /// accepts any id.
CancelThread { kind: CancelThreadKind, id: i64 }, CancelLooseEnd { kind: CancelLooseEndKind, id: i64 },
} }
/// Responses on a per-agent socket. /// Responses on a per-agent socket.
@ -379,9 +379,9 @@ pub enum AgentResponse {
/// `Ask` result: the queued question id. The answer lands later /// `Ask` result: the queued question id. The answer lands later
/// as `HelperEvent::QuestionAnswered` in this agent's inbox. /// as `HelperEvent::QuestionAnswered` in this agent's inbox.
QuestionQueued { id: i64 }, QuestionQueued { id: i64 },
/// `GetOpenThreads` result: list of loose ends pending against /// `GetLooseEnds` result: list of loose ends pending against
/// this agent. Ordered newest-first within each kind. /// this agent. Ordered newest-first within each kind.
OpenThreads { threads: Vec<OpenThread> }, LooseEnds { loose_ends: Vec<LooseEnd> },
/// `CountPendingReminders` result. /// `CountPendingReminders` result.
PendingRemindersCount { count: u64 }, PendingRemindersCount { count: u64 },
/// `Whoami` result: identity + role + the current hyperhive rev /// `Whoami` result: identity + role + the current hyperhive rev
@ -655,9 +655,9 @@ pub enum ManagerRequest {
/// Hive-wide loose-ends view: EVERY pending approval + EVERY /// Hive-wide loose-ends view: EVERY pending approval + EVERY
/// unanswered question in the swarm. Used by the manager to scan /// unanswered question in the swarm. Used by the manager to scan
/// for stalled coordination — the per-agent equivalent on the /// for stalled coordination — the per-agent equivalent on the
/// sub-agent surface is `AgentRequest::GetOpenThreads` which /// sub-agent surface is `AgentRequest::GetLooseEnds` which
/// only returns rows where the agent itself is asker / target. /// only returns rows where the agent itself is asker / target.
GetOpenThreads, GetLooseEnds,
/// Count of the manager's own pending reminders. Mirror of /// Count of the manager's own pending reminders. Mirror of
/// `AgentRequest::CountPendingReminders` on the manager surface. /// `AgentRequest::CountPendingReminders` on the manager surface.
CountPendingReminders, CountPendingReminders,
@ -666,8 +666,8 @@ pub enum ManagerRequest {
Whoami, Whoami,
/// Cancel an open thread (question or reminder). Manager surface /// Cancel an open thread (question or reminder). Manager surface
/// can cancel any row (no owner check) — same dispatch as /// can cancel any row (no owner check) — same dispatch as
/// `AgentRequest::CancelThread` but with privileged auth. /// `AgentRequest::CancelLooseEnd` but with privileged auth.
CancelThread { kind: CancelThreadKind, id: i64 }, CancelLooseEnd { kind: CancelLooseEndKind, id: i64 },
} }
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Serialize, Deserialize)]
@ -700,11 +700,11 @@ pub enum ManagerResponse {
Logs { Logs {
content: String, content: String,
}, },
/// `GetOpenThreads` result: hive-wide loose ends (approvals + /// `GetLooseEnds` result: hive-wide loose ends (approvals +
/// unanswered questions). Same `OpenThread` variants as the /// unanswered questions). Same `LooseEnd` variants as the
/// agent surface; the manager's view is unfiltered. /// agent surface; the manager's view is unfiltered.
OpenThreads { LooseEnds {
threads: Vec<OpenThread>, loose_ends: Vec<LooseEnd>,
}, },
/// `CountPendingReminders` result. /// `CountPendingReminders` result.
PendingRemindersCount { PendingRemindersCount {