use serde::{Deserialize, Serialize}; use std::fs; #[derive(Debug, Clone, Serialize, Deserialize)] #[serde(rename_all = "camelCase")] pub struct EnvConflict { pub var_name: String, pub var_value: String, pub source_type: String, // "system" | "file" pub source_path: String, // Registry path or file path } #[cfg(target_os = "windows")] use winreg::enums::*; #[cfg(target_os = "windows")] use winreg::RegKey; /// Check environment variables for conflicts pub fn check_env_conflicts(app: &str) -> Result, String> { let keywords = get_keywords_for_app(app); let mut conflicts = Vec::new(); // Check system environment variables conflicts.extend(check_system_env(&keywords)?); // Check shell configuration files (Unix only) #[cfg(not(target_os = "windows"))] conflicts.extend(check_shell_configs(&keywords)?); Ok(conflicts) } /// Get relevant keywords for each app fn get_keywords_for_app(app: &str) -> Vec<&str> { match app.to_lowercase().as_str() { "claude" => vec!["ANTHROPIC"], "codex" => vec!["OPENAI"], _ => vec![], } } /// Check system environment variables (Windows Registry or Unix env) #[cfg(target_os = "windows")] fn check_system_env(keywords: &[&str]) -> Result, String> { let mut conflicts = Vec::new(); // Check HKEY_CURRENT_USER\Environment if let Ok(hkcu) = RegKey::predef(HKEY_CURRENT_USER).open_subkey("Environment") { for (name, value) in hkcu.enum_values().filter_map(Result::ok) { if keywords.iter().any(|k| name.to_uppercase().contains(k)) { if let Ok(val) = value.to_string() { conflicts.push(EnvConflict { var_name: name.clone(), var_value: val, source_type: "system".to_string(), source_path: "HKEY_CURRENT_USER\\Environment".to_string(), }); } } } } // Check HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Session Manager\Environment if let Ok(hklm) = RegKey::predef(HKEY_LOCAL_MACHINE) .open_subkey("SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment") { for (name, value) in hklm.enum_values().filter_map(Result::ok) { if keywords.iter().any(|k| name.to_uppercase().contains(k)) { if let Ok(val) = value.to_string() { conflicts.push(EnvConflict { var_name: name.clone(), var_value: val, source_type: "system".to_string(), source_path: "HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment".to_string(), }); } } } } Ok(conflicts) } #[cfg(not(target_os = "windows"))] fn check_system_env(keywords: &[&str]) -> Result, String> { let mut conflicts = Vec::new(); // Check current process environment for (key, value) in std::env::vars() { if keywords.iter().any(|k| key.to_uppercase().contains(k)) { conflicts.push(EnvConflict { var_name: key, var_value: value, source_type: "system".to_string(), source_path: "Process Environment".to_string(), }); } } Ok(conflicts) } /// Check shell configuration files for environment variable exports (Unix only) #[cfg(not(target_os = "windows"))] fn check_shell_configs(keywords: &[&str]) -> Result, String> { let mut conflicts = Vec::new(); let home = std::env::var("HOME").unwrap_or_else(|_| "/tmp".to_string()); let config_files = vec![ format!("{}/.bashrc", home), format!("{}/.bash_profile", home), format!("{}/.zshrc", home), format!("{}/.zprofile", home), format!("{}/.profile", home), "/etc/profile".to_string(), "/etc/bashrc".to_string(), ]; for file_path in config_files { if let Ok(content) = fs::read_to_string(&file_path) { // Parse lines for export statements for (line_num, line) in content.lines().enumerate() { let trimmed = line.trim(); // Match patterns like: export VAR=value or VAR=value if trimmed.starts_with("export ") || (!trimmed.starts_with('#') && trimmed.contains('=')) { let export_line = trimmed.strip_prefix("export ").unwrap_or(trimmed); if let Some(eq_pos) = export_line.find('=') { let var_name = export_line[..eq_pos].trim(); let var_value = export_line[eq_pos + 1..].trim(); // Check if variable name contains any keyword if keywords.iter().any(|k| var_name.to_uppercase().contains(k)) { conflicts.push(EnvConflict { var_name: var_name.to_string(), var_value: var_value.trim_matches('"').trim_matches('\'').to_string(), source_type: "file".to_string(), source_path: format!("{}:{}", file_path, line_num + 1), }); } } } } } } Ok(conflicts) } #[cfg(test)] mod tests { use super::*; #[test] fn test_get_keywords() { assert_eq!(get_keywords_for_app("claude"), vec!["ANTHROPIC"]); assert_eq!(get_keywords_for_app("codex"), vec!["OPENAI"]); assert_eq!(get_keywords_for_app("unknown"), Vec::<&str>::new()); } }