use rmcp::model::ErrorData;
use crate::error_meta;
fn validate_parent_in_root(
path: &str,
root: &std::path::Path,
) -> Result<std::path::PathBuf, ErrorData> {
let p = std::path::Path::new(path);
let file_name = p.file_name().ok_or_else(|| {
ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path must include a filename component".to_string(),
Some(error_meta(
"validation",
false,
"provide a path with a filename, not ending in '..' or '/'",
)),
)
})?;
let parent = p.parent().unwrap_or(std::path::Path::new(""));
let parent_path = if parent.as_os_str().is_empty() || parent == std::path::Path::new(".") {
root.to_path_buf()
} else {
root.join(parent)
};
let canonical_parent = std::fs::canonicalize(&parent_path).map_err(|e| {
io_error_to_path_error(
&e,
parent.to_str().unwrap_or("(invalid utf-8)"),
"provide a valid parent directory within the working directory",
)
})?;
if !canonical_parent.starts_with(root) {
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a path within the working directory",
)),
));
}
if !std::fs::metadata(&canonical_parent)
.map(|m| m.is_dir())
.unwrap_or(false)
{
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"parent path is not a directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a path whose parent is a directory",
)),
));
}
let resolved_path = canonical_parent.join(file_name);
if !resolved_path.starts_with(root) {
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a path within the working directory",
)),
));
}
Ok(resolved_path)
}
pub(crate) fn validate_path(
path: &str,
require_exists: bool,
) -> Result<std::path::PathBuf, ErrorData> {
let cwd = std::env::current_dir().map_err(|_| {
ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"ensure the working directory is accessible",
)),
)
})?;
let allowed_root = std::fs::canonicalize(&cwd).map_err(|_| {
ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"ensure the working directory is accessible",
)),
)
})?;
let canonical_path = if require_exists {
std::fs::canonicalize(path).map_err(|e| {
let msg = match e.kind() {
std::io::ErrorKind::NotFound => "path not found".to_string(),
std::io::ErrorKind::PermissionDenied => "permission denied".to_string(),
_ => "path is outside the working directory".to_string(),
};
ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
msg,
Some(error_meta(
"validation",
false,
"provide a valid path within the working directory",
)),
)
})?
} else {
validate_parent_in_root(path, &allowed_root)?
};
if !canonical_path.starts_with(&allowed_root) {
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a path within the current working directory",
)),
));
}
Ok(canonical_path)
}
pub(crate) fn io_error_to_path_error(
err: &std::io::Error,
path_context: &str,
suggested_action: &'static str,
) -> ErrorData {
let msg = match err.kind() {
std::io::ErrorKind::NotFound => format!("path not found: {path_context}"),
std::io::ErrorKind::PermissionDenied => format!("permission denied: {path_context}"),
_ => format!("path is invalid: {path_context}"),
};
let mut meta = error_meta("validation", false, suggested_action);
if let Some(obj) = meta.as_object_mut() {
obj.insert(
"ioErrorKind".to_string(),
serde_json::json!(format!("{:?}", err.kind())),
);
obj.insert(
"ioErrorSource".to_string(),
serde_json::json!(err.to_string()),
);
}
ErrorData::new(rmcp::model::ErrorCode::INVALID_PARAMS, msg, Some(meta))
}
pub(crate) fn validate_path_in_dir(
path: &str,
require_exists: bool,
working_dir: &std::path::Path,
) -> Result<std::path::PathBuf, ErrorData> {
let canonical_working_dir = std::fs::canonicalize(working_dir).map_err(|e| {
io_error_to_path_error(&e, "working_dir", "provide a valid working directory")
})?;
if !std::fs::metadata(&canonical_working_dir)
.map(|m| m.is_dir())
.unwrap_or(false)
{
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"working_dir must be a directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a valid directory path",
)),
));
}
let canonical_path = if require_exists {
let target_path = canonical_working_dir.join(path);
std::fs::canonicalize(&target_path).map_err(|e| {
io_error_to_path_error(
&e,
path,
"provide a valid path within the working directory",
)
})?
} else {
validate_parent_in_root(path, &canonical_working_dir)?
};
if !canonical_path.starts_with(&canonical_working_dir) {
return Err(ErrorData::new(
rmcp::model::ErrorCode::INVALID_PARAMS,
"path is outside the working directory".to_string(),
Some(error_meta(
"validation",
false,
"provide a path within the working directory",
)),
));
}
Ok(canonical_path)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn validate_path_no_trailing_slash() {
let input = "subdir/new_file.txt";
let result = validate_path(input, false);
if let Ok(resolved) = result {
let path_str = resolved.to_string_lossy();
assert!(
!path_str.ends_with('/'),
"resolved path must not end with trailing slash: {path_str}"
);
assert_eq!(
resolved.extension(),
Some(std::ffi::OsStr::new("txt")),
"file extension should be txt, path has trailing separator"
);
}
}
}