use super::Tool;
use crate::{PawanError, Result};
use async_trait::async_trait;
use serde_json::{json, Value};
use std::path::PathBuf;
use std::process::Stdio;
use tokio::io::AsyncReadExt;
use tokio::process::Command;
use std::io::Write;
use tracing;
pub struct SpawnAgentTool {
workspace_root: PathBuf,
}
impl SpawnAgentTool {
pub fn new(workspace_root: PathBuf) -> Self {
Self { workspace_root }
}
fn find_pawan_binary(&self) -> String {
for candidate in &[
self.workspace_root.join("target/release/pawan"),
self.workspace_root.join("target/debug/pawan"),
] {
if candidate.exists() {
return candidate.to_string_lossy().to_string();
}
}
"pawan".to_string()
}
}
#[async_trait]
impl Tool for SpawnAgentTool {
fn name(&self) -> &str {
"spawn_agent"
}
fn description(&self) -> &str {
"Spawn a sub-agent (pawan subprocess) to handle a task independently. \
Returns the agent's response as JSON. Use this for parallel or delegated tasks."
}
fn mutating(&self) -> bool {
true }
fn parameters_schema(&self) -> Value {
json!({
"type": "object",
"properties": {
"prompt": {
"type": "string",
"description": "The task/prompt for the sub-agent"
},
"model": {
"type": "string",
"description": "Model to use (optional, defaults to parent's model)"
},
"timeout": {
"type": "integer",
"description": "Timeout in seconds (default: 120)"
},
"workspace": {
"type": "string",
"description": "Workspace directory for the sub-agent (default: same as parent)"
},
"retries": {
"type": "integer",
"description": "Number of retry attempts on failure (default: 0, max: 2)"
}
},
"required": ["prompt"]
})
}
fn thulp_definition(&self) -> thulp_core::ToolDefinition {
use thulp_core::{Parameter, ParameterType};
thulp_core::ToolDefinition::builder("spawn_agent")
.description(self.description())
.parameter(Parameter::builder("prompt").param_type(ParameterType::String).required(true)
.description("The task/prompt for the sub-agent").build())
.parameter(Parameter::builder("model").param_type(ParameterType::String).required(false)
.description("Model to use (optional, defaults to parent's model)").build())
.parameter(Parameter::builder("timeout").param_type(ParameterType::Integer).required(false)
.description("Timeout in seconds (default: 120)").build())
.parameter(Parameter::builder("workspace").param_type(ParameterType::String).required(false)
.description("Workspace directory for the sub-agent (default: same as parent)").build())
.parameter(Parameter::builder("retries").param_type(ParameterType::Integer).required(false)
.description("Number of retry attempts on failure (default: 0, max: 2)").build())
.build()
}
async fn execute(&self, args: Value) -> Result<Value> {
let prompt = args["prompt"]
.as_str()
.ok_or_else(|| PawanError::Tool("prompt is required for spawn_agent".into()))?;
let timeout = args["timeout"].as_u64().unwrap_or(120);
let model = args["model"].as_str();
let workspace = args["workspace"]
.as_str()
.map(PathBuf::from)
.unwrap_or_else(|| self.workspace_root.clone());
let max_retries = args["retries"].as_u64().unwrap_or(0).min(2) as usize;
let agent_id = uuid::Uuid::new_v4().to_string()[..8].to_string();
let status_path = format!("/tmp/pawan-agent-{}.status", agent_id);
let started_at = chrono::Utc::now().to_rfc3339();
let pawan_bin = self.find_pawan_binary();
for attempt in 0..=max_retries {
let mut cmd = Command::new(&pawan_bin);
cmd.arg("run")
.arg("-o")
.arg("json")
.arg("--timeout")
.arg(timeout.to_string())
.arg("-w")
.arg(workspace.to_string_lossy().to_string());
if let Some(m) = model {
cmd.arg("-m").arg(m);
}
cmd.arg(prompt);
cmd.stdout(Stdio::piped())
.stderr(Stdio::piped())
.stdin(Stdio::null());
if let Ok(mut f) = std::fs::File::create(&status_path) {
let _ = write!(f, r#"{{"state":"running","prompt":"{}","started_at":"{}","attempt":{}}}"#,
prompt.chars().take(100).collect::<String>().replace('"', "'"), started_at, attempt + 1);
}
let mut child = cmd.spawn().map_err(|e| {
PawanError::Tool(format!(
"Failed to spawn sub-agent: {}. Binary: {}",
e, pawan_bin
))
})?;
let mut stdout = String::new();
let mut stderr = String::new();
if let Some(mut handle) = child.stdout.take() {
handle.read_to_string(&mut stdout).await.ok();
}
if let Some(mut handle) = child.stderr.take() {
handle.read_to_string(&mut stderr).await.ok();
}
let status = child.wait().await.map_err(PawanError::Io)?;
let result = if let Ok(json_result) = serde_json::from_str::<Value>(&stdout) {
json_result
} else {
json!({
"content": stdout.trim(),
"raw_output": true
})
};
if status.success() || attempt == max_retries {
let duration_ms = chrono::Utc::now().signed_duration_since(chrono::DateTime::parse_from_rfc3339(&started_at).unwrap_or_default()).num_milliseconds();
if let Ok(mut f) = std::fs::File::create(&status_path) {
let state = if status.success() { "done" } else { "failed" };
let _ = write!(f, r#"{{"state":"{}","exit_code":{},"duration_ms":{},"attempt":{}}}"#,
state, status.code().unwrap_or(-1), duration_ms, attempt + 1);
}
return Ok(json!({
"success": status.success(),
"attempt": attempt + 1,
"total_attempts": attempt + 1,
"result": result,
"stderr": stderr.trim(),
}));
}
tracing::warn!(attempt = attempt + 1, "spawn_agent attempt failed, retrying");
}
Err(PawanError::Tool("spawn_agent: all retry attempts exhausted".into()))
}
}
pub struct SpawnAgentsTool {
workspace_root: PathBuf,
}
impl SpawnAgentsTool {
pub fn new(workspace_root: PathBuf) -> Self {
Self { workspace_root }
}
}
#[async_trait]
impl Tool for SpawnAgentsTool {
fn name(&self) -> &str {
"spawn_agents"
}
fn description(&self) -> &str {
"Spawn multiple sub-agents in parallel. Each task runs concurrently and results are returned as an array."
}
fn parameters_schema(&self) -> Value {
json!({
"type": "object",
"properties": {
"tasks": {
"type": "array",
"items": {
"type": "object",
"properties": {
"prompt": {"type": "string"},
"model": {"type": "string"},
"timeout": {"type": "integer"},
"workspace": {"type": "string"}
},
"required": ["prompt"]
}
}
},
"required": ["tasks"]
})
}
fn thulp_definition(&self) -> thulp_core::ToolDefinition {
use thulp_core::{Parameter, ParameterType};
thulp_core::ToolDefinition::builder("spawn_agents")
.description(self.description())
.parameter(Parameter::builder("tasks").param_type(ParameterType::Array).required(true)
.description("Array of task objects, each with prompt (required), model, timeout, workspace").build())
.build()
}
async fn execute(&self, args: Value) -> Result<Value> {
let tasks = args["tasks"]
.as_array()
.ok_or_else(|| PawanError::Tool("tasks array is required for spawn_agents".into()))?;
let single_tool = SpawnAgentTool::new(self.workspace_root.clone());
let futures: Vec<_> = tasks
.iter()
.map(|task| single_tool.execute(task.clone()))
.collect();
let results = futures::future::join_all(futures).await;
let output: Vec<Value> = results
.into_iter()
.map(|r| match r {
Ok(v) => v,
Err(e) => json!({"success": false, "error": e.to_string()}),
})
.collect();
Ok(json!({
"success": true,
"results": output,
"total_tasks": tasks.len(),
}))
}
}
#[cfg(test)]
mod tests {
use super::*;
use tempfile::TempDir;
#[test]
fn test_spawn_agent_tool_name() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
assert_eq!(tool.name(), "spawn_agent");
}
#[test]
fn test_spawn_agents_tool_name() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
assert_eq!(tool.name(), "spawn_agents");
}
#[test]
fn test_spawn_agent_schema_has_prompt() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let schema = tool.parameters_schema();
assert!(schema["properties"]["prompt"].is_object());
assert!(schema["required"].as_array().unwrap().iter().any(|v| v == "prompt"));
}
#[test]
fn test_find_pawan_binary_prefers_release_over_debug() {
let tmp = TempDir::new().unwrap();
std::fs::create_dir_all(tmp.path().join("target/release")).unwrap();
std::fs::create_dir_all(tmp.path().join("target/debug")).unwrap();
let release = tmp.path().join("target/release/pawan");
let debug = tmp.path().join("target/debug/pawan");
std::fs::write(&release, "#!/bin/sh\necho release").unwrap();
std::fs::write(&debug, "#!/bin/sh\necho debug").unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let binary = tool.find_pawan_binary();
assert_eq!(
binary,
release.to_string_lossy().to_string(),
"release binary must win over debug"
);
}
#[test]
fn test_find_pawan_binary_falls_back_to_debug_when_no_release() {
let tmp = TempDir::new().unwrap();
std::fs::create_dir_all(tmp.path().join("target/debug")).unwrap();
let debug = tmp.path().join("target/debug/pawan");
std::fs::write(&debug, "#!/bin/sh\necho debug").unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let binary = tool.find_pawan_binary();
assert_eq!(binary, debug.to_string_lossy().to_string());
}
#[test]
fn test_find_pawan_binary_falls_through_to_path_when_nothing_in_workspace() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let binary = tool.find_pawan_binary();
assert_eq!(binary, "pawan");
}
#[tokio::test]
async fn test_spawn_agent_missing_prompt_errors() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let result = tool.execute(json!({ "model": "test-model" })).await;
assert!(result.is_err(), "missing prompt must error");
let err = format!("{}", result.unwrap_err());
assert!(
err.contains("prompt"),
"error message should mention prompt, got: {}",
err
);
}
#[test]
fn test_spawn_agents_schema_requires_tasks_array() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let schema = tool.parameters_schema();
let required = schema["required"].as_array().unwrap();
assert!(required.iter().any(|v| v == "tasks"), "tasks must be required");
let tasks_type = schema["properties"]["tasks"]["type"].as_str();
assert_eq!(tasks_type, Some("array"));
}
#[tokio::test]
async fn test_spawn_agents_empty_tasks_succeeds_with_zero_results() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let result = tool.execute(json!({ "tasks": [] })).await.unwrap();
assert_eq!(result["success"], true);
assert_eq!(result["total_tasks"], 0);
assert_eq!(result["results"].as_array().unwrap().len(), 0);
}
#[tokio::test]
async fn test_spawn_agents_missing_tasks_errors() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let result = tool.execute(json!({})).await;
assert!(result.is_err());
let err = format!("{}", result.unwrap_err());
assert!(err.contains("tasks"));
}
#[tokio::test]
async fn test_spawn_agent_prompt_non_string_errors() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let result = tool.execute(json!({ "prompt": 42 })).await;
assert!(result.is_err(), "non-string prompt must error");
let err = format!("{}", result.unwrap_err());
assert!(err.contains("prompt"), "error should mention 'prompt', got: {}", err);
}
#[tokio::test]
async fn test_spawn_agents_tasks_non_array_errors() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let result = tool.execute(json!({ "tasks": "not an array" })).await;
assert!(result.is_err(), "non-array tasks must error");
let err = format!("{}", result.unwrap_err());
assert!(err.contains("tasks"), "error should mention 'tasks', got: {}", err);
}
#[test]
fn test_spawn_agent_schema_lists_all_optional_params() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let schema = tool.parameters_schema();
let props = schema["properties"].as_object().unwrap();
for p in &["prompt", "model", "timeout", "workspace", "retries"] {
assert!(props.contains_key(*p), "schema missing '{}'", p);
}
let required = schema["required"].as_array().unwrap();
assert_eq!(required.len(), 1);
assert_eq!(required[0], "prompt");
}
#[test]
fn test_spawn_agents_schema_tasks_items_has_prompt_required() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let schema = tool.parameters_schema();
let items_required = schema["properties"]["tasks"]["items"]["required"]
.as_array()
.expect("tasks.items.required should exist");
assert!(items_required.iter().any(|v| v == "prompt"));
}
#[test]
fn test_spawn_agent_thulp_definition_has_all_5_params() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentTool::new(tmp.path().to_path_buf());
let def = tool.thulp_definition();
assert_eq!(def.name, "spawn_agent");
let param_names: Vec<&str> = def.parameters.iter().map(|p| p.name.as_str()).collect();
for p in &["prompt", "model", "timeout", "workspace", "retries"] {
assert!(param_names.contains(p), "thulp definition missing '{}'", p);
}
let required_count = def.parameters.iter().filter(|p| p.required).count();
assert_eq!(required_count, 1, "only prompt should be required");
}
#[test]
fn test_spawn_agents_thulp_definition_has_tasks_param() {
let tmp = TempDir::new().unwrap();
let tool = SpawnAgentsTool::new(tmp.path().to_path_buf());
let def = tool.thulp_definition();
assert_eq!(def.name, "spawn_agents");
assert_eq!(def.parameters.len(), 1);
let tasks_param = &def.parameters[0];
assert_eq!(tasks_param.name, "tasks");
assert!(tasks_param.required);
}
}