Merge pull request #1 from Iam54r1n4/main

Add deep link support and few other things
This commit is contained in:
Hiddify 2023-01-24 10:22:13 +01:00 committed by GitHub
commit cb65eec84d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 576 additions and 146 deletions

View file

@ -1,4 +0,0 @@
#!/bin/sh
. "$(dirname "$0")/_/husky.sh"
yarn pretty-quick --staged

View file

@ -1,12 +1,12 @@
{
"name": "clash-verge",
"name": "HiddifyDesktop",
"version": "1.2.1",
"license": "GPL-3.0",
"scripts": {
"dev": "tauri dev",
"dev:diff": "tauri dev -f verge-dev",
"build": "tauri build",
"tauri": "tauri",
"dev": "cargo tauri dev",
"dev:diff": "cargo tauri dev -f verge-dev",
"build": "cargo tauri build",
"tauri": "cargo tauri",
"web:dev": "vite",
"web:build": "tsc && vite build",
"web:serve": "vite preview",

303
src-tauri/Cargo.lock generated
View file

@ -82,6 +82,16 @@ version = "1.5.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "983cd8b9d4b02a6dc6ffa557262eb5858a27a0038ffffe21a0f133eaa819a164"
[[package]]
name = "async-attributes"
version = "1.1.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a3203e79f4dd9bdda415ed03cf14dae5a2bf775c683a00f94e9cd1faf0f596e5"
dependencies = [
"quote",
"syn",
]
[[package]]
name = "async-channel"
version = "1.8.0"
@ -119,6 +129,22 @@ dependencies = [
"futures-lite",
]
[[package]]
name = "async-global-executor"
version = "2.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f1b6f5d7df27bd294849f8eec66ecfc63d11814df7a4f5d74168a2394467b776"
dependencies = [
"async-channel",
"async-executor",
"async-io",
"async-lock",
"blocking",
"futures-lite",
"once_cell",
"tokio",
]
[[package]]
name = "async-io"
version = "1.12.0"
@ -179,6 +205,33 @@ dependencies = [
"windows-sys 0.42.0",
]
[[package]]
name = "async-std"
version = "1.12.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "62565bb4402e926b29953c785397c6dc0391b7b446e45008b0049eb43cec6f5d"
dependencies = [
"async-attributes",
"async-channel",
"async-global-executor",
"async-io",
"async-lock",
"crossbeam-utils",
"futures-channel",
"futures-core",
"futures-io",
"futures-lite",
"gloo-timers",
"kv-log-macro",
"log 0.4.17",
"memchr",
"once_cell",
"pin-project-lite",
"pin-utils",
"slab",
"wasm-bindgen-futures",
]
[[package]]
name = "async-task"
version = "4.3.0"
@ -226,22 +279,6 @@ version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "065374052e7df7ee4047b1160cca5e1467a12351a40b3da123c870ba0b8eda2a"
[[package]]
name = "attohttpc"
version = "0.22.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1fcf00bc6d5abb29b5f97e3c61a90b6d3caa12f3faf897d4a3e3607c050a35a7"
dependencies = [
"flate2",
"http",
"log 0.4.17",
"native-tls",
"serde",
"serde_json",
"serde_urlencoded",
"url",
]
[[package]]
name = "auto-launch"
version = "0.4.0"
@ -370,6 +407,12 @@ version = "3.11.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "572f695136211188308f16ad2ca5c851a712c464060ae6974944458eb83880ba"
[[package]]
name = "bytecount"
version = "0.6.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2c676a478f63e9fa2dd5368a42f28bba0d6c560b775f38583c8bbaa7fcd67c9c"
[[package]]
name = "bytemuck"
version = "1.12.3"
@ -494,6 +537,7 @@ name = "clash-verge"
version = "0.1.0"
dependencies = [
"anyhow",
"async-std",
"auto-launch",
"chrono",
"ctrlc",
@ -501,12 +545,15 @@ dependencies = [
"delay_timer",
"dirs 4.0.0",
"dunce",
"futures 0.3.25",
"interprocess",
"log 0.4.17",
"log4rs",
"nanoid",
"once_cell",
"open",
"parking_lot",
"patch",
"port_scanner",
"reqwest",
"rquickjs",
@ -519,6 +566,7 @@ dependencies = [
"tauri",
"tauri-build",
"tauri-runtime-wry",
"tauri-utils",
"tokio",
"warp",
"which 4.3.0",
@ -902,7 +950,7 @@ dependencies = [
"cron_clock",
"dashmap",
"event-listener",
"futures",
"futures 0.3.25",
"log 0.4.17",
"lru",
"once_cell",
@ -1169,6 +1217,12 @@ dependencies = [
"new_debug_unreachable",
]
[[package]]
name = "futures"
version = "0.1.31"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3a471a38ef8ed83cd6e40aa59c1ffe17db6855c18e3604d9c4ed8c08ebc28678"
[[package]]
name = "futures"
version = "0.3.25"
@ -1261,6 +1315,7 @@ version = "0.3.25"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "197676987abd2f9cadff84926f410af1c183608d36641465df73ae8211dc65d6"
dependencies = [
"futures 0.1.31",
"futures-channel",
"futures-core",
"futures-io",
@ -1364,7 +1419,7 @@ dependencies = [
"libc",
"log 0.4.17",
"rustversion",
"windows 0.39.0",
"windows",
]
[[package]]
@ -1499,6 +1554,18 @@ dependencies = [
"regex 1.7.0",
]
[[package]]
name = "gloo-timers"
version = "0.2.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9b995a66bb87bebce9a0f4a95aed01daca4872c050bfcb21653361c03bc35e5c"
dependencies = [
"futures-channel",
"futures-core",
"js-sys",
"wasm-bindgen",
]
[[package]]
name = "gobject-sys"
version = "0.15.10"
@ -1885,6 +1952,32 @@ dependencies = [
"serde_derive",
]
[[package]]
name = "interprocess"
version = "1.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "81f2533f3be42fffe3b5e63b71aeca416c1c3bc33e4e27be018521e76b1f38fb"
dependencies = [
"blocking",
"cfg-if 1.0.0",
"futures-core",
"futures-io",
"intmap",
"libc",
"once_cell",
"rustc_version 0.4.0",
"spinning",
"thiserror",
"to_method",
"winapi",
]
[[package]]
name = "intmap"
version = "0.7.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ae52f28f45ac2bc96edb7714de995cffc174a395fb0abf5bff453587c980d7b9"
[[package]]
name = "ipnet"
version = "2.6.0"
@ -1978,6 +2071,15 @@ dependencies = [
"selectors",
]
[[package]]
name = "kv-log-macro"
version = "1.0.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0de8b303297635ad57c9f5059fd9cee7a47f8e8daa09df0fcd07dd39fb22977f"
dependencies = [
"log 0.4.17",
]
[[package]]
name = "lazy_static"
version = "0.2.11"
@ -2081,6 +2183,7 @@ checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e"
dependencies = [
"cfg-if 1.0.0",
"serde",
"value-bag",
]
[[package]]
@ -2235,12 +2338,6 @@ version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a"
[[package]]
name = "minisign-verify"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "933dca44d65cdd53b355d0b73d380a2ff5da71f87f036053188bf1eab6a19881"
[[package]]
name = "miniz_oxide"
version = "0.5.4"
@ -2401,6 +2498,17 @@ dependencies = [
"minimal-lexical",
]
[[package]]
name = "nom_locate"
version = "4.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b1e299bf5ea7b212e811e71174c5d1a5d065c4c0ad0c8691ecb1f97e3e66025e"
dependencies = [
"bytecount",
"memchr",
"nom 7.1.1",
]
[[package]]
name = "ntapi"
version = "0.4.0"
@ -2502,17 +2610,6 @@ dependencies = [
"objc_exception",
]
[[package]]
name = "objc-foundation"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1add1b659e36c9607c7aab864a76c7a4c2760cd0cd2e120f3fb8b952c7e22bf9"
dependencies = [
"block",
"objc",
"objc_id",
]
[[package]]
name = "objc_exception"
version = "0.1.2"
@ -2692,6 +2789,17 @@ version = "1.0.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b1de2e551fb905ac83f73f7aedf2f0cb4a0da7e35efa24a202a936269f1f18e1"
[[package]]
name = "patch"
version = "0.7.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "15c07fdcdd8b05bdcf2a25bc195b6c34cbd52762ada9dba88bf81e7686d14e7a"
dependencies = [
"chrono",
"nom 7.1.1",
"nom_locate",
]
[[package]]
name = "pathdiff"
version = "0.2.1"
@ -3215,30 +3323,6 @@ dependencies = [
"winreg",
]
[[package]]
name = "rfd"
version = "0.10.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0149778bd99b6959285b0933288206090c50e2327f47a9c463bfdbf45c8823ea"
dependencies = [
"block",
"dispatch",
"glib-sys",
"gobject-sys",
"gtk-sys",
"js-sys",
"lazy_static 1.4.0",
"log 0.4.17",
"objc",
"objc-foundation",
"objc_id",
"raw-window-handle",
"wasm-bindgen",
"wasm-bindgen-futures",
"web-sys",
"windows 0.37.0",
]
[[package]]
name = "rquickjs"
version = "0.1.7"
@ -3787,6 +3871,15 @@ dependencies = [
"system-deps 5.0.0",
]
[[package]]
name = "spinning"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2d4f0e86297cad2658d92a707320d87bf4e6ae1050287f51d19b67ef3f153a7b"
dependencies = [
"lock_api",
]
[[package]]
name = "stable_deref_trait"
version = "1.2.0"
@ -3940,7 +4033,7 @@ dependencies = [
"serde",
"unicode-segmentation",
"uuid 1.2.2",
"windows 0.39.0",
"windows",
"windows-implement",
"x11-dl",
]
@ -3963,8 +4056,6 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d8ea1d785ab2164373703817bff144c4610a69ad3f659becaca0e1ea004b98d8"
dependencies = [
"anyhow",
"attohttpc",
"base64",
"cocoa",
"dirs-next",
"embed_plist",
@ -3977,7 +4068,6 @@ dependencies = [
"heck 0.4.0",
"http",
"ignore",
"minisign-verify",
"objc",
"once_cell",
"open",
@ -3986,7 +4076,6 @@ dependencies = [
"rand 0.8.5",
"raw-window-handle",
"regex 1.7.0",
"rfd",
"semver 1.0.14",
"serde",
"serde_json",
@ -4001,14 +4090,12 @@ dependencies = [
"tauri-utils",
"tempfile",
"thiserror",
"time 0.3.17",
"tokio",
"url",
"uuid 1.2.2",
"webkit2gtk",
"webview2-com",
"windows 0.39.0",
"zip",
"windows",
]
[[package]]
@ -4084,7 +4171,7 @@ dependencies = [
"thiserror",
"uuid 1.2.2",
"webview2-com",
"windows 0.39.0",
"windows",
]
[[package]]
@ -4103,7 +4190,7 @@ dependencies = [
"uuid 1.2.2",
"webkit2gtk",
"webview2-com",
"windows 0.39.0",
"windows",
"wry",
]
@ -4132,7 +4219,7 @@ dependencies = [
"thiserror",
"url",
"walkdir",
"windows 0.39.0",
"windows",
]
[[package]]
@ -4331,6 +4418,12 @@ version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cda74da7e1a664f795bb1f8a87ec406fb89a02522cf6e50620d016add6dbbf5c"
[[package]]
name = "to_method"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c7c4ceeeca15c8384bbc3e011dbd8fccb7f068a440b752b7d9b32ceb0ca0e2e8"
[[package]]
name = "tokio"
version = "1.23.0"
@ -4665,6 +4758,16 @@ version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d"
[[package]]
name = "value-bag"
version = "1.0.0-alpha.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2209b78d1249f7e6f3293657c9779fe31ced465df091bbd433a1cf88e916ec55"
dependencies = [
"ctor",
"version_check",
]
[[package]]
name = "vcpkg"
version = "0.2.15"
@ -4905,7 +5008,7 @@ checksum = "b4a769c9f1a64a8734bde70caafac2b96cada12cd4aefa49196b3a386b8b4178"
dependencies = [
"webview2-com-macros",
"webview2-com-sys",
"windows 0.39.0",
"windows",
"windows-implement",
]
@ -4930,7 +5033,7 @@ dependencies = [
"serde",
"serde_json",
"thiserror",
"windows 0.39.0",
"windows",
"windows-bindgen",
"windows-metadata",
]
@ -5020,19 +5123,6 @@ dependencies = [
"windows-sys 0.42.0",
]
[[package]]
name = "windows"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "57b543186b344cc61c85b5aab0d2e3adf4e0f99bc076eff9aa5927bcc0b8a647"
dependencies = [
"windows_aarch64_msvc 0.37.0",
"windows_i686_gnu 0.37.0",
"windows_i686_msvc 0.37.0",
"windows_x86_64_gnu 0.37.0",
"windows_x86_64_msvc 0.37.0",
]
[[package]]
name = "windows"
version = "0.39.0"
@ -5119,12 +5209,6 @@ version = "0.36.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9bb8c3fd39ade2d67e9874ac4f3db21f0d710bee00fe7cab16949ec184eeaa47"
[[package]]
name = "windows_aarch64_msvc"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2623277cb2d1c216ba3b578c0f3cf9cdebeddb6e66b1b218bb33596ea7769c3a"
[[package]]
name = "windows_aarch64_msvc"
version = "0.39.0"
@ -5143,12 +5227,6 @@ version = "0.36.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "180e6ccf01daf4c426b846dfc66db1fc518f074baa793aa7d9b9aaeffad6a3b6"
[[package]]
name = "windows_i686_gnu"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d3925fd0b0b804730d44d4b6278c50f9699703ec49bcd628020f46f4ba07d9e1"
[[package]]
name = "windows_i686_gnu"
version = "0.39.0"
@ -5167,12 +5245,6 @@ version = "0.36.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e2e7917148b2812d1eeafaeb22a97e4813dfa60a3f8f78ebe204bcc88f12f024"
[[package]]
name = "windows_i686_msvc"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ce907ac74fe331b524c1298683efbf598bb031bc84d5e274db2083696d07c57c"
[[package]]
name = "windows_i686_msvc"
version = "0.39.0"
@ -5191,12 +5263,6 @@ version = "0.36.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4dcd171b8776c41b97521e5da127a2d86ad280114807d0b2ab1e462bc764d9e1"
[[package]]
name = "windows_x86_64_gnu"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2babfba0828f2e6b32457d5341427dcbb577ceef556273229959ac23a10af33d"
[[package]]
name = "windows_x86_64_gnu"
version = "0.39.0"
@ -5221,12 +5287,6 @@ version = "0.36.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c811ca4a8c853ef420abd8592ba53ddbbac90410fab6903b3e79972a631f7680"
[[package]]
name = "windows_x86_64_msvc"
version = "0.37.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f4dd6dc7df2d84cf7b33822ed5b86318fb1781948e9663bacd047fc9dd52259d"
[[package]]
name = "windows_x86_64_msvc"
version = "0.39.0"
@ -5291,7 +5351,7 @@ dependencies = [
"webkit2gtk",
"webkit2gtk-sys",
"webview2-com",
"windows 0.39.0",
"windows",
"windows-implement",
]
@ -5339,14 +5399,3 @@ checksum = "56c1936c4cc7a1c9ab21a1ebb602eb942ba868cbd44a99cb7cdc5892335e1c85"
dependencies = [
"linked-hash-map",
]
[[package]]
name = "zip"
version = "0.6.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "537ce7411d25e54e8ae21a7ce0b15840e7bfcff15b51d697ec3266cc76bdf080"
dependencies = [
"byteorder",
"crc32fast",
"crossbeam-utils",
]

View file

@ -35,12 +35,17 @@ port_scanner = "0.1.5"
delay_timer = "0.11.1"
parking_lot = "0.12.0"
tokio = { version = "1", features = ["full"] }
async-std = { version = "1", features = ["attributes", "tokio1"] }
serde = { version = "1.0", features = ["derive"] }
reqwest = { version = "0.11", features = ["json"] }
tauri = { version = "1.1.1", features = ["global-shortcut-all", "process-all", "shell-all", "system-tray", "updater", "window-all"] }
tauri = { version = "1.1.1", features = ["global-shortcut-all", "process-all", "shell-all", "system-tray", "window-all"] }
tauri-runtime-wry = { version = "0.12" }
window-vibrancy = { version = "0.3.0" }
window-shadows = { version = "0.2.0" }
patch = "0.7.0"
interprocess = "1.2.1"
tauri-utils = "1.2.1"
futures = { version = "0.3", features = ["compat"] }
[target.'cfg(windows)'.dependencies]
runas = "0.2.1"

View file

@ -0,0 +1,143 @@
use std::{
fs::{create_dir_all, remove_file, File},
io::{Error, ErrorKind, Read, Result, Write},
os::unix::net::{UnixListener, UnixStream},
process::Command,
};
use dirs_next::data_dir;
use super::ID;
pub async fn register<F,Fut>(scheme: &str, handler: F) -> Result<()>
where
F: FnMut(String) -> Fut + Send + 'static,
Fut: Future<Output = ()> + Send + 'static,
{
listen(handler);
let mut target = data_dir()
.ok_or_else(|| Error::new(ErrorKind::NotFound, "data directory not found."))?
.join("applications");
create_dir_all(&target)?;
let exe = tauri_utils::platform::current_exe()?;
let file_name = format!(
"{}-handler.desktop",
exe.file_name()
.ok_or_else(|| Error::new(
ErrorKind::NotFound,
"Couldn't get file name of curent executable.",
))?
.to_string_lossy()
);
target.push(&file_name);
let mime_types = format!("x-scheme-handler/{};", scheme);
let mut file = File::create(&target)?;
file.write_all(
format!(
include_str!("template.desktop"),
name = ID
.get()
.expect("Called register() before prepare()")
.split('.')
.last()
.unwrap(),
exec = exe.to_string_lossy(),
mime_types = mime_types
)
.as_bytes(),
)?;
target.pop();
Command::new("update-desktop-database")
.arg(target)
.status()?;
Command::new("xdg-mime")
.args(["default", &file_name, scheme])
.status()?;
Ok(())
}
pub fn unregister(_scheme: &str) -> Result<()> {
let mut target =
data_dir().ok_or_else(|| Error::new(ErrorKind::NotFound, "data directory not found."))?;
target.push("applications");
target.push(format!(
"{}-handler.desktop",
tauri_utils::platform::current_exe()?
.file_name()
.ok_or_else(|| Error::new(
ErrorKind::NotFound,
"Couldn't get file name of curent executable.",
))?
.to_string_lossy()
));
remove_file(&target)?;
Ok(())
}
pub fn listen<F: FnMut(String) + Send + 'static>(mut handler: F) {
std::thread::spawn(move || {
let addr = format!(
"/tmp/{}-deep-link.sock",
ID.get().expect("listen() called before prepare()")
);
let listener = UnixListener::bind(addr).expect("Can't create listener");
for stream in listener.incoming() {
match stream {
Ok(mut stream) => {
let mut buffer = String::new();
if let Err(io_err) = stream.read_to_string(&mut buffer) {
log::error!("Error reading incoming connection: {}", io_err.to_string());
};
handler(dbg!(buffer));
}
Err(err) => {
log::error!("Incoming connection failed: {}", err);
continue;
}
}
}
});
}
pub fn prepare(identifier: &str) {
let addr = format!("/tmp/{}-deep-link.sock", identifier);
match UnixStream::connect(&addr) {
Ok(mut stream) => {
if let Err(io_err) =
stream.write_all(std::env::args().nth(1).unwrap_or_default().as_bytes())
{
log::error!(
"Error sending message to primary instance: {}",
io_err.to_string()
);
};
std::process::exit(0);
}
Err(err) => {
log::error!("Error creating socket listener: {}", err.to_string());
if err.kind() == ErrorKind::ConnectionRefused {
let _ = remove_file(&addr);
}
}
};
ID.set(identifier.to_string())
.expect("prepare() called more than once with different identifiers.");
}

View file

@ -0,0 +1,11 @@
pub fn register<F: FnMut(String) + Send + 'static>(
identifier: &str,
scheme: &str,
handler: F,
) -> Result<(), std::io::Error> {
unimplemented!()
}
pub fn unregister(scheme: &str) -> Result<(), std::io::Error> {
unimplemented!()
}

View file

@ -0,0 +1,27 @@
use once_cell::sync::OnceCell;
#[cfg(target_os = "linux")]
mod linux;
#[cfg(target_os = "linux")]
pub use linux::*;
#[cfg(target_os = "windows")]
mod windows;
#[cfg(target_os = "windows")]
pub use windows::*;
#[cfg(target_os = "macos")]
mod macos;
#[cfg(target_os = "macos")]
pub use macos::*;
static ID: OnceCell<String> = OnceCell::new();
/// This function is meant for use-cases where the default [`prepare`] function can't be used.
///
/// # Errors
/// If ID was already set this functions returns an error containing the ID as String.
#[allow(unused)]
pub fn set_identifier(identifier: &str) -> Result<(), String> {
ID.set(identifier.to_string())
}

View file

@ -0,0 +1,101 @@
use std::{
io::{BufRead, BufReader, Write},
path::Path,
};
use interprocess::local_socket::{LocalSocketListener, LocalSocketStream};
use warp::Future;
use winreg::{enums::HKEY_CURRENT_USER, RegKey};
use super::ID;
// Consider adding a function to register without starting the listener.
// Plugin needs linux and macOS support before making decisions.
pub async fn register<F,Fut>(
scheme: &str,
handler: F,
) -> Result<(), std::io::Error>
where
F: FnMut(String) -> Fut + Send + 'static,
Fut: Future<Output = ()> + Send + 'static,
{
listen(handler);
let hkcu = RegKey::predef(HKEY_CURRENT_USER);
let base = Path::new("Software").join("Classes").join(scheme);
let exe = tauri_utils::platform::current_exe()?
.to_string_lossy()
.replace("\\\\?\\", "");
let (key, _) = hkcu.create_subkey(&base)?;
key.set_value(
"",
&format!(
"URL:{}",
ID.get().expect("register() called before prepare()")
),
)?;
key.set_value("URL Protocol", &"")?;
let (icon, _) = hkcu.create_subkey(base.join("DefaultIcon"))?;
icon.set_value("", &format!("{},0", &exe))?;
let (cmd, _) = hkcu.create_subkey(base.join("shell").join("open").join("command"))?;
cmd.set_value("", &format!("{} \"%1\"", &exe))?;
Ok(())
}
#[allow(unused)]
pub fn unregister(scheme: &str) -> Result<(), std::io::Error> {
let hkcu = RegKey::predef(HKEY_CURRENT_USER);
let base = Path::new("Software").join("Classes").join(scheme);
hkcu.delete_subkey_all(base)?;
Ok(())
}
pub fn listen<F,Fut>(mut handler: F) -> ()
where
F: FnMut(String) -> Fut + Send + 'static ,
Fut: Future<Output = ()> + Send + 'static,
{
let task_to_do = async move {
let listener =
LocalSocketListener::bind(ID.get().expect("listen() called before prepare()").as_str())
.expect("Can't create listener");
for conn in listener.incoming().filter_map(|c| {c.map_err(|error| log::error!("Incoming connection failed: {}", error)).ok()})
{
let mut conn = BufReader::new(conn);
let mut buffer = String::new();
if let Err(io_err) = conn.read_line(&mut buffer) {
log::error!("Error reading incoming connection: {}", io_err.to_string());
};
buffer.pop();
handler(buffer).await;
}
};
tokio::spawn(async move{
task_to_do.await;
});
}
pub fn prepare(identifier: &str) {
if let Ok(mut conn) = LocalSocketStream::connect(identifier) {
if let Err(io_err) = conn.write_all(std::env::args().nth(1).unwrap_or_default().as_bytes())
{
log::error!(
"Error sending message to primary instance: {}",
io_err.to_string()
);
};
let _ = conn.write_all(b"\n");
std::process::exit(0);
};
ID.set(identifier.to_string())
.expect("prepare() called more than once with different identifiers.");
}

View file

@ -9,11 +9,47 @@ mod core;
mod enhance;
mod feat;
mod utils;
mod deep_link;
use crate::utils::{init, resolve, server};
use crate::utils::{init, resolve, server, help};
use crate::core::handle::Handle;
use tauri::{api, SystemTray};
fn main() -> std::io::Result<()> {
#[tokio::main]
async fn main() -> std::io::Result<()> {
// Deep linking
deep_link::prepare("app.clashverge");
// Define handler
let handler = | deep_link | async move {
// Convert deep link to something that import_profile can use
let profile_url_and_name = help::convert_deeplink_to_url_for_import_profile(&deep_link);
// If deep link is invalid, we pop up a message to user
if profile_url_and_name.is_err(){
Handle::notice_message("set_config::error", "Profile url is invalid");
}
// Import profile
let import_result = cmds::import_profile(profile_url_and_name.unwrap(), None).await;
// If we couldn't import profile& we pop up a message to user
if import_result.is_err(){
Handle::notice_message("set_config::error",format!("Profile url is invalid | {}", import_result.err().unwrap()));
}
Handle::notice_message("set_config::ok", "Profile added.");
};
// Register "clash" scheme
let deep_link_register_result = deep_link::register("clash",handler.clone()).await;
// If we couldn't register, we log it
if deep_link_register_result.is_err(){
println!("We can't register \"clash\" scheme for program | {}",deep_link_register_result.err().unwrap())
}
// Register "clashmeta" scheme
// deep_link_register_result = deep_link::register("clashmeta", handler).await;
// If we couldn't register, we log it
// if deep_link_register_result.is_err(){
// println!("We can't register \"clashmeta\" scheme for program | {}",deep_link_register_result.err().unwrap())
// }
// 单例检测
if server::check_singleton().is_err() {
println!("app exists");

View file

@ -106,7 +106,7 @@ pub fn verge_path() -> Result<PathBuf> {
pub fn profiles_path() -> Result<PathBuf> {
Ok(app_home_dir()?.join(PROFILE_YAML))
}
#[allow(unused)]
pub fn app_res_dir() -> Result<PathBuf> {
unsafe {
Ok(RESOURCE_DIR

View file

@ -105,6 +105,50 @@ pub fn open_file(path: PathBuf) -> Result<()> {
Ok(())
}
#[derive(Debug)]
pub enum ExtractDeeplinkError{
InvalidInput
}
// pub fn extract_url_and_profile_name_from_deep_link(deep_link:&String) -> Result<(String,String),ExtractDeeplinkError>{
// // Sample: clash://install-config?url=https://mysite.com/all.yml&name=profilename
// let (url,profile) = {
// let pruned = deep_link.split("url=").collect::<Vec<_>>();
// if pruned.len() < 2 {
// return Err(ExtractDeeplinkError::InvalidInput)
// }
// let url_and_profile_name = pruned[1].split("&").collect::<Vec<_>>();
// if url_and_profile_name.len() < 2{
// return Err(ExtractDeeplinkError::InvalidInput)
// }
// let url = url_and_profile_name[0].to_string();
// let profile_name = {
// let splitted: Vec<_> = url_and_profile_name[1].split("=").collect();
// if splitted.len() < 2 {
// return Err(ExtractDeeplinkError::InvalidInput)
// }
// splitted[1].to_string()
// };
// (url,profile_name)
// };
// return Ok((url,profile));
// }
pub fn convert_deeplink_to_url_for_import_profile(deep_link:&String) -> Result<String,ExtractDeeplinkError>{
// Sample: clash://install-config?url=https://mysite.com/all.yml&name=profilename
let import_profile_url_raw = {
let url_part:Vec<_> = deep_link.split("url=").collect();
if url_part.len() < 2{
return Err(ExtractDeeplinkError::InvalidInput)
}
url_part
};
// Convert url to something that import_profile functin can use
let import_profile_url = import_profile_url_raw[1].replacen('&', "?", 1);
Ok(import_profile_url)
}
#[macro_export]
macro_rules! error {
($result: expr) => {
@ -169,3 +213,20 @@ fn test_parse_value() {
assert_eq!(parse_str::<usize>(test_1, "expire1="), None);
assert_eq!(parse_str::<usize>(test_2, "attachment="), None);
}
//#[test]
// fn test_extract_url_and_profile_name_from_deep_link(){
// let s = "clash://install-config?url=https://mysite.com/all.yml&name=profilename";
// let (url,prof_name) = extract_url_and_profile_name_from_deep_link(&s.to_string()).unwrap();
// assert_eq!(url,"https://mysite.com/all.yml");
// assert_eq!(prof_name,"profilename");
// }
#[test]
fn test_convert_deeplink_to_url_for_import_profile(){
let s = "clashy://install-config?url=https://antyfilter.aeycia.cl/80467cf865c2ef1af111716ddf30dd29/80467cf865c2ef1af111716ddf30dd29/clash/all.yml&name=all_antyfilter.aeycia.cl";
let res = convert_deeplink_to_url_for_import_profile(&s.to_string());
if res.is_err(){
println!("Test failed: {:?}",res.err().unwrap())
}else{
panic!("Test successfully completed")
}
}

View file

@ -1,6 +1,6 @@
{
"package": {
"productName": "Clash Verge",
"productName": "HiddifyDesktop",
"version": "1.2.1"
},
"build": {
@ -26,7 +26,7 @@
"icons/icon.ico"
],
"resources": ["resources"],
"externalBin": ["sidecar/clash", "sidecar/clash-meta"],
"externalBin": ["sidecar/clash-meta", "sidecar/clash"],
"copyright": "© 2022 zzzgydi All Rights Reserved",
"category": "DeveloperTool",
"shortDescription": "A Clash GUI based on tauri.",
@ -51,7 +51,7 @@
}
},
"updater": {
"active": true,
"active": false,
"endpoints": [
"https://github.com/zzzgydi/clash-verge/releases/download/updater/update.json",
"https://hub.fastgit.xyz/zzzgydi/clash-verge/releases/download/updater/update-proxy.json"

View file

@ -78,7 +78,7 @@ const ProfilePage = () => {
// init selected array
const { selected = [] } = profile;
const selectedMap = Object.fromEntries(
h selected.map((each) => [each.name!, each.now!])
selected.map((each) => [each.name!, each.now!])
);
let hasChange = false;

View file

@ -5,7 +5,7 @@
"useDefineForClassFields": true,
"lib": ["DOM", "DOM.Iterable", "ESNext"],
"allowJs": false,
"skipLibCheck": false,
"skipLibCheck": true,
"esModuleInterop": false,
"allowSyntheticDefaultImports": true,
"strict": true,
@ -16,10 +16,11 @@
"isolatedModules": true,
"noEmit": true,
"jsx": "react-jsx",
"noImplicitAny": false,
"paths": {
"@/*": ["src/*"],
"@root/*": ["./*"]
}
},
},
"include": ["./src"]
}