From dfc8f80ba575d29f35f7a7fde456485518287592 Mon Sep 17 00:00:00 2001
From: zhom <2717306+zhom@users.noreply.github.com>
Date: Mon, 13 Apr 2026 02:47:16 +0400
Subject: [PATCH] refactor: wayfern launch
---
next-env.d.ts | 2 +-
src-tauri/src/proxy_server.rs | 23 ++++-
src-tauri/src/wayfern_manager.rs | 151 +++++++++++++++++--------------
3 files changed, 105 insertions(+), 71 deletions(-)
diff --git a/next-env.d.ts b/next-env.d.ts
index 9edff1c..b87975d 100644
--- a/next-env.d.ts
+++ b/next-env.d.ts
@@ -1,6 +1,6 @@
///
///
-import "./.next/types/routes.d.ts";
+import "./dist/dev/types/routes.d.ts";
// NOTE: This file should not be edited
// see https://nextjs.org/docs/app/api-reference/config/typescript for more information.
diff --git a/src-tauri/src/proxy_server.rs b/src-tauri/src/proxy_server.rs
index b2cafe4..5ad45fb 100644
--- a/src-tauri/src/proxy_server.rs
+++ b/src-tauri/src/proxy_server.rs
@@ -1435,10 +1435,26 @@ async fn handle_connect_from_buffer(
// Returns a BoxedAsyncStream so all upstream types (plain TCP, SOCKS,
// Shadowsocks) share the same bidirectional-copy tunnel code below.
let should_bypass = bypass_matcher.should_bypass(target_host);
+ // Helper: configure outbound TCP to match browser TCP fingerprint
+ let configure_tcp = |stream: &TcpStream| {
+ let _ = stream.set_nodelay(true);
+ };
let target_stream: BoxedAsyncStream = match upstream_url.as_ref() {
- None => Box::new(TcpStream::connect((target_host, target_port)).await?),
- Some(url) if url == "DIRECT" => Box::new(TcpStream::connect((target_host, target_port)).await?),
- _ if should_bypass => Box::new(TcpStream::connect((target_host, target_port)).await?),
+ None => {
+ let s = TcpStream::connect((target_host, target_port)).await?;
+ configure_tcp(&s);
+ Box::new(s)
+ }
+ Some(url) if url == "DIRECT" => {
+ let s = TcpStream::connect((target_host, target_port)).await?;
+ configure_tcp(&s);
+ Box::new(s)
+ }
+ _ if should_bypass => {
+ let s = TcpStream::connect((target_host, target_port)).await?;
+ configure_tcp(&s);
+ Box::new(s)
+ }
Some(upstream_url_str) => {
let upstream = Url::parse(upstream_url_str)?;
let scheme = upstream.scheme();
@@ -1448,6 +1464,7 @@ async fn handle_connect_from_buffer(
let proxy_host = upstream.host_str().unwrap_or("127.0.0.1");
let proxy_port = upstream.port().unwrap_or(8080);
let mut proxy_stream = TcpStream::connect((proxy_host, proxy_port)).await?;
+ configure_tcp(&proxy_stream);
let mut connect_req = format!(
"CONNECT {}:{} HTTP/1.1\r\nHost: {}:{}\r\n",
diff --git a/src-tauri/src/wayfern_manager.rs b/src-tauri/src/wayfern_manager.rs
index d7904c9..5b998a7 100644
--- a/src-tauri/src/wayfern_manager.rs
+++ b/src-tauri/src/wayfern_manager.rs
@@ -1,5 +1,6 @@
use crate::browser_runner::BrowserRunner;
use crate::profile::BrowserProfile;
+use playwright::api::Playwright;
use reqwest::Client;
use serde::{Deserialize, Serialize};
use serde_json::json;
@@ -53,14 +54,14 @@ pub struct WayfernLaunchResult {
pub cdp_port: Option,
}
-#[derive(Debug)]
struct WayfernInstance {
- #[allow(dead_code)]
id: String,
process_id: Option,
profile_path: Option,
url: Option,
cdp_port: Option,
+ playwright_context: Option,
+ playwright_runtime: Option,
}
struct WayfernManagerInner {
@@ -86,14 +87,6 @@ impl WayfernManager {
inner: Arc::new(AsyncMutex::new(WayfernManagerInner {
instances: HashMap::new(),
})),
- // Every request this client makes goes to a local `http://127.0.0.1:`
- // endpoint that Wayfern is still bringing up. Without a per-request timeout,
- // a single hanging connect or a stuck HTTP response will block
- // `wait_for_cdp_ready` indefinitely — its 120-attempt poll loop depends on
- // each request returning fast. A 2-second per-request timeout turns that
- // into a worst-case ~60-second bounded wait, and `generate_fingerprint_config`
- // can then return a real error instead of hanging the profile-creation UI
- // forever.
http_client: Client::builder()
.timeout(Duration::from_secs(2))
.build()
@@ -101,6 +94,16 @@ impl WayfernManager {
}
}
+ async fn create_playwright(
+ &self,
+ ) -> Result> {
+ Playwright::initialize()
+ .await
+ .map_err(|e| -> Box {
+ format!("Failed to initialize Playwright: {e}").into()
+ })
+ }
+
pub fn instance() -> &'static WayfernManager {
&WAYFERN_MANAGER
}
@@ -593,7 +596,6 @@ impl WayfernManager {
let mut args = vec![
format!("--remote-debugging-port={port}"),
"--remote-debugging-address=127.0.0.1".to_string(),
- format!("--user-data-dir={}", profile_path),
"--no-first-run".to_string(),
"--no-default-browser-check".to_string(),
"--disable-background-mode".to_string(),
@@ -604,7 +606,7 @@ impl WayfernManager {
"--disable-session-crashed-bubble".to_string(),
"--hide-crash-restore-bubble".to_string(),
"--disable-infobars".to_string(),
- "--disable-features=DialMediaRouteProvider".to_string(),
+ "--disable-features=DialMediaRouteProvider,DnsOverHttps,AsyncDns".to_string(),
"--use-mock-keychain".to_string(),
"--password-store=basic".to_string(),
];
@@ -616,10 +618,6 @@ impl WayfernManager {
args.push("--disable-dev-shm-usage".to_string());
}
- if let Some(proxy) = proxy_url {
- args.push(format!("--proxy-server={proxy}"));
- }
-
if ephemeral {
args.push("--disk-cache-size=1".to_string());
args.push("--disable-breakpad".to_string());
@@ -632,8 +630,17 @@ impl WayfernManager {
args.push(format!("--load-extension={}", extension_paths.join(",")));
}
- // Pass wayfern token as CLI flag so the browser can gate CDP features
- let wayfern_token = crate::cloud_auth::CLOUD_AUTH.get_wayfern_token().await;
+ let mut wayfern_token = crate::cloud_auth::CLOUD_AUTH.get_wayfern_token().await;
+ if wayfern_token.is_none() {
+ log::info!("Wayfern token not ready, waiting...");
+ for _ in 0..15 {
+ tokio::time::sleep(Duration::from_secs(1)).await;
+ wayfern_token = crate::cloud_auth::CLOUD_AUTH.get_wayfern_token().await;
+ if wayfern_token.is_some() {
+ break;
+ }
+ }
+ }
if let Some(ref token) = wayfern_token {
args.push(format!("--wayfern-token={token}"));
log::info!("Wayfern token passed as CLI flag (length: {})", token.len());
@@ -641,28 +648,61 @@ impl WayfernManager {
log::warn!("No wayfern token available — CDP gated methods will be blocked");
}
- // Don't add URL to args - we'll navigate via CDP after setting fingerprint
- // This ensures fingerprint is applied at navigation commit time
+ if let Some(proxy) = proxy_url {
+ let pac_data = format!(
+ "data:application/x-ns-proxy-autoconfig,function FindProxyForURL(url,host){{return \"PROXY {}\";}}",
+ proxy.trim_start_matches("http://").trim_start_matches("https://")
+ );
+ args.push(format!("--proxy-pac-url={pac_data}"));
+ args.push("--dns-prefetch-disable".to_string());
+ }
- let mut cmd = TokioCommand::new(&executable_path);
- cmd.args(&args);
- cmd.stdout(Stdio::piped());
- cmd.stderr(Stdio::piped());
+ let pw = self.create_playwright().await?;
+ let chromium = pw.chromium();
+ let profile_path_ref = std::path::Path::new(profile_path);
+ let mut launcher = chromium.persistent_context_launcher(profile_path_ref);
+ launcher = launcher.executable(executable_path.as_ref());
+ launcher = launcher.headless(false);
+ launcher = launcher.chromium_sandbox(true);
+ launcher = launcher.args(&args);
+ launcher = launcher.timeout(0.0);
- let child = cmd.spawn().map_err(|e| {
- let hint = if e.raw_os_error() == Some(14001) {
- ". This usually means the Visual C++ Redistributable is not installed. \
- Download it from https://aka.ms/vs/17/release/vc_redist.x64.exe"
- } else {
- ""
- };
- format!("Failed to launch Wayfern: {e}{hint}")
- })?;
- let process_id = child.id();
+ let pw_context =
+ launcher
+ .launch()
+ .await
+ .map_err(|e| -> Box {
+ let hint = if format!("{e}").contains("14001") {
+ ". This usually means the Visual C++ Redistributable is not installed. \
+ Download it from https://aka.ms/vs/17/release/vc_redist.x64.exe"
+ } else {
+ ""
+ };
+ format!("Failed to launch Wayfern: {e}{hint}").into()
+ })?;
- self.wait_for_cdp_ready(port).await?;
+ let process_id = {
+ use sysinfo::{ProcessRefreshKind, RefreshKind, System};
+ let system = System::new_with_specifics(
+ RefreshKind::nothing().with_processes(ProcessRefreshKind::everything()),
+ );
+ let mut found: Option = None;
+ for (pid, process) in system.processes() {
+ let cmd_str = process
+ .cmd()
+ .iter()
+ .map(|s| s.to_string_lossy().to_string())
+ .collect::>()
+ .join(" ");
+ if cmd_str.contains(&format!("--remote-debugging-port={port}")) {
+ found = Some(pid.as_u32());
+ break;
+ }
+ }
+ found
+ };
+ let pw_runtime = pw;
- // Get CDP targets first - needed for both fingerprint and navigation
let targets = self.get_cdp_targets(port).await?;
log::info!("Found {} CDP targets", targets.len());
@@ -761,37 +801,7 @@ impl WayfernManager {
log::warn!("No fingerprint found in config, browser will use default fingerprint");
}
- // Set geolocation override via CDP so navigator.geolocation.getCurrentPosition() matches
- if let Some(fingerprint_json) = &config.fingerprint {
- if let Ok(fp) = serde_json::from_str::(fingerprint_json) {
- let fp_obj = if fp.get("fingerprint").is_some() {
- fp.get("fingerprint").unwrap()
- } else {
- &fp
- };
- if let (Some(lat), Some(lng)) = (
- fp_obj.get("latitude").and_then(|v| v.as_f64()),
- fp_obj.get("longitude").and_then(|v| v.as_f64()),
- ) {
- let accuracy = fp_obj
- .get("accuracy")
- .and_then(|v| v.as_f64())
- .unwrap_or(100.0);
- if let Some(target) = page_targets.first() {
- if let Some(ws_url) = &target.websocket_debugger_url {
- let _ = self
- .send_cdp_command(
- ws_url,
- "Emulation.setGeolocationOverride",
- json!({ "latitude": lat, "longitude": lng, "accuracy": accuracy }),
- )
- .await;
- log::info!("Set geolocation override: lat={lat}, lng={lng}");
- }
- }
- }
- }
- }
+ // Geolocation is handled internally by the browser binary.
// Navigate to URL via CDP - fingerprint will be applied at navigation commit time
if let Some(url) = url {
@@ -816,6 +826,8 @@ impl WayfernManager {
profile_path: Some(profile_path.to_string()),
url: url.map(|s| s.to_string()),
cdp_port: Some(port),
+ playwright_context: Some(pw_context),
+ playwright_runtime: Some(pw_runtime),
};
let mut inner = self.inner.lock().await;
@@ -837,6 +849,9 @@ impl WayfernManager {
let mut inner = self.inner.lock().await;
if let Some(instance) = inner.instances.remove(id) {
+ log::info!("Cleaning up Wayfern instance {}", instance.id);
+ drop(instance.playwright_context);
+ drop(instance.playwright_runtime);
if let Some(pid) = instance.process_id {
#[cfg(unix)]
{
@@ -991,6 +1006,8 @@ impl WayfernManager {
profile_path: Some(found_profile_path.clone()),
url: None,
cdp_port,
+ playwright_context: None,
+ playwright_runtime: None,
},
);