270 lines
7.6 KiB
Rust
270 lines
7.6 KiB
Rust
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<bool> = 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<PathBuf> {
|
|
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) => {
|
|
log::error!(target: "app", "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<PathBuf> {
|
|
// 避免在Handle未初始化时崩溃
|
|
let app_handle = handle::Handle::app_handle();
|
|
|
|
match app_handle.path().resource_dir() {
|
|
Ok(dir) => Ok(dir.join("resources")),
|
|
Err(e) => {
|
|
log::error!(target: "app", "Failed to get the resource directory: {e}");
|
|
Err(anyhow::anyhow!("Failed to get the resource directory"))
|
|
}
|
|
}
|
|
}
|
|
|
|
/// profiles dir
|
|
pub fn app_profiles_dir() -> Result<PathBuf> {
|
|
Ok(app_home_dir()?.join("profiles"))
|
|
}
|
|
|
|
/// icons dir
|
|
pub fn app_icons_dir() -> Result<PathBuf> {
|
|
Ok(app_home_dir()?.join("icons"))
|
|
}
|
|
|
|
pub fn find_target_icons(target: &str) -> Result<Option<String>> {
|
|
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<PathBuf> {
|
|
Ok(app_home_dir()?.join("logs"))
|
|
}
|
|
|
|
// latest verge log
|
|
pub fn app_latest_log() -> Result<PathBuf> {
|
|
Ok(app_logs_dir()?.join("latest.log"))
|
|
}
|
|
|
|
/// local backups dir
|
|
pub fn local_backup_dir() -> Result<PathBuf> {
|
|
let dir = app_home_dir()?.join(BACKUP_DIR);
|
|
fs::create_dir_all(&dir)?;
|
|
Ok(dir)
|
|
}
|
|
|
|
pub fn clash_path() -> Result<PathBuf> {
|
|
Ok(app_home_dir()?.join(CLASH_CONFIG))
|
|
}
|
|
|
|
pub fn verge_path() -> Result<PathBuf> {
|
|
Ok(app_home_dir()?.join(VERGE_CONFIG))
|
|
}
|
|
|
|
pub fn profiles_path() -> Result<PathBuf> {
|
|
Ok(app_home_dir()?.join(PROFILE_YAML))
|
|
}
|
|
|
|
#[cfg(target_os = "macos")]
|
|
pub fn service_path() -> Result<PathBuf> {
|
|
let res_dir = app_resources_dir()?;
|
|
Ok(res_dir.join("clash-verge-service"))
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
pub fn service_path() -> Result<PathBuf> {
|
|
let res_dir = app_resources_dir()?;
|
|
Ok(res_dir.join("clash-verge-service.exe"))
|
|
}
|
|
|
|
pub fn sidecar_log_dir() -> Result<PathBuf> {
|
|
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<PathBuf> {
|
|
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<PathBuf> {
|
|
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<Vec<u8>> {
|
|
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<PathBuf> {
|
|
["/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 {
|
|
log::error!(target: "app", "Failed to create safe directory: {home_config:?}");
|
|
None
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
pub fn ipc_path() -> Result<PathBuf> {
|
|
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<PathBuf> {
|
|
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(())
|
|
}
|
|
}
|