use thiserror::Error;
#[derive(Debug, Error)]
pub enum ToolError {
#[error("Tool not found: {0}")]
NotFound(String),
#[error("Invalid input: {0}")]
InvalidInput(String),
#[error("Execution error: {0}")]
Execution(String),
#[error("Permission denied: {0}")]
PermissionDenied(String),
#[error("Tool requires approval: {0}")]
ApprovalRequired(String),
#[error("File not found: {0}")]
FileNotFound(String),
#[error("I/O error: {0}")]
Io(#[from] std::io::Error),
#[error("JSON error: {0}")]
Json(#[from] serde_json::Error),
#[error("Tool execution timed out after {0}s")]
Timeout(u64),
#[error("Internal error: {0}")]
Internal(String),
}
impl ToolError {
pub fn is_pre_execution_miss(&self) -> bool {
matches!(self, ToolError::NotFound(_) | ToolError::InvalidInput(_))
}
}
pub type Result<T> = std::result::Result<T, ToolError>;
pub fn expand_tilde(path: &str) -> std::path::PathBuf {
use std::path::PathBuf;
if let Some(rest) = path.strip_prefix("~/")
&& let Some(home) = dirs::home_dir()
{
return home.join(rest);
}
if path == "~"
&& let Some(home) = dirs::home_dir()
{
return home;
}
PathBuf::from(path)
}
pub fn collapse_home(path: &std::path::Path) -> String {
if let Some(home) = dirs::home_dir()
&& let Ok(rest) = path.strip_prefix(&home)
{
if rest.as_os_str().is_empty() {
return "~".to_string();
}
let suffix = rest.display().to_string();
if let Some(stripped) = suffix.strip_prefix('/') {
return format!("~/{}", stripped);
}
return format!("~/{}", suffix);
}
path.display().to_string()
}
pub fn resolve_tool_path(
requested_path: &str,
working_directory: &std::path::Path,
) -> std::path::PathBuf {
let expanded = expand_tilde(requested_path);
if expanded.is_absolute() {
expanded
} else {
working_directory.join(expanded)
}
}
pub fn validate_path_safety(
requested_path: &str,
working_directory: &std::path::Path,
) -> Result<std::path::PathBuf> {
let path = resolve_tool_path(requested_path, working_directory);
if !path.exists() {
let parent = path
.parent()
.ok_or_else(|| ToolError::InvalidInput("Invalid path: no parent directory".into()))?;
if !parent.exists() {
return Err(ToolError::InvalidInput(format!(
"Parent directory does not exist: {}",
parent.display()
)));
}
}
Ok(path)
}
pub fn validate_file_path(
requested_path: &str,
working_directory: &std::path::Path,
) -> std::result::Result<std::path::PathBuf, String> {
let path = match validate_path_safety(requested_path, working_directory) {
Ok(p) => p,
Err(ToolError::InvalidInput(msg)) => {
return Err(format!("Invalid path: {}", msg));
}
Err(e) => {
return Err(format!("Path validation failed: {}", e));
}
};
if !path.exists() {
return Err(format!("File not found: {}", path.display()));
}
if !path.is_file() {
return Err(format!("Path is not a file: {}", path.display()));
}
Ok(path)
}
pub fn validate_directory_path(
requested_path: &str,
working_directory: &std::path::Path,
) -> std::result::Result<std::path::PathBuf, String> {
let path = match validate_path_safety(requested_path, working_directory) {
Ok(p) => p,
Err(ToolError::InvalidInput(msg)) => {
return Err(format!("Invalid path: {}", msg));
}
Err(e) => {
return Err(format!("Path validation failed: {}", e));
}
};
if !path.exists() {
return Err(format!("Directory not found: {}", path.display()));
}
if !path.is_dir() {
return Err(format!("Path is not a directory: {}", path.display()));
}
Ok(path)
}