use glob::glob;
use std::fmt::Write as FmtWrite;
use std::fs::{self, File};
use std::io::Write;
use std::path::{Path, PathBuf};
use std::sync::Arc;
use tokio::sync::Mutex;
use tracing::{debug, warn};
use crate::errors::{Result, WinxError};
use crate::state::bash_state::BashState;
use crate::state::persistence::{
load_bash_state_from_path, save_bash_state_to_path, BashStateSnapshot,
};
use crate::types::ContextSave;
use crate::utils::path::expand_user;
pub async fn handle_tool_call(
bash_state: &Arc<Mutex<Option<BashState>>>,
args: ContextSave,
) -> Result<String> {
let bash_state_guard = bash_state.lock().await;
let bash_state = bash_state_guard.as_ref().ok_or(WinxError::BashStateNotInitialized)?;
let result = save_context(bash_state, args)?;
if should_open_context_file(&result) {
if let Err(e) = try_open_file(&result) {
debug!("Failed to open the context file: {}", e);
}
}
Ok(result)
}
fn save_context(bash_state: &BashState, mut context: ContextSave) -> Result<String> {
normalize_context(&mut context)?;
let (relevant_files, warnings) = collect_relevant_files(&context)?;
let memory_dir = resolve_memory_dir()?;
let relevant_files_data = read_files_content(&relevant_files, 10_000)?;
let memory_data = format_memory(&context, &relevant_files_data);
let safe_id = sanitize_filename(&context.id);
let (memory_file_path, state_file_path) = context_paths_from_dir(&memory_dir, &safe_id);
if let Some(response) = write_memory_file(&memory_file_path, &memory_data, &context)? {
return Ok(response);
}
write_bash_state_file(&state_file_path, bash_state);
Ok(context_save_response(&relevant_files, &warnings, &context, &memory_file_path))
}
fn normalize_context(context: &mut ContextSave) -> Result<()> {
if !context.project_root_path.is_empty() {
context.project_root_path = expand_user(&context.project_root_path);
}
if context.id.is_empty() {
return Err(WinxError::ArgumentParseError("Task ID cannot be empty".to_string()));
}
Ok(())
}
fn collect_relevant_files(context: &ContextSave) -> Result<(Vec<PathBuf>, Vec<String>)> {
let mut relevant_files = Vec::new();
let mut warnings = Vec::new();
for glob_pattern in &context.relevant_file_globs {
let expanded_glob = expand_user(glob_pattern);
let final_glob =
if !Path::new(&expanded_glob).is_absolute() && !context.project_root_path.is_empty() {
PathBuf::from(&context.project_root_path)
.join(expanded_glob)
.to_string_lossy()
.to_string()
} else {
expanded_glob
};
debug!("Processing glob pattern: {}", final_glob);
let matches = glob(&final_glob).map_err(|e| {
WinxError::ArgumentParseError(format!("Invalid glob pattern '{final_glob}': {e}"))
})?;
let mut found_files = false;
for entry in matches {
match entry {
Ok(path) => {
if path.is_file() {
relevant_files.push(path);
found_files = true;
if relevant_files.len() >= 1000 {
warn!("Reached limit of 1000 files for glob '{}'", final_glob);
break;
}
}
}
Err(e) => {
warn!("Error matching glob '{}': {}", final_glob, e);
}
}
}
if !found_files {
warnings.push(format!("Warning: No files found for the glob: {glob_pattern}"));
}
}
debug!("Found {} relevant files", relevant_files.len());
Ok((relevant_files, warnings))
}
fn resolve_memory_dir() -> Result<PathBuf> {
let app_dir = match get_app_dir_xdg() {
Ok(dir) => dir,
Err(e) => {
debug!("Failed to get primary app directory: {:?}", e);
let fallback = std::env::temp_dir().join("winx-memory");
debug!("Using fallback directory: {}", fallback.display());
fs::create_dir_all(&fallback).map_err(|e2| WinxError::FileAccessError {
path: fallback.clone(),
message: format!(
"Failed to create fallback directory: {e2} (after previous error: {e:?})"
),
})?;
fallback
}
};
let mut memory_dir = app_dir.join("memory");
match fs::create_dir_all(&memory_dir) {
Ok(()) => {}
Err(e) => {
debug!("Failed to create memory directory: {}", e);
debug!("Using app_dir directly as memory_dir due to failed subdirectory creation");
memory_dir = app_dir;
}
}
Ok(memory_dir)
}
pub(crate) fn context_paths(id: &str) -> Result<(PathBuf, PathBuf)> {
let safe_id = sanitize_filename(id);
let memory_dir = resolve_memory_dir()?;
Ok(context_paths_from_dir(&memory_dir, &safe_id))
}
fn context_paths_from_dir(memory_dir: &Path, safe_id: &str) -> (PathBuf, PathBuf) {
(
memory_dir.join(format!("{safe_id}.txt")),
memory_dir.join(format!("{safe_id}_bash_state.json")),
)
}
pub(crate) fn load_saved_context(id: &str) -> Result<Option<(String, Option<BashStateSnapshot>)>> {
if id.is_empty() {
return Ok(None);
}
let (memory_file_path, state_file_path) = context_paths(id)?;
if !memory_file_path.exists() {
return Ok(None);
}
let memory_data =
fs::read_to_string(&memory_file_path).map_err(|e| WinxError::FileAccessError {
path: memory_file_path.clone(),
message: format!("Failed to read saved context: {e}"),
})?;
let state = load_bash_state_from_path(&state_file_path).map_err(|e| {
WinxError::SerializationError(format!("Failed to load saved bash state: {e}"))
})?;
Ok(Some((memory_data, state)))
}
fn write_memory_file(
memory_file_path: &Path,
memory_data: &str,
context: &ContextSave,
) -> Result<Option<String>> {
match File::create(memory_file_path) {
Ok(mut file) => {
if let Err(e) = file.write_all(memory_data.as_bytes()) {
warn!("Failed to write memory data: {}", e);
return Ok(Some(save_to_temp_file(memory_data, context)?));
}
}
Err(e) => {
warn!("Failed to create memory file: {}", e);
return Ok(Some(save_to_temp_file(memory_data, context)?));
}
}
Ok(None)
}
fn write_bash_state_file(state_file_path: &Path, bash_state: &BashState) {
if let Err(e) = save_bash_state_to_path(state_file_path, &bash_state.snapshot()) {
warn!("Failed to write bash state data: {}", e);
}
}
fn context_save_response(
relevant_files: &[PathBuf],
warnings: &[String],
context: &ContextSave,
memory_file_path: &Path,
) -> String {
let memory_file_path_str = memory_file_path.to_string_lossy().to_string();
if !relevant_files.is_empty() || context.relevant_file_globs.is_empty() {
if warnings.is_empty() {
memory_file_path_str
} else {
format!(
"{}\n\nContext file successfully saved at {}",
warnings.join("\n"),
memory_file_path_str
)
}
} else {
format!(
"Error: No files found for the given globs. Context file successfully saved at \"{memory_file_path_str}\", but please fix the error."
)
}
}
fn format_memory(context: &ContextSave, relevant_files_data: &str) -> String {
let mut memory_data = String::new();
if !context.project_root_path.is_empty() {
let _ = write!(memory_data, "Project root path: {}\n\n", context.project_root_path);
}
memory_data.push_str(&context.description);
memory_data.push_str("\n\n");
let _ =
write!(memory_data, "Relevant file globs: {}\n\n", context.relevant_file_globs.join(", "));
memory_data.push_str("File contents:\n\n");
memory_data.push_str(relevant_files_data);
memory_data
}
fn get_app_dir_xdg() -> Result<PathBuf> {
let app_dir = get_primary_app_dir().or_else(|_| get_fallback_app_dir())?;
debug!("Using app directory: {}", app_dir.display());
Ok(app_dir)
}
fn get_primary_app_dir() -> Result<PathBuf> {
if let Ok(xdg_path) = std::env::var("XDG_DATA_HOME") {
let app_dir = PathBuf::from(xdg_path).join("winx");
if let Ok(()) = fs::create_dir_all(&app_dir) {
return Ok(app_dir);
}
}
if let Ok(home) = std::env::var("HOME") {
let app_dir = PathBuf::from(home).join(".local/share/winx");
if let Ok(()) = fs::create_dir_all(&app_dir) {
return Ok(app_dir);
}
}
Err(WinxError::FileAccessError {
path: PathBuf::from("<primary-paths>"),
message: "Could not create app directory in primary locations".to_string(),
})
}
fn get_fallback_app_dir() -> Result<PathBuf> {
let current_dir = std::env::current_dir().unwrap_or_else(|_| PathBuf::from("."));
let app_dir = current_dir.join(".winx-data");
if let Ok(()) = fs::create_dir_all(&app_dir) {
return Ok(app_dir);
}
let temp_dir = std::env::temp_dir().join("winx-data");
fs::create_dir_all(&temp_dir).map_err(|e| WinxError::FileAccessError {
path: temp_dir.clone(),
message: format!("Failed to create app directory in any location: {e}"),
})?;
Ok(temp_dir)
}
fn read_files_content(file_paths: &[PathBuf], max_files: usize) -> Result<String> {
let mut result = String::new();
let mut skipped_binary = Vec::new();
for (i, path) in file_paths.iter().take(max_files).enumerate() {
match fs::read_to_string(path) {
Ok(file_content) => {
let _ = writeln!(result, "--- File {}: {} ---", i + 1, path.display());
result.push_str(&file_content);
result.push_str("\n\n");
}
Err(e) if e.kind() == std::io::ErrorKind::InvalidData => {
skipped_binary.push(path.display().to_string());
}
Err(e) => {
return Err(WinxError::FileAccessError {
path: path.clone(),
message: format!("Failed to read file: {e}"),
});
}
}
}
if !skipped_binary.is_empty() {
let _ = write!(
result,
"Note: Skipped {} binary/non-text file(s): {}\n\n",
skipped_binary.len(),
skipped_binary.join(", ")
);
}
if file_paths.len() > max_files {
let _ = writeln!(
result,
"Note: Only showing the first {} files out of {}.",
max_files,
file_paths.len()
);
}
Ok(result)
}
fn try_open_file(file_path: &str) -> Result<()> {
if std::env::consts::OS != "macos" && std::env::consts::OS != "linux" {
return Ok(());
}
let cmd = if std::env::consts::OS == "macos" {
"open"
} else {
for cmd in &["xdg-open", "gnome-open", "kde-open"] {
let status = std::process::Command::new("which")
.arg(cmd)
.stdout(std::process::Stdio::null())
.stderr(std::process::Stdio::null())
.status();
if let Ok(status) = status {
if status.success() {
let _ = std::process::Command::new(cmd)
.arg(file_path)
.stdout(std::process::Stdio::null())
.stderr(std::process::Stdio::null())
.spawn()
.map_err(|e| {
WinxError::CommandExecutionError(format!(
"Failed to spawn open command: {e}"
))
})?;
return Ok(());
}
}
}
return Ok(());
};
let _ = std::process::Command::new(cmd)
.arg(file_path)
.stdout(std::process::Stdio::null())
.stderr(std::process::Stdio::null())
.spawn()
.map_err(|e| {
WinxError::CommandExecutionError(format!("Failed to spawn open command: {e}"))
})?;
Ok(())
}
fn should_open_context_file(file_path: &str) -> bool {
std::env::var("WINX_OPEN_CONTEXT").is_ok_and(|value| value == "1" || value == "true")
&& Path::new(file_path).is_file()
&& !cfg!(test)
}
fn save_to_temp_file(memory_data: &str, context: &ContextSave) -> Result<String> {
let temp_dir = std::env::temp_dir();
let safe_id = sanitize_filename(&context.id);
let temp_file_path = temp_dir.join(format!("winx-{safe_id}.txt"));
let mut file = File::create(&temp_file_path).map_err(|e| WinxError::FileAccessError {
path: temp_file_path.clone(),
message: format!("Failed to create temporary file: {e}"),
})?;
file.write_all(memory_data.as_bytes()).map_err(|e| WinxError::FileAccessError {
path: temp_file_path.clone(),
message: format!("Failed to write to temporary file: {e}"),
})?;
let path_str = temp_file_path.to_string_lossy().to_string();
Ok(format!(
"Context was saved to temporary file at {path_str} due to permission issues with regular locations."
))
}
fn sanitize_filename(input: &str) -> String {
let invalid_chars = vec!['/', '\\', ':', '*', '?', '"', '<', '>', '|'];
let mut result = input.to_string();
for c in invalid_chars {
result = result.replace(c, "_");
}
if result.len() > 50 {
use rand::RngExt;
result = format!("{}-{}", &result[0..45], rand::rng().random_range(1000..9999));
}
result
}