use rmcp::model::Tool;
use rmcp::ErrorData;
use serde_json::{Map, Value};
pub struct ToolOutput {
pub text: String,
pub original_tokens: usize,
pub saved_tokens: usize,
pub mode: Option<String>,
pub path: Option<String>,
}
impl ToolOutput {
pub fn simple(text: String) -> Self {
Self {
text,
original_tokens: 0,
saved_tokens: 0,
mode: None,
path: None,
}
}
pub fn with_savings(text: String, original: usize, saved: usize) -> Self {
Self {
text,
original_tokens: original,
saved_tokens: saved,
mode: None,
path: None,
}
}
}
pub trait McpTool: Send + Sync {
fn name(&self) -> &'static str;
fn tool_def(&self) -> Tool;
fn handle(&self, args: &Map<String, Value>, ctx: &ToolContext)
-> Result<ToolOutput, ErrorData>;
}
pub struct ToolContext {
pub project_root: String,
pub minimal: bool,
pub resolved_paths: std::collections::HashMap<String, String>,
pub crp_mode: crate::tools::CrpMode,
pub cache: Option<crate::tools::SharedCache>,
pub session: Option<std::sync::Arc<tokio::sync::RwLock<crate::core::session::SessionState>>>,
pub tool_calls:
Option<std::sync::Arc<tokio::sync::RwLock<Vec<crate::core::protocol::ToolCallRecord>>>>,
pub agent_id: Option<std::sync::Arc<tokio::sync::RwLock<Option<String>>>>,
pub workflow:
Option<std::sync::Arc<tokio::sync::RwLock<Option<crate::core::workflow::WorkflowRun>>>>,
pub ledger:
Option<std::sync::Arc<tokio::sync::RwLock<crate::core::context_ledger::ContextLedger>>>,
pub client_name: Option<std::sync::Arc<tokio::sync::RwLock<String>>>,
pub pipeline_stats:
Option<std::sync::Arc<tokio::sync::RwLock<crate::core::pipeline::PipelineStats>>>,
pub call_count: Option<std::sync::Arc<std::sync::atomic::AtomicUsize>>,
pub autonomy: Option<std::sync::Arc<crate::tools::autonomy::AutonomyState>>,
}
impl ToolContext {
pub fn resolved_path(&self, arg: &str) -> Option<&str> {
self.resolved_paths.get(arg).map(String::as_str)
}
pub fn resolve_path_sync(&self, path: &str) -> Result<String, String> {
let normalized = crate::core::pathutil::normalize_tool_path(path);
if normalized.is_empty() || normalized == "." {
return Ok(normalized);
}
let p = std::path::Path::new(&normalized);
let resolved = if p.is_absolute() || p.exists() {
std::path::PathBuf::from(&normalized)
} else {
let joined = std::path::Path::new(&self.project_root).join(&normalized);
if joined.exists() {
joined
} else {
std::path::Path::new(&self.project_root).join(&normalized)
}
};
let jail_root = std::path::Path::new(&self.project_root);
let jailed = crate::core::pathjail::jail_path(&resolved, jail_root)?;
crate::core::io_boundary::check_secret_path_for_tool("resolve_path", &jailed)?;
Ok(crate::core::pathutil::normalize_tool_path(
&jailed.to_string_lossy().replace('\\', "/"),
))
}
}
pub fn get_str(args: &Map<String, Value>, key: &str) -> Option<String> {
args.get(key).and_then(|v| v.as_str()).map(String::from)
}
pub fn get_int(args: &Map<String, Value>, key: &str) -> Option<i64> {
args.get(key).and_then(serde_json::Value::as_i64)
}
pub fn get_bool(args: &Map<String, Value>, key: &str) -> Option<bool> {
args.get(key).and_then(serde_json::Value::as_bool)
}
pub fn get_str_array(args: &Map<String, Value>, key: &str) -> Option<Vec<String>> {
args.get(key).and_then(|v| v.as_array()).map(|arr| {
arr.iter()
.filter_map(|v| v.as_str().map(String::from))
.collect()
})
}