use crate::{ core::{ClashInfo, Core, PrfItem, PrfOption, Profiles, Verge}, utils::{dirs, help, sysopt::SysProxyConfig}, }; use crate::{log_if_err, ret_err, wrap_err}; use anyhow::Result; use serde_yaml::Mapping; use tauri::{api, State}; type CmdResult = Result; /// get all profiles from `profiles.yaml` #[tauri::command] pub fn get_profiles(core: State<'_, Core>) -> CmdResult { let profiles = core.profiles.lock(); Ok(profiles.clone()) } /// manually exec enhanced profile #[tauri::command] pub fn enhance_profiles(core: State<'_, Core>) -> CmdResult { wrap_err!(core.activate_enhanced(false)) } /// import the profile from url /// and save to `profiles.yaml` #[tauri::command] pub async fn import_profile( url: String, option: Option, core: State<'_, Core>, ) -> CmdResult { let item = wrap_err!(PrfItem::from_url(&url, None, None, option).await)?; let mut profiles = core.profiles.lock(); wrap_err!(profiles.append_item(item)) } /// new a profile /// append a temp profile item file to the `profiles` dir /// view the temp profile file by using vscode or other editor #[tauri::command] pub async fn create_profile( item: PrfItem, // partial file_data: Option, core: State<'_, Core>, ) -> CmdResult { let item = wrap_err!(PrfItem::from(item, file_data).await)?; let mut profiles = core.profiles.lock(); wrap_err!(profiles.append_item(item)) } /// Update the profile #[tauri::command] pub async fn update_profile( index: String, option: Option, core: State<'_, Core>, ) -> CmdResult { let (url, opt) = { // must release the lock here let profiles = core.profiles.lock(); let item = wrap_err!(profiles.get_item(&index))?; // check the profile type if let Some(typ) = item.itype.as_ref() { if *typ != "remote" { ret_err!(format!("could not update the `{typ}` profile")); } } if item.url.is_none() { ret_err!("failed to get the item url"); } (item.url.clone().unwrap(), item.option.clone()) }; let fetch_opt = PrfOption::merge(opt, option); let item = wrap_err!(PrfItem::from_url(&url, None, None, fetch_opt).await)?; let mut profiles = core.profiles.lock(); wrap_err!(profiles.update_item(index.clone(), item))?; // reactivate the profile if Some(index) == profiles.get_current() { drop(profiles); log_if_err!(core.activate_enhanced(false)); } Ok(()) } /// change the current profile #[tauri::command] pub fn select_profile(index: String, core: State<'_, Core>) -> CmdResult { let mut profiles = core.profiles.lock(); wrap_err!(profiles.put_current(index))?; drop(profiles); wrap_err!(core.activate_enhanced(false)) } /// change the profile chain #[tauri::command] pub fn change_profile_chain(chain: Option>, core: State<'_, Core>) -> CmdResult { let mut profiles = core.profiles.lock(); profiles.put_chain(chain); drop(profiles); wrap_err!(core.activate_enhanced(false)) } /// change the profile valid fields #[tauri::command] pub fn change_profile_valid(valid: Option>, core: State) -> CmdResult { let mut profiles = core.profiles.lock(); profiles.put_valid(valid); drop(profiles); wrap_err!(core.activate_enhanced(false)) } /// delete profile item #[tauri::command] pub fn delete_profile(index: String, core: State<'_, Core>) -> CmdResult { let mut profiles = core.profiles.lock(); if wrap_err!(profiles.delete_item(index))? { drop(profiles); log_if_err!(core.activate_enhanced(false)); } Ok(()) } /// patch the profile config #[tauri::command] pub fn patch_profile(index: String, profile: PrfItem, core: State<'_, Core>) -> CmdResult { let mut profiles = core.profiles.lock(); wrap_err!(profiles.patch_item(index, profile)) } /// run vscode command to edit the profile #[tauri::command] pub fn view_profile(index: String, core: State<'_, Core>) -> CmdResult { let profiles = core.profiles.lock(); let item = wrap_err!(profiles.get_item(&index))?; let file = item.file.clone(); if file.is_none() { ret_err!("the file is null"); } let path = dirs::app_profiles_dir().join(file.unwrap()); if !path.exists() { ret_err!("the file not found"); } wrap_err!(help::open_file(path)) } /// read the profile item file data #[tauri::command] pub fn read_profile_file(index: String, core: State<'_, Core>) -> CmdResult { let profiles = core.profiles.lock(); let item = wrap_err!(profiles.get_item(&index))?; let data = wrap_err!(item.read_file())?; Ok(data) } /// save the profile item file data #[tauri::command] pub fn save_profile_file( index: String, file_data: Option, core: State<'_, Core>, ) -> CmdResult { if file_data.is_none() { return Ok(()); } let profiles = core.profiles.lock(); let item = wrap_err!(profiles.get_item(&index))?; wrap_err!(item.save_file(file_data.unwrap())) } /// get the clash core info from the state /// the caller can also get the infomation by clash's api #[tauri::command] pub fn get_clash_info(core: State<'_, Core>) -> CmdResult { let clash = core.clash.lock(); Ok(clash.info.clone()) } /// update the clash core config /// after putting the change to the clash core /// then we should save the latest config #[tauri::command] pub fn patch_clash_config(payload: Mapping, core: State<'_, Core>) -> CmdResult { wrap_err!(core.patch_clash(payload)) } /// get the verge config #[tauri::command] pub fn get_verge_config(core: State<'_, Core>) -> CmdResult { let verge = core.verge.lock(); let config = verge.clone(); Ok(config) } /// patch the verge config /// this command only save the config and not responsible for other things #[tauri::command] pub fn patch_verge_config( payload: Verge, app_handle: tauri::AppHandle, core: State<'_, Core>, ) -> Result<(), String> { wrap_err!(core.patch_verge(payload, &app_handle)) } /// restart the sidecar #[tauri::command] pub fn restart_sidecar(core: State<'_, Core>) -> CmdResult { wrap_err!(core.restart_clash()) } /// kill all sidecars when update app #[tauri::command] pub fn kill_sidecar() { api::process::kill_children(); } /// get the system proxy #[tauri::command] pub fn get_sys_proxy() -> Result { wrap_err!(SysProxyConfig::get_sys()) } /// get the current proxy config /// which may not the same as system proxy #[tauri::command] pub fn get_cur_proxy(core: State<'_, Core>) -> CmdResult> { let sysopt = core.sysopt.lock(); wrap_err!(sysopt.get_sysproxy()) } /// open app config dir #[tauri::command] pub fn open_app_dir() -> Result<(), String> { let app_dir = dirs::app_home_dir(); wrap_err!(open::that(app_dir)) } /// open logs dir #[tauri::command] pub fn open_logs_dir() -> Result<(), String> { let log_dir = dirs::app_logs_dir(); wrap_err!(open::that(log_dir)) }