use crate::{ core::{CoreManager, handle, manager::RunningMode}, logging, utils::logging::Type, }; use anyhow::Result; use async_trait::async_trait; use once_cell::sync::OnceCell; use std::{fs, path::PathBuf}; use tauri::Manager; #[cfg(not(feature = "verge-dev"))] pub static APP_ID: &str = "io.github.clash-verge-rev.clash-verge-rev"; #[cfg(not(feature = "verge-dev"))] pub static BACKUP_DIR: &str = "clash-verge-rev-backup"; #[cfg(feature = "verge-dev")] pub static APP_ID: &str = "io.github.clash-verge-rev.clash-verge-rev.dev"; #[cfg(feature = "verge-dev")] pub static BACKUP_DIR: &str = "clash-verge-rev-backup-dev"; pub static PORTABLE_FLAG: OnceCell = OnceCell::new(); pub static CLASH_CONFIG: &str = "config.yaml"; pub static VERGE_CONFIG: &str = "verge.yaml"; pub static PROFILE_YAML: &str = "profiles.yaml"; pub static DNS_CONFIG: &str = "dns_config.yaml"; /// init portable flag pub fn init_portable_flag() -> Result<()> { use tauri::utils::platform::current_exe; let app_exe = current_exe()?; if let Some(dir) = app_exe.parent() { let dir = PathBuf::from(dir).join(".config/PORTABLE"); if dir.exists() { PORTABLE_FLAG.get_or_init(|| true); } } PORTABLE_FLAG.get_or_init(|| false); Ok(()) } /// get the verge app home dir pub fn app_home_dir() -> Result { use tauri::utils::platform::current_exe; let flag = PORTABLE_FLAG.get().unwrap_or(&false); if *flag { let app_exe = current_exe()?; let app_exe = dunce::canonicalize(app_exe)?; let app_dir = app_exe .parent() .ok_or_else(|| anyhow::anyhow!("failed to get the portable app dir"))?; return Ok(PathBuf::from(app_dir).join(".config").join(APP_ID)); } // 避免在Handle未初始化时崩溃 let app_handle = handle::Handle::app_handle(); match app_handle.path().data_dir() { Ok(dir) => Ok(dir.join(APP_ID)), Err(e) => { logging!( error, Type::File, "Failed to get the app home directory: {e}" ); Err(anyhow::anyhow!("Failed to get the app homedirectory")) } } } /// get the resources dir pub fn app_resources_dir() -> Result { // 避免在Handle未初始化时崩溃 let app_handle = handle::Handle::app_handle(); match app_handle.path().resource_dir() { Ok(dir) => Ok(dir.join("resources")), Err(e) => { logging!( error, Type::File, "Failed to get the resource directory: {e}" ); Err(anyhow::anyhow!("Failed to get the resource directory")) } } } /// profiles dir pub fn app_profiles_dir() -> Result { Ok(app_home_dir()?.join("profiles")) } /// icons dir pub fn app_icons_dir() -> Result { Ok(app_home_dir()?.join("icons")) } pub fn find_target_icons(target: &str) -> Result> { let icons_dir = app_icons_dir()?; let mut matching_files = Vec::new(); for entry in fs::read_dir(icons_dir)? { let entry = entry?; let path = entry.path(); if let Some(file_name) = path.file_name().and_then(|n| n.to_str()) && file_name.starts_with(target) && (file_name.ends_with(".ico") || file_name.ends_with(".png")) { matching_files.push(path); } } if matching_files.is_empty() { Ok(None) } else { match matching_files.first() { Some(first_path) => { let first = path_to_str(first_path)?; Ok(Some(first.into())) } None => Ok(None), } } } /// logs dir pub fn app_logs_dir() -> Result { Ok(app_home_dir()?.join("logs")) } // latest verge log pub fn app_latest_log() -> Result { Ok(app_logs_dir()?.join("latest.log")) } /// local backups dir pub fn local_backup_dir() -> Result { let dir = app_home_dir()?.join(BACKUP_DIR); fs::create_dir_all(&dir)?; Ok(dir) } pub fn clash_path() -> Result { Ok(app_home_dir()?.join(CLASH_CONFIG)) } pub fn verge_path() -> Result { Ok(app_home_dir()?.join(VERGE_CONFIG)) } pub fn profiles_path() -> Result { Ok(app_home_dir()?.join(PROFILE_YAML)) } #[cfg(target_os = "macos")] pub fn service_path() -> Result { let res_dir = app_resources_dir()?; Ok(res_dir.join("clash-verge-service")) } #[cfg(windows)] pub fn service_path() -> Result { let res_dir = app_resources_dir()?; Ok(res_dir.join("clash-verge-service.exe")) } pub fn sidecar_log_dir() -> Result { let log_dir = app_logs_dir()?.join("sidecar"); let _ = std::fs::create_dir_all(&log_dir); Ok(log_dir) } pub fn service_log_dir() -> Result { let log_dir = app_logs_dir()?.join("service"); let _ = std::fs::create_dir_all(&log_dir); Ok(log_dir) } pub fn clash_latest_log() -> Result { match *CoreManager::global().get_running_mode() { RunningMode::Service => Ok(service_log_dir()?.join("service_latest.log")), RunningMode::Sidecar | RunningMode::NotRunning => { Ok(sidecar_log_dir()?.join("sidecar_latest.log")) } } } pub fn path_to_str(path: &PathBuf) -> Result<&str> { let path_str = path .as_os_str() .to_str() .ok_or_else(|| anyhow::anyhow!("failed to get path from {:?}", path))?; Ok(path_str) } pub fn get_encryption_key() -> Result> { let app_dir = app_home_dir()?; let key_path = app_dir.join(".encryption_key"); if key_path.exists() { // Read existing key fs::read(&key_path).map_err(|e| anyhow::anyhow!("Failed to read encryption key: {}", e)) } else { // Generate and save new key let mut key = vec![0u8; 32]; getrandom::fill(&mut key)?; // Ensure directory exists if let Some(parent) = key_path.parent() { fs::create_dir_all(parent) .map_err(|e| anyhow::anyhow!("Failed to create key directory: {}", e))?; } // Save key fs::write(&key_path, &key) .map_err(|e| anyhow::anyhow!("Failed to save encryption key: {}", e))?; Ok(key) } } #[cfg(unix)] pub fn ensure_mihomo_safe_dir() -> Option { ["/tmp"] .iter() .map(PathBuf::from) .find(|path| path.exists()) .or_else(|| { std::env::var_os("HOME").and_then(|home| { let home_config = PathBuf::from(home).join(".config"); if home_config.exists() || fs::create_dir_all(&home_config).is_ok() { Some(home_config) } else { logging!( error, Type::File, "Failed to create safe directory: {home_config:?}" ); None } }) }) } #[cfg(unix)] pub fn ipc_path() -> Result { ensure_mihomo_safe_dir() .map(|base_dir| base_dir.join("verge").join("verge-mihomo.sock")) .or_else(|| { app_home_dir() .ok() .map(|dir| dir.join("verge").join("verge-mihomo.sock")) }) .ok_or_else(|| anyhow::anyhow!("Failed to determine ipc path")) } #[cfg(target_os = "windows")] pub fn ipc_path() -> Result { Ok(PathBuf::from(r"\\.\pipe\verge-mihomo")) } #[async_trait] pub trait PathBufExec { async fn remove_if_exists(&self) -> Result<()>; } #[async_trait] impl PathBufExec for PathBuf { async fn remove_if_exists(&self) -> Result<()> { if self.exists() { tokio::fs::remove_file(self).await?; logging!(info, Type::File, "Removed file: {:?}", self); } Ok(()) } }