feat.rs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. //!
  2. //! feat mod 里的函数主要用于
  3. //! - hotkey 快捷键
  4. //! - timer 定时器
  5. //! - cmds 页面调用
  6. //!
  7. use crate::config::*;
  8. use crate::core::*;
  9. use crate::log_err;
  10. use crate::utils::resolve;
  11. use anyhow::{bail, Result};
  12. use serde_yaml::{Mapping, Value};
  13. use wry::application::clipboard::Clipboard;
  14. // 打开面板
  15. pub fn open_dashboard() {
  16. let handle = handle::Handle::global();
  17. let app_handle = handle.app_handle.lock();
  18. if let Some(app_handle) = app_handle.as_ref() {
  19. resolve::create_window(app_handle);
  20. }
  21. }
  22. // 重启clash
  23. pub fn restart_clash_core() {
  24. tauri::async_runtime::spawn(async {
  25. match CoreManager::global().run_core().await {
  26. Ok(_) => {
  27. handle::Handle::refresh_clash();
  28. handle::Handle::notice_message("set_config::ok", "ok");
  29. }
  30. Err(err) => {
  31. handle::Handle::notice_message("set_config::error", format!("{err}"));
  32. log::error!(target:"app", "{err}");
  33. }
  34. }
  35. });
  36. }
  37. // 切换模式 rule/global/direct/script mode
  38. pub fn change_clash_mode(mode: String) {
  39. let mut mapping = Mapping::new();
  40. mapping.insert(Value::from("mode"), mode.clone().into());
  41. tauri::async_runtime::spawn(async move {
  42. log::debug!(target: "app", "change clash mode to {mode}");
  43. match clash_api::patch_configs(&mapping).await {
  44. Ok(_) => {
  45. // 更新配置
  46. Config::clash().data().patch_config(mapping);
  47. if Config::clash().data().save_config().is_ok() {
  48. handle::Handle::refresh_clash();
  49. log_err!(handle::Handle::update_systray_part());
  50. }
  51. }
  52. Err(err) => log::error!(target: "app", "{err}"),
  53. }
  54. });
  55. }
  56. // 切换系统代理
  57. pub fn toggle_system_proxy() {
  58. let enable = Config::verge().draft().enable_system_proxy.clone();
  59. let enable = enable.unwrap_or(false);
  60. tauri::async_runtime::spawn(async move {
  61. match patch_verge(IVerge {
  62. enable_system_proxy: Some(!enable),
  63. ..IVerge::default()
  64. })
  65. .await
  66. {
  67. Ok(_) => handle::Handle::refresh_verge(),
  68. Err(err) => log::error!(target: "app", "{err}"),
  69. }
  70. });
  71. }
  72. // 打开系统代理
  73. pub fn enable_system_proxy() {
  74. tauri::async_runtime::spawn(async {
  75. match patch_verge(IVerge {
  76. enable_system_proxy: Some(true),
  77. ..IVerge::default()
  78. })
  79. .await
  80. {
  81. Ok(_) => handle::Handle::refresh_verge(),
  82. Err(err) => log::error!(target: "app", "{err}"),
  83. }
  84. });
  85. }
  86. // 关闭系统代理
  87. pub fn disable_system_proxy() {
  88. tauri::async_runtime::spawn(async {
  89. match patch_verge(IVerge {
  90. enable_system_proxy: Some(false),
  91. ..IVerge::default()
  92. })
  93. .await
  94. {
  95. Ok(_) => handle::Handle::refresh_verge(),
  96. Err(err) => log::error!(target: "app", "{err}"),
  97. }
  98. });
  99. }
  100. // 切换tun模式
  101. pub fn toggle_tun_mode() {
  102. let enable = Config::verge().data().enable_tun_mode.clone();
  103. let enable = enable.unwrap_or(false);
  104. tauri::async_runtime::spawn(async move {
  105. match patch_verge(IVerge {
  106. enable_tun_mode: Some(!enable),
  107. ..IVerge::default()
  108. })
  109. .await
  110. {
  111. Ok(_) => handle::Handle::refresh_verge(),
  112. Err(err) => log::error!(target: "app", "{err}"),
  113. }
  114. });
  115. }
  116. // 打开tun模式
  117. pub fn enable_tun_mode() {
  118. tauri::async_runtime::spawn(async {
  119. match patch_verge(IVerge {
  120. enable_tun_mode: Some(true),
  121. ..IVerge::default()
  122. })
  123. .await
  124. {
  125. Ok(_) => handle::Handle::refresh_verge(),
  126. Err(err) => log::error!(target: "app", "{err}"),
  127. }
  128. });
  129. }
  130. // 关闭tun模式
  131. pub fn disable_tun_mode() {
  132. tauri::async_runtime::spawn(async {
  133. match patch_verge(IVerge {
  134. enable_tun_mode: Some(false),
  135. ..IVerge::default()
  136. })
  137. .await
  138. {
  139. Ok(_) => handle::Handle::refresh_verge(),
  140. Err(err) => log::error!(target: "app", "{err}"),
  141. }
  142. });
  143. }
  144. /// 修改clash的配置
  145. pub async fn patch_clash(patch: Mapping) -> Result<()> {
  146. Config::clash().draft().patch_config(patch.clone());
  147. match {
  148. let mixed_port = patch.get("mixed-port");
  149. if mixed_port.is_some() {
  150. let changed = mixed_port != Config::clash().data().0.get("mixed-port");
  151. // 检查端口占用
  152. if changed {
  153. if let Some(port) = mixed_port.clone().unwrap().as_u64() {
  154. if !port_scanner::local_port_available(port as u16) {
  155. Config::clash().discard();
  156. bail!("port already in use");
  157. }
  158. }
  159. }
  160. };
  161. // 激活配置
  162. if mixed_port.is_some()
  163. || patch.get("secret").is_some()
  164. || patch.get("external-controller").is_some()
  165. {
  166. Config::generate()?;
  167. CoreManager::global().run_core().await?;
  168. handle::Handle::refresh_clash();
  169. }
  170. // 更新系统代理
  171. if mixed_port.is_some() {
  172. log_err!(sysopt::Sysopt::global().init_sysproxy());
  173. }
  174. if patch.get("mode").is_some() {
  175. log_err!(handle::Handle::update_systray_part());
  176. }
  177. Config::runtime().latest().patch_config(patch);
  178. <Result<()>>::Ok(())
  179. } {
  180. Ok(()) => {
  181. Config::clash().apply();
  182. Config::clash().data().save_config()?;
  183. Ok(())
  184. }
  185. Err(err) => {
  186. Config::clash().discard();
  187. Err(err)
  188. }
  189. }
  190. }
  191. /// 修改verge的配置
  192. /// 一般都是一个个的修改
  193. pub async fn patch_verge(patch: IVerge) -> Result<()> {
  194. Config::verge().draft().patch_config(patch.clone());
  195. let tun_mode = patch.enable_tun_mode;
  196. let auto_launch = patch.enable_auto_launch;
  197. let system_proxy = patch.enable_system_proxy;
  198. let proxy_bypass = patch.system_proxy_bypass;
  199. let language = patch.language;
  200. match {
  201. #[cfg(target_os = "windows")]
  202. {
  203. let service_mode = patch.enable_service_mode;
  204. if service_mode.is_some() {
  205. log::debug!(target: "app", "change service mode to {}", service_mode.unwrap());
  206. Config::generate()?;
  207. CoreManager::global().run_core().await?;
  208. } else if tun_mode.is_some() {
  209. update_core_config().await?;
  210. }
  211. }
  212. #[cfg(not(target_os = "windows"))]
  213. if tun_mode.is_some() {
  214. update_core_config().await?;
  215. }
  216. if auto_launch.is_some() {
  217. sysopt::Sysopt::global().update_launch()?;
  218. }
  219. if system_proxy.is_some() || proxy_bypass.is_some() {
  220. sysopt::Sysopt::global().update_sysproxy()?;
  221. sysopt::Sysopt::global().guard_proxy();
  222. }
  223. if let Some(true) = patch.enable_proxy_guard {
  224. sysopt::Sysopt::global().guard_proxy();
  225. }
  226. if let Some(hotkeys) = patch.hotkeys {
  227. hotkey::Hotkey::global().update(hotkeys)?;
  228. }
  229. if language.is_some() {
  230. handle::Handle::update_systray()?;
  231. } else if system_proxy.or(tun_mode).is_some() {
  232. handle::Handle::update_systray_part()?;
  233. }
  234. <Result<()>>::Ok(())
  235. } {
  236. Ok(()) => {
  237. Config::verge().apply();
  238. Config::verge().data().save_file()?;
  239. Ok(())
  240. }
  241. Err(err) => {
  242. Config::verge().discard();
  243. Err(err)
  244. }
  245. }
  246. }
  247. /// 更新某个profile
  248. /// 如果更新当前配置就激活配置
  249. pub async fn update_profile(uid: String, option: Option<PrfOption>) -> Result<()> {
  250. let url_opt = {
  251. let profiles = Config::profiles();
  252. let profiles = profiles.latest();
  253. let item = profiles.get_item(&uid)?;
  254. let is_remote = item.itype.as_ref().map_or(false, |s| s == "remote");
  255. if !is_remote {
  256. None // 直接更新
  257. } else if item.url.is_none() {
  258. bail!("failed to get the profile item url");
  259. } else {
  260. Some((item.url.clone().unwrap(), item.option.clone()))
  261. }
  262. };
  263. let should_update = match url_opt {
  264. Some((url, opt)) => {
  265. let merged_opt = PrfOption::merge(opt, option);
  266. let item = PrfItem::from_url(&url, None, None, merged_opt).await?;
  267. let profiles = Config::profiles();
  268. let mut profiles = profiles.latest();
  269. profiles.update_item(uid.clone(), item)?;
  270. Some(uid) == profiles.get_current()
  271. }
  272. None => true,
  273. };
  274. if should_update {
  275. update_core_config().await?;
  276. }
  277. Ok(())
  278. }
  279. /// 更新配置
  280. async fn update_core_config() -> Result<()> {
  281. match CoreManager::global().update_config().await {
  282. Ok(_) => {
  283. handle::Handle::refresh_clash();
  284. handle::Handle::notice_message("set_config::ok", "ok");
  285. Ok(())
  286. }
  287. Err(err) => {
  288. handle::Handle::notice_message("set_config::error", format!("{err}"));
  289. Err(err)
  290. }
  291. }
  292. }
  293. /// copy env variable
  294. pub fn copy_clash_env() {
  295. let port = { Config::clash().data().get_client_info().port };
  296. let text = format!("export https_proxy=http://127.0.0.1:{port} http_proxy=http://127.0.0.1:{port} all_proxy=socks5://127.0.0.1:{port}");
  297. let mut cliboard = Clipboard::new();
  298. cliboard.write_text(text);
  299. }