Files
clash-proxy/src-tauri/src/feat.rs

443 lines
14 KiB
Rust
Raw Normal View History

2022-11-18 18:18:41 +08:00
//
//! feat mod 里的函数主要用于
//! - hotkey 快捷键
//! - timer 定时器
//! - cmds 页面调用
//!
2022-11-14 01:26:33 +08:00
use crate::config::*;
2022-09-14 01:19:02 +08:00
use crate::core::*;
2022-11-14 01:26:33 +08:00
use crate::log_err;
use crate::utils::resolve;
2022-11-17 17:07:13 +08:00
use anyhow::{bail, Result};
2022-11-14 01:26:33 +08:00
use serde_yaml::{Mapping, Value};
use tauri::{AppHandle, ClipboardManager, Manager};
2022-09-14 01:19:02 +08:00
// 打开面板
pub fn open_or_close_dashboard() {
let handle = handle::Handle::global();
let app_handle = handle.app_handle.lock();
if let Some(app_handle) = app_handle.as_ref() {
if let Some(window) = app_handle.get_window("main") {
if let Ok(true) = window.is_focused() {
let _ = window.close();
return;
}
}
resolve::create_window(app_handle);
}
}
2022-10-28 00:40:29 +08:00
// 重启clash
pub fn restart_clash_core() {
2022-11-14 01:26:33 +08:00
tauri::async_runtime::spawn(async {
2022-11-17 20:19:40 +08:00
match CoreManager::global().run_core().await {
2022-11-18 18:18:41 +08:00
Ok(_) => {
handle::Handle::refresh_clash();
handle::Handle::notice_message("set_config::ok", "ok");
}
Err(err) => {
handle::Handle::notice_message("set_config::error", format!("{err}"));
log::error!(target:"app", "{err}");
}
2022-11-17 20:19:40 +08:00
}
2022-11-14 01:26:33 +08:00
});
2022-10-28 00:40:29 +08:00
}
2022-11-14 01:26:33 +08:00
// 切换模式 rule/global/direct/script mode
pub fn change_clash_mode(mode: String) {
let mut mapping = Mapping::new();
mapping.insert(Value::from("mode"), mode.clone().into());
tauri::async_runtime::spawn(async move {
2022-11-23 16:04:25 +08:00
log::debug!(target: "app", "change clash mode to {mode}");
2022-11-14 01:26:33 +08:00
match clash_api::patch_configs(&mapping).await {
Ok(_) => {
// 更新订阅
2022-11-17 17:07:13 +08:00
Config::clash().data().patch_config(mapping);
2022-11-14 01:26:33 +08:00
2022-11-23 16:04:25 +08:00
if Config::clash().data().save_config().is_ok() {
2022-11-14 01:26:33 +08:00
handle::Handle::refresh_clash();
log_err!(handle::Handle::update_systray_part());
}
}
2022-11-18 20:15:34 +08:00
Err(err) => log::error!(target: "app", "{err}"),
2022-11-14 01:26:33 +08:00
}
});
2022-09-14 01:19:02 +08:00
}
// 切换系统代理
2022-11-17 17:07:13 +08:00
pub fn toggle_system_proxy() {
2024-01-10 17:36:35 +08:00
let enable = Config::verge().draft().enable_system_proxy;
2022-11-17 17:07:13 +08:00
let enable = enable.unwrap_or(false);
tauri::async_runtime::spawn(async move {
match patch_verge(IVerge {
enable_system_proxy: Some(!enable),
..IVerge::default()
})
.await
{
Ok(_) => handle::Handle::refresh_verge(),
Err(err) => log::error!(target: "app", "{err}"),
}
});
2022-09-14 01:19:02 +08:00
}
// 切换tun模式
2022-11-17 17:07:13 +08:00
pub fn toggle_tun_mode() {
2024-01-10 17:36:35 +08:00
let enable = Config::verge().data().enable_tun_mode;
2022-11-17 17:07:13 +08:00
let enable = enable.unwrap_or(false);
tauri::async_runtime::spawn(async move {
if !enable {
if let Ok(res) = service::check_service().await {
if res.code == 0 {
match patch_verge(IVerge {
enable_tun_mode: Some(!enable),
..IVerge::default()
})
.await
{
Ok(_) => handle::Handle::refresh_verge(),
Err(err) => log::error!(target: "app", "{err}"),
}
return;
}
}
tauri::api::dialog::message(
None::<&tauri::Window>,
"Please install and enable service mode",
"Service mode is required for Tun mode",
);
} else {
match patch_verge(IVerge {
enable_tun_mode: Some(!enable),
..IVerge::default()
})
.await
{
Ok(_) => handle::Handle::refresh_verge(),
Err(err) => log::error!(target: "app", "{err}"),
}
2022-11-17 17:07:13 +08:00
}
});
2022-09-14 01:19:02 +08:00
}
/// 修改clash的订阅
2022-11-17 17:07:13 +08:00
pub async fn patch_clash(patch: Mapping) -> Result<()> {
Config::clash().draft().patch_config(patch.clone());
2022-11-14 01:26:33 +08:00
2024-06-12 10:00:22 +08:00
let res = {
let redir_port = patch.get("redir-port");
let tproxy_port = patch.get("tproxy-port");
2022-11-17 17:07:13 +08:00
let mixed_port = patch.get("mixed-port");
2024-02-05 16:47:40 +08:00
let socks_port = patch.get("socks-port");
let port = patch.get("port");
2023-12-01 12:56:18 +08:00
let enable_random_port = Config::verge().latest().enable_random_port.unwrap_or(false);
if mixed_port.is_some() && !enable_random_port {
2024-01-10 17:36:35 +08:00
let changed = mixed_port.unwrap()
2023-12-01 12:56:18 +08:00
!= Config::verge()
.latest()
.verge_mixed_port
.unwrap_or(Config::clash().data().get_mixed_port());
2022-11-17 17:07:13 +08:00
// 检查端口占用
if changed {
2024-01-10 17:36:35 +08:00
if let Some(port) = mixed_port.unwrap().as_u64() {
2022-11-17 17:07:13 +08:00
if !port_scanner::local_port_available(port as u16) {
Config::clash().discard();
bail!("port already in use");
2022-11-17 17:07:13 +08:00
}
}
}
};
2022-11-14 01:26:33 +08:00
// 激活订阅
if redir_port.is_some()
|| tproxy_port.is_some()
|| mixed_port.is_some()
2024-02-05 16:47:40 +08:00
|| socks_port.is_some()
|| port.is_some()
2022-11-17 22:53:41 +08:00
|| patch.get("secret").is_some()
|| patch.get("external-controller").is_some()
{
2022-11-18 18:37:17 +08:00
Config::generate()?;
2022-11-18 18:18:41 +08:00
CoreManager::global().run_core().await?;
2022-11-18 18:37:17 +08:00
handle::Handle::refresh_clash();
2022-11-17 22:53:41 +08:00
}
2022-11-14 01:26:33 +08:00
2022-11-17 17:07:13 +08:00
// 更新系统代理
if mixed_port.is_some() {
log_err!(sysopt::Sysopt::global().init_sysproxy());
2022-11-14 01:26:33 +08:00
}
2022-11-17 17:07:13 +08:00
if patch.get("mode").is_some() {
log_err!(handle::Handle::update_systray_part());
}
2022-11-14 01:26:33 +08:00
Config::runtime().latest().patch_config(patch);
2022-11-17 17:07:13 +08:00
<Result<()>>::Ok(())
2024-06-12 10:00:22 +08:00
};
match res {
2022-11-17 17:07:13 +08:00
Ok(()) => {
Config::clash().apply();
Config::clash().data().save_config()?;
Ok(())
2022-11-14 01:26:33 +08:00
}
2022-11-17 17:07:13 +08:00
Err(err) => {
Config::clash().discard();
Err(err)
}
}
2022-11-14 01:26:33 +08:00
}
/// 修改verge的订阅
2022-11-14 01:26:33 +08:00
/// 一般都是一个个的修改
2022-11-17 17:07:13 +08:00
pub async fn patch_verge(patch: IVerge) -> Result<()> {
Config::verge().draft().patch_config(patch.clone());
2022-11-14 01:26:33 +08:00
let tun_mode = patch.enable_tun_mode;
let auto_launch = patch.enable_auto_launch;
let system_proxy = patch.enable_system_proxy;
2024-05-26 17:59:39 +08:00
let pac = patch.proxy_auto_config;
let pac_content = patch.pac_file_content;
2022-11-14 01:26:33 +08:00
let proxy_bypass = patch.system_proxy_bypass;
let language = patch.language;
let port = patch.verge_mixed_port;
2024-05-15 12:00:38 +08:00
#[cfg(target_os = "macos")]
let tray_icon = patch.tray_icon;
2024-02-24 00:52:21 +08:00
let common_tray_icon = patch.common_tray_icon;
let sysproxy_tray_icon = patch.sysproxy_tray_icon;
let tun_tray_icon = patch.tun_tray_icon;
2024-05-09 09:01:04 +08:00
#[cfg(not(target_os = "windows"))]
let redir_enabled = patch.verge_redir_enabled;
#[cfg(target_os = "linux")]
let tproxy_enabled = patch.verge_tproxy_enabled;
let socks_enabled = patch.verge_socks_enabled;
let http_enabled = patch.verge_http_enabled;
2024-06-12 10:00:22 +08:00
let res = {
let service_mode = patch.enable_service_mode;
2022-11-22 22:01:34 +08:00
if service_mode.is_some() {
log::debug!(target: "app", "change service mode to {}", service_mode.unwrap());
2022-11-14 01:26:33 +08:00
Config::generate()?;
CoreManager::global().run_core().await?;
} else if tun_mode.is_some() {
2022-11-18 18:18:41 +08:00
update_core_config().await?;
2022-11-17 17:07:13 +08:00
}
2024-05-09 09:01:04 +08:00
#[cfg(not(target_os = "windows"))]
if redir_enabled.is_some() {
Config::generate()?;
CoreManager::global().run_core().await?;
}
#[cfg(target_os = "linux")]
if tproxy_enabled.is_some() {
Config::generate()?;
CoreManager::global().run_core().await?;
}
if socks_enabled.is_some() || http_enabled.is_some() {
Config::generate()?;
CoreManager::global().run_core().await?;
}
2022-11-17 17:07:13 +08:00
if auto_launch.is_some() {
sysopt::Sysopt::global().update_launch()?;
}
2024-05-26 17:59:39 +08:00
if system_proxy.is_some()
|| proxy_bypass.is_some()
|| port.is_some()
|| pac.is_some()
|| pac_content.is_some()
{
2022-11-17 17:07:13 +08:00
sysopt::Sysopt::global().update_sysproxy()?;
sysopt::Sysopt::global().guard_proxy();
}
2022-11-14 01:26:33 +08:00
2022-11-17 17:07:13 +08:00
if let Some(true) = patch.enable_proxy_guard {
sysopt::Sysopt::global().guard_proxy();
}
2022-11-14 01:26:33 +08:00
2022-11-17 17:07:13 +08:00
if let Some(hotkeys) = patch.hotkeys {
hotkey::Hotkey::global().update(hotkeys)?;
}
2022-11-14 01:26:33 +08:00
2022-11-17 17:07:13 +08:00
if language.is_some() {
handle::Handle::update_systray()?;
2024-02-24 00:52:21 +08:00
} else if system_proxy.is_some()
|| tun_mode.is_some()
|| common_tray_icon.is_some()
|| sysproxy_tray_icon.is_some()
|| tun_tray_icon.is_some()
{
2022-11-17 17:07:13 +08:00
handle::Handle::update_systray_part()?;
}
2024-05-15 12:00:38 +08:00
#[cfg(target_os = "macos")]
if tray_icon.is_some() {
handle::Handle::update_systray_part()?;
}
2022-11-17 17:07:13 +08:00
<Result<()>>::Ok(())
2024-06-12 10:00:22 +08:00
};
match res {
2022-11-17 17:07:13 +08:00
Ok(()) => {
Config::verge().apply();
Config::verge().data().save_file()?;
Ok(())
}
Err(err) => {
Config::verge().discard();
Err(err)
}
}
2022-11-14 01:26:33 +08:00
}
2022-11-12 11:37:23 +08:00
2022-11-16 01:26:41 +08:00
/// 更新某个profile
/// 如果更新当前订阅就激活订阅
2022-11-16 01:26:41 +08:00
pub async fn update_profile(uid: String, option: Option<PrfOption>) -> Result<()> {
2022-11-17 17:07:13 +08:00
let url_opt = {
let profiles = Config::profiles();
let profiles = profiles.latest();
let item = profiles.get_item(&uid)?;
let is_remote = item.itype.as_ref().map_or(false, |s| s == "remote");
if !is_remote {
None // 直接更新
} else if item.url.is_none() {
bail!("failed to get the profile item url");
} else {
Some((item.url.clone().unwrap(), item.option.clone()))
}
};
let should_update = match url_opt {
Some((url, opt)) => {
let merged_opt = PrfOption::merge(opt, option);
let item = PrfItem::from_url(&url, None, None, merged_opt).await?;
let profiles = Config::profiles();
let mut profiles = profiles.latest();
profiles.update_item(uid.clone(), item)?;
Some(uid) == profiles.get_current()
}
None => true,
};
if should_update {
2022-11-18 18:18:41 +08:00
update_core_config().await?;
2022-11-17 17:07:13 +08:00
}
2022-11-16 01:26:41 +08:00
Ok(())
}
2022-11-18 18:18:41 +08:00
/// 更新订阅
2022-11-18 18:18:41 +08:00
async fn update_core_config() -> Result<()> {
match CoreManager::global().update_config().await {
Ok(_) => {
handle::Handle::refresh_clash();
handle::Handle::notice_message("set_config::ok", "ok");
Ok(())
}
Err(err) => {
handle::Handle::notice_message("set_config::error", format!("{err}"));
Err(err)
}
}
}
/// copy env variable
pub fn copy_clash_env(app_handle: &AppHandle) {
2023-12-07 14:52:14 +08:00
let port = { Config::verge().latest().verge_mixed_port.unwrap_or(7897) };
2023-11-09 10:52:52 +08:00
let http_proxy = format!("http://127.0.0.1:{}", port);
let socks5_proxy = format!("socks5://127.0.0.1:{}", port);
2023-12-01 12:56:18 +08:00
let sh =
format!("export https_proxy={http_proxy} http_proxy={http_proxy} all_proxy={socks5_proxy}");
2024-05-03 12:50:04 +08:00
let cmd: String = format!("set http_proxy={http_proxy}\r\nset https_proxy={http_proxy}");
2023-11-09 10:52:52 +08:00
let ps: String = format!("$env:HTTP_PROXY=\"{http_proxy}\"; $env:HTTPS_PROXY=\"{http_proxy}\"");
2023-12-01 15:52:49 +08:00
let mut cliboard = app_handle.clipboard_manager();
let env_type = { Config::verge().latest().env_type.clone() };
let env_type = match env_type {
Some(env_type) => env_type,
None => {
#[cfg(not(target_os = "windows"))]
let default = "bash";
#[cfg(target_os = "windows")]
let default = "powershell";
default.to_string()
}
};
match env_type.as_str() {
"bash" => cliboard.write_text(sh).unwrap_or_default(),
2023-12-01 15:52:49 +08:00
"cmd" => cliboard.write_text(cmd).unwrap_or_default(),
"powershell" => cliboard.write_text(ps).unwrap_or_default(),
_ => log::error!(target: "app", "copy_clash_env: Invalid env type! {env_type}"),
2023-12-01 15:52:49 +08:00
};
}
2024-01-17 11:02:17 +08:00
pub async fn test_delay(url: String) -> Result<u32> {
use tokio::time::{Duration, Instant};
let mut builder = reqwest::ClientBuilder::new().use_rustls_tls().no_proxy();
let port = Config::verge()
.latest()
.verge_mixed_port
.unwrap_or(Config::clash().data().get_mixed_port());
let tun_mode = Config::verge().latest().enable_tun_mode.unwrap_or(false);
2024-01-17 11:02:17 +08:00
let proxy_scheme = format!("http://127.0.0.1:{port}");
if !tun_mode {
if let Ok(proxy) = reqwest::Proxy::http(&proxy_scheme) {
builder = builder.proxy(proxy);
}
if let Ok(proxy) = reqwest::Proxy::https(&proxy_scheme) {
builder = builder.proxy(proxy);
}
if let Ok(proxy) = reqwest::Proxy::all(&proxy_scheme) {
builder = builder.proxy(proxy);
}
2024-01-17 11:02:17 +08:00
}
let request = builder
.timeout(Duration::from_millis(10000))
.build()?
.get(url).header("User-Agent", "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");
2024-01-17 11:02:17 +08:00
let start = Instant::now();
2024-06-05 19:17:43 +08:00
let response = request.send().await;
match response {
Ok(response) => {
log::trace!(target: "app", "test_delay response: {:#?}", response);
if response.status().is_success() {
Ok(start.elapsed().as_millis() as u32)
} else {
Ok(10000u32)
}
}
Err(err) => {
log::trace!(target: "app", "test_delay error: {:#?}", err);
Err(err.into())
}
2024-01-17 11:02:17 +08:00
}
}
pub fn check_permission() -> Result<()> {
#[cfg(target_os = "windows")]
{
let hklm = winreg::RegKey::predef(winreg::enums::HKEY_LOCAL_MACHINE);
if let Ok(reg) = hklm.open_subkey_with_flags(
"SOFTWARE\\WOW6432Node\\Microsoft\\Windows\\CurrentVersion\\Run",
winreg::enums::KEY_SET_VALUE,
) {
reg.delete_value("Clash Verge").unwrap_or_default();
return Ok(());
}
}
Err(anyhow::anyhow!("permission denied"))
}