style: resolve formatting problem

This commit is contained in:
limsanity 2022-07-13 00:54:47 +08:00
parent fbb17a0ba5
commit c278f1af00
6 changed files with 1379 additions and 1379 deletions

View File

@ -1,6 +1,6 @@
use crate::{ use crate::{
core::{ClashInfo, Core, PrfItem, PrfOption, Profiles, Verge}, core::{ClashInfo, Core, PrfItem, PrfOption, Profiles, Verge},
utils::{dirs, help, sysopt::SysProxyConfig}, utils::{dirs, help, sysopt::SysProxyConfig},
}; };
use crate::{log_if_err, ret_err, wrap_err}; use crate::{log_if_err, ret_err, wrap_err};
use anyhow::Result; use anyhow::Result;
@ -12,28 +12,28 @@ type CmdResult<T = ()> = Result<T, String>;
/// get all profiles from `profiles.yaml` /// get all profiles from `profiles.yaml`
#[tauri::command] #[tauri::command]
pub fn get_profiles(core: State<'_, Core>) -> CmdResult<Profiles> { pub fn get_profiles(core: State<'_, Core>) -> CmdResult<Profiles> {
let profiles = core.profiles.lock(); let profiles = core.profiles.lock();
Ok(profiles.clone()) Ok(profiles.clone())
} }
/// manually exec enhanced profile /// manually exec enhanced profile
#[tauri::command] #[tauri::command]
pub fn enhance_profiles(core: State<'_, Core>) -> CmdResult { pub fn enhance_profiles(core: State<'_, Core>) -> CmdResult {
wrap_err!(core.activate_enhanced(false)) wrap_err!(core.activate_enhanced(false))
} }
/// import the profile from url /// import the profile from url
/// and save to `profiles.yaml` /// and save to `profiles.yaml`
#[tauri::command] #[tauri::command]
pub async fn import_profile( pub async fn import_profile(
url: String, url: String,
option: Option<PrfOption>, option: Option<PrfOption>,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
let item = wrap_err!(PrfItem::from_url(&url, None, None, option).await)?; let item = wrap_err!(PrfItem::from_url(&url, None, None, option).await)?;
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
wrap_err!(profiles.append_item(item)) wrap_err!(profiles.append_item(item))
} }
/// new a profile /// new a profile
@ -41,137 +41,137 @@ pub async fn import_profile(
/// view the temp profile file by using vscode or other editor /// view the temp profile file by using vscode or other editor
#[tauri::command] #[tauri::command]
pub async fn create_profile( pub async fn create_profile(
item: PrfItem, // partial item: PrfItem, // partial
file_data: Option<String>, file_data: Option<String>,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
let item = wrap_err!(PrfItem::from(item, file_data).await)?; let item = wrap_err!(PrfItem::from(item, file_data).await)?;
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
wrap_err!(profiles.append_item(item)) wrap_err!(profiles.append_item(item))
} }
/// Update the profile /// Update the profile
#[tauri::command] #[tauri::command]
pub async fn update_profile( pub async fn update_profile(
index: String, index: String,
option: Option<PrfOption>, option: Option<PrfOption>,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
wrap_err!(Core::update_profile_item(core.inner().clone(), index, option).await) wrap_err!(Core::update_profile_item(core.inner().clone(), index, option).await)
} }
/// change the current profile /// change the current profile
#[tauri::command] #[tauri::command]
pub fn select_profile(index: String, core: State<'_, Core>) -> CmdResult { pub fn select_profile(index: String, core: State<'_, Core>) -> CmdResult {
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
wrap_err!(profiles.put_current(index))?; wrap_err!(profiles.put_current(index))?;
drop(profiles); drop(profiles);
wrap_err!(core.activate_enhanced(false)) wrap_err!(core.activate_enhanced(false))
} }
/// change the profile chain /// change the profile chain
#[tauri::command] #[tauri::command]
pub fn change_profile_chain(chain: Option<Vec<String>>, core: State<'_, Core>) -> CmdResult { pub fn change_profile_chain(chain: Option<Vec<String>>, core: State<'_, Core>) -> CmdResult {
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
profiles.put_chain(chain); profiles.put_chain(chain);
drop(profiles); drop(profiles);
wrap_err!(core.activate_enhanced(false)) wrap_err!(core.activate_enhanced(false))
} }
/// change the profile valid fields /// change the profile valid fields
#[tauri::command] #[tauri::command]
pub fn change_profile_valid(valid: Option<Vec<String>>, core: State<Core>) -> CmdResult { pub fn change_profile_valid(valid: Option<Vec<String>>, core: State<Core>) -> CmdResult {
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
profiles.put_valid(valid); profiles.put_valid(valid);
drop(profiles); drop(profiles);
wrap_err!(core.activate_enhanced(false)) wrap_err!(core.activate_enhanced(false))
} }
/// delete profile item /// delete profile item
#[tauri::command] #[tauri::command]
pub fn delete_profile(index: String, core: State<'_, Core>) -> CmdResult { pub fn delete_profile(index: String, core: State<'_, Core>) -> CmdResult {
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
if wrap_err!(profiles.delete_item(index))? { if wrap_err!(profiles.delete_item(index))? {
drop(profiles); drop(profiles);
log_if_err!(core.activate_enhanced(false)); log_if_err!(core.activate_enhanced(false));
} }
Ok(()) Ok(())
} }
/// patch the profile config /// patch the profile config
#[tauri::command] #[tauri::command]
pub fn patch_profile(index: String, profile: PrfItem, core: State<'_, Core>) -> CmdResult { pub fn patch_profile(index: String, profile: PrfItem, core: State<'_, Core>) -> CmdResult {
let mut profiles = core.profiles.lock(); let mut profiles = core.profiles.lock();
wrap_err!(profiles.patch_item(index, profile))?; wrap_err!(profiles.patch_item(index, profile))?;
drop(profiles); drop(profiles);
// update cron task // update cron task
let mut timer = core.timer.lock(); let mut timer = core.timer.lock();
wrap_err!(timer.refresh()) wrap_err!(timer.refresh())
} }
/// run vscode command to edit the profile /// run vscode command to edit the profile
#[tauri::command] #[tauri::command]
pub fn view_profile(index: String, core: State<'_, Core>) -> CmdResult { pub fn view_profile(index: String, core: State<'_, Core>) -> CmdResult {
let profiles = core.profiles.lock(); let profiles = core.profiles.lock();
let item = wrap_err!(profiles.get_item(&index))?; let item = wrap_err!(profiles.get_item(&index))?;
let file = item.file.clone(); let file = item.file.clone();
if file.is_none() { if file.is_none() {
ret_err!("the file is null"); ret_err!("the file is null");
} }
let path = dirs::app_profiles_dir().join(file.unwrap()); let path = dirs::app_profiles_dir().join(file.unwrap());
if !path.exists() { if !path.exists() {
ret_err!("the file not found"); ret_err!("the file not found");
} }
wrap_err!(help::open_file(path)) wrap_err!(help::open_file(path))
} }
/// read the profile item file data /// read the profile item file data
#[tauri::command] #[tauri::command]
pub fn read_profile_file(index: String, core: State<'_, Core>) -> CmdResult<String> { pub fn read_profile_file(index: String, core: State<'_, Core>) -> CmdResult<String> {
let profiles = core.profiles.lock(); let profiles = core.profiles.lock();
let item = wrap_err!(profiles.get_item(&index))?; let item = wrap_err!(profiles.get_item(&index))?;
let data = wrap_err!(item.read_file())?; let data = wrap_err!(item.read_file())?;
Ok(data) Ok(data)
} }
/// save the profile item file data /// save the profile item file data
#[tauri::command] #[tauri::command]
pub fn save_profile_file( pub fn save_profile_file(
index: String, index: String,
file_data: Option<String>, file_data: Option<String>,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
if file_data.is_none() { if file_data.is_none() {
return Ok(()); return Ok(());
} }
let profiles = core.profiles.lock(); let profiles = core.profiles.lock();
let item = wrap_err!(profiles.get_item(&index))?; let item = wrap_err!(profiles.get_item(&index))?;
wrap_err!(item.save_file(file_data.unwrap())) wrap_err!(item.save_file(file_data.unwrap()))
} }
/// get the clash core info from the state /// get the clash core info from the state
/// the caller can also get the infomation by clash's api /// the caller can also get the infomation by clash's api
#[tauri::command] #[tauri::command]
pub fn get_clash_info(core: State<'_, Core>) -> CmdResult<ClashInfo> { pub fn get_clash_info(core: State<'_, Core>) -> CmdResult<ClashInfo> {
let clash = core.clash.lock(); let clash = core.clash.lock();
Ok(clash.info.clone()) Ok(clash.info.clone())
} }
/// update the clash core config /// update the clash core config
@ -179,138 +179,138 @@ pub fn get_clash_info(core: State<'_, Core>) -> CmdResult<ClashInfo> {
/// then we should save the latest config /// then we should save the latest config
#[tauri::command] #[tauri::command]
pub fn patch_clash_config( pub fn patch_clash_config(
payload: Mapping, payload: Mapping,
app_handle: tauri::AppHandle, app_handle: tauri::AppHandle,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
wrap_err!(core.patch_clash(payload, &app_handle)) wrap_err!(core.patch_clash(payload, &app_handle))
} }
/// get the verge config /// get the verge config
#[tauri::command] #[tauri::command]
pub fn get_verge_config(core: State<'_, Core>) -> CmdResult<Verge> { pub fn get_verge_config(core: State<'_, Core>) -> CmdResult<Verge> {
let verge = core.verge.lock(); let verge = core.verge.lock();
Ok(verge.clone()) Ok(verge.clone())
} }
/// patch the verge config /// patch the verge config
/// this command only save the config and not responsible for other things /// this command only save the config and not responsible for other things
#[tauri::command] #[tauri::command]
pub fn patch_verge_config( pub fn patch_verge_config(
payload: Verge, payload: Verge,
app_handle: tauri::AppHandle, app_handle: tauri::AppHandle,
core: State<'_, Core>, core: State<'_, Core>,
) -> CmdResult { ) -> CmdResult {
wrap_err!(core.patch_verge(payload, &app_handle)) wrap_err!(core.patch_verge(payload, &app_handle))
} }
/// change clash core /// change clash core
#[tauri::command] #[tauri::command]
pub fn change_clash_core(core: State<'_, Core>, clash_core: Option<String>) -> CmdResult { pub fn change_clash_core(core: State<'_, Core>, clash_core: Option<String>) -> CmdResult {
wrap_err!(core.change_core(clash_core)) wrap_err!(core.change_core(clash_core))
} }
/// restart the sidecar /// restart the sidecar
#[tauri::command] #[tauri::command]
pub fn restart_sidecar(core: State<'_, Core>) -> CmdResult { pub fn restart_sidecar(core: State<'_, Core>) -> CmdResult {
wrap_err!(core.restart_clash()) wrap_err!(core.restart_clash())
} }
/// kill all sidecars when update app /// kill all sidecars when update app
#[tauri::command] #[tauri::command]
pub fn kill_sidecar() { pub fn kill_sidecar() {
api::process::kill_children(); api::process::kill_children();
} }
/// get the system proxy /// get the system proxy
#[tauri::command] #[tauri::command]
pub fn get_sys_proxy() -> Result<SysProxyConfig, String> { pub fn get_sys_proxy() -> Result<SysProxyConfig, String> {
wrap_err!(SysProxyConfig::get_sys()) wrap_err!(SysProxyConfig::get_sys())
} }
/// get the current proxy config /// get the current proxy config
/// which may not the same as system proxy /// which may not the same as system proxy
#[tauri::command] #[tauri::command]
pub fn get_cur_proxy(core: State<'_, Core>) -> CmdResult<Option<SysProxyConfig>> { pub fn get_cur_proxy(core: State<'_, Core>) -> CmdResult<Option<SysProxyConfig>> {
let sysopt = core.sysopt.lock(); let sysopt = core.sysopt.lock();
wrap_err!(sysopt.get_sysproxy()) wrap_err!(sysopt.get_sysproxy())
} }
/// open app config dir /// open app config dir
#[tauri::command] #[tauri::command]
pub fn open_app_dir() -> Result<(), String> { pub fn open_app_dir() -> Result<(), String> {
let app_dir = dirs::app_home_dir(); let app_dir = dirs::app_home_dir();
wrap_err!(open::that(app_dir)) wrap_err!(open::that(app_dir))
} }
/// open logs dir /// open logs dir
#[tauri::command] #[tauri::command]
pub fn open_logs_dir() -> Result<(), String> { pub fn open_logs_dir() -> Result<(), String> {
let log_dir = dirs::app_logs_dir(); let log_dir = dirs::app_logs_dir();
wrap_err!(open::that(log_dir)) wrap_err!(open::that(log_dir))
} }
/// service mode /// service mode
#[cfg(windows)] #[cfg(windows)]
pub mod service { pub mod service {
use super::*; use super::*;
use crate::core::win_service::JsonResponse; use crate::core::win_service::JsonResponse;
#[tauri::command] #[tauri::command]
pub async fn start_service() -> Result<(), String> { pub async fn start_service() -> Result<(), String> {
wrap_err!(crate::core::Service::start_service().await) wrap_err!(crate::core::Service::start_service().await)
} }
#[tauri::command] #[tauri::command]
pub async fn stop_service() -> Result<(), String> { pub async fn stop_service() -> Result<(), String> {
wrap_err!(crate::core::Service::stop_service().await) wrap_err!(crate::core::Service::stop_service().await)
} }
#[tauri::command] #[tauri::command]
pub async fn check_service() -> Result<JsonResponse, String> { pub async fn check_service() -> Result<JsonResponse, String> {
// no log // no log
match crate::core::Service::check_service().await { match crate::core::Service::check_service().await {
Ok(res) => Ok(res), Ok(res) => Ok(res),
Err(err) => Err(err.to_string()), Err(err) => Err(err.to_string()),
}
} }
}
#[tauri::command] #[tauri::command]
pub async fn install_service() -> Result<(), String> { pub async fn install_service() -> Result<(), String> {
wrap_err!(crate::core::Service::install_service().await) wrap_err!(crate::core::Service::install_service().await)
} }
#[tauri::command] #[tauri::command]
pub async fn uninstall_service() -> Result<(), String> { pub async fn uninstall_service() -> Result<(), String> {
wrap_err!(crate::core::Service::uninstall_service().await) wrap_err!(crate::core::Service::uninstall_service().await)
} }
} }
#[cfg(not(windows))] #[cfg(not(windows))]
pub mod service { pub mod service {
use super::*; use super::*;
#[tauri::command] #[tauri::command]
pub async fn start_service() -> Result<(), String> { pub async fn start_service() -> Result<(), String> {
Ok(()) Ok(())
} }
#[tauri::command] #[tauri::command]
pub async fn stop_service() -> Result<(), String> { pub async fn stop_service() -> Result<(), String> {
Ok(()) Ok(())
} }
#[tauri::command] #[tauri::command]
pub async fn check_service() -> Result<(), String> { pub async fn check_service() -> Result<(), String> {
Ok(()) Ok(())
} }
#[tauri::command] #[tauri::command]
pub async fn install_service() -> Result<(), String> { pub async fn install_service() -> Result<(), String> {
Ok(()) Ok(())
} }
#[tauri::command] #[tauri::command]
pub async fn uninstall_service() -> Result<(), String> { pub async fn uninstall_service() -> Result<(), String> {
Ok(()) Ok(())
} }
} }

