Files
worldmonitor/src-tauri/src/main.rs
Elie Habib 5b1f980b70 Fix Windows settings window: async command, no menu bar, no white flash
- Make open_settings_window_command async to prevent WebView2 deadlock on Windows
- Create settings window with visible(false) to avoid white flash before content loads
- Remove menu bar from settings window on Windows/Linux (macOS uses screen-level menu)
- Frontend calls plugin:window|show + set_focus after init completes
2026-02-15 00:15:23 +04:00

603 lines
19 KiB
Rust

#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
use std::fs::{self, File, OpenOptions};
use std::io::Write;
use std::path::{Path, PathBuf};
use std::process::{Child, Command, Stdio};
use std::sync::Mutex;
use std::time::{SystemTime, UNIX_EPOCH};
use std::env;
use keyring::Entry;
use serde_json::{Map, Value};
use tauri::menu::{AboutMetadata, Menu, MenuItem, PredefinedMenuItem, Submenu};
use tauri::{AppHandle, Manager, RunEvent, WebviewUrl, WebviewWindowBuilder};
const LOCAL_API_PORT: &str = "46123";
const KEYRING_SERVICE: &str = "world-monitor";
const LOCAL_API_LOG_FILE: &str = "local-api.log";
const DESKTOP_LOG_FILE: &str = "desktop.log";
const MENU_FILE_SETTINGS_ID: &str = "file.settings";
const MENU_HELP_GITHUB_ID: &str = "help.github";
const MENU_HELP_DEVTOOLS_ID: &str = "help.devtools";
const SUPPORTED_SECRET_KEYS: [&str; 15] = [
"GROQ_API_KEY",
"OPENROUTER_API_KEY",
"FRED_API_KEY",
"EIA_API_KEY",
"CLOUDFLARE_API_TOKEN",
"ACLED_ACCESS_TOKEN",
"WINGBITS_API_KEY",
"WS_RELAY_URL",
"VITE_OPENSKY_RELAY_URL",
"OPENSKY_CLIENT_ID",
"OPENSKY_CLIENT_SECRET",
"AISSTREAM_API_KEY",
"VITE_WS_RELAY_URL",
"FINNHUB_API_KEY",
"NASA_FIRMS_API_KEY",
];
#[derive(Default)]
struct LocalApiState {
child: Mutex<Option<Child>>,
token: Mutex<Option<String>>,
}
fn secret_entry(key: &str) -> Result<Entry, String> {
if !SUPPORTED_SECRET_KEYS.contains(&key) {
return Err(format!("Unsupported secret key: {key}"));
}
Entry::new(KEYRING_SERVICE, key).map_err(|e| format!("Keyring init failed: {e}"))
}
fn generate_local_token() -> String {
use std::collections::hash_map::RandomState;
use std::hash::{BuildHasher, Hasher};
let state = RandomState::new();
let mut h1 = state.build_hasher();
h1.write_u64(std::process::id() as u64);
let a = h1.finish();
let mut h2 = state.build_hasher();
let nanos = SystemTime::now()
.duration_since(UNIX_EPOCH)
.map(|d| d.as_nanos())
.unwrap_or(0);
h2.write_u128(nanos);
let b = h2.finish();
format!("{a:016x}{b:016x}")
}
#[tauri::command]
fn get_local_api_token(state: tauri::State<'_, LocalApiState>) -> Result<String, String> {
let token = state
.token
.lock()
.map_err(|_| "Failed to lock local API token".to_string())?;
token.clone().ok_or_else(|| "Token not generated".to_string())
}
#[tauri::command]
fn list_supported_secret_keys() -> Vec<String> {
SUPPORTED_SECRET_KEYS.iter().map(|key| (*key).to_string()).collect()
}
#[tauri::command]
fn get_secret(key: String) -> Result<Option<String>, String> {
let entry = secret_entry(&key)?;
match entry.get_password() {
Ok(value) => Ok(Some(value)),
Err(keyring::Error::NoEntry) => Ok(None),
Err(err) => Err(format!("Failed to read keyring secret: {err}")),
}
}
#[tauri::command]
fn set_secret(key: String, value: String) -> Result<(), String> {
let entry = secret_entry(&key)?;
entry
.set_password(&value)
.map_err(|e| format!("Failed to write keyring secret: {e}"))
}
#[tauri::command]
fn delete_secret(key: String) -> Result<(), String> {
let entry = secret_entry(&key)?;
match entry.delete_credential() {
Ok(_) => Ok(()),
Err(keyring::Error::NoEntry) => Ok(()),
Err(err) => Err(format!("Failed to delete keyring secret: {err}")),
}
}
fn cache_file_path(app: &AppHandle) -> Result<PathBuf, String> {
let dir = app
.path()
.app_data_dir()
.map_err(|e| format!("Failed to resolve app data dir: {e}"))?;
std::fs::create_dir_all(&dir)
.map_err(|e| format!("Failed to create app data directory {}: {e}", dir.display()))?;
Ok(dir.join("persistent-cache.json"))
}
#[tauri::command]
fn read_cache_entry(app: AppHandle, key: String) -> Result<Option<Value>, String> {
let path = cache_file_path(&app)?;
if !path.exists() {
return Ok(None);
}
let contents = std::fs::read_to_string(&path)
.map_err(|e| format!("Failed to read cache store {}: {e}", path.display()))?;
let parsed: Value = serde_json::from_str(&contents).unwrap_or_else(|_| Value::Object(Map::new()));
let Some(root) = parsed.as_object() else {
return Ok(None);
};
Ok(root.get(&key).cloned())
}
#[tauri::command]
fn write_cache_entry(app: AppHandle, key: String, value: String) -> Result<(), String> {
let path = cache_file_path(&app)?;
let mut root: Map<String, Value> = if path.exists() {
let contents = std::fs::read_to_string(&path)
.map_err(|e| format!("Failed to read cache store {}: {e}", path.display()))?;
serde_json::from_str::<Value>(&contents)
.ok()
.and_then(|v| v.as_object().cloned())
.unwrap_or_default()
} else {
Map::new()
};
let parsed_value: Value = serde_json::from_str(&value)
.map_err(|e| format!("Invalid cache payload JSON: {e}"))?;
root.insert(key, parsed_value);
let serialized = serde_json::to_string_pretty(&Value::Object(root))
.map_err(|e| format!("Failed to serialize cache store: {e}"))?;
std::fs::write(&path, serialized)
.map_err(|e| format!("Failed to write cache store {}: {e}", path.display()))
}
fn logs_dir_path(app: &AppHandle) -> Result<PathBuf, String> {
let dir = app
.path()
.app_log_dir()
.map_err(|e| format!("Failed to resolve app log dir: {e}"))?;
fs::create_dir_all(&dir)
.map_err(|e| format!("Failed to create app log dir {}: {e}", dir.display()))?;
Ok(dir)
}
fn sidecar_log_path(app: &AppHandle) -> Result<PathBuf, String> {
Ok(logs_dir_path(app)?.join(LOCAL_API_LOG_FILE))
}
fn desktop_log_path(app: &AppHandle) -> Result<PathBuf, String> {
Ok(logs_dir_path(app)?.join(DESKTOP_LOG_FILE))
}
fn append_desktop_log(app: &AppHandle, level: &str, message: &str) {
let Ok(path) = desktop_log_path(app) else {
return;
};
let Ok(mut file) = OpenOptions::new().create(true).append(true).open(path) else {
return;
};
let timestamp = SystemTime::now()
.duration_since(UNIX_EPOCH)
.map(|d| d.as_secs())
.unwrap_or(0);
let _ = writeln!(file, "[{timestamp}][{level}] {message}");
}
fn open_in_shell(arg: &str) -> Result<(), String> {
#[cfg(target_os = "macos")]
let mut command = {
let mut cmd = Command::new("open");
cmd.arg(arg);
cmd
};
#[cfg(target_os = "windows")]
let mut command = {
let mut cmd = Command::new("explorer");
cmd.arg(arg);
cmd
};
#[cfg(all(unix, not(target_os = "macos")))]
let mut command = {
let mut cmd = Command::new("xdg-open");
cmd.arg(arg);
cmd
};
command
.spawn()
.map(|_| ())
.map_err(|e| format!("Failed to open {}: {e}", arg))
}
fn open_path_in_shell(path: &Path) -> Result<(), String> {
open_in_shell(&path.to_string_lossy())
}
#[tauri::command]
fn open_url(url: String) -> Result<(), String> {
if !url.starts_with("https://") && !url.starts_with("http://") {
return Err("Only http/https URLs are allowed".to_string());
}
open_in_shell(&url)
}
fn open_logs_folder_impl(app: &AppHandle) -> Result<PathBuf, String> {
let dir = logs_dir_path(app)?;
open_path_in_shell(&dir)?;
Ok(dir)
}
fn open_sidecar_log_impl(app: &AppHandle) -> Result<PathBuf, String> {
let log_path = sidecar_log_path(app)?;
if !log_path.exists() {
File::create(&log_path)
.map_err(|e| format!("Failed to create sidecar log {}: {e}", log_path.display()))?;
}
open_path_in_shell(&log_path)?;
Ok(log_path)
}
#[tauri::command]
fn open_logs_folder(app: AppHandle) -> Result<String, String> {
open_logs_folder_impl(&app).map(|path| path.display().to_string())
}
#[tauri::command]
fn open_sidecar_log_file(app: AppHandle) -> Result<String, String> {
open_sidecar_log_impl(&app).map(|path| path.display().to_string())
}
#[tauri::command]
async fn open_settings_window_command(app: AppHandle) -> Result<(), String> {
open_settings_window(&app)
}
#[tauri::command]
fn close_settings_window(app: AppHandle) -> Result<(), String> {
if let Some(window) = app.get_webview_window("settings") {
window.close().map_err(|e| format!("Failed to close settings window: {e}"))?;
}
Ok(())
}
/// Fetch JSON from Polymarket Gamma API using native TLS (bypasses Cloudflare JA3 blocking).
/// Called from frontend when browser CORS and sidecar Node.js TLS both fail.
#[tauri::command]
async fn fetch_polymarket(path: String, params: String) -> Result<String, String> {
let allowed = ["events", "markets", "tags"];
let segment = path.trim_start_matches('/');
if !allowed.iter().any(|a| segment.starts_with(a)) {
return Err("Invalid Polymarket path".into());
}
let url = format!("https://gamma-api.polymarket.com/{}?{}", segment, params);
let client = reqwest::Client::builder()
.use_native_tls()
.build()
.map_err(|e| format!("HTTP client error: {e}"))?;
let resp = client
.get(&url)
.header("Accept", "application/json")
.timeout(std::time::Duration::from_secs(10))
.send()
.await
.map_err(|e| format!("Polymarket fetch failed: {e}"))?;
if !resp.status().is_success() {
return Err(format!("Polymarket HTTP {}", resp.status()));
}
resp.text().await.map_err(|e| format!("Read body failed: {e}"))
}
fn open_settings_window(app: &AppHandle) -> Result<(), String> {
if let Some(window) = app.get_webview_window("settings") {
let _ = window.show();
window
.set_focus()
.map_err(|e| format!("Failed to focus settings window: {e}"))?;
return Ok(());
}
let _settings_window = WebviewWindowBuilder::new(app, "settings", WebviewUrl::App("settings.html".into()))
.title("World Monitor Settings")
.inner_size(980.0, 760.0)
.min_inner_size(820.0, 620.0)
.resizable(true)
.visible(false)
.build()
.map_err(|e| format!("Failed to create settings window: {e}"))?;
// On Windows/Linux, menus are per-window. Remove the inherited app menu
// from the settings window (macOS uses a shared app-wide menu bar instead).
#[cfg(not(target_os = "macos"))]
let _ = _settings_window.remove_menu();
Ok(())
}
fn build_app_menu(handle: &AppHandle) -> tauri::Result<Menu<tauri::Wry>> {
let settings_item = MenuItem::with_id(
handle,
MENU_FILE_SETTINGS_ID,
"Settings...",
true,
Some("CmdOrCtrl+,"),
)?;
let separator = PredefinedMenuItem::separator(handle)?;
let quit_item = PredefinedMenuItem::quit(handle, Some("Quit"))?;
let file_menu =
Submenu::with_items(handle, "File", true, &[&settings_item, &separator, &quit_item])?;
let about_metadata = AboutMetadata {
name: Some("World Monitor".into()),
version: Some(env!("CARGO_PKG_VERSION").into()),
copyright: Some("\u{00a9} 2025 Elie Habib".into()),
website: Some("https://worldmonitor.app".into()),
website_label: Some("worldmonitor.app".into()),
..Default::default()
};
let about_item = PredefinedMenuItem::about(handle, Some("About World Monitor"), Some(about_metadata))?;
let github_item = MenuItem::with_id(
handle,
MENU_HELP_GITHUB_ID,
"GitHub Repository",
true,
None::<&str>,
)?;
let devtools_item = MenuItem::with_id(
handle,
MENU_HELP_DEVTOOLS_ID,
"Toggle Developer Tools",
true,
Some("CmdOrCtrl+Alt+I"),
)?;
let help_separator = PredefinedMenuItem::separator(handle)?;
let help_menu = Submenu::with_items(
handle,
"Help",
true,
&[&about_item, &help_separator, &github_item, &devtools_item],
)?;
Menu::with_items(handle, &[&file_menu, &help_menu])
}
fn handle_menu_event(app: &AppHandle, event: tauri::menu::MenuEvent) {
match event.id().as_ref() {
MENU_FILE_SETTINGS_ID => {
if let Err(err) = open_settings_window(app) {
append_desktop_log(app, "ERROR", &format!("settings menu failed: {err}"));
eprintln!("[tauri] settings menu failed: {err}");
}
}
MENU_HELP_GITHUB_ID => {
let _ = open_in_shell("https://github.com/koala73/worldmonitor");
}
MENU_HELP_DEVTOOLS_ID => {
if let Some(window) = app.get_webview_window("main") {
if window.is_devtools_open() {
window.close_devtools();
} else {
window.open_devtools();
}
}
}
_ => {}
}
}
fn local_api_paths(app: &AppHandle) -> (PathBuf, PathBuf) {
let resource_dir = app
.path()
.resource_dir()
.unwrap_or_else(|_| PathBuf::from("."));
let sidecar_script = if cfg!(debug_assertions) {
PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("sidecar/local-api-server.mjs")
} else {
resource_dir.join("sidecar/local-api-server.mjs")
};
let api_dir_root = if cfg!(debug_assertions) {
PathBuf::from(env!("CARGO_MANIFEST_DIR"))
.parent()
.map(PathBuf::from)
.unwrap_or_else(|| PathBuf::from("."))
} else {
let direct_api = resource_dir.join("api");
let lifted_root = resource_dir.join("_up_");
let lifted_api = lifted_root.join("api");
if direct_api.exists() {
resource_dir
} else if lifted_api.exists() {
lifted_root
} else {
resource_dir
}
};
(sidecar_script, api_dir_root)
}
fn resolve_node_binary() -> Option<PathBuf> {
if let Ok(explicit) = env::var("LOCAL_API_NODE_BIN") {
let explicit_path = PathBuf::from(explicit);
if explicit_path.exists() {
return Some(explicit_path);
}
}
let node_name = if cfg!(windows) { "node.exe" } else { "node" };
if let Some(path_var) = env::var_os("PATH") {
for dir in env::split_paths(&path_var) {
let candidate = dir.join(node_name);
if candidate.exists() {
return Some(candidate);
}
}
}
let common_locations = if cfg!(windows) {
vec![
PathBuf::from(r"C:\Program Files\nodejs\node.exe"),
PathBuf::from(r"C:\Program Files (x86)\nodejs\node.exe"),
]
} else {
vec![
PathBuf::from("/opt/homebrew/bin/node"),
PathBuf::from("/usr/local/bin/node"),
PathBuf::from("/usr/bin/node"),
PathBuf::from("/opt/local/bin/node"),
]
};
common_locations.into_iter().find(|path| path.exists())
}
fn start_local_api(app: &AppHandle) -> Result<(), String> {
let state = app.state::<LocalApiState>();
let mut slot = state
.child
.lock()
.map_err(|_| "Failed to lock local API state".to_string())?;
if slot.is_some() {
return Ok(());
}
let (script, resource_root) = local_api_paths(app);
if !script.exists() {
return Err(format!(
"Local API sidecar script missing at {}",
script.display()
));
}
let node_binary = resolve_node_binary().ok_or_else(|| {
"Node.js executable not found. Install Node 18+ or set LOCAL_API_NODE_BIN".to_string()
})?;
let log_path = sidecar_log_path(app)?;
let log_file = OpenOptions::new()
.create(true)
.append(true)
.open(&log_path)
.map_err(|e| format!("Failed to open local API log {}: {e}", log_path.display()))?;
let log_file_err = log_file
.try_clone()
.map_err(|e| format!("Failed to clone local API log handle: {e}"))?;
append_desktop_log(
app,
"INFO",
&format!(
"starting local API sidecar script={} resource_root={} log={}",
script.display(),
resource_root.display(),
log_path.display()
),
);
append_desktop_log(app, "INFO", &format!("resolved node binary={}", node_binary.display()));
// Generate a unique token for local API auth (prevents other local processes from accessing sidecar)
let mut token_slot = state.token.lock().map_err(|_| "Failed to lock token slot")?;
if token_slot.is_none() {
*token_slot = Some(generate_local_token());
}
let local_api_token = token_slot.clone().unwrap();
drop(token_slot);
let mut cmd = Command::new(&node_binary);
cmd.arg(&script)
.env("LOCAL_API_PORT", LOCAL_API_PORT)
.env("LOCAL_API_RESOURCE_DIR", resource_root)
.env("LOCAL_API_MODE", "tauri-sidecar")
.env("LOCAL_API_TOKEN", &local_api_token)
.stdout(Stdio::from(log_file))
.stderr(Stdio::from(log_file_err));
// Pass keychain secrets to sidecar as env vars
let mut secret_count = 0u32;
for key in SUPPORTED_SECRET_KEYS.iter() {
if let Ok(entry) = Entry::new(KEYRING_SERVICE, key) {
if let Ok(value) = entry.get_password() {
if !value.trim().is_empty() {
cmd.env(key, value.trim());
secret_count += 1;
}
}
}
}
append_desktop_log(app, "INFO", &format!("injected {secret_count} keychain secrets into sidecar env"));
let child = cmd
.spawn()
.map_err(|e| format!("Failed to launch local API: {e}"))?;
append_desktop_log(app, "INFO", &format!("local API sidecar started pid={}", child.id()));
*slot = Some(child);
Ok(())
}
fn stop_local_api(app: &AppHandle) {
if let Ok(state) = app.try_state::<LocalApiState>().ok_or(()) {
if let Ok(mut slot) = state.child.lock() {
if let Some(mut child) = slot.take() {
let _ = child.kill();
append_desktop_log(app, "INFO", "local API sidecar stopped");
}
}
}
}
fn main() {
tauri::Builder::default()
.menu(build_app_menu)
.on_menu_event(handle_menu_event)
.manage(LocalApiState::default())
.invoke_handler(tauri::generate_handler![
list_supported_secret_keys,
get_secret,
set_secret,
delete_secret,
get_local_api_token,
read_cache_entry,
write_cache_entry,
open_logs_folder,
open_sidecar_log_file,
open_settings_window_command,
close_settings_window,
open_url,
fetch_polymarket
])
.setup(|app| {
if let Err(err) = start_local_api(&app.handle()) {
append_desktop_log(
&app.handle(),
"ERROR",
&format!("local API sidecar failed to start: {err}"),
);
eprintln!("[tauri] local API sidecar failed to start: {err}");
}
Ok(())
})
.build(tauri::generate_context!())
.expect("error while running world-monitor tauri application")
.run(|app, event| {
if matches!(event, RunEvent::ExitRequested { .. } | RunEvent::Exit) {
stop_local_api(&app);
}
});
}