mirror of
https://github.com/clash-verge-rev/clash-verge-rev.git
synced 2026-04-14 22:10:32 +08:00
156 lines
4.7 KiB
Rust
156 lines
4.7 KiB
Rust
use crate::{
|
||
config::Config,
|
||
core::{CoreManager, handle, tray},
|
||
feat::clean_async,
|
||
process::AsyncHandler,
|
||
utils::{self, resolve::reset_resolve_done},
|
||
};
|
||
use clash_verge_logging::{Type, logging, logging_error};
|
||
use serde_yaml_ng::{Mapping, Value};
|
||
use smartstring::alias::String;
|
||
|
||
/// Restart the Clash core
|
||
pub async fn restart_clash_core() {
|
||
match CoreManager::global().restart_core().await {
|
||
Ok(_) => {
|
||
handle::Handle::refresh_clash();
|
||
handle::Handle::notice_message("set_config::ok", "ok");
|
||
}
|
||
Err(err) => {
|
||
handle::Handle::notice_message("set_config::error", format!("{err}"));
|
||
logging!(error, Type::Core, "{err}");
|
||
}
|
||
}
|
||
}
|
||
|
||
/// Restart the application
|
||
pub async fn restart_app() {
|
||
logging!(debug, Type::System, "启动重启应用流程");
|
||
utils::server::shutdown_embedded_server();
|
||
Config::apply_all_and_save_file().await;
|
||
|
||
// 设置退出标志
|
||
handle::Handle::global().set_is_exiting();
|
||
|
||
logging!(info, Type::System, "开始异步清理资源");
|
||
let cleanup_result = clean_async().await;
|
||
|
||
logging!(
|
||
info,
|
||
Type::System,
|
||
"资源清理完成,退出代码: {}",
|
||
if cleanup_result { 0 } else { 1 }
|
||
);
|
||
|
||
reset_resolve_done();
|
||
let app_handle = handle::Handle::app_handle();
|
||
app_handle.restart();
|
||
}
|
||
|
||
fn after_change_clash_mode() {
|
||
AsyncHandler::spawn(move || async {
|
||
let mihomo = handle::Handle::mihomo().await;
|
||
match mihomo.get_connections().await {
|
||
Ok(connections) => {
|
||
if let Some(connections_array) = connections.connections {
|
||
for connection in connections_array {
|
||
let _ = mihomo.close_connection(&connection.id).await;
|
||
}
|
||
drop(mihomo);
|
||
}
|
||
}
|
||
Err(err) => {
|
||
logging!(error, Type::Core, "Failed to get connections: {err}");
|
||
}
|
||
}
|
||
});
|
||
}
|
||
|
||
/// Change Clash mode (rule/global/direct/script)
|
||
pub async fn change_clash_mode(mode: String) {
|
||
let mut mapping = Mapping::new();
|
||
mapping.insert(Value::from("mode"), Value::from(mode.as_str()));
|
||
// Convert YAML mapping to JSON Value
|
||
let json_value = serde_json::json!({
|
||
"mode": mode
|
||
});
|
||
logging!(debug, Type::Core, "change clash mode to {mode}");
|
||
match handle::Handle::mihomo()
|
||
.await
|
||
.patch_base_config(&json_value)
|
||
.await
|
||
{
|
||
Ok(_) => {
|
||
// 更新订阅
|
||
Config::clash()
|
||
.await
|
||
.edit_draft(|d| d.patch_config(mapping));
|
||
|
||
// 分离数据获取和异步调用
|
||
let clash_data = Config::clash().await.data_arc();
|
||
if clash_data.save_config().await.is_ok() {
|
||
handle::Handle::refresh_clash();
|
||
logging_error!(Type::Tray, tray::Tray::global().update_menu().await);
|
||
logging_error!(
|
||
Type::Tray,
|
||
tray::Tray::global()
|
||
.update_icon(&Config::verge().await.data_arc())
|
||
.await
|
||
);
|
||
}
|
||
|
||
let is_auto_close_connection = Config::verge()
|
||
.await
|
||
.data_arc()
|
||
.auto_close_connection
|
||
.unwrap_or(false);
|
||
if is_auto_close_connection {
|
||
after_change_clash_mode();
|
||
}
|
||
}
|
||
Err(err) => logging!(error, Type::Core, "{err}"),
|
||
}
|
||
}
|
||
|
||
/// Test connection delay to a URL
|
||
pub async fn test_delay(url: String) -> anyhow::Result<u32> {
|
||
use crate::utils::network::{NetworkManager, ProxyType};
|
||
use tokio::time::Instant;
|
||
|
||
let tun_mode = Config::verge()
|
||
.await
|
||
.latest_arc()
|
||
.enable_tun_mode
|
||
.unwrap_or(false);
|
||
|
||
// 如果是TUN模式,不使用代理,否则使用自身代理
|
||
let proxy_type = if !tun_mode {
|
||
ProxyType::Localhost
|
||
} else {
|
||
ProxyType::None
|
||
};
|
||
|
||
let user_agent = Some("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36 Edg/120.0.0.0".into());
|
||
|
||
let start = Instant::now();
|
||
|
||
let response = NetworkManager::new()
|
||
.get_with_interrupt(&url, proxy_type, Some(10), user_agent, false)
|
||
.await;
|
||
|
||
match response {
|
||
Ok(response) => {
|
||
logging!(trace, Type::Network, "test_delay response: {response:#?}");
|
||
if response.status().is_success() {
|
||
Ok(start.elapsed().as_millis() as u32)
|
||
} else {
|
||
Ok(10000u32)
|
||
}
|
||
}
|
||
Err(err) => {
|
||
logging!(trace, Type::Network, "test_delay error: {err:#?}");
|
||
Err(err)
|
||
}
|
||
}
|
||
}
|