View File

@ -5,304 +5,304 @@ use serde_yaml::{Mapping, Value};
#[derive(Default, Debug, Clone, Deserialize, Serialize)] #[derive(Default, Debug, Clone, Deserialize, Serialize)]
pub struct ClashInfo { pub struct ClashInfo {
/// clash sidecar status /// clash sidecar status
pub status: String, pub status: String,
/// clash core port /// clash core port
pub port: Option<String>, pub port: Option<String>,
/// same as `external-controller` /// same as `external-controller`
pub server: Option<String>, pub server: Option<String>,
/// clash secret /// clash secret
pub secret: Option<String>, pub secret: Option<String>,
/// mode /// mode
pub mode: Option<String>, pub mode: Option<String>,
} }
impl ClashInfo { impl ClashInfo {
/// parse the clash's config.yaml /// parse the clash's config.yaml
/// get some information /// get some information
pub fn from(config: &Mapping) -> ClashInfo { pub fn from(config: &Mapping) -> ClashInfo {
let key_port_1 = Value::from("port"); let key_port_1 = Value::from("port");
let key_port_2 = Value::from("mixed-port"); let key_port_2 = Value::from("mixed-port");
let key_server = Value::from("external-controller"); let key_server = Value::from("external-controller");
let key_secret = Value::from("secret"); let key_secret = Value::from("secret");
let key_mode = Value::from("mode"); let key_mode = Value::from("mode");
let port = match config.get(&key_port_1) { let port = match config.get(&key_port_1) {
Some(value) => match value { Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()), Value::String(val_str) => Some(val_str.clone()),
Value::Number(val_num) => Some(val_num.to_string()), Value::Number(val_num) => Some(val_num.to_string()),
_ => None, _ => None,
}, },
_ => None, _ => None,
}; };
let port = match port { let port = match port {
Some(_) => port, Some(_) => port,
None => match config.get(&key_port_2) { None => match config.get(&key_port_2) {
Some(value) => match value { Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()), Value::String(val_str) => Some(val_str.clone()),
Value::Number(val_num) => Some(val_num.to_string()), Value::Number(val_num) => Some(val_num.to_string()),
_ => None, _ => None,
}, },
_ => None, _ => None,
}, },
}; };
// `external-controller` could be // `external-controller` could be
// "127.0.0.1:9090" or ":9090" // "127.0.0.1:9090" or ":9090"
let server = match config.get(&key_server) { let server = match config.get(&key_server) {
Some(value) => { Some(value) => {
let val_str = value.as_str().unwrap_or(""); let val_str = value.as_str().unwrap_or("");
if val_str.starts_with(":") { if val_str.starts_with(":") {
Some(format!("127.0.0.1{val_str}")) Some(format!("127.0.0.1{val_str}"))
} else { } else {
Some(val_str.into()) Some(val_str.into())
}
}
_ => None,
};
let secret = match config.get(&key_secret) {
Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()),
Value::Bool(val_bool) => Some(val_bool.to_string()),
Value::Number(val_num) => Some(val_num.to_string()),
_ => None,
},
_ => None,
};
let mode = match config.get(&key_mode) {
Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()),
_ => None,
},
_ => None,
};
ClashInfo {
status: "init".into(),
port,
server,
secret,
mode,
} }
}
_ => None,
};
let secret = match config.get(&key_secret) {
Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()),
Value::Bool(val_bool) => Some(val_bool.to_string()),
Value::Number(val_num) => Some(val_num.to_string()),
_ => None,
},
_ => None,
};
let mode = match config.get(&key_mode) {
Some(value) => match value {
Value::String(val_str) => Some(val_str.clone()),
_ => None,
},
_ => None,
};
ClashInfo {
status: "init".into(),
port,
server,
secret,
mode,
} }
}
} }
pub struct Clash { pub struct Clash {
/// maintain the clash config /// maintain the clash config
pub config: Mapping, pub config: Mapping,
/// some info /// some info
pub info: ClashInfo, pub info: ClashInfo,
} }
impl Clash { impl Clash {
pub fn new() -> Clash { pub fn new() -> Clash {
let config = Clash::read_config(); let config = Clash::read_config();
let info = ClashInfo::from(&config); let info = ClashInfo::from(&config);
Clash { config, info } Clash { config, info }
}
/// get clash config
pub fn read_config() -> Mapping {
config::read_yaml::<Mapping>(dirs::clash_path())
}
/// save the clash config
pub fn save_config(&self) -> Result<()> {
config::save_yaml(
dirs::clash_path(),
&self.config,
Some("# Default Config For Clash Core\n\n"),
)
}
/// patch update the clash config
/// if the port is changed then return true
pub fn patch_config(&mut self, patch: Mapping) -> Result<(bool, bool)> {
let port_key = Value::from("mixed-port");
let server_key = Value::from("external-controller");
let secret_key = Value::from("secret");
let mode_key = Value::from("mode");
let mut change_port = false;
let mut change_info = false;
let mut change_mode = false;
for (key, value) in patch.into_iter() {
if key == port_key {
change_port = true;
}
if key == mode_key {
change_mode = true;
}
if key == port_key || key == server_key || key == secret_key || key == mode_key {
change_info = true;
}
self.config.insert(key, value);
} }
/// get clash config if change_info {
pub fn read_config() -> Mapping { self.info = ClashInfo::from(&self.config);
config::read_yaml::<Mapping>(dirs::clash_path())
} }
/// save the clash config self.save_config()?;
pub fn save_config(&self) -> Result<()> {
config::save_yaml( Ok((change_port, change_mode))
dirs::clash_path(), }
&self.config,
Some("# Default Config For Clash Core\n\n"), /// revise the `tun` and `dns` config
) pub fn _tun_mode(mut config: Mapping, enable: bool) -> Mapping {
macro_rules! revise {
($map: expr, $key: expr, $val: expr) => {
let ret_key = Value::String($key.into());
$map.insert(ret_key, Value::from($val));
};
} }
/// patch update the clash config // if key not exists then append value
/// if the port is changed then return true macro_rules! append {
pub fn patch_config(&mut self, patch: Mapping) -> Result<(bool, bool)> { ($map: expr, $key: expr, $val: expr) => {
let port_key = Value::from("mixed-port"); let ret_key = Value::String($key.into());
let server_key = Value::from("external-controller"); if !$map.contains_key(&ret_key) {
let secret_key = Value::from("secret"); $map.insert(ret_key, Value::from($val));
let mode_key = Value::from("mode");
let mut change_port = false;
let mut change_info = false;
let mut change_mode = false;
for (key, value) in patch.into_iter() {
if key == port_key {
change_port = true;
}
if key == mode_key {
change_mode = true;
}
if key == port_key || key == server_key || key == secret_key || key == mode_key {
change_info = true;
}
self.config.insert(key, value);
} }
};
if change_info {
self.info = ClashInfo::from(&self.config);
}
self.save_config()?;
Ok((change_port, change_mode))
} }
/// revise the `tun` and `dns` config // tun config
pub fn _tun_mode(mut config: Mapping, enable: bool) -> Mapping { let tun_val = config.get(&Value::from("tun"));
macro_rules! revise { let mut new_tun = Mapping::new();
($map: expr, $key: expr, $val: expr) => {
let ret_key = Value::String($key.into());
$map.insert(ret_key, Value::from($val));
};
}
// if key not exists then append value if tun_val.is_some() && tun_val.as_ref().unwrap().is_mapping() {
macro_rules! append { new_tun = tun_val.as_ref().unwrap().as_mapping().unwrap().clone();
($map: expr, $key: expr, $val: expr) => {
let ret_key = Value::String($key.into());
if !$map.contains_key(&ret_key) {
$map.insert(ret_key, Value::from($val));
}
};
}
// tun config
let tun_val = config.get(&Value::from("tun"));
let mut new_tun = Mapping::new();
if tun_val.is_some() && tun_val.as_ref().unwrap().is_mapping() {
new_tun = tun_val.as_ref().unwrap().as_mapping().unwrap().clone();
}
revise!(new_tun, "enable", enable);
if enable {
append!(new_tun, "stack", "gvisor");
append!(new_tun, "dns-hijack", vec!["198.18.0.2:53"]);
append!(new_tun, "auto-route", true);
append!(new_tun, "auto-detect-interface", true);
}
revise!(config, "tun", new_tun);
if enable {
// dns config
let dns_val = config.get(&Value::from("dns"));
let mut new_dns = Mapping::new();
if dns_val.is_some() && dns_val.as_ref().unwrap().is_mapping() {
new_dns = dns_val.as_ref().unwrap().as_mapping().unwrap().clone();
}
revise!(new_dns, "enable", enable);
// 借鉴cfw的默认配置
append!(new_dns, "enhanced-mode", "fake-ip");
append!(
new_dns,
"nameserver",
vec!["114.114.114.114", "223.5.5.5", "8.8.8.8"]
);
append!(new_dns, "fallback", vec![] as Vec<&str>);
#[cfg(target_os = "windows")]
append!(
new_dns,
"fake-ip-filter",
vec![
"dns.msftncsi.com",
"www.msftncsi.com",
"www.msftconnecttest.com"
]
);
revise!(config, "dns", new_dns);
}
config
} }
/// only 5 default fields available (clash config fields) revise!(new_tun, "enable", enable);
/// convert to lowercase
pub fn strict_filter(config: Mapping) -> Mapping {
// Only the following fields are allowed:
// proxies/proxy-providers/proxy-groups/rule-providers/rules
let valid_keys = vec![
"proxies",
"proxy-providers",
"proxy-groups",
"rules",
"rule-providers",
];
let mut new_config = Mapping::new(); if enable {
append!(new_tun, "stack", "gvisor");
for (key, value) in config.into_iter() { append!(new_tun, "dns-hijack", vec!["198.18.0.2:53"]);
key.as_str().map(|key_str| { append!(new_tun, "auto-route", true);
// change to lowercase append!(new_tun, "auto-detect-interface", true);
let mut key_str = String::from(key_str);
key_str.make_ascii_lowercase();
// filter
if valid_keys.contains(&&*key_str) {
new_config.insert(Value::String(key_str), value);
}
});
}
new_config
} }
/// more clash config fields available revise!(config, "tun", new_tun);
/// convert to lowercase
pub fn loose_filter(config: Mapping) -> Mapping {
// all of these can not be revised by script or merge
// http/https/socks port should be under control
let not_allow = vec![
"port",
"socks-port",
"mixed-port",
"allow-lan",
"mode",
"external-controller",
"secret",
"log-level",
];
let mut new_config = Mapping::new(); if enable {
// dns config
let dns_val = config.get(&Value::from("dns"));
let mut new_dns = Mapping::new();
for (key, value) in config.into_iter() { if dns_val.is_some() && dns_val.as_ref().unwrap().is_mapping() {
key.as_str().map(|key_str| { new_dns = dns_val.as_ref().unwrap().as_mapping().unwrap().clone();
// change to lowercase }
let mut key_str = String::from(key_str); revise!(new_dns, "enable", enable);
key_str.make_ascii_lowercase();
// filter // 借鉴cfw的默认配置
if !not_allow.contains(&&*key_str) { append!(new_dns, "enhanced-mode", "fake-ip");
new_config.insert(Value::String(key_str), value); append!(
} new_dns,
}); "nameserver",
} vec!["114.114.114.114", "223.5.5.5", "8.8.8.8"]
);
append!(new_dns, "fallback", vec![] as Vec<&str>);
new_config #[cfg(target_os = "windows")]
append!(
new_dns,
"fake-ip-filter",
vec![
"dns.msftncsi.com",
"www.msftncsi.com",
"www.msftconnecttest.com"
]
);
revise!(config, "dns", new_dns);
} }
config
}
/// only 5 default fields available (clash config fields)
/// convert to lowercase
pub fn strict_filter(config: Mapping) -> Mapping {
// Only the following fields are allowed:
// proxies/proxy-providers/proxy-groups/rule-providers/rules
let valid_keys = vec![
"proxies",
"proxy-providers",
"proxy-groups",
"rules",
"rule-providers",
];
let mut new_config = Mapping::new();
for (key, value) in config.into_iter() {
key.as_str().map(|key_str| {
// change to lowercase
let mut key_str = String::from(key_str);
key_str.make_ascii_lowercase();
// filter
if valid_keys.contains(&&*key_str) {
new_config.insert(Value::String(key_str), value);
}
});
}
new_config
}
/// more clash config fields available
/// convert to lowercase
pub fn loose_filter(config: Mapping) -> Mapping {
// all of these can not be revised by script or merge
// http/https/socks port should be under control
let not_allow = vec![
"port",
"socks-port",
"mixed-port",
"allow-lan",
"mode",
"external-controller",
"secret",
"log-level",
];
let mut new_config = Mapping::new();
for (key, value) in config.into_iter() {
key.as_str().map(|key_str| {
// change to lowercase
let mut key_str = String::from(key_str);
key_str.make_ascii_lowercase();
// filter
if !not_allow.contains(&&*key_str) {
new_config.insert(Value::String(key_str), value);
}
});
}
new_config
}
} }
impl Default for Clash { impl Default for Clash {
fn default() -> Self { fn default() -> Self {
Clash::new() Clash::new()
} }
} }

