feat: add optional description to request_apply_commit and request_spawn
This commit is contained in:
parent
a6d1464071
commit
4a8a668348
9 changed files with 97 additions and 30 deletions
|
|
@ -4,12 +4,12 @@ Tools (hyperhive surface):
|
||||||
|
|
||||||
- `mcp__hyperhive__recv(wait_seconds?)` — drain one more message from your inbox. Without `wait_seconds` (or with `0`) it returns immediately — a cheap inbox peek you can drop between actions. To **wait** when you have nothing else to do, call with a long wait (e.g. `wait_seconds: 180`, the max) — you'll wake instantly on new work, otherwise return after the timeout. Use that instead of ending the turn or sleeping in a Bash command.
|
- `mcp__hyperhive__recv(wait_seconds?)` — drain one more message from your inbox. Without `wait_seconds` (or with `0`) it returns immediately — a cheap inbox peek you can drop between actions. To **wait** when you have nothing else to do, call with a long wait (e.g. `wait_seconds: 180`, the max) — you'll wake instantly on new work, otherwise return after the timeout. Use that instead of ending the turn or sleeping in a Bash command.
|
||||||
- `mcp__hyperhive__send(to, body)` — message an agent (by name), another peer, or the operator (`operator` surfaces in the dashboard). Use `to: "*"` to broadcast to all agents (they receive a hint that it's a broadcast and may not need action).
|
- `mcp__hyperhive__send(to, body)` — message an agent (by name), another peer, or the operator (`operator` surfaces in the dashboard). Use `to: "*"` to broadcast to all agents (they receive a hint that it's a broadcast and may not need action).
|
||||||
- `mcp__hyperhive__request_spawn(name)` — queue a brand-new sub-agent for operator approval (≤9 char name).
|
- `mcp__hyperhive__request_spawn(name, description?)` — queue a brand-new sub-agent for operator approval (≤9 char name). Pass an optional `description` and it appears on the dashboard approval card — no need to send a separate message explaining the request.
|
||||||
- `mcp__hyperhive__kill(name)` — graceful stop on a sub-agent. No approval required.
|
- `mcp__hyperhive__kill(name)` — graceful stop on a sub-agent. No approval required.
|
||||||
- `mcp__hyperhive__start(name)` — start a stopped sub-agent. No approval required.
|
- `mcp__hyperhive__start(name)` — start a stopped sub-agent. No approval required.
|
||||||
- `mcp__hyperhive__restart(name)` — stop + start a sub-agent. No approval required.
|
- `mcp__hyperhive__restart(name)` — stop + start a sub-agent. No approval required.
|
||||||
- `mcp__hyperhive__update(name)` — rebuild a sub-agent (re-applies the current hyperhive flake + agent.nix, restarts the container). No approval required — idempotent. Use when you receive a `needs_update` system event.
|
- `mcp__hyperhive__update(name)` — rebuild a sub-agent (re-applies the current hyperhive flake + agent.nix, restarts the container). No approval required — idempotent. Use when you receive a `needs_update` system event.
|
||||||
- `mcp__hyperhive__request_apply_commit(agent, commit_ref)` — submit a config change for any agent (`hm1nd` for self) for operator approval. 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_operator(question, options?, multi?, ttl_seconds?)` — surface a question on the dashboard. Returns immediately with a question id; the operator's answer arrives later as a system `operator_answered` 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 — useful when the decision becomes moot if the operator hasn't responded in time; on expiry the answer is `[expired]`. Do not poll inside the same turn — finish the current work and react when the event lands.
|
- `mcp__hyperhive__ask_operator(question, options?, multi?, ttl_seconds?)` — surface a question on the dashboard. Returns immediately with a question id; the operator's answer arrives later as a system `operator_answered` 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 — useful when the decision becomes moot if the operator hasn't responded in time; on expiry the answer is `[expired]`. Do not poll inside the same turn — finish the current work and react when the event lands.
|
||||||
|
|
||||||
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.
|
||||||
|
|
|
||||||
|
|
@ -265,6 +265,10 @@ pub struct RequestSpawnArgs {
|
||||||
/// New sub-agent name (≤9 chars). Queues a Spawn approval; the
|
/// New sub-agent name (≤9 chars). Queues a Spawn approval; the
|
||||||
/// operator approves on the dashboard before the container is created.
|
/// operator approves on the dashboard before the container is created.
|
||||||
pub name: String,
|
pub name: String,
|
||||||
|
/// Optional description shown on the dashboard approval card so the
|
||||||
|
/// operator knows what the new agent is for without a separate message.
|
||||||
|
#[serde(default)]
|
||||||
|
pub description: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, schemars::JsonSchema)]
|
#[derive(Debug, serde::Deserialize, schemars::JsonSchema)]
|
||||||
|
|
@ -320,6 +324,10 @@ pub struct RequestApplyCommitArgs {
|
||||||
pub agent: String,
|
pub agent: String,
|
||||||
/// Git sha (full or short) pointing at the proposed `agent.nix`.
|
/// Git sha (full or short) pointing at the proposed `agent.nix`.
|
||||||
pub commit_ref: String,
|
pub commit_ref: String,
|
||||||
|
/// Optional description shown on the dashboard approval card so the
|
||||||
|
/// operator knows what the change does without opening the diff.
|
||||||
|
#[serde(default)]
|
||||||
|
pub description: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
|
@ -395,7 +403,10 @@ impl ManagerServer {
|
||||||
let name = args.name.clone();
|
let name = args.name.clone();
|
||||||
run_tool_envelope("request_spawn", log, async move {
|
run_tool_envelope("request_spawn", log, async move {
|
||||||
let resp = self
|
let resp = self
|
||||||
.dispatch(hive_sh4re::ManagerRequest::RequestSpawn { name: args.name })
|
.dispatch(hive_sh4re::ManagerRequest::RequestSpawn {
|
||||||
|
name: args.name,
|
||||||
|
description: args.description,
|
||||||
|
})
|
||||||
.await;
|
.await;
|
||||||
format_ack(
|
format_ack(
|
||||||
resp,
|
resp,
|
||||||
|
|
@ -521,6 +532,7 @@ impl ManagerServer {
|
||||||
.dispatch(hive_sh4re::ManagerRequest::RequestApplyCommit {
|
.dispatch(hive_sh4re::ManagerRequest::RequestApplyCommit {
|
||||||
agent: args.agent,
|
agent: args.agent,
|
||||||
commit_ref: args.commit_ref,
|
commit_ref: args.commit_ref,
|
||||||
|
description: args.description,
|
||||||
})
|
})
|
||||||
.await;
|
.await;
|
||||||
format_ack(
|
format_ack(
|
||||||
|
|
|
||||||
|
|
@ -684,6 +684,9 @@
|
||||||
'new sub-agent — container will be created on approve'),
|
'new sub-agent — container will be created on approve'),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
if (a.description) {
|
||||||
|
li.append(el('div', { class: 'approval-description' }, a.description));
|
||||||
|
}
|
||||||
// Deny prompts the operator for an optional reason; the
|
// Deny prompts the operator for an optional reason; the
|
||||||
// submit handler stashes it into a hidden `note` input that
|
// submit handler stashes it into a hidden `note` input that
|
||||||
// rides along on the POST and is surfaced to the manager via
|
// rides along on the POST and is surfaced to the manager via
|
||||||
|
|
|
||||||
|
|
@ -258,6 +258,7 @@ code {
|
||||||
}
|
}
|
||||||
.approvals .row { display: flex; align-items: center; flex-wrap: wrap; gap: 0.4em; }
|
.approvals .row { display: flex; align-items: center; flex-wrap: wrap; gap: 0.4em; }
|
||||||
.approvals form.inline { display: inline; margin-left: 0.4em; }
|
.approvals form.inline { display: inline; margin-left: 0.4em; }
|
||||||
|
.approval-description { font-size: 0.85em; color: var(--fg-dim, #888); margin: 0.2em 0 0.4em 1.2em; }
|
||||||
.approval-tabs {
|
.approval-tabs {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.4em;
|
gap: 0.4em;
|
||||||
|
|
|
||||||
|
|
@ -24,6 +24,20 @@ CREATE INDEX IF NOT EXISTS idx_approvals_pending
|
||||||
ON approvals (id) WHERE status = 'pending';
|
ON approvals (id) WHERE status = 'pending';
|
||||||
";
|
";
|
||||||
|
|
||||||
|
/// Add the `description` column to pre-description databases. Manager-supplied
|
||||||
|
/// note shown on the dashboard approval card at submission time (distinct from
|
||||||
|
/// `note` which is set on denial/failure).
|
||||||
|
fn ensure_description_column(conn: &Connection) -> Result<()> {
|
||||||
|
let has: bool = conn
|
||||||
|
.prepare("SELECT 1 FROM pragma_table_info('approvals') WHERE name = 'description'")?
|
||||||
|
.exists([])?;
|
||||||
|
if !has {
|
||||||
|
conn.execute_batch("ALTER TABLE approvals ADD COLUMN description TEXT;")
|
||||||
|
.context("add approvals.description column")?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Add the `kind` column to pre-Phase-8 databases. ALTER TABLE ADD COLUMN is
|
/// Add the `kind` column to pre-Phase-8 databases. ALTER TABLE ADD COLUMN is
|
||||||
/// idempotent here only via a column-existence check (sqlite doesn't support
|
/// idempotent here only via a column-existence check (sqlite doesn't support
|
||||||
/// IF NOT EXISTS on ADD COLUMN). Defaults legacy rows to `apply_commit`,
|
/// IF NOT EXISTS on ADD COLUMN). Defaults legacy rows to `apply_commit`,
|
||||||
|
|
@ -72,21 +86,24 @@ impl Approvals {
|
||||||
.context("apply approvals schema")?;
|
.context("apply approvals schema")?;
|
||||||
ensure_kind_column(&conn).context("migrate approvals.kind")?;
|
ensure_kind_column(&conn).context("migrate approvals.kind")?;
|
||||||
ensure_fetched_sha_column(&conn).context("migrate approvals.fetched_sha")?;
|
ensure_fetched_sha_column(&conn).context("migrate approvals.fetched_sha")?;
|
||||||
|
ensure_description_column(&conn).context("migrate approvals.description")?;
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
conn: Mutex::new(conn),
|
conn: Mutex::new(conn),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn submit(&self, agent: &str, commit_ref: &str) -> Result<i64> {
|
pub fn submit_kind(
|
||||||
self.submit_kind(agent, ApprovalKind::ApplyCommit, commit_ref)
|
&self,
|
||||||
}
|
agent: &str,
|
||||||
|
kind: ApprovalKind,
|
||||||
pub fn submit_kind(&self, agent: &str, kind: ApprovalKind, commit_ref: &str) -> Result<i64> {
|
commit_ref: &str,
|
||||||
|
description: Option<&str>,
|
||||||
|
) -> Result<i64> {
|
||||||
let conn = self.conn.lock().unwrap();
|
let conn = self.conn.lock().unwrap();
|
||||||
conn.execute(
|
conn.execute(
|
||||||
"INSERT INTO approvals (agent, kind, commit_ref, requested_at, status)
|
"INSERT INTO approvals (agent, kind, commit_ref, requested_at, status, description)
|
||||||
VALUES (?1, ?2, ?3, ?4, 'pending')",
|
VALUES (?1, ?2, ?3, ?4, 'pending', ?5)",
|
||||||
params![agent, kind_to_str(kind), commit_ref, now_unix()],
|
params![agent, kind_to_str(kind), commit_ref, now_unix(), description],
|
||||||
)?;
|
)?;
|
||||||
Ok(conn.last_insert_rowid())
|
Ok(conn.last_insert_rowid())
|
||||||
}
|
}
|
||||||
|
|
@ -107,7 +124,7 @@ impl Approvals {
|
||||||
pub fn recent_resolved(&self, limit: u64) -> Result<Vec<Approval>> {
|
pub fn recent_resolved(&self, limit: u64) -> Result<Vec<Approval>> {
|
||||||
let conn = self.conn.lock().unwrap();
|
let conn = self.conn.lock().unwrap();
|
||||||
let mut stmt = conn.prepare(
|
let mut stmt = conn.prepare(
|
||||||
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha
|
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha, description
|
||||||
FROM approvals
|
FROM approvals
|
||||||
WHERE status IN ('approved', 'denied', 'failed')
|
WHERE status IN ('approved', 'denied', 'failed')
|
||||||
ORDER BY resolved_at DESC, id DESC
|
ORDER BY resolved_at DESC, id DESC
|
||||||
|
|
@ -121,7 +138,7 @@ impl Approvals {
|
||||||
pub fn pending(&self) -> Result<Vec<Approval>> {
|
pub fn pending(&self) -> Result<Vec<Approval>> {
|
||||||
let conn = self.conn.lock().unwrap();
|
let conn = self.conn.lock().unwrap();
|
||||||
let mut stmt = conn.prepare(
|
let mut stmt = conn.prepare(
|
||||||
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha
|
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha, description
|
||||||
FROM approvals
|
FROM approvals
|
||||||
WHERE status = 'pending'
|
WHERE status = 'pending'
|
||||||
ORDER BY id ASC",
|
ORDER BY id ASC",
|
||||||
|
|
@ -134,7 +151,7 @@ impl Approvals {
|
||||||
pub fn get(&self, id: i64) -> Result<Option<Approval>> {
|
pub fn get(&self, id: i64) -> Result<Option<Approval>> {
|
||||||
let conn = self.conn.lock().unwrap();
|
let conn = self.conn.lock().unwrap();
|
||||||
conn.query_row(
|
conn.query_row(
|
||||||
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha
|
"SELECT id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha, description
|
||||||
FROM approvals WHERE id = ?1",
|
FROM approvals WHERE id = ?1",
|
||||||
params![id],
|
params![id],
|
||||||
row_to_approval,
|
row_to_approval,
|
||||||
|
|
@ -147,9 +164,9 @@ impl Approvals {
|
||||||
/// approval so the caller can run the action and pass the agent name.
|
/// approval so the caller can run the action and pass the agent name.
|
||||||
pub fn mark_approved(&self, id: i64) -> Result<Approval> {
|
pub fn mark_approved(&self, id: i64) -> Result<Approval> {
|
||||||
let conn = self.conn.lock().unwrap();
|
let conn = self.conn.lock().unwrap();
|
||||||
let current: Option<(String, String, String, i64, String, Option<String>)> = conn
|
let current: Option<(String, String, String, i64, String, Option<String>, Option<String>)> =
|
||||||
.query_row(
|
conn.query_row(
|
||||||
"SELECT agent, kind, commit_ref, requested_at, status, fetched_sha
|
"SELECT agent, kind, commit_ref, requested_at, status, fetched_sha, description
|
||||||
FROM approvals WHERE id = ?1",
|
FROM approvals WHERE id = ?1",
|
||||||
params![id],
|
params![id],
|
||||||
|row| {
|
|row| {
|
||||||
|
|
@ -160,11 +177,14 @@ impl Approvals {
|
||||||
row.get(3)?,
|
row.get(3)?,
|
||||||
row.get(4)?,
|
row.get(4)?,
|
||||||
row.get(5)?,
|
row.get(5)?,
|
||||||
|
row.get(6)?,
|
||||||
))
|
))
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
.optional()?;
|
.optional()?;
|
||||||
let Some((agent, kind, commit_ref, requested_at, status, fetched_sha)) = current else {
|
let Some((agent, kind, commit_ref, requested_at, status, fetched_sha, description)) =
|
||||||
|
current
|
||||||
|
else {
|
||||||
bail!("approval {id} not found");
|
bail!("approval {id} not found");
|
||||||
};
|
};
|
||||||
if status != "pending" {
|
if status != "pending" {
|
||||||
|
|
@ -185,6 +205,7 @@ impl Approvals {
|
||||||
resolved_at: Some(resolved_at),
|
resolved_at: Some(resolved_at),
|
||||||
note: None,
|
note: None,
|
||||||
fetched_sha,
|
fetched_sha,
|
||||||
|
description,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -224,7 +245,7 @@ impl Approvals {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn row_to_approval(row: &rusqlite::Row<'_>) -> rusqlite::Result<Approval> {
|
fn row_to_approval(row: &rusqlite::Row<'_>) -> rusqlite::Result<Approval> {
|
||||||
// Column order: id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha.
|
// Column order: id, agent, kind, commit_ref, requested_at, status, resolved_at, note, fetched_sha, description.
|
||||||
let kind: String = row.get(2)?;
|
let kind: String = row.get(2)?;
|
||||||
let kind = match kind.as_str() {
|
let kind = match kind.as_str() {
|
||||||
"apply_commit" => ApprovalKind::ApplyCommit,
|
"apply_commit" => ApprovalKind::ApplyCommit,
|
||||||
|
|
@ -261,6 +282,7 @@ fn row_to_approval(row: &rusqlite::Row<'_>) -> rusqlite::Result<Approval> {
|
||||||
resolved_at: row.get(6)?,
|
resolved_at: row.get(6)?,
|
||||||
note: row.get(7)?,
|
note: row.get(7)?,
|
||||||
fetched_sha: row.get(8)?,
|
fetched_sha: row.get(8)?,
|
||||||
|
description: row.get(9)?,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -237,6 +237,9 @@ struct ApprovalView {
|
||||||
sha_short: Option<String>,
|
sha_short: Option<String>,
|
||||||
/// Pre-rendered syntax-coloured diff HTML, for `ApplyCommit` only.
|
/// Pre-rendered syntax-coloured diff HTML, for `ApplyCommit` only.
|
||||||
diff_html: Option<String>,
|
diff_html: Option<String>,
|
||||||
|
/// Manager-supplied description shown on the approval card.
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
description: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Replace silent `.unwrap_or_default()` on the data sources behind
|
/// Replace silent `.unwrap_or_default()` on the data sources behind
|
||||||
|
|
@ -605,10 +608,11 @@ async fn build_approval_views(approvals: Vec<Approval>) -> Vec<ApprovalView> {
|
||||||
let diff = approval_diff(&a.agent, a.id).await;
|
let diff = approval_diff(&a.agent, a.id).await;
|
||||||
ApprovalView {
|
ApprovalView {
|
||||||
id: a.id,
|
id: a.id,
|
||||||
agent: a.agent,
|
agent: a.agent.clone(),
|
||||||
kind: "apply_commit",
|
kind: "apply_commit",
|
||||||
sha_short: Some(sha),
|
sha_short: Some(sha),
|
||||||
diff_html: Some(render_diff_lines(&diff)),
|
diff_html: Some(render_diff_lines(&diff)),
|
||||||
|
description: a.description,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hive_sh4re::ApprovalKind::Spawn => ApprovalView {
|
hive_sh4re::ApprovalKind::Spawn => ApprovalView {
|
||||||
|
|
@ -617,6 +621,7 @@ async fn build_approval_views(approvals: Vec<Approval>) -> Vec<ApprovalView> {
|
||||||
kind: "spawn",
|
kind: "spawn",
|
||||||
sha_short: None,
|
sha_short: None,
|
||||||
diff_html: None,
|
diff_html: None,
|
||||||
|
description: a.description,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
@ -1065,7 +1070,7 @@ async fn post_request_spawn(
|
||||||
match state
|
match state
|
||||||
.coord
|
.coord
|
||||||
.approvals
|
.approvals
|
||||||
.submit_kind(&name, hive_sh4re::ApprovalKind::Spawn, "")
|
.submit_kind(&name, hive_sh4re::ApprovalKind::Spawn, "", None)
|
||||||
{
|
{
|
||||||
Ok(id) => {
|
Ok(id) => {
|
||||||
tracing::info!(%id, %name, "operator: spawn approval queued via dashboard");
|
tracing::info!(%id, %name, "operator: spawn approval queued via dashboard");
|
||||||
|
|
|
||||||
|
|
@ -131,12 +131,14 @@ async fn dispatch(req: &ManagerRequest, coord: &Arc<Coordinator>) -> ManagerResp
|
||||||
message: format!("{e:#}"),
|
message: format!("{e:#}"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
ManagerRequest::RequestSpawn { name } => {
|
ManagerRequest::RequestSpawn { name, description } => {
|
||||||
tracing::info!(%name, "manager: request_spawn");
|
tracing::info!(%name, "manager: request_spawn");
|
||||||
match coord
|
match coord.approvals.submit_kind(
|
||||||
.approvals
|
name,
|
||||||
.submit_kind(name, hive_sh4re::ApprovalKind::Spawn, "")
|
hive_sh4re::ApprovalKind::Spawn,
|
||||||
{
|
"",
|
||||||
|
description.as_deref(),
|
||||||
|
) {
|
||||||
Ok(id) => {
|
Ok(id) => {
|
||||||
tracing::info!(%id, %name, "spawn approval queued");
|
tracing::info!(%id, %name, "spawn approval queued");
|
||||||
ManagerResponse::Ok
|
ManagerResponse::Ok
|
||||||
|
|
@ -257,9 +259,13 @@ async fn dispatch(req: &ManagerRequest, coord: &Arc<Coordinator>) -> ManagerResp
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ManagerRequest::RequestApplyCommit { agent, commit_ref } => {
|
ManagerRequest::RequestApplyCommit {
|
||||||
|
agent,
|
||||||
|
commit_ref,
|
||||||
|
description,
|
||||||
|
} => {
|
||||||
tracing::info!(%agent, %commit_ref, "manager: request_apply_commit");
|
tracing::info!(%agent, %commit_ref, "manager: request_apply_commit");
|
||||||
match submit_apply_commit(coord, agent, commit_ref).await {
|
match submit_apply_commit(coord, agent, commit_ref, description.as_deref()).await {
|
||||||
Ok((id, sha)) => {
|
Ok((id, sha)) => {
|
||||||
tracing::info!(%id, %agent, manager_ref = %commit_ref, %sha, "approval queued + proposal tag planted");
|
tracing::info!(%id, %agent, manager_ref = %commit_ref, %sha, "approval queued + proposal tag planted");
|
||||||
ManagerResponse::Ok
|
ManagerResponse::Ok
|
||||||
|
|
@ -287,6 +293,7 @@ async fn submit_apply_commit(
|
||||||
coord: &Arc<Coordinator>,
|
coord: &Arc<Coordinator>,
|
||||||
agent: &str,
|
agent: &str,
|
||||||
commit_ref: &str,
|
commit_ref: &str,
|
||||||
|
description: Option<&str>,
|
||||||
) -> anyhow::Result<(i64, String)> {
|
) -> anyhow::Result<(i64, String)> {
|
||||||
let proposed_dir = crate::coordinator::Coordinator::agent_proposed_dir(agent);
|
let proposed_dir = crate::coordinator::Coordinator::agent_proposed_dir(agent);
|
||||||
let applied_dir = crate::coordinator::Coordinator::agent_applied_dir(agent);
|
let applied_dir = crate::coordinator::Coordinator::agent_applied_dir(agent);
|
||||||
|
|
@ -304,7 +311,12 @@ async fn submit_apply_commit(
|
||||||
}
|
}
|
||||||
let id = coord
|
let id = coord
|
||||||
.approvals
|
.approvals
|
||||||
.submit(agent, commit_ref)
|
.submit_kind(
|
||||||
|
agent,
|
||||||
|
hive_sh4re::ApprovalKind::ApplyCommit,
|
||||||
|
commit_ref,
|
||||||
|
description,
|
||||||
|
)
|
||||||
.map_err(|e| anyhow::anyhow!("queue approval row: {e:#}"))?;
|
.map_err(|e| anyhow::anyhow!("queue approval row: {e:#}"))?;
|
||||||
let tag = format!("proposal/{id}");
|
let tag = format!("proposal/{id}");
|
||||||
let sha = match crate::lifecycle::git_fetch_to_tag(
|
let sha = match crate::lifecycle::git_fetch_to_tag(
|
||||||
|
|
|
||||||
|
|
@ -106,7 +106,7 @@ async fn dispatch(req: &HostRequest, coord: Arc<Coordinator>) -> HostResponse {
|
||||||
tracing::info!(%name, "request_spawn");
|
tracing::info!(%name, "request_spawn");
|
||||||
let id = coord
|
let id = coord
|
||||||
.approvals
|
.approvals
|
||||||
.submit_kind(name, hive_sh4re::ApprovalKind::Spawn, "")?;
|
.submit_kind(name, hive_sh4re::ApprovalKind::Spawn, "", None)?;
|
||||||
tracing::info!(%id, %name, "spawn approval queued");
|
tracing::info!(%id, %name, "spawn approval queued");
|
||||||
HostResponse::success()
|
HostResponse::success()
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -83,6 +83,11 @@ pub struct Approval {
|
||||||
pub resolved_at: Option<i64>,
|
pub resolved_at: Option<i64>,
|
||||||
#[serde(default, skip_serializing_if = "Option::is_none")]
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
pub note: Option<String>,
|
pub note: Option<String>,
|
||||||
|
/// Optional free-text description the manager attached at submission
|
||||||
|
/// time — shown on the dashboard approval card so the operator can
|
||||||
|
/// understand the change without opening the diff.
|
||||||
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
|
pub description: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// What action the approval, when granted, will trigger.
|
/// What action the approval, when granted, will trigger.
|
||||||
|
|
@ -414,6 +419,9 @@ pub enum ManagerRequest {
|
||||||
/// agent of the same name already exists, the approval will fail.
|
/// agent of the same name already exists, the approval will fail.
|
||||||
RequestSpawn {
|
RequestSpawn {
|
||||||
name: String,
|
name: String,
|
||||||
|
/// Optional description shown on the dashboard approval card.
|
||||||
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
|
description: Option<String>,
|
||||||
},
|
},
|
||||||
/// Stop a sub-agent (graceful).
|
/// Stop a sub-agent (graceful).
|
||||||
Kill {
|
Kill {
|
||||||
|
|
@ -439,6 +447,10 @@ pub enum ManagerRequest {
|
||||||
RequestApplyCommit {
|
RequestApplyCommit {
|
||||||
agent: String,
|
agent: String,
|
||||||
commit_ref: String,
|
commit_ref: String,
|
||||||
|
/// Optional description shown on the dashboard approval card so the
|
||||||
|
/// operator knows what the change does without opening the diff.
|
||||||
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
|
description: Option<String>,
|
||||||
},
|
},
|
||||||
/// Ask the operator a question. Returns immediately with the queued
|
/// Ask the operator a question. Returns immediately with the queued
|
||||||
/// question id; the operator's answer arrives later as a
|
/// question id; the operator's answer arrives later as a
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue