use rmcp::model::Tool;
use rmcp::ErrorData;
use serde_json::{json, Map, Value};
use crate::server::tool_trait::{
get_bool, get_int, get_str, require_resolved_path, McpTool, ToolContext, ToolOutput,
};
use crate::tool_defs::tool_def;
pub struct CtxEditTool;
impl McpTool for CtxEditTool {
fn name(&self) -> &'static str {
"ctx_edit"
}
fn tool_def(&self) -> Tool {
tool_def(
"ctx_edit",
"Edit a file via search-and-replace. Works without native Read/Edit tools. Use this when the IDE's Edit tool requires Read but Read is unavailable.",
json!({
"type": "object",
"properties": {
"path": { "type": "string", "description": "Absolute file path" },
"old_string": { "type": "string", "description": "Exact text to find and replace (must be unique unless replace_all=true)" },
"new_string": { "type": "string", "description": "Replacement text" },
"replace_all": { "type": "boolean", "description": "Replace all occurrences (default: false)", "default": false },
"create": { "type": "boolean", "description": "Create a new file with new_string as content (ignores old_string)", "default": false }
},
"required": ["path", "new_string"]
}),
)
}
fn handle(
&self,
args: &Map<String, Value>,
ctx: &ToolContext,
) -> Result<ToolOutput, ErrorData> {
let path = require_resolved_path(ctx, args, "path")?;
let old_string = get_str(args, "old_string").unwrap_or_default();
let new_string = get_str(args, "new_string")
.ok_or_else(|| ErrorData::invalid_params("new_string is required", None))?;
let replace_all = get_bool(args, "replace_all").unwrap_or(false);
let create = get_bool(args, "create").unwrap_or(false);
let expected_md5 = get_str(args, "expected_md5");
let expected_size = get_int(args, "expected_size").and_then(|v| u64::try_from(v).ok());
let expected_mtime_ms =
get_int(args, "expected_mtime_ms").and_then(|v| u64::try_from(v).ok());
let backup = get_bool(args, "backup").unwrap_or(false);
let backup_path = get_str(args, "backup_path")
.map(|p| ctx.resolved_paths.get("backup_path").cloned().unwrap_or(p));
let evidence = get_bool(args, "evidence").unwrap_or(true);
let diff_max_lines = get_int(args, "diff_max_lines")
.and_then(|v| usize::try_from(v.max(0)).ok())
.unwrap_or(200);
let allow_lossy_utf8 = get_bool(args, "allow_lossy_utf8").unwrap_or(false);
let edit_params = crate::tools::ctx_edit::EditParams {
path: path.clone(),
old_string,
new_string,
replace_all,
create,
expected_md5,
expected_size,
expected_mtime_ms,
backup,
backup_path,
evidence,
diff_max_lines,
allow_lossy_utf8,
};
tokio::task::block_in_place(|| {
let cache_lock = ctx
.cache
.as_ref()
.ok_or_else(|| ErrorData::internal_error("cache not available", None))?;
let rt = tokio::runtime::Handle::current();
let file_lock = crate::core::path_locks::per_file_lock(&path);
let _file_guard = {
let deadline = std::time::Instant::now() + std::time::Duration::from_secs(30);
loop {
if let Ok(guard) = file_lock.try_lock() {
break guard;
}
if std::time::Instant::now() >= deadline {
return Err(ErrorData::internal_error(
format!("per-file edit lock contention for {path} — another edit to the same file is in progress, retry in a moment"),
None,
));
}
std::thread::sleep(std::time::Duration::from_millis(20));
}
};
let last_mode = match rt.block_on(tokio::time::timeout(
std::time::Duration::from_secs(5),
cache_lock.read(),
)) {
Ok(cache) => cache
.get(&path)
.map(|e| e.last_mode.clone())
.unwrap_or_default(),
Err(_) => String::new(),
};
let (output, effect) = crate::tools::ctx_edit::run_io(&edit_params, &last_mode);
if !matches!(effect, crate::tools::ctx_edit::CacheEffect::None) {
match rt.block_on(tokio::time::timeout(
std::time::Duration::from_secs(5),
cache_lock.write(),
)) {
Ok(mut cache) => {
crate::tools::ctx_edit::apply_cache_effect(&mut cache, &path, effect);
}
Err(_) => {
tracing::warn!(
"ctx_edit: cache write-lock timeout (5s) applying post-edit cache effect for {path}"
);
}
}
}
if let Some(session_lock) = ctx.session.as_ref() {
let guard = rt.block_on(tokio::time::timeout(
std::time::Duration::from_secs(5),
session_lock.write(),
));
if let Ok(mut session) = guard {
session.mark_modified(&path);
}
}
Ok(ToolOutput {
text: output,
original_tokens: 0,
saved_tokens: 0,
mode: None,
path: Some(path),
changed: false,
})
})
}
}