View File

@ -35,461 +35,467 @@ static mut WINDOW_CLOSABLE: bool = true;
#[derive(Clone)] #[derive(Clone)]
pub struct Core { pub struct Core {
pub clash: Arc<Mutex<Clash>>, pub clash: Arc<Mutex<Clash>>,
pub verge: Arc<Mutex<Verge>>, pub verge: Arc<Mutex<Verge>>,
pub profiles: Arc<Mutex<Profiles>>, pub profiles: Arc<Mutex<Profiles>>,
pub service: Arc<Mutex<Service>>, pub service: Arc<Mutex<Service>>,
pub sysopt: Arc<Mutex<Sysopt>>, pub sysopt: Arc<Mutex<Sysopt>>,
pub timer: Arc<Mutex<Timer>>, pub timer: Arc<Mutex<Timer>>,
pub window: Arc<Mutex<Option<Window>>>, pub window: Arc<Mutex<Option<Window>>>,
} }
impl Core { impl Core {
pub fn new() -> Core { pub fn new() -> Core {
let clash = Clash::new(); let clash = Clash::new();
let verge = Verge::new(); let verge = Verge::new();
let profiles = Profiles::new(); let profiles = Profiles::new();
let service = Service::new(); let service = Service::new();
Core { Core {
clash: Arc::new(Mutex::new(clash)), clash: Arc::new(Mutex::new(clash)),
verge: Arc::new(Mutex::new(verge)), verge: Arc::new(Mutex::new(verge)),
profiles: Arc::new(Mutex::new(profiles)), profiles: Arc::new(Mutex::new(profiles)),
service: Arc::new(Mutex::new(service)), service: Arc::new(Mutex::new(service)),
sysopt: Arc::new(Mutex::new(Sysopt::new())), sysopt: Arc::new(Mutex::new(Sysopt::new())),
timer: Arc::new(Mutex::new(Timer::new())), timer: Arc::new(Mutex::new(Timer::new())),
window: Arc::new(Mutex::new(None)), window: Arc::new(Mutex::new(None)),
} }
}
/// initialize the core state
pub fn init(&self, app_handle: tauri::AppHandle) {
let verge = self.verge.lock();
let clash_core = verge.clash_core.clone();
let mut service = self.service.lock();
service.set_core(clash_core);
#[cfg(windows)]
{
let enable = verge.enable_service_mode.clone();
service.set_mode(enable.unwrap_or(false));
} }
/// initialize the core state log_if_err!(service.start());
pub fn init(&self, app_handle: tauri::AppHandle) { drop(verge);
let verge = self.verge.lock(); drop(service);
let clash_core = verge.clash_core.clone();
let mut service = self.service.lock(); log_if_err!(self.activate());
service.set_core(clash_core);
#[cfg(windows)] let clash = self.clash.lock();
{ let verge = self.verge.lock();
let enable = verge.enable_service_mode.clone();
service.set_mode(enable.unwrap_or(false));
}
log_if_err!(service.start()); let silent_start = verge.enable_silent_start.clone();
drop(verge); let auto_launch = verge.enable_auto_launch.clone();
drop(service);
log_if_err!(self.activate()); // silent start
if silent_start.unwrap_or(false) {
let clash = self.clash.lock(); let window = self.window.lock();
let verge = self.verge.lock(); window.as_ref().map(|win| {
win.hide().unwrap();
let silent_start = verge.enable_silent_start.clone(); });
let auto_launch = verge.enable_auto_launch.clone();
// silent start
if silent_start.unwrap_or(false) {
let window = self.window.lock();
window.as_ref().map(|win| {
win.hide().unwrap();
});
}
let mut sysopt = self.sysopt.lock();
sysopt.init_sysproxy(clash.info.port.clone(), &verge);
drop(clash);
drop(verge);
log_if_err!(sysopt.init_launch(auto_launch));
log_if_err!(self.update_systray(&app_handle));
// wait the window setup during resolve app
let core = self.clone();
tauri::async_runtime::spawn(async move {
sleep(Duration::from_secs(2)).await;
log_if_err!(core.activate_enhanced(true));
});
// timer initialize
let mut timer = self.timer.lock();
timer.set_core(self.clone());
log_if_err!(timer.refresh());
} }
/// save the window instance let mut sysopt = self.sysopt.lock();
pub fn set_win(&self, win: Option<Window>) {
let mut window = self.window.lock(); sysopt.init_sysproxy(clash.info.port.clone(), &verge);
*window = win;
drop(clash);
drop(verge);
log_if_err!(sysopt.init_launch(auto_launch));
log_if_err!(self.update_systray(&app_handle));
// wait the window setup during resolve app
let core = self.clone();
tauri::async_runtime::spawn(async move {
sleep(Duration::from_secs(2)).await;
log_if_err!(core.activate_enhanced(true));
});
// timer initialize
let mut timer = self.timer.lock();
timer.set_core(self.clone());
log_if_err!(timer.refresh());
}
/// save the window instance
pub fn set_win(&self, win: Option<Window>) {
let mut window = self.window.lock();
*window = win;
}
/// restart the clash sidecar
pub fn restart_clash(&self) -> Result<()> {
let mut service = self.service.lock();
service.restart()?;
drop(service);
self.activate()?;
self.activate_enhanced(true)
}
/// change the clash core
pub fn change_core(&self, clash_core: Option<String>) -> Result<()> {
let clash_core = clash_core.unwrap_or("clash".into());
if &clash_core != "clash" && &clash_core != "clash-meta" {
bail!("invalid clash core name \"{clash_core}\"");
} }
/// restart the clash sidecar let mut verge = self.verge.lock();
pub fn restart_clash(&self) -> Result<()> { verge.patch_config(Verge {
let mut service = self.service.lock(); clash_core: Some(clash_core.clone()),
service.restart()?; ..Verge::default()
drop(service); })?;
drop(verge);
self.activate()?; let mut service = self.service.lock();
self.activate_enhanced(true) service.stop()?;
service.set_core(Some(clash_core));
service.start()?;
drop(service);
self.activate()?;
self.activate_enhanced(true)
}
/// Patch Clash
/// handle the clash config changed
pub fn patch_clash(&self, patch: Mapping, app_handle: &AppHandle) -> Result<()> {
let ((changed_port, changed_mode), port) = {
let mut clash = self.clash.lock();
(clash.patch_config(patch)?, clash.info.port.clone())
};
// todo: port check
if changed_port {
let mut service = self.service.lock();
service.restart()?;
drop(service);
self.activate()?;
self.activate_enhanced(true)?;
let mut sysopt = self.sysopt.lock();
let verge = self.verge.lock();
sysopt.init_sysproxy(port, &verge);
} }
/// change the clash core if changed_mode {
pub fn change_core(&self, clash_core: Option<String>) -> Result<()> { self.update_systray(app_handle)?;
let clash_core = clash_core.unwrap_or("clash".into()); }
if &clash_core != "clash" && &clash_core != "clash-meta" { Ok(())
bail!("invalid clash core name \"{clash_core}\""); }
}
let mut verge = self.verge.lock(); /// Patch Verge
verge.patch_config(Verge { pub fn patch_verge(&self, patch: Verge, app_handle: &AppHandle) -> Result<()> {
clash_core: Some(clash_core.clone()), let tun_mode = patch.enable_tun_mode.clone();
..Verge::default() let auto_launch = patch.enable_auto_launch.clone();
})?; let system_proxy = patch.enable_system_proxy.clone();
drop(verge); let proxy_bypass = patch.system_proxy_bypass.clone();
let proxy_guard = patch.enable_proxy_guard.clone();
#[cfg(windows)]
{
let service_mode = patch.enable_service_mode.clone();
if service_mode.is_some() {
let service_mode = service_mode.unwrap();
let mut service = self.service.lock(); let mut service = self.service.lock();
service.stop()?; service.stop()?;
service.set_core(Some(clash_core)); service.set_mode(service_mode);
service.start()?; service.start()?;
drop(service); drop(service);
self.activate()?; self.activate_enhanced(false)?;
self.activate_enhanced(true) }
} }
/// Patch Clash if auto_launch.is_some() {
/// handle the clash config changed let mut sysopt = self.sysopt.lock();
pub fn patch_clash(&self, patch: Mapping, app_handle: &AppHandle) -> Result<()> { sysopt.update_launch(auto_launch)?;
let ((changed_port, changed_mode), port) = {
let mut clash = self.clash.lock();
(clash.patch_config(patch)?, clash.info.port.clone())
};
// todo: port check
if changed_port {
let mut service = self.service.lock();
service.restart()?;
drop(service);
self.activate()?;
self.activate_enhanced(true)?;
let mut sysopt = self.sysopt.lock();
let verge = self.verge.lock();
sysopt.init_sysproxy(port, &verge);
}
if changed_mode {
self.update_systray(app_handle)?;
}
Ok(())
} }
/// Patch Verge if system_proxy.is_some() || proxy_bypass.is_some() {
pub fn patch_verge(&self, patch: Verge, app_handle: &AppHandle) -> Result<()> { let mut sysopt = self.sysopt.lock();
let tun_mode = patch.enable_tun_mode.clone(); sysopt.update_sysproxy(system_proxy.clone(), proxy_bypass)?;
let auto_launch = patch.enable_auto_launch.clone(); sysopt.guard_proxy();
let system_proxy = patch.enable_system_proxy.clone();
let proxy_bypass = patch.system_proxy_bypass.clone();
let proxy_guard = patch.enable_proxy_guard.clone();
#[cfg(windows)]
{
let service_mode = patch.enable_service_mode.clone();
if service_mode.is_some() {
let service_mode = service_mode.unwrap();
let mut service = self.service.lock();
service.stop()?;
service.set_mode(service_mode);
service.start()?;
drop(service);
self.activate_enhanced(false)?;
}
}
if auto_launch.is_some() {
let mut sysopt = self.sysopt.lock();
sysopt.update_launch(auto_launch)?;
}
if system_proxy.is_some() || proxy_bypass.is_some() {
let mut sysopt = self.sysopt.lock();
sysopt.update_sysproxy(system_proxy.clone(), proxy_bypass)?;
sysopt.guard_proxy();
}
if proxy_guard.unwrap_or(false) {
let sysopt = self.sysopt.lock();
sysopt.guard_proxy();
}
#[cfg(target_os = "windows")]
if tun_mode.is_some() && *tun_mode.as_ref().unwrap_or(&false) {
let wintun_dll = dirs::app_home_dir().join("wintun.dll");
if !wintun_dll.exists() {
bail!("failed to enable TUN for missing `wintun.dll`");
}
}
// save the patch
let mut verge = self.verge.lock();
verge.patch_config(patch)?;
drop(verge);
if system_proxy.is_some() || tun_mode.is_some() {
self.update_systray(app_handle)?;
}
if tun_mode.is_some() {
self.activate_enhanced(false)?;
}
Ok(())
} }
/// update the system tray state if proxy_guard.unwrap_or(false) {
pub fn update_systray(&self, app_handle: &AppHandle) -> Result<()> { let sysopt = self.sysopt.lock();
let clash = self.clash.lock(); sysopt.guard_proxy();
let info = clash.info.clone();
let mode = info.mode.as_ref();
let verge = self.verge.lock();
let tray = app_handle.tray_handle();
let system_proxy = verge.enable_system_proxy.as_ref();
let tun_mode = verge.enable_tun_mode.as_ref();
tray.get_item("rule_mode")
.set_selected((*mode.unwrap()).eq("rule"))?;
tray.get_item("global_mode")
.set_selected((*mode.unwrap()).eq("global"))?;
tray.get_item("direct_mode")
.set_selected((*mode.unwrap()).eq("direct"))?;
tray.get_item("script_mode")
.set_selected((*mode.unwrap()).eq("script"))?;
tray.get_item("system_proxy")
.set_selected(*system_proxy.unwrap_or(&false))?;
tray.get_item("tun_mode")
.set_selected(*tun_mode.unwrap_or(&false))?;
// update verge config
let window = app_handle.get_window("main");
let notice = Notice::from(window);
notice.refresh_verge();
Ok(())
} }
/// activate the profile #[cfg(target_os = "windows")]
/// auto activate enhanced profile if tun_mode.is_some() && *tun_mode.as_ref().unwrap_or(&false) {
pub fn activate(&self) -> Result<()> { let wintun_dll = dirs::app_home_dir().join("wintun.dll");
let data = { if !wintun_dll.exists() {
let profiles = self.profiles.lock(); bail!("failed to enable TUN for missing `wintun.dll`");
let data = profiles.gen_activate()?; }
Clash::strict_filter(data)
};
let (mut config, info) = {
let clash = self.clash.lock();
let config = clash.config.clone();
let info = clash.info.clone();
(config, info)
};
for (key, value) in data.into_iter() {
config.insert(key, value);
}
let config = {
let verge = self.verge.lock();
let tun_mode = verge.enable_tun_mode.unwrap_or(false);
Clash::_tun_mode(config, tun_mode)
};
let notice = {
let window = self.window.lock();
Notice::from(window.clone())
};
let service = self.service.lock();
service.set_config(info, config, notice)
} }
/// Enhanced // save the patch
/// enhanced profiles mode let mut verge = self.verge.lock();
pub fn activate_enhanced(&self, skip: bool) -> Result<()> { verge.patch_config(patch)?;
let window = self.window.lock(); drop(verge);
if window.is_none() {
bail!("failed to get the main window");
}
let event_name = help::get_uid("e"); if system_proxy.is_some() || tun_mode.is_some() {
let event_name = format!("enhanced-cb-{event_name}"); self.update_systray(app_handle)?;
// generate the payload
let payload = {
let profiles = self.profiles.lock();
profiles.gen_enhanced(event_name.clone())?
};
// do not run enhanced
if payload.chain.len() == 0 {
if skip {
return Ok(());
}
drop(window);
return self.activate();
}
let tun_mode = {
let verge = self.verge.lock();
verge.enable_tun_mode.unwrap_or(false)
};
let info = {
let clash = self.clash.lock();
clash.info.clone()
};
let notice = Notice::from(window.clone());
let service = self.service.clone();
let window = window.clone().unwrap();
window.once(&event_name, move |event| {
let result = event.payload();
if result.is_none() {
log::warn!("event payload result is none");
return;
}
let result = result.unwrap();
let result: PrfEnhancedResult = serde_json::from_str(result).unwrap();
if let Some(data) = result.data {
let mut config = Clash::read_config();
let filter_data = Clash::loose_filter(data); // loose filter
for (key, value) in filter_data.into_iter() {
config.insert(key, value);
}
let config = Clash::_tun_mode(config, tun_mode);
let service = service.lock();
log_if_err!(service.set_config(info, config, notice));
log::info!("profile enhanced status {}", result.status);
}
result.error.map(|err| log::error!("{err}"));
});
let verge = self.verge.lock();
let silent_start = verge.enable_silent_start.clone();
let closable = unsafe { WINDOW_CLOSABLE };
if silent_start.unwrap_or(false) && closable {
unsafe {
WINDOW_CLOSABLE = false;
}
window.emit("script-handler-close", payload).unwrap();
} else {
window.emit("script-handler", payload).unwrap();
}
Ok(())
} }
// update rule/global/direct/script mode if tun_mode.is_some() {
pub fn update_mode(&self, app_handle: &AppHandle, mode: &str) -> Result<()> { self.activate_enhanced(false)?;
let mut mapping = Mapping::new();
mapping.insert(Value::from("mode"), Value::from(mode));
self.patch_clash(mapping, app_handle);
let (config, info) = {
let clash = self.clash.lock();
let config = clash.config.clone();
let info = clash.info.clone();
(config, info)
};
let notice = {
let window = self.window.lock();
Notice::from(window.clone())
};
let service = self.service.lock();
service.patch_config(info, config, notice);
Ok(())
} }
Ok(())
}
/// update the system tray state
pub fn update_systray(&self, app_handle: &AppHandle) -> Result<()> {
let clash = self.clash.lock();
let info = clash.info.clone();
let mode = info.mode.as_ref();
let verge = self.verge.lock();
let tray = app_handle.tray_handle();
let system_proxy = verge.enable_system_proxy.as_ref();
let tun_mode = verge.enable_tun_mode.as_ref();
tray
.get_item("rule_mode")
.set_selected((*mode.unwrap()).eq("rule"))?;
tray
.get_item("global_mode")
.set_selected((*mode.unwrap()).eq("global"))?;
tray
.get_item("direct_mode")
.set_selected((*mode.unwrap()).eq("direct"))?;
tray
.get_item("script_mode")
.set_selected((*mode.unwrap()).eq("script"))?;
tray
.get_item("system_proxy")
.set_selected(*system_proxy.unwrap_or(&false))?;
tray
.get_item("tun_mode")
.set_selected(*tun_mode.unwrap_or(&false))?;
// update verge config
let window = app_handle.get_window("main");
let notice = Notice::from(window);
notice.refresh_verge();
Ok(())
}
/// activate the profile
/// auto activate enhanced profile
pub fn activate(&self) -> Result<()> {
let data = {
let profiles = self.profiles.lock();
let data = profiles.gen_activate()?;
Clash::strict_filter(data)
};
let (mut config, info) = {
let clash = self.clash.lock();
let config = clash.config.clone();
let info = clash.info.clone();
(config, info)
};
for (key, value) in data.into_iter() {
config.insert(key, value);
}
let config = {
let verge = self.verge.lock();
let tun_mode = verge.enable_tun_mode.unwrap_or(false);
Clash::_tun_mode(config, tun_mode)
};
let notice = {
let window = self.window.lock();
Notice::from(window.clone())
};
let service = self.service.lock();
service.set_config(info, config, notice)
}
/// Enhanced
/// enhanced profiles mode
pub fn activate_enhanced(&self, skip: bool) -> Result<()> {
let window = self.window.lock();
if window.is_none() {
bail!("failed to get the main window");
}
let event_name = help::get_uid("e");
let event_name = format!("enhanced-cb-{event_name}");
// generate the payload
let payload = {
let profiles = self.profiles.lock();
profiles.gen_enhanced(event_name.clone())?
};
// do not run enhanced
if payload.chain.len() == 0 {
if skip {
return Ok(());
}
drop(window);
return self.activate();
}
let tun_mode = {
let verge = self.verge.lock();
verge.enable_tun_mode.unwrap_or(false)
};
let info = {
let clash = self.clash.lock();
clash.info.clone()
};
let notice = Notice::from(window.clone());
let service = self.service.clone();
let window = window.clone().unwrap();
window.once(&event_name, move |event| {
let result = event.payload();
if result.is_none() {
log::warn!("event payload result is none");
return;
}
let result = result.unwrap();
let result: PrfEnhancedResult = serde_json::from_str(result).unwrap();
if let Some(data) = result.data {
let mut config = Clash::read_config();
let filter_data = Clash::loose_filter(data); // loose filter
for (key, value) in filter_data.into_iter() {
config.insert(key, value);
}
let config = Clash::_tun_mode(config, tun_mode);
let service = service.lock();
log_if_err!(service.set_config(info, config, notice));
log::info!("profile enhanced status {}", result.status);
}
result.error.map(|err| log::error!("{err}"));
});
let verge = self.verge.lock();
let silent_start = verge.enable_silent_start.clone();
let closable = unsafe { WINDOW_CLOSABLE };
if silent_start.unwrap_or(false) && closable {
unsafe {
WINDOW_CLOSABLE = false;
}
window.emit("script-handler-close", payload).unwrap();
} else {
window.emit("script-handler", payload).unwrap();
}
Ok(())
}
// update rule/global/direct/script mode
pub fn update_mode(&self, app_handle: &AppHandle, mode: &str) -> Result<()> {
let mut mapping = Mapping::new();
mapping.insert(Value::from("mode"), Value::from(mode));
self.patch_clash(mapping, app_handle);
let (config, info) = {
let clash = self.clash.lock();
let config = clash.config.clone();
let info = clash.info.clone();
(config, info)
};
let notice = {
let window = self.window.lock();
Notice::from(window.clone())
};
let service = self.service.lock();
service.patch_config(info, config, notice);
Ok(())
}
} }
impl Core { impl Core {
/// Static function /// Static function
/// update profile item /// update profile item
pub async fn update_profile_item( pub async fn update_profile_item(
core: Core, core: Core,
uid: String, uid: String,
option: Option<PrfOption>, option: Option<PrfOption>,
) -> Result<()> { ) -> Result<()> {
let (url, opt) = { let (url, opt) = {
let profiles = core.profiles.lock(); let profiles = core.profiles.lock();
let item = profiles.get_item(&uid)?; let item = profiles.get_item(&uid)?;
if let Some(typ) = item.itype.as_ref() { if let Some(typ) = item.itype.as_ref() {
// maybe only valid for `local` profile // maybe only valid for `local` profile
if *typ != "remote" { if *typ != "remote" {
// reactivate the config // reactivate the config
if Some(uid) == profiles.get_current() { if Some(uid) == profiles.get_current() {
drop(profiles);
return core.activate_enhanced(false);
}
return Ok(());
}
}
if item.url.is_none() {
bail!("failed to get the profile item url");
}
(item.url.clone().unwrap(), item.option.clone())
};
let merged_opt = PrfOption::merge(opt, option);
let item = PrfItem::from_url(&url, None, None, merged_opt).await?;
let mut profiles = core.profiles.lock();
profiles.update_item(uid.clone(), item)?;
// reactivate the profile
if Some(uid) == profiles.get_current() {
drop(profiles); drop(profiles);
core.activate_enhanced(false)?; return core.activate_enhanced(false);
} }
Ok(()) return Ok(());
}
}
if item.url.is_none() {
bail!("failed to get the profile item url");
}
(item.url.clone().unwrap(), item.option.clone())
};
let merged_opt = PrfOption::merge(opt, option);
let item = PrfItem::from_url(&url, None, None, merged_opt).await?;
let mut profiles = core.profiles.lock();
profiles.update_item(uid.clone(), item)?;
// reactivate the profile
if Some(uid) == profiles.get_current() {
drop(profiles);
core.activate_enhanced(false)?;
} }
Ok(())
}
} }

