use std::{ collections::HashMap, sync::{Arc, Mutex}, time::Duration, }; #[cfg(not(any(target_os = "ios")))] use crate::Connection; use hbb_common::{ config::{Config, LocalConfig}, tokio::{self, sync::broadcast, time::Instant}, }; use serde::{Deserialize, Serialize}; use serde_json::{json, Value}; const TIME_HEARTBEAT: Duration = Duration::from_secs(15); const UPLOAD_SYSINFO_TIMEOUT: Duration = Duration::from_secs(120); const TIME_CONN: Duration = Duration::from_secs(3); #[cfg(not(any(target_os = "ios")))] lazy_static::lazy_static! { static ref SENDER : Mutex>> = Mutex::new(start_hbbs_sync()); static ref PRO: Arc> = Default::default(); } #[cfg(not(any(target_os = "ios")))] pub fn start() { let _sender = SENDER.lock().unwrap(); } #[cfg(not(target_os = "ios"))] pub fn signal_receiver() -> broadcast::Receiver> { SENDER.lock().unwrap().subscribe() } #[cfg(not(any(target_os = "ios")))] fn start_hbbs_sync() -> broadcast::Sender> { let (tx, _rx) = broadcast::channel::>(16); std::thread::spawn(move || start_hbbs_sync_async()); return tx; } #[derive(Debug, Serialize, Deserialize)] pub struct StrategyOptions { #[serde(default, skip_serializing_if = "HashMap::is_empty")] pub config_options: HashMap, #[serde(default, skip_serializing_if = "HashMap::is_empty")] pub extra: HashMap, } #[cfg(not(any(target_os = "ios")))] #[tokio::main(flavor = "current_thread")] async fn start_hbbs_sync_async() { let mut interval = tokio::time::interval_at(Instant::now() + TIME_CONN, TIME_CONN); let mut last_sent: Option = None; let mut info_uploaded: (bool, String, Option) = (false, "".to_owned(), None); loop { tokio::select! { _ = interval.tick() => { let url = heartbeat_url(); if url.is_empty() { *PRO.lock().unwrap() = false; continue; } if !Config::get_option("stop-service").is_empty() { continue; } let conns = Connection::alive_conns(); if info_uploaded.0 && url != info_uploaded.1 { info_uploaded.0 = false; *PRO.lock().unwrap() = false; } if !info_uploaded.0 && info_uploaded.2.map(|x| x.elapsed() >= UPLOAD_SYSINFO_TIMEOUT).unwrap_or(true){ let mut v = crate::get_sysinfo(); v["version"] = json!(crate::VERSION); v["id"] = json!(Config::get_id()); v["uuid"] = json!(crate::encode64(hbb_common::get_uuid())); match crate::post_request(url.replace("heartbeat", "sysinfo"), v.to_string(), "").await { Ok(x) => { if x == "SYSINFO_UPDATED" { info_uploaded = (true, url.clone(), None); hbb_common::log::info!("sysinfo updated"); *PRO.lock().unwrap() = true; } else if x == "ID_NOT_FOUND" { info_uploaded.2 = None; // next heartbeat will upload sysinfo again } else { info_uploaded.2 = Some(Instant::now()); } } _ => { info_uploaded.2 = Some(Instant::now()); } } } if conns.is_empty() && last_sent.map(|x| x.elapsed() < TIME_HEARTBEAT).unwrap_or(false){ continue; } last_sent = Some(Instant::now()); let mut v = Value::default(); v["id"] = json!(Config::get_id()); v["uuid"] = json!(crate::encode64(hbb_common::get_uuid())); v["ver"] = json!(hbb_common::get_version_number(crate::VERSION)); if !conns.is_empty() { v["conns"] = json!(conns); } let modified_at = LocalConfig::get_option("strategy_timestamp").parse::().unwrap_or(0); v["modified_at"] = json!(modified_at); if let Ok(s) = crate::post_request(url.clone(), v.to_string(), "").await { if let Ok(mut rsp) = serde_json::from_str::>(&s) { if let Some(conns) = rsp.remove("disconnect") { if let Ok(conns) = serde_json::from_value::>(conns) { SENDER.lock().unwrap().send(conns).ok(); } } if let Some(rsp_modified_at) = rsp.remove("modified_at") { if let Ok(rsp_modified_at) = serde_json::from_value::(rsp_modified_at) { if rsp_modified_at != modified_at { LocalConfig::set_option("strategy_timestamp".to_string(), rsp_modified_at.to_string()); } } } if let Some(strategy) = rsp.remove("strategy") { if let Ok(strategy) = serde_json::from_value::(strategy) { handle_config_options(strategy.config_options); } } } } } } } } fn heartbeat_url() -> String { let url = crate::common::get_api_server( Config::get_option("api-server"), Config::get_option("custom-rendezvous-server"), ); if url.is_empty() || url.contains("rustdesk.com") { return "".to_owned(); } format!("{}/api/heartbeat", url) } fn handle_config_options(config_options: HashMap) { let mut options = Config::get_options(); config_options .iter() .map(|(k, v)| { if k == "allow-share-rdp" { // only changes made after installation take effect. #[cfg(windows)] if crate::ui_interface::is_rdp_service_open() { let current = crate::ui_interface::is_share_rdp(); let set = v == "Y"; if current != set { crate::platform::windows::set_share_rdp(set); } } } else { if v.is_empty() { options.remove(k); } else { options.insert(k.to_string(), v.to_string()); } } }) .count(); Config::set_options(options); } #[allow(unused)] #[cfg(not(any(target_os = "ios")))] pub fn is_pro() -> bool { PRO.lock().unwrap().clone() }