use serde_json::{json, Value};
use crate::{Result, RuntimeError};
use super::super::{Tool, ToolContext};
pub struct SubagentStatusTool;
#[async_trait::async_trait]
impl Tool for SubagentStatusTool {
fn name(&self) -> &str { "subagent_status" }
fn description(&self) -> &str {
"Poll the current state of a reactive subagent. Returns status \
(running/finished/timed_out/failed), the last 500 characters of output \
produced so far, elapsed time in seconds, and the number of tool calls \
made. Non-blocking — returns immediately."
}
fn parameters(&self) -> Value {
json!({
"type": "object",
"properties": {
"handle_id": {
"type": "string",
"description": "Handle ID returned by subagent_start (e.g. \"sa_3\")."
}
},
"required": ["handle_id"]
})
}
async fn execute(&self, params: Value, ctx: ToolContext) -> Result<String> {
let handle_id = params["handle_id"].as_str()
.ok_or_else(|| RuntimeError::Tool("Missing 'handle_id' parameter".to_string()))?
.to_string();
let registry = ctx.capabilities.subagent_registry.as_ref()
.ok_or_else(|| RuntimeError::Tool(
"SubagentRegistry not available on this ToolContext".to_string()
))?;
let reg = registry.lock().unwrap();
let handle = reg.get(&handle_id)
.ok_or_else(|| RuntimeError::Tool(
format!("No subagent found with handle_id '{}'", handle_id)
))?;
let full: String = handle.partial_output();
let agent_name = handle.agent_name.clone();
let status_str = handle.status().as_str().to_string();
let elapsed = handle.elapsed_secs();
let tool_count = handle.tool_log().len();
let _ = handle;
drop(reg);
let char_count = full.chars().count();
let partial_output: String = if char_count > 500 {
full.chars().skip(char_count - 500).collect()
} else {
full
};
Ok(json!({
"handle_id": handle_id,
"agent_name": agent_name,
"status": status_str,
"partial_output": partial_output,
"elapsed_secs": (elapsed * 10.0).round() / 10.0,
"tool_count": tool_count
}).to_string())
}
}