use rmcp::model::Tool;
use rmcp::ErrorData;
use serde_json::{json, Map, Value};
use crate::server::tool_trait::{get_str, McpTool, ToolContext, ToolOutput};
use crate::tool_defs::tool_def;
pub struct CtxComposeTool;
impl McpTool for CtxComposeTool {
fn name(&self) -> &'static str {
"ctx_compose"
}
fn tool_def(&self) -> Tool {
tool_def(
"ctx_compose",
"Task composer: one call returns keywords + semantically ranked files + exact match locations + the top symbol's body inline. Replaces the search→read→outline→read chain.",
json!({
"type": "object",
"properties": {
"task": { "type": "string", "description": "Natural-language task or question" },
"path": { "type": "string", "description": "Project root (default: .)" }
},
"required": ["task"]
}),
)
}
fn handle(
&self,
args: &Map<String, Value>,
ctx: &ToolContext,
) -> Result<ToolOutput, ErrorData> {
let task = get_str(args, "task")
.ok_or_else(|| ErrorData::invalid_params("task is required", None))?;
let path = if let Some(p) = ctx.resolved_path("path") {
p.to_string()
} else if let Some(err) = ctx.path_error("path") {
return Err(ErrorData::invalid_params(format!("path: {err}"), None));
} else {
ctx.project_root.clone()
};
if let Some(ref cache) = ctx.bm25_cache {
crate::tools::ctx_semantic_search::set_thread_cache(cache.clone());
}
let (text, sent) = tokio::task::block_in_place(|| {
crate::tools::ctx_compose::handle(&task, &path, ctx.crp_mode)
});
if text.starts_with("ERROR") {
return Err(ErrorData::invalid_params(text, None));
}
Ok(ToolOutput {
text,
original_tokens: sent,
saved_tokens: 0,
mode: Some("compose".to_string()),
path: Some(path),
changed: false,
})
}
}