From cf78bb368611186e8253ddaa0d6c39d73568e64d Mon Sep 17 00:00:00 2001 From: wonfen Date: Tue, 25 Mar 2025 00:51:26 +0800 Subject: [PATCH] refactor: service reinstallation logic on detection failure --- src-tauri/src/cmd/core.rs | 8 + src-tauri/src/cmd/mod.rs | 2 + src-tauri/src/config/verge.rs | 7 + src-tauri/src/core/core.rs | 21 ++- src-tauri/src/core/service.rs | 268 ++++++++++++++++++++++++++++++++-- src-tauri/src/lib.rs | 1 + 6 files changed, 292 insertions(+), 15 deletions(-) create mode 100644 src-tauri/src/cmd/core.rs diff --git a/src-tauri/src/cmd/core.rs b/src-tauri/src/cmd/core.rs new file mode 100644 index 00000000..0b1c44c6 --- /dev/null +++ b/src-tauri/src/cmd/core.rs @@ -0,0 +1,8 @@ +use super::CmdResult; +use crate::{core::CoreManager, wrap_err}; + +/// 修复系统服务 +#[tauri::command] +pub async fn repair_service() -> CmdResult { + wrap_err!(CoreManager::global().repair_service().await) +} \ No newline at end of file diff --git a/src-tauri/src/cmd/mod.rs b/src-tauri/src/cmd/mod.rs index 61374546..cbfd4a87 100644 --- a/src-tauri/src/cmd/mod.rs +++ b/src-tauri/src/cmd/mod.rs @@ -6,6 +6,7 @@ pub type CmdResult = Result; // Command modules pub mod app; pub mod clash; +pub mod core; pub mod media_unlock_checker; pub mod network; pub mod profile; @@ -22,6 +23,7 @@ pub mod lighteweight; // Re-export all command functions for backwards compatibility pub use app::*; pub use clash::*; +pub use core::*; pub use media_unlock_checker::*; pub use network::*; pub use profile::*; diff --git a/src-tauri/src/config/verge.rs b/src-tauri/src/config/verge.rs index c1010cc1..1fb31a2c 100644 --- a/src-tauri/src/config/verge.rs +++ b/src-tauri/src/config/verge.rs @@ -196,6 +196,9 @@ pub struct IVerge { /// 自动进入轻量模式的延迟(分钟) pub auto_light_weight_minutes: Option, + + /// 服务状态跟踪 + pub service_state: Option, } #[derive(Default, Debug, Clone, Deserialize, Serialize)] @@ -303,6 +306,7 @@ impl IVerge { auto_light_weight_minutes: Some(10), enable_dns_settings: Some(true), home_cards: None, + service_state: None, ..Self::default() } } @@ -389,6 +393,7 @@ impl IVerge { patch!(auto_light_weight_minutes); patch!(enable_dns_settings); patch!(home_cards); + patch!(service_state); } /// 在初始化前尝试拿到单例端口的值 @@ -482,6 +487,7 @@ pub struct IVergeResponse { pub auto_light_weight_minutes: Option, pub enable_dns_settings: Option, pub home_cards: Option, + pub service_state: Option, } impl From for IVergeResponse { @@ -549,6 +555,7 @@ impl From for IVergeResponse { auto_light_weight_minutes: verge.auto_light_weight_minutes, enable_dns_settings: verge.enable_dns_settings, home_cards: verge.home_cards, + service_state: verge.service_state, } } } diff --git a/src-tauri/src/core/core.rs b/src-tauri/src/core/core.rs index f2d03112..880634b1 100644 --- a/src-tauri/src/core/core.rs +++ b/src-tauri/src/core/core.rs @@ -141,15 +141,17 @@ impl CoreManager { let config_path = Config::generate_file(ConfigType::Run)?; - // 先尝试服务模式 - if service::check_service().await.is_ok() { + // 先检查服务状态 + let service_available = service::check_service().await.is_ok(); + + if service_available { log::info!(target: "app", "try to run core in service mode"); match service::run_core_by_service(&config_path).await { Ok(_) => { log::info!(target: "app", "core started successfully in service mode"); } Err(err) => { - // 服务启动失败,尝试sidecar模式 + // 服务启动失败,直接尝试sidecar模式,不再尝试重装服务 log::warn!(target: "app", "failed to start core in service mode: {}", err); log::info!(target: "app", "trying to run core in sidecar mode"); self.run_core_by_sidecar(&config_path).await?; @@ -308,6 +310,19 @@ impl CoreManager { Ok(()) } + /// 强制重新安装服务(供UI调用,用户主动修复服务) + pub async fn repair_service(&self) -> Result<()> { + log::info!(target: "app", "user requested service repair"); + + // 调用强制重装服务 + service::force_reinstall_service().await?; + + // 重启核心 + self.restart_core().await?; + + Ok(()) + } + /// 使用默认配置 pub async fn use_default_config(&self, msg_type: &str, msg_content: &str) -> Result<()> { let runtime_path = dirs::app_home_dir()?.join(RUNTIME_CONFIG); diff --git a/src-tauri/src/core/service.rs b/src-tauri/src/core/service.rs index 949621f1..feef40d3 100644 --- a/src-tauri/src/core/service.rs +++ b/src-tauri/src/core/service.rs @@ -1,13 +1,82 @@ use crate::{config::Config, utils::dirs}; use anyhow::{bail, Context, Result}; use serde::{Deserialize, Serialize}; -use std::{collections::HashMap, env::current_exe, path::PathBuf, process::Command as StdCommand}; +use std::{collections::HashMap, env::current_exe, path::PathBuf, process::Command as StdCommand, time::{SystemTime, UNIX_EPOCH}}; use tokio::time::Duration; // Windows only const SERVICE_URL: &str = "http://127.0.0.1:33211"; -const REQUIRED_SERVICE_VERSION: &str = "1.0.4"; // 定义所需的服务版本号 +const REQUIRED_SERVICE_VERSION: &str = "1.0.5"; // 定义所需的服务版本号 + +// 限制重装时间和次数的常量 +const REINSTALL_COOLDOWN_SECS: u64 = 300; // 5分钟冷却期 +const MAX_REINSTALLS_PER_DAY: u32 = 3; // 每24小时最多重装3次 +const ONE_DAY_SECS: u64 = 86400; // 24小时的秒数 + +#[derive(Debug, Deserialize, Serialize, Clone, Default)] +pub struct ServiceState { + pub last_install_time: u64, // 上次安装时间戳 (Unix 时间戳,秒) + pub install_count: u32, // 24小时内安装次数 + pub last_check_time: u64, // 上次检查时间 + pub last_error: Option, // 上次错误信息 +} + +impl ServiceState { + // 获取当前的服务状态 + pub fn get() -> Self { + if let Some(state) = Config::verge().latest().service_state.clone() { + return state; + } + Self::default() + } + + // 保存服务状态 + pub fn save(&self) -> Result<()> { + let config = Config::verge(); + let mut latest = config.latest().clone(); + latest.service_state = Some(self.clone()); + *config.draft() = latest; + config.apply(); + Config::verge().latest().save_file() + } + + // 更新安装信息 + pub fn record_install(&mut self) { + let now = SystemTime::now() + .duration_since(UNIX_EPOCH) + .unwrap_or_default() + .as_secs(); + + // 检查是否需要重置计数器(24小时已过) + if now - self.last_install_time > ONE_DAY_SECS { + self.install_count = 0; + } + + self.last_install_time = now; + self.install_count += 1; + } + + // 检查是否可以重新安装 + pub fn can_reinstall(&self) -> bool { + let now = SystemTime::now() + .duration_since(UNIX_EPOCH) + .unwrap_or_default() + .as_secs(); + + // 如果在冷却期内,不允许重装 + if now - self.last_install_time < REINSTALL_COOLDOWN_SECS { + return false; + } + + // 如果24小时内安装次数过多,也不允许 + if now - self.last_install_time < ONE_DAY_SECS && self.install_count >= MAX_REINSTALLS_PER_DAY { + return false; + } + + true + } +} #[derive(Debug, Deserialize, Serialize, Clone)] pub struct ResponseBody { @@ -41,6 +110,15 @@ pub struct VersionJsonResponse { pub async fn reinstall_service() -> Result<()> { log::info!(target:"app", "reinstall service"); + // 获取当前服务状态 + let mut service_state = ServiceState::get(); + + // 检查是否允许重装 + if !service_state.can_reinstall() { + log::warn!(target:"app", "service reinstall rejected: cooldown period or max attempts reached"); + bail!("Service reinstallation is rate limited. Please try again later."); + } + use deelevate::{PrivilegeLevel, Token}; use runas::Command as RunasCommand; use std::os::windows::process::CommandExt; @@ -74,12 +152,20 @@ pub async fn reinstall_service() -> Result<()> { }; if !status.success() { - bail!( + let error = format!( "failed to install service with status {}", status.code().unwrap() ); + service_state.last_error = Some(error.clone()); + service_state.save()?; + bail!(error); } + // 记录安装信息并保存 + service_state.record_install(); + service_state.last_error = None; + service_state.save()?; + Ok(()) } @@ -226,19 +312,54 @@ pub async fn check_service_version() -> Result { /// check if service needs to be reinstalled pub async fn check_service_needs_reinstall() -> bool { + // 获取当前服务状态 + let service_state = ServiceState::get(); + + // 首先检查是否在冷却期或超过重装次数限制 + if !service_state.can_reinstall() { + log::info!(target: "app", "service reinstall check: in cooldown period or max attempts reached"); + return false; + } + + // 然后才检查版本和可用性 match check_service_version().await { - Ok(version) => version != REQUIRED_SERVICE_VERSION, - Err(_) => true, // 如果无法获取版本或服务未运行,也需要重新安装 + Ok(version) => { + // 打印更详细的日志,方便排查问题 + log::info!(target: "app", "服务版本检测:当前={}, 要求={}", version, REQUIRED_SERVICE_VERSION); + + let needs_reinstall = version != REQUIRED_SERVICE_VERSION; + if needs_reinstall { + log::warn!(target: "app", "发现服务版本不匹配,需要重装! 当前={}, 要求={}", + version, REQUIRED_SERVICE_VERSION); + + // 打印版本字符串的原始字节,确认没有隐藏字符 + log::debug!(target: "app", "当前版本字节: {:?}", version.as_bytes()); + log::debug!(target: "app", "要求版本字节: {:?}", REQUIRED_SERVICE_VERSION.as_bytes()); + } else { + log::info!(target: "app", "服务版本匹配,无需重装"); + } + + needs_reinstall + }, + Err(err) => { + // 检查服务是否可用,如果可用但版本检查失败,可能只是版本API有问题 + match is_service_running().await { + Ok(true) => { + log::info!(target: "app", "service is running but version check failed: {}", err); + false // 服务在运行,不需要重装 + } + _ => { + log::info!(target: "app", "service is not running or unavailable"); + true // 服务不可用,需要重装 + } + } + } } } -/// start the clash by service -pub(super) async fn run_core_by_service(config_file: &PathBuf) -> Result<()> { - // 检查服务版本,如果不匹配则重新安装 - if check_service_needs_reinstall().await { - log::info!(target: "app", "service version mismatch, reinstalling"); - reinstall_service().await?; - } +/// 尝试使用现有服务启动核心,不进行重装 +pub(super) async fn start_with_existing_service(config_file: &PathBuf) -> Result<()> { + log::info!(target:"app", "attempting to start core with existing service"); let clash_core = { Config::verge().latest().clash_core.clone() }; let clash_core = clash_core.unwrap_or("verge-mihomo".into()); @@ -280,6 +401,106 @@ pub(super) async fn run_core_by_service(config_file: &PathBuf) -> Result<()> { Ok(()) } +/// start the clash by service +pub(super) async fn run_core_by_service(config_file: &PathBuf) -> Result<()> { + log::info!(target: "app", "正在尝试通过服务启动核心"); + + // 先检查服务版本,不受冷却期限制 + let version_check = match check_service_version().await { + Ok(version) => { + log::info!(target: "app", "检测到服务版本: {}, 要求版本: {}", + version, REQUIRED_SERVICE_VERSION); + + // 通过字节比较确保完全匹配 + if version.as_bytes() != REQUIRED_SERVICE_VERSION.as_bytes() { + log::warn!(target: "app", "服务版本不匹配,需要重装"); + false // 版本不匹配 + } else { + log::info!(target: "app", "服务版本匹配"); + true // 版本匹配 + } + }, + Err(err) => { + log::warn!(target: "app", "无法获取服务版本: {}", err); + false // 无法获取版本 + } + }; + + // 先尝试直接启动服务,如果服务可用且版本匹配 + if version_check { + if let Ok(true) = is_service_running().await { + // 服务正在运行且版本匹配,直接使用 + log::info!(target: "app", "服务已在运行且版本匹配,尝试使用"); + return start_with_existing_service(config_file).await; + } + } + + // 强制执行版本检查,如果版本不匹配则重装 + if !version_check { + log::info!(target: "app", "服务版本不匹配,尝试重装"); + + // 获取服务状态,检查是否可以重装 + let service_state = ServiceState::get(); + if !service_state.can_reinstall() { + log::warn!(target: "app", "由于限制无法重装服务"); + // 尝试直接启动,即使版本不匹配 + if let Ok(()) = start_with_existing_service(config_file).await { + log::info!(target: "app", "尽管版本不匹配,但成功启动了服务"); + return Ok(()); + } else { + bail!("服务版本不匹配且无法重装,启动失败"); + } + } + + // 尝试重装 + log::info!(target: "app", "开始重装服务"); + if let Err(err) = reinstall_service().await { + log::warn!(target: "app", "服务重装失败: {}", err); + + // 尝试使用现有服务 + log::info!(target: "app", "尝试使用现有服务"); + return start_with_existing_service(config_file).await; + } + + // 重装成功,尝试启动 + log::info!(target: "app", "服务重装成功,尝试启动"); + return start_with_existing_service(config_file).await; + } + + // 检查服务状态 + match check_service().await { + Ok(_) => { + // 服务可访问但可能没有运行核心,尝试直接启动 + log::info!(target: "app", "服务可用但未运行核心,尝试启动"); + if let Ok(()) = start_with_existing_service(config_file).await { + return Ok(()); + } + }, + Err(err) => { + log::warn!(target: "app", "服务检查失败: {}", err); + } + } + + // 服务不可用或启动失败,检查是否需要重装 + if check_service_needs_reinstall().await { + log::info!(target: "app", "服务需要重装"); + + // 尝试重装 + if let Err(err) = reinstall_service().await { + log::warn!(target: "app", "服务重装失败: {}", err); + bail!("Failed to reinstall service: {}", err); + } + + // 重装后再次尝试启动 + log::info!(target: "app", "服务重装完成,尝试启动核心"); + start_with_existing_service(config_file).await + } else { + // 不需要或不能重装,返回错误 + log::warn!(target: "app", "服务不可用且无法重装"); + bail!("Service is not available and cannot be reinstalled at this time") + } +} + /// stop the clash by service pub(super) async fn stop_core_by_service() -> Result<()> { let url = format!("{SERVICE_URL}/stop_clash"); @@ -305,3 +526,26 @@ pub async fn is_service_running() -> Result { Ok(false) } } + +/// 强制重装服务(用于UI中的修复服务按钮) +pub async fn force_reinstall_service() -> Result<()> { + log::info!(target: "app", "用户请求强制重装服务"); + + // 创建默认服务状态(重置所有限制) + let service_state = ServiceState::default(); + service_state.save()?; + + log::info!(target: "app", "已重置服务状态,开始执行重装"); + + // 执行重装 + match reinstall_service().await { + Ok(()) => { + log::info!(target: "app", "服务重装成功"); + Ok(()) + }, + Err(err) => { + log::error!(target: "app", "强制重装服务失败: {}", err); + bail!("强制重装服务失败: {}", err) + } + } +} diff --git a/src-tauri/src/lib.rs b/src-tauri/src/lib.rs index 20013f24..aeeabe6b 100644 --- a/src-tauri/src/lib.rs +++ b/src-tauri/src/lib.rs @@ -152,6 +152,7 @@ pub fn run() { // 添加新的命令 cmd::get_running_mode, cmd::install_service, + cmd::repair_service, cmd::get_app_uptime, cmd::get_auto_launch_status, // clash