View File

@ -3,33 +3,33 @@ use tauri::Window;
#[derive(Debug, Default, Clone)] #[derive(Debug, Default, Clone)]
pub struct Notice { pub struct Notice {
win: Option<Window>, win: Option<Window>,
} }
impl Notice { impl Notice {
pub fn from(win: Option<Window>) -> Notice { pub fn from(win: Option<Window>) -> Notice {
Notice { win } Notice { win }
} }
pub fn set_win(&mut self, win: Option<Window>) { pub fn set_win(&mut self, win: Option<Window>) {
self.win = win; self.win = win;
} }
pub fn refresh_clash(&self) { pub fn refresh_clash(&self) {
if let Some(window) = self.win.as_ref() { if let Some(window) = self.win.as_ref() {
log_if_err!(window.emit("verge://refresh-clash-config", "yes")); log_if_err!(window.emit("verge://refresh-clash-config", "yes"));
}
} }
}
pub fn refresh_verge(&self) { pub fn refresh_verge(&self) {
if let Some(window) = self.win.as_ref() { if let Some(window) = self.win.as_ref() {
log_if_err!(window.emit("verge://refresh-verge-config", "yes")); log_if_err!(window.emit("verge://refresh-verge-config", "yes"));
}
} }
}
pub fn refresh_profiles(&self) { pub fn refresh_profiles(&self) {
if let Some(window) = self.win.as_ref() { if let Some(window) = self.win.as_ref() {
log_if_err!(window.emit("verge://refresh-profiles-config", "yes")); log_if_err!(window.emit("verge://refresh-profiles-config", "yes"));
}
} }
}
} }

View File

@ -14,462 +14,456 @@ static mut CLASH_CORE: &str = "clash";
#[derive(Debug)] #[derive(Debug)]
pub struct Service { pub struct Service {
sidecar: Option<CommandChild>, sidecar: Option<CommandChild>,
#[allow(unused)] #[allow(unused)]
service_mode: bool, service_mode: bool,
} }
impl Service { impl Service {
pub fn new() -> Service { pub fn new() -> Service {
Service { Service {
sidecar: None, sidecar: None,
service_mode: false, service_mode: false,
}
}
pub fn set_core(&mut self, clash_core: Option<String>) {
unsafe {
CLASH_CORE = Box::leak(clash_core.unwrap_or("clash".into()).into_boxed_str());
}
}
#[allow(unused)]
pub fn set_mode(&mut self, enable: bool) {
self.service_mode = enable;
}
#[cfg(not(windows))]
pub fn start(&mut self) -> Result<()> {
self.start_clash_by_sidecar()
}
#[cfg(windows)]
pub fn start(&mut self) -> Result<()> {
if !self.service_mode {
return self.start_clash_by_sidecar();
}
tauri::async_runtime::spawn(async move {
match Self::check_service().await {
Ok(status) => {
// 未启动clash
if status.code != 0 {
if let Err(err) = Self::start_clash_by_service().await {
log::error!("{err}");
}
}
} }
Err(err) => log::error!("{err}"),
}
});
Ok(())
}
#[cfg(not(windows))]
pub fn stop(&mut self) -> Result<()> {
self.stop_clash_by_sidecar()
}
#[cfg(windows)]
pub fn stop(&mut self) -> Result<()> {
if !self.service_mode {
return self.stop_clash_by_sidecar();
} }
pub fn set_core(&mut self, clash_core: Option<String>) { tauri::async_runtime::spawn(async move {
unsafe { if let Err(err) = Self::stop_clash_by_service().await {
CLASH_CORE = Box::leak(clash_core.unwrap_or("clash".into()).into_boxed_str()); log::error!("{err}");
}
});
Ok(())
}
pub fn restart(&mut self) -> Result<()> {
self.stop()?;
self.start()
}
/// start the clash sidecar
fn start_clash_by_sidecar(&mut self) -> Result<()> {
if self.sidecar.is_some() {
bail!("could not run clash sidecar twice");
}
let app_dir = dirs::app_home_dir();
let app_dir = app_dir.as_os_str().to_str().unwrap();
let clash_core = unsafe { CLASH_CORE };
let cmd = Command::new_sidecar(clash_core)?;
let (mut rx, cmd_child) = cmd.args(["-d", app_dir]).spawn()?;
self.sidecar = Some(cmd_child);
// clash log
tauri::async_runtime::spawn(async move {
while let Some(event) = rx.recv().await {
match event {
CommandEvent::Stdout(line) => log::info!("[clash]: {}", line),
CommandEvent::Stderr(err) => log::error!("[clash]: {}", err),
_ => {}
} }
}
});
Ok(())
}
/// stop the clash sidecar
fn stop_clash_by_sidecar(&mut self) -> Result<()> {
if let Some(sidecar) = self.sidecar.take() {
sidecar.kill()?;
}
Ok(())
}
/// update clash config
/// using PUT methods
pub fn set_config(&self, info: ClashInfo, config: Mapping, notice: Notice) -> Result<()> {
if !self.service_mode && self.sidecar.is_none() {
bail!("did not start sidecar");
} }
#[allow(unused)] let temp_path = dirs::profiles_temp_path();
pub fn set_mode(&mut self, enable: bool) { config::save_yaml(temp_path.clone(), &config, Some("# Clash Verge Temp File"))?;
self.service_mode = enable;
if info.server.is_none() {
if info.port.is_none() {
bail!("failed to parse config.yaml file");
} else {
bail!("failed to parse the server");
}
} }
#[cfg(not(windows))] let server = info.server.unwrap();
pub fn start(&mut self) -> Result<()> { let server = format!("http://{server}/configs");
self.start_clash_by_sidecar()
let mut headers = HeaderMap::new();
headers.insert("Content-Type", "application/json".parse().unwrap());
if let Some(secret) = info.secret.as_ref() {
let secret = format!("Bearer {}", secret.clone()).parse().unwrap();
headers.insert("Authorization", secret);
} }
#[cfg(windows)] tauri::async_runtime::spawn(async move {
pub fn start(&mut self) -> Result<()> { let mut data = HashMap::new();
if !self.service_mode { data.insert("path", temp_path.as_os_str().to_str().unwrap());
return self.start_clash_by_sidecar();
}
tauri::async_runtime::spawn(async move { // retry 5 times
match Self::check_service().await { for _ in 0..5 {
Ok(status) => { match reqwest::ClientBuilder::new().no_proxy().build() {
// 未启动clash Ok(client) => {
if status.code != 0 { let builder = client.put(&server).headers(headers.clone()).json(&data);
if let Err(err) = Self::start_clash_by_service().await {
log::error!("{err}"); match builder.send().await {
} Ok(resp) => {
} if resp.status() != 204 {
log::error!("failed to activate clash with status \"{}\"", resp.status());
} }
Err(err) => log::error!("{err}"),
notice.refresh_clash();
// do not retry
break;
}
Err(err) => log::error!("failed to activate for `{err}`"),
} }
}); }
Err(err) => log::error!("failed to activate for `{err}`"),
Ok(())
}
#[cfg(not(windows))]
pub fn stop(&mut self) -> Result<()> {
self.stop_clash_by_sidecar()
}
#[cfg(windows)]
pub fn stop(&mut self) -> Result<()> {
if !self.service_mode {
return self.stop_clash_by_sidecar();
} }
sleep(Duration::from_millis(500)).await;
}
});
tauri::async_runtime::spawn(async move { Ok(())
if let Err(err) = Self::stop_clash_by_service().await { }
log::error!("{err}");
}
});
Ok(()) /// patch clash config
pub fn patch_config(&self, info: ClashInfo, config: Mapping, notice: Notice) -> Result<()> {
if !self.service_mode && self.sidecar.is_none() {
bail!("did not start sidecar");
} }
pub fn restart(&mut self) -> Result<()> { if info.server.is_none() {
self.stop()?; if info.port.is_none() {
self.start() bail!("failed to parse config.yaml file");
} else {
bail!("failed to parse the server");
}
} }
/// start the clash sidecar let server = info.server.unwrap();
fn start_clash_by_sidecar(&mut self) -> Result<()> { let server = format!("http://{server}/configs");
if self.sidecar.is_some() {
bail!("could not run clash sidecar twice");
}
let app_dir = dirs::app_home_dir(); let mut headers = HeaderMap::new();
let app_dir = app_dir.as_os_str().to_str().unwrap(); headers.insert("Content-Type", "application/json".parse().unwrap());
let clash_core = unsafe { CLASH_CORE }; if let Some(secret) = info.secret.as_ref() {
let cmd = Command::new_sidecar(clash_core)?; let secret = format!("Bearer {}", secret.clone()).parse().unwrap();
let (mut rx, cmd_child) = cmd.args(["-d", app_dir]).spawn()?; headers.insert("Authorization", secret);
}
self.sidecar = Some(cmd_child); tauri::async_runtime::spawn(async move {
// retry 5 times
for _ in 0..5 {
match reqwest::ClientBuilder::new().no_proxy().build() {
Ok(client) => {
let builder = client.patch(&server).headers(headers.clone()).json(&config);
// clash log match builder.send().await {
tauri::async_runtime::spawn(async move { Ok(resp) => {
while let Some(event) = rx.recv().await { if resp.status() != 204 {
match event { log::error!("failed to activate clash with status \"{}\"", resp.status());
CommandEvent::Stdout(line) => log::info!("[clash]: {}", line),
CommandEvent::Stderr(err) => log::error!("[clash]: {}", err),
_ => {}
} }
notice.refresh_clash();
// do not retry
break;
}
Err(err) => log::error!("failed to activate for `{err}`"),
} }
}); }
Err(err) => log::error!("failed to activate for `{err}`"),
Ok(())
}
/// stop the clash sidecar
fn stop_clash_by_sidecar(&mut self) -> Result<()> {
if let Some(sidecar) = self.sidecar.take() {
sidecar.kill()?;
} }
Ok(()) sleep(Duration::from_millis(500)).await;
} }
});
/// update clash config Ok(())
/// using PUT methods }
pub fn set_config(&self, info: ClashInfo, config: Mapping, notice: Notice) -> Result<()> {
if !self.service_mode && self.sidecar.is_none() {
bail!("did not start sidecar");
}
let temp_path = dirs::profiles_temp_path();
config::save_yaml(temp_path.clone(), &config, Some("# Clash Verge Temp File"))?;
if info.server.is_none() {
if info.port.is_none() {
bail!("failed to parse config.yaml file");
} else {
bail!("failed to parse the server");
}
}
let server = info.server.unwrap();
let server = format!("http://{server}/configs");
let mut headers = HeaderMap::new();
headers.insert("Content-Type", "application/json".parse().unwrap());
if let Some(secret) = info.secret.as_ref() {
let secret = format!("Bearer {}", secret.clone()).parse().unwrap();
headers.insert("Authorization", secret);
}
tauri::async_runtime::spawn(async move {
let mut data = HashMap::new();
data.insert("path", temp_path.as_os_str().to_str().unwrap());
// retry 5 times
for _ in 0..5 {
match reqwest::ClientBuilder::new().no_proxy().build() {
Ok(client) => {
let builder = client.put(&server).headers(headers.clone()).json(&data);
match builder.send().await {
Ok(resp) => {
if resp.status() != 204 {
log::error!(
"failed to activate clash with status \"{}\"",
resp.status()
);
}
notice.refresh_clash();
// do not retry
break;
}
Err(err) => log::error!("failed to activate for `{err}`"),
}
}
Err(err) => log::error!("failed to activate for `{err}`"),
}
sleep(Duration::from_millis(500)).await;
}
});
Ok(())
}
/// patch clash config
pub fn patch_config(&self, info: ClashInfo, config: Mapping, notice: Notice) -> Result<()> {
if !self.service_mode && self.sidecar.is_none() {
bail!("did not start sidecar");
}
if info.server.is_none() {
if info.port.is_none() {
bail!("failed to parse config.yaml file");
} else {
bail!("failed to parse the server");
}
}
let server = info.server.unwrap();
let server = format!("http://{server}/configs");
let mut headers = HeaderMap::new();
headers.insert("Content-Type", "application/json".parse().unwrap());
if let Some(secret) = info.secret.as_ref() {
let secret = format!("Bearer {}", secret.clone()).parse().unwrap();
headers.insert("Authorization", secret);
}
tauri::async_runtime::spawn(async move {
// retry 5 times
for _ in 0..5 {
match reqwest::ClientBuilder::new().no_proxy().build() {
Ok(client) => {
let builder = client.patch(&server).headers(headers.clone()).json(&config);
match builder.send().await {
Ok(resp) => {
if resp.status() != 204 {
log::error!(
"failed to activate clash with status \"{}\"",
resp.status()
);
}
notice.refresh_clash();
// do not retry
break;
}
Err(err) => log::error!("failed to activate for `{err}`"),
}
}
Err(err) => log::error!("failed to activate for `{err}`"),
}
sleep(Duration::from_millis(500)).await;
}
});
Ok(())
}
} }
impl Drop for Service { impl Drop for Service {
fn drop(&mut self) { fn drop(&mut self) {
log_if_err!(self.stop()); log_if_err!(self.stop());
} }
} }
/// ### Service Mode /// ### Service Mode
/// ///
#[cfg(windows)] #[cfg(windows)]
pub mod win_service { pub mod win_service {
use super::*; use super::*;
use anyhow::Context; use anyhow::Context;
use deelevate::{PrivilegeLevel, Token}; use deelevate::{PrivilegeLevel, Token};
use runas::Command as RunasCommand; use runas::Command as RunasCommand;
use std::os::windows::process::CommandExt; use std::os::windows::process::CommandExt;
use std::{env::current_exe, process::Command as StdCommand}; use std::{env::current_exe, process::Command as StdCommand};
const SERVICE_NAME: &str = "clash_verge_service"; const SERVICE_NAME: &str = "clash_verge_service";
const SERVICE_URL: &str = "http://127.0.0.1:33211"; const SERVICE_URL: &str = "http://127.0.0.1:33211";
#[derive(Debug, Deserialize, Serialize, Clone)] #[derive(Debug, Deserialize, Serialize, Clone)]
pub struct ResponseBody { pub struct ResponseBody {
pub bin_path: String, pub bin_path: String,
pub config_dir: String, pub config_dir: String,
pub log_file: String, pub log_file: String,
}
#[derive(Debug, Deserialize, Serialize, Clone)]
pub struct JsonResponse {
pub code: u64,
pub msg: String,
pub data: Option<ResponseBody>,
}
impl Service {
/// Install the Clash Verge Service
/// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程
pub async fn install_service() -> Result<()> {
let binary_path = dirs::service_path();
let install_path = binary_path.with_file_name("install-service.exe");
if !install_path.exists() {
bail!("installer exe not found");
}
let token = Token::with_current_process()?;
let level = token.privilege_level()?;
let status = match level {
PrivilegeLevel::NotPrivileged => RunasCommand::new(install_path).status()?,
_ => StdCommand::new(install_path)
.creation_flags(0x08000000)
.status()?,
};
if !status.success() {
bail!(
"failed to install service with status {}",
status.code().unwrap()
);
}
Ok(())
} }
#[derive(Debug, Deserialize, Serialize, Clone)] /// Uninstall the Clash Verge Service
pub struct JsonResponse { /// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程
pub code: u64, pub async fn uninstall_service() -> Result<()> {
pub msg: String, let binary_path = dirs::service_path();
pub data: Option<ResponseBody>, let uninstall_path = binary_path.with_file_name("uninstall-service.exe");
if !uninstall_path.exists() {
bail!("uninstaller exe not found");
}
let token = Token::with_current_process()?;
let level = token.privilege_level()?;
let status = match level {
PrivilegeLevel::NotPrivileged => RunasCommand::new(uninstall_path).status()?,
_ => StdCommand::new(uninstall_path)
.creation_flags(0x08000000)
.status()?,
};
if !status.success() {
bail!(
"failed to uninstall service with status {}",
status.code().unwrap()
);
}
Ok(())
} }
impl Service { /// [deprecated]
/// Install the Clash Verge Service /// start service
/// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程 /// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程
pub async fn install_service() -> Result<()> { pub async fn start_service() -> Result<()> {
let binary_path = dirs::service_path(); let token = Token::with_current_process()?;
let install_path = binary_path.with_file_name("install-service.exe"); let level = token.privilege_level()?;
if !install_path.exists() { let args = ["start", SERVICE_NAME];
bail!("installer exe not found");
}
let token = Token::with_current_process()?; let status = match level {
let level = token.privilege_level()?; PrivilegeLevel::NotPrivileged => RunasCommand::new("sc").args(&args).status()?,
_ => StdCommand::new("sc").args(&args).status()?,
};
let status = match level { match status.success() {
PrivilegeLevel::NotPrivileged => RunasCommand::new(install_path).status()?, true => Ok(()),
_ => StdCommand::new(install_path) false => bail!(
.creation_flags(0x08000000) "failed to start service with status {}",
.status()?, status.code().unwrap()
}; ),
}
if !status.success() {
bail!(
"failed to install service with status {}",
status.code().unwrap()
);
}
Ok(())
}
/// Uninstall the Clash Verge Service
/// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程
pub async fn uninstall_service() -> Result<()> {
let binary_path = dirs::service_path();
let uninstall_path = binary_path.with_file_name("uninstall-service.exe");
if !uninstall_path.exists() {
bail!("uninstaller exe not found");
}
let token = Token::with_current_process()?;
let level = token.privilege_level()?;
let status = match level {
PrivilegeLevel::NotPrivileged => RunasCommand::new(uninstall_path).status()?,
_ => StdCommand::new(uninstall_path)
.creation_flags(0x08000000)
.status()?,
};
if !status.success() {
bail!(
"failed to uninstall service with status {}",
status.code().unwrap()
);
}
Ok(())
}
/// [deprecated]
/// start service
/// 该函数应该在协程或者线程中执行避免UAC弹窗阻塞主线程
pub async fn start_service() -> Result<()> {
let token = Token::with_current_process()?;
let level = token.privilege_level()?;
let args = ["start", SERVICE_NAME];
let status = match level {
PrivilegeLevel::NotPrivileged => RunasCommand::new("sc").args(&args).status()?,
_ => StdCommand::new("sc").args(&args).status()?,
};
match status.success() {
true => Ok(()),
false => bail!(
"failed to start service with status {}",
status.code().unwrap()
),
}
}
/// stop service
pub async fn stop_service() -> Result<()> {
let url = format!("{SERVICE_URL}/stop_service");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
/// check the windows service status
pub async fn check_service() -> Result<JsonResponse> {
let url = format!("{SERVICE_URL}/get_clash");
let response = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.get(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
Ok(response)
}
/// start the clash by service
pub(super) async fn start_clash_by_service() -> Result<()> {
let status = Self::check_service().await?;
if status.code == 0 {
Self::stop_clash_by_service().await?;
sleep(Duration::from_secs(1)).await;
}
let clash_core = unsafe { CLASH_CORE };
let clash_bin = format!("{clash_core}.exe");
let bin_path = current_exe().unwrap().with_file_name(clash_bin);
let bin_path = bin_path.as_os_str().to_str().unwrap();
let config_dir = dirs::app_home_dir();
let config_dir = config_dir.as_os_str().to_str().unwrap();
let log_path = dirs::service_log_file();
let log_path = log_path.as_os_str().to_str().unwrap();
let mut map = HashMap::new();
map.insert("bin_path", bin_path);
map.insert("config_dir", config_dir);
map.insert("log_file", log_path);
let url = format!("{SERVICE_URL}/start_clash");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.json(&map)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
/// stop the clash by service
pub(super) async fn stop_clash_by_service() -> Result<()> {
let url = format!("{SERVICE_URL}/stop_clash");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
} }
/// stop service
pub async fn stop_service() -> Result<()> {
let url = format!("{SERVICE_URL}/stop_service");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
/// check the windows service status
pub async fn check_service() -> Result<JsonResponse> {
let url = format!("{SERVICE_URL}/get_clash");
let response = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.get(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
Ok(response)
}
/// start the clash by service
pub(super) async fn start_clash_by_service() -> Result<()> {
let status = Self::check_service().await?;
if status.code == 0 {
Self::stop_clash_by_service().await?;
sleep(Duration::from_secs(1)).await;
}
let clash_core = unsafe { CLASH_CORE };
let clash_bin = format!("{clash_core}.exe");
let bin_path = current_exe().unwrap().with_file_name(clash_bin);
let bin_path = bin_path.as_os_str().to_str().unwrap();
let config_dir = dirs::app_home_dir();
let config_dir = config_dir.as_os_str().to_str().unwrap();
let log_path = dirs::service_log_file();
let log_path = log_path.as_os_str().to_str().unwrap();
let mut map = HashMap::new();
map.insert("bin_path", bin_path);
map.insert("config_dir", config_dir);
map.insert("log_file", log_path);
let url = format!("{SERVICE_URL}/start_clash");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.json(&map)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
/// stop the clash by service
pub(super) async fn stop_clash_by_service() -> Result<()> {
let url = format!("{SERVICE_URL}/stop_clash");
let res = reqwest::ClientBuilder::new()
.no_proxy()
.build()?
.post(url)
.send()
.await?
.json::<JsonResponse>()
.await
.context("failed to connect to the Clash Verge Service")?;
if res.code != 0 {
bail!(res.msg);
}
Ok(())
}
}
} }

View File

@ -1,6 +1,6 @@
#![cfg_attr( #![cfg_attr(
all(not(debug_assertions), target_os = "windows"), all(not(debug_assertions), target_os = "windows"),
windows_subsystem = "windows" windows_subsystem = "windows"
)] )]
mod cmds; mod cmds;
@ -8,177 +8,177 @@ mod core;
mod utils; mod utils;
use crate::{ use crate::{
core::Verge, core::Verge,
utils::{resolve, server}, utils::{resolve, server},
}; };
use serde_yaml::{Mapping, Value}; use serde_yaml::{Mapping, Value};
use tauri::{ use tauri::{
api, CustomMenuItem, Manager, SystemTray, SystemTrayEvent, SystemTrayMenu, SystemTrayMenuItem, api, CustomMenuItem, Manager, SystemTray, SystemTrayEvent, SystemTrayMenu, SystemTrayMenuItem,
}; };
fn main() -> std::io::Result<()> { fn main() -> std::io::Result<()> {
if server::check_singleton().is_err() { if server::check_singleton().is_err() {
println!("app exists"); println!("app exists");
return Ok(()); return Ok(());
} }
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
unsafe { unsafe {
use crate::utils::dirs; use crate::utils::dirs;
dirs::init_portable_flag(); dirs::init_portable_flag();
} }
let tray_menu = SystemTrayMenu::new() let tray_menu = SystemTrayMenu::new()
.add_item(CustomMenuItem::new("open_window", "Show")) .add_item(CustomMenuItem::new("open_window", "Show"))
.add_item(CustomMenuItem::new("rule_mode", "Rule Mode")) .add_item(CustomMenuItem::new("rule_mode", "Rule Mode"))
.add_item(CustomMenuItem::new("global_mode", "Global Mode")) .add_item(CustomMenuItem::new("global_mode", "Global Mode"))
.add_item(CustomMenuItem::new("direct_mode", "Direct Mode")) .add_item(CustomMenuItem::new("direct_mode", "Direct Mode"))
.add_item(CustomMenuItem::new("script_mode", "Script Mode")) .add_item(CustomMenuItem::new("script_mode", "Script Mode"))
.add_item(CustomMenuItem::new("system_proxy", "System Proxy")) .add_item(CustomMenuItem::new("system_proxy", "System Proxy"))
.add_item(CustomMenuItem::new("tun_mode", "Tun Mode")) .add_item(CustomMenuItem::new("tun_mode", "Tun Mode"))
.add_item(CustomMenuItem::new("restart_clash", "Restart Clash")) .add_item(CustomMenuItem::new("restart_clash", "Restart Clash"))
.add_native_item(SystemTrayMenuItem::Separator) .add_native_item(SystemTrayMenuItem::Separator)
.add_item(CustomMenuItem::new("quit", "Quit").accelerator("CmdOrControl+Q")); .add_item(CustomMenuItem::new("quit", "Quit").accelerator("CmdOrControl+Q"));
#[allow(unused_mut)] #[allow(unused_mut)]
let mut builder = tauri::Builder::default() let mut builder = tauri::Builder::default()
.setup(|app| Ok(resolve::resolve_setup(app))) .setup(|app| Ok(resolve::resolve_setup(app)))
.system_tray(SystemTray::new().with_menu(tray_menu)) .system_tray(SystemTray::new().with_menu(tray_menu))
.on_system_tray_event(move |app_handle, event| match event { .on_system_tray_event(move |app_handle, event| match event {
SystemTrayEvent::MenuItemClick { id, .. } => match id.as_str() { SystemTrayEvent::MenuItemClick { id, .. } => match id.as_str() {
"open_window" => { "open_window" => {
resolve::create_window(app_handle); resolve::create_window(app_handle);
} }
"rule_mode" => { "rule_mode" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
crate::log_if_err!(core.update_mode(app_handle, "rule")); crate::log_if_err!(core.update_mode(app_handle, "rule"));
} }
"global_mode" => { "global_mode" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
crate::log_if_err!(core.update_mode(app_handle, "global")); crate::log_if_err!(core.update_mode(app_handle, "global"));
} }
"direct_mode" => { "direct_mode" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
crate::log_if_err!(core.update_mode(app_handle, "direct")); crate::log_if_err!(core.update_mode(app_handle, "direct"));
} }
"script_mode" => { "script_mode" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
crate::log_if_err!(core.update_mode(app_handle, "script")); crate::log_if_err!(core.update_mode(app_handle, "script"));
} }
"system_proxy" => { "system_proxy" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
let new_value = { let new_value = {
let verge = core.verge.lock(); let verge = core.verge.lock();
!verge.enable_system_proxy.clone().unwrap_or(false) !verge.enable_system_proxy.clone().unwrap_or(false)
}; };
let patch = Verge { let patch = Verge {
enable_system_proxy: Some(new_value), enable_system_proxy: Some(new_value),
..Verge::default() ..Verge::default()
}; };
crate::log_if_err!(core.patch_verge(patch, app_handle)); crate::log_if_err!(core.patch_verge(patch, app_handle));
} }
"tun_mode" => { "tun_mode" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
let new_value = { let new_value = {
let verge = core.verge.lock(); let verge = core.verge.lock();
!verge.enable_tun_mode.clone().unwrap_or(false) !verge.enable_tun_mode.clone().unwrap_or(false)
}; };
let patch = Verge { let patch = Verge {
enable_tun_mode: Some(new_value), enable_tun_mode: Some(new_value),
..Verge::default() ..Verge::default()
}; };
crate::log_if_err!(core.patch_verge(patch, app_handle)); crate::log_if_err!(core.patch_verge(patch, app_handle));
} }
"restart_clash" => { "restart_clash" => {
let core = app_handle.state::<core::Core>(); let core = app_handle.state::<core::Core>();
crate::log_if_err!(core.restart_clash()); crate::log_if_err!(core.restart_clash());
} }
"quit" => { "quit" => {
resolve::resolve_reset(app_handle); resolve::resolve_reset(app_handle);
app_handle.exit(0); app_handle.exit(0);
} }
_ => {} _ => {}
}, },
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
SystemTrayEvent::LeftClick { .. } => { SystemTrayEvent::LeftClick { .. } => {
resolve::create_window(app_handle); resolve::create_window(app_handle);
} }
_ => {} _ => {}
}) })
.invoke_handler(tauri::generate_handler![ .invoke_handler(tauri::generate_handler![
// common // common
cmds::get_sys_proxy, cmds::get_sys_proxy,
cmds::get_cur_proxy, cmds::get_cur_proxy,
cmds::open_app_dir, cmds::open_app_dir,
cmds::open_logs_dir, cmds::open_logs_dir,
cmds::kill_sidecar, cmds::kill_sidecar,
cmds::restart_sidecar, cmds::restart_sidecar,
// clash // clash
cmds::get_clash_info, cmds::get_clash_info,
cmds::patch_clash_config, cmds::patch_clash_config,
cmds::change_clash_core, cmds::change_clash_core,
// verge // verge
cmds::get_verge_config, cmds::get_verge_config,
cmds::patch_verge_config, cmds::patch_verge_config,
// profile // profile
cmds::view_profile, cmds::view_profile,
cmds::patch_profile, cmds::patch_profile,
cmds::create_profile, cmds::create_profile,
cmds::import_profile, cmds::import_profile,
cmds::update_profile, cmds::update_profile,
cmds::delete_profile, cmds::delete_profile,
cmds::select_profile, cmds::select_profile,
cmds::get_profiles, cmds::get_profiles,
cmds::enhance_profiles, cmds::enhance_profiles,
cmds::change_profile_chain, cmds::change_profile_chain,
cmds::change_profile_valid, cmds::change_profile_valid,
cmds::read_profile_file, cmds::read_profile_file,
cmds::save_profile_file, cmds::save_profile_file,
// service mode // service mode
cmds::service::start_service, cmds::service::start_service,
cmds::service::stop_service, cmds::service::stop_service,
cmds::service::check_service, cmds::service::check_service,
cmds::service::install_service, cmds::service::install_service,
cmds::service::uninstall_service, cmds::service::uninstall_service,
]); ]);
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
{ {
use tauri::{Menu, MenuItem, Submenu}; use tauri::{Menu, MenuItem, Submenu};
let submenu_file = Submenu::new( let submenu_file = Submenu::new(
"File", "File",
Menu::new() Menu::new()
.add_native_item(MenuItem::Undo) .add_native_item(MenuItem::Undo)
.add_native_item(MenuItem::Redo) .add_native_item(MenuItem::Redo)
.add_native_item(MenuItem::Copy) .add_native_item(MenuItem::Copy)
.add_native_item(MenuItem::Paste) .add_native_item(MenuItem::Paste)
.add_native_item(MenuItem::Cut) .add_native_item(MenuItem::Cut)
.add_native_item(MenuItem::SelectAll), .add_native_item(MenuItem::SelectAll),
); );
builder = builder.menu(Menu::new().add_submenu(submenu_file)); builder = builder.menu(Menu::new().add_submenu(submenu_file));
} }
builder builder
.build(tauri::generate_context!()) .build(tauri::generate_context!())
.expect("error while running tauri application") .expect("error while running tauri application")
.run(|app_handle, e| match e { .run(|app_handle, e| match e {
tauri::RunEvent::ExitRequested { api, .. } => { tauri::RunEvent::ExitRequested { api, .. } => {
api.prevent_exit(); api.prevent_exit();
} }
tauri::RunEvent::Exit => { tauri::RunEvent::Exit => {
resolve::resolve_reset(app_handle); resolve::resolve_reset(app_handle);
api::process::kill_children(); api::process::kill_children();
} }
_ => {} _ => {}
}); });
Ok(()) Ok(())
} }