629 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			629 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| use std::{
 | |
|     collections::HashMap,
 | |
|     net::SocketAddr,
 | |
|     sync::{Arc, Mutex, RwLock, Weak},
 | |
|     time::Duration,
 | |
| };
 | |
| 
 | |
| use bytes::Bytes;
 | |
| 
 | |
| pub use connection::*;
 | |
| #[cfg(not(any(target_os = "android", target_os = "ios")))]
 | |
| use hbb_common::config::Config2;
 | |
| use hbb_common::tcp::{self, new_listener};
 | |
| use hbb_common::{
 | |
|     allow_err,
 | |
|     anyhow::Context,
 | |
|     bail,
 | |
|     config::{Config, CONNECT_TIMEOUT, RELAY_PORT},
 | |
|     log,
 | |
|     message_proto::*,
 | |
|     protobuf::{Enum, Message as _},
 | |
|     rendezvous_proto::*,
 | |
|     socket_client,
 | |
|     sodiumoxide::crypto::{box_, sign},
 | |
|     timeout, tokio, ResultType, Stream,
 | |
| };
 | |
| #[cfg(not(any(target_os = "android", target_os = "ios")))]
 | |
| use service::ServiceTmpl;
 | |
| use service::{EmptyExtraFieldService, GenericService, Service, Subscriber};
 | |
| 
 | |
| use crate::ipc::Data;
 | |
| 
 | |
| pub mod audio_service;
 | |
| cfg_if::cfg_if! {
 | |
| if #[cfg(not(any(target_os = "android", target_os = "ios")))] {
 | |
| mod clipboard_service;
 | |
| #[cfg(target_os = "linux")]
 | |
| pub(crate) mod wayland;
 | |
| #[cfg(target_os = "linux")]
 | |
| pub mod uinput;
 | |
| #[cfg(target_os = "linux")]
 | |
| pub mod rdp_input;
 | |
| #[cfg(target_os = "linux")]
 | |
| pub mod dbus;
 | |
| pub mod input_service;
 | |
| } else {
 | |
| mod clipboard_service {
 | |
| pub const NAME: &'static str = "";
 | |
| }
 | |
| pub mod input_service {
 | |
| pub const NAME_CURSOR: &'static str = "";
 | |
| pub const NAME_POS: &'static str = "";
 | |
| pub const NAME_WINDOW_FOCUS: &'static str = "";
 | |
| }
 | |
| }
 | |
| }
 | |
| 
 | |
| mod connection;
 | |
| pub mod display_service;
 | |
| #[cfg(windows)]
 | |
| pub mod portable_service;
 | |
| mod service;
 | |
| mod video_qos;
 | |
| pub mod video_service;
 | |
| 
 | |
| pub type Childs = Arc<Mutex<Vec<std::process::Child>>>;
 | |
| type ConnMap = HashMap<i32, ConnInner>;
 | |
| 
 | |
| #[cfg(any(target_os = "macos", target_os = "linux"))]
 | |
| const CONFIG_SYNC_INTERVAL_SECS: f32 = 0.3;
 | |
| 
 | |
| lazy_static::lazy_static! {
 | |
|     pub static ref CHILD_PROCESS: Childs = Default::default();
 | |
|     pub static ref CONN_COUNT: Arc<Mutex<usize>> = Default::default();
 | |
|     // A client server used to provide local services(audio, video, clipboard, etc.)
 | |
|     // for all initiative connections.
 | |
|     //
 | |
|     // [Note]
 | |
|     // ugly
 | |
|     // Now we use this [`CLIENT_SERVER`] to do following operations:
 | |
|     // - record local audio, and send to remote
 | |
|     pub static ref CLIENT_SERVER: ServerPtr = new();
 | |
| }
 | |
| 
 | |
| pub struct Server {
 | |
|     connections: ConnMap,
 | |
|     services: HashMap<String, Box<dyn Service>>,
 | |
|     id_count: i32,
 | |
| }
 | |
| 
 | |
| pub type ServerPtr = Arc<RwLock<Server>>;
 | |
| pub type ServerPtrWeak = Weak<RwLock<Server>>;
 | |
| 
 | |
| pub fn new() -> ServerPtr {
 | |
|     let mut server = Server {
 | |
|         connections: HashMap::new(),
 | |
|         services: HashMap::new(),
 | |
|         id_count: hbb_common::rand::random::<i32>() % 1000 + 1000, // ensure positive
 | |
|     };
 | |
|     server.add_service(Box::new(audio_service::new()));
 | |
|     #[cfg(not(target_os = "ios"))]
 | |
|     server.add_service(Box::new(display_service::new()));
 | |
|     #[cfg(not(any(target_os = "android", target_os = "ios")))]
 | |
|     {
 | |
|         server.add_service(Box::new(clipboard_service::new()));
 | |
|         if !display_service::capture_cursor_embedded() {
 | |
|             server.add_service(Box::new(input_service::new_cursor()));
 | |
|             server.add_service(Box::new(input_service::new_pos()));
 | |
|             server.add_service(Box::new(input_service::new_window_focus()));
 | |
|         }
 | |
|     }
 | |
|     Arc::new(RwLock::new(server))
 | |
| }
 | |
| 
 | |
| async fn accept_connection_(server: ServerPtr, socket: Stream, secure: bool) -> ResultType<()> {
 | |
|     let local_addr = socket.local_addr();
 | |
|     drop(socket);
 | |
|     // even we drop socket, below still may fail if not use reuse_addr,
 | |
|     // there is TIME_WAIT before socket really released, so sometimes we
 | |
|     // see “Only one usage of each socket address is normally permitted” on windows sometimes,
 | |
|     let listener = new_listener(local_addr, true).await?;
 | |
|     log::info!("Server listening on: {}", &listener.local_addr()?);
 | |
|     if let Ok((stream, addr)) = timeout(CONNECT_TIMEOUT, listener.accept()).await? {
 | |
|         stream.set_nodelay(true).ok();
 | |
|         let stream_addr = stream.local_addr()?;
 | |
|         create_tcp_connection(server, Stream::from(stream, stream_addr), addr, secure).await?;
 | |
|     }
 | |
|     Ok(())
 | |
| }
 | |
| 
 | |
| pub async fn create_tcp_connection(
 | |
|     server: ServerPtr,
 | |
|     stream: Stream,
 | |
|     addr: SocketAddr,
 | |
|     secure: bool,
 | |
| ) -> ResultType<()> {
 | |
|     let mut stream = stream;
 | |
|     let id = server.write().unwrap().get_new_id();
 | |
|     let (sk, pk) = Config::get_key_pair();
 | |
|     if secure && pk.len() == sign::PUBLICKEYBYTES && sk.len() == sign::SECRETKEYBYTES {
 | |
|         let mut sk_ = [0u8; sign::SECRETKEYBYTES];
 | |
|         sk_[..].copy_from_slice(&sk);
 | |
|         let sk = sign::SecretKey(sk_);
 | |
|         let mut msg_out = Message::new();
 | |
|         let (our_pk_b, our_sk_b) = box_::gen_keypair();
 | |
|         msg_out.set_signed_id(SignedId {
 | |
|             id: sign::sign(
 | |
|                 &IdPk {
 | |
|                     id: Config::get_id(),
 | |
|                     pk: Bytes::from(our_pk_b.0.to_vec()),
 | |
|                     ..Default::default()
 | |
|                 }
 | |
|                 .write_to_bytes()
 | |
|                 .unwrap_or_default(),
 | |
|                 &sk,
 | |
|             )
 | |
|             .into(),
 | |
|             ..Default::default()
 | |
|         });
 | |
|         timeout(CONNECT_TIMEOUT, stream.send(&msg_out)).await??;
 | |
|         match timeout(CONNECT_TIMEOUT, stream.next()).await? {
 | |
|             Some(res) => {
 | |
|                 let bytes = res?;
 | |
|                 if let Ok(msg_in) = Message::parse_from_bytes(&bytes) {
 | |
|                     if let Some(message::Union::PublicKey(pk)) = msg_in.union {
 | |
|                         if pk.asymmetric_value.len() == box_::PUBLICKEYBYTES {
 | |
|                             stream.set_key(tcp::Encrypt::decode(
 | |
|                                 &pk.symmetric_value,
 | |
|                                 &pk.asymmetric_value,
 | |
|                                 &our_sk_b,
 | |
|                             )?);
 | |
|                         } else if pk.asymmetric_value.is_empty() {
 | |
|                             Config::set_key_confirmed(false);
 | |
|                             log::info!("Force to update pk");
 | |
|                         } else {
 | |
|                             bail!("Handshake failed: invalid public sign key length from peer");
 | |
|                         }
 | |
|                     } else {
 | |
|                         log::error!("Handshake failed: invalid message type");
 | |
|                     }
 | |
|                 } else {
 | |
|                     bail!("Handshake failed: invalid message format");
 | |
|                 }
 | |
|             }
 | |
|             None => {
 | |
|                 bail!("Failed to receive public key");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[cfg(target_os = "macos")]
 | |
|     {
 | |
|         use std::process::Command;
 | |
|         Command::new("/usr/bin/caffeinate")
 | |
|             .arg("-u")
 | |
|             .arg("-t 5")
 | |
|             .spawn()
 | |
|             .ok();
 | |
|         log::info!("wake up macos");
 | |
|     }
 | |
|     Connection::start(addr, stream, id, Arc::downgrade(&server)).await;
 | |
|     Ok(())
 | |
| }
 | |
| 
 | |
| pub async fn accept_connection(
 | |
|     server: ServerPtr,
 | |
|     socket: Stream,
 | |
|     peer_addr: SocketAddr,
 | |
|     secure: bool,
 | |
| ) {
 | |
|     if let Err(err) = accept_connection_(server, socket, secure).await {
 | |
|         log::error!("Failed to accept connection from {}: {}", peer_addr, err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| pub async fn create_relay_connection(
 | |
|     server: ServerPtr,
 | |
|     relay_server: String,
 | |
|     uuid: String,
 | |
|     peer_addr: SocketAddr,
 | |
|     secure: bool,
 | |
|     ipv4: bool,
 | |
| ) {
 | |
|     if let Err(err) =
 | |
|         create_relay_connection_(server, relay_server, uuid.clone(), peer_addr, secure, ipv4).await
 | |
|     {
 | |
|         log::error!(
 | |
|             "Failed to create relay connection for {} with uuid {}: {}",
 | |
|             peer_addr,
 | |
|             uuid,
 | |
|             err
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| async fn create_relay_connection_(
 | |
|     server: ServerPtr,
 | |
|     relay_server: String,
 | |
|     uuid: String,
 | |
|     peer_addr: SocketAddr,
 | |
|     secure: bool,
 | |
|     ipv4: bool,
 | |
| ) -> ResultType<()> {
 | |
|     let mut stream = socket_client::connect_tcp(
 | |
|         socket_client::ipv4_to_ipv6(crate::check_port(relay_server, RELAY_PORT), ipv4),
 | |
|         CONNECT_TIMEOUT,
 | |
|     )
 | |
|     .await?;
 | |
|     let mut msg_out = RendezvousMessage::new();
 | |
|     let licence_key = crate::get_key(true).await;
 | |
|     msg_out.set_request_relay(RequestRelay {
 | |
|         licence_key,
 | |
|         uuid,
 | |
|         ..Default::default()
 | |
|     });
 | |
|     stream.send(&msg_out).await?;
 | |
|     create_tcp_connection(server, stream, peer_addr, secure).await?;
 | |
|     Ok(())
 | |
| }
 | |
| 
 | |
| impl Server {
 | |
|     fn is_video_service_name(name: &str) -> bool {
 | |
|         name.starts_with(video_service::NAME)
 | |
|     }
 | |
| 
 | |
|     pub fn try_add_primay_video_service(&mut self) {
 | |
|         let primary_video_service_name =
 | |
|             video_service::get_service_name(*display_service::PRIMARY_DISPLAY_IDX);
 | |
|         if !self.contains(&primary_video_service_name) {
 | |
|             self.add_service(Box::new(video_service::new(
 | |
|                 *display_service::PRIMARY_DISPLAY_IDX,
 | |
|             )));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     pub fn add_connection(&mut self, conn: ConnInner, noperms: &Vec<&'static str>) {
 | |
|         let primary_video_service_name =
 | |
|             video_service::get_service_name(*display_service::PRIMARY_DISPLAY_IDX);
 | |
|         for s in self.services.values() {
 | |
|             let name = s.name();
 | |
|             if Self::is_video_service_name(&name) && name != primary_video_service_name {
 | |
|                 continue;
 | |
|             }
 | |
|             if !noperms.contains(&(&name as _)) {
 | |
|                 s.on_subscribe(conn.clone());
 | |
|             }
 | |
|         }
 | |
|         #[cfg(target_os = "macos")]
 | |
|         self.update_enable_retina();
 | |
|         self.connections.insert(conn.id(), conn);
 | |
|         *CONN_COUNT.lock().unwrap() = self.connections.len();
 | |
|     }
 | |
| 
 | |
|     pub fn remove_connection(&mut self, conn: &ConnInner) {
 | |
|         for s in self.services.values() {
 | |
|             s.on_unsubscribe(conn.id());
 | |
|         }
 | |
|         self.connections.remove(&conn.id());
 | |
|         *CONN_COUNT.lock().unwrap() = self.connections.len();
 | |
|         #[cfg(target_os = "macos")]
 | |
|         self.update_enable_retina();
 | |
|     }
 | |
| 
 | |
|     pub fn close_connections(&mut self) {
 | |
|         let conn_inners: Vec<_> = self.connections.values_mut().collect();
 | |
|         for c in conn_inners {
 | |
|             let mut misc = Misc::new();
 | |
|             misc.set_stop_service(true);
 | |
|             let mut msg = Message::new();
 | |
|             msg.set_misc(misc);
 | |
|             c.send(Arc::new(msg));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn add_service(&mut self, service: Box<dyn Service>) {
 | |
|         let name = service.name();
 | |
|         self.services.insert(name, service);
 | |
|     }
 | |
| 
 | |
|     pub fn contains(&self, name: &str) -> bool {
 | |
|         self.services.contains_key(name)
 | |
|     }
 | |
| 
 | |
|     pub fn subscribe(&mut self, name: &str, conn: ConnInner, sub: bool) {
 | |
|         if let Some(s) = self.services.get(name) {
 | |
|             if s.is_subed(conn.id()) == sub {
 | |
|                 return;
 | |
|             }
 | |
|             if sub {
 | |
|                 s.on_subscribe(conn.clone());
 | |
|             } else {
 | |
|                 s.on_unsubscribe(conn.id());
 | |
|             }
 | |
|             #[cfg(target_os = "macos")]
 | |
|             self.update_enable_retina();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // get a new unique id
 | |
|     pub fn get_new_id(&mut self) -> i32 {
 | |
|         self.id_count += 1;
 | |
|         self.id_count
 | |
|     }
 | |
| 
 | |
|     pub fn set_video_service_opt(&self, display: Option<usize>, opt: &str, value: &str) {
 | |
|         for (k, v) in self.services.iter() {
 | |
|             if let Some(display) = display {
 | |
|                 if k != &video_service::get_service_name(display) {
 | |
|                     continue;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if Self::is_video_service_name(k) {
 | |
|                 v.set_option(opt, value);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     fn get_subbed_displays_count(&self, conn_id: i32) -> usize {
 | |
|         self.services
 | |
|             .keys()
 | |
|             .filter(|k| {
 | |
|                 Self::is_video_service_name(k) && self.services.get(*k).unwrap().is_subed(conn_id)
 | |
|             })
 | |
|             .count()
 | |
|     }
 | |
| 
 | |
|     fn capture_displays(
 | |
|         &mut self,
 | |
|         conn: ConnInner,
 | |
|         displays: &[usize],
 | |
|         include: bool,
 | |
|         exclude: bool,
 | |
|     ) {
 | |
|         let displays = displays
 | |
|             .iter()
 | |
|             .map(|d| video_service::get_service_name(*d))
 | |
|             .collect::<Vec<_>>();
 | |
|         let keys = self.services.keys().cloned().collect::<Vec<_>>();
 | |
|         for name in keys.iter() {
 | |
|             if Self::is_video_service_name(&name) {
 | |
|                 if displays.contains(&name) {
 | |
|                     if include {
 | |
|                         self.subscribe(&name, conn.clone(), true);
 | |
|                     }
 | |
|                 } else {
 | |
|                     if exclude {
 | |
|                         self.subscribe(&name, conn.clone(), false);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[cfg(target_os = "macos")]
 | |
|     fn update_enable_retina(&self) {
 | |
|         let mut video_service_count = 0;
 | |
|         for (name, service) in self.services.iter() {
 | |
|             if Self::is_video_service_name(&name) && service.ok() {
 | |
|                 video_service_count += 1;
 | |
|             }
 | |
|         }
 | |
|         *scrap::quartz::ENABLE_RETINA.lock().unwrap() = video_service_count < 2;
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl Drop for Server {
 | |
|     fn drop(&mut self) {
 | |
|         for s in self.services.values() {
 | |
|             s.join();
 | |
|         }
 | |
|         #[cfg(target_os = "linux")]
 | |
|         wayland::clear();
 | |
|     }
 | |
| }
 | |
| 
 | |
| pub fn check_zombie() {
 | |
|     std::thread::spawn(|| loop {
 | |
|         let mut lock = CHILD_PROCESS.lock().unwrap();
 | |
|         let mut i = 0;
 | |
|         while i != lock.len() {
 | |
|             let c = &mut (*lock)[i];
 | |
|             if let Ok(Some(_)) = c.try_wait() {
 | |
|                 lock.remove(i);
 | |
|             } else {
 | |
|                 i += 1;
 | |
|             }
 | |
|         }
 | |
|         drop(lock);
 | |
|         std::thread::sleep(Duration::from_millis(100));
 | |
|     });
 | |
| }
 | |
| 
 | |
| /// Start the host server that allows the remote peer to control the current machine.
 | |
| ///
 | |
| /// # Arguments
 | |
| ///
 | |
| /// * `is_server` - Whether the current client is definitely the server.
 | |
| /// If true, the server will be started.
 | |
| /// Otherwise, client will check if there's already a server and start one if not.
 | |
| #[cfg(any(target_os = "android", target_os = "ios"))]
 | |
| #[tokio::main]
 | |
| pub async fn start_server(_is_server: bool) {
 | |
|     crate::RendezvousMediator::start_all().await;
 | |
| }
 | |
| 
 | |
| /// Start the host server that allows the remote peer to control the current machine.
 | |
| ///
 | |
| /// # Arguments
 | |
| ///
 | |
| /// * `is_server` - Whether the current client is definitely the server.
 | |
| /// If true, the server will be started.
 | |
| /// Otherwise, client will check if there's already a server and start one if not.
 | |
| #[cfg(not(any(target_os = "android", target_os = "ios")))]
 | |
| #[tokio::main]
 | |
| pub async fn start_server(is_server: bool) {
 | |
|     #[cfg(target_os = "linux")]
 | |
|     {
 | |
|         log::info!("DISPLAY={:?}", std::env::var("DISPLAY"));
 | |
|         log::info!("XAUTHORITY={:?}", std::env::var("XAUTHORITY"));
 | |
|     }
 | |
|     #[cfg(feature = "hwcodec")]
 | |
|     #[cfg(any(target_os = "windows", target_os = "linux"))]
 | |
|     scrap::hwcodec::start_check_process(false);
 | |
|     #[cfg(windows)]
 | |
|     hbb_common::platform::windows::start_cpu_performance_monitor();
 | |
| 
 | |
|     if is_server {
 | |
|         crate::common::set_server_running(true);
 | |
|         std::thread::spawn(move || {
 | |
|             if let Err(err) = crate::ipc::start("") {
 | |
|                 log::error!("Failed to start ipc: {}", err);
 | |
|                 std::process::exit(-1);
 | |
|             }
 | |
|         });
 | |
|         input_service::fix_key_down_timeout_loop();
 | |
|         #[cfg(target_os = "linux")]
 | |
|         if crate::platform::current_is_wayland() {
 | |
|             allow_err!(input_service::setup_uinput(0, 1920, 0, 1080).await);
 | |
|         }
 | |
|         #[cfg(any(target_os = "macos", target_os = "linux"))]
 | |
|         tokio::spawn(async { sync_and_watch_config_dir().await });
 | |
|         #[cfg(target_os = "windows")]
 | |
|         crate::platform::try_kill_broker();
 | |
|         crate::RendezvousMediator::start_all().await;
 | |
|     } else {
 | |
|         match crate::ipc::connect(1000, "").await {
 | |
|             Ok(mut conn) => {
 | |
|                 if conn.send(&Data::SyncConfig(None)).await.is_ok() {
 | |
|                     if let Ok(Some(data)) = conn.next_timeout(1000).await {
 | |
|                         match data {
 | |
|                             Data::SyncConfig(Some(configs)) => {
 | |
|                                 let (config, config2) = *configs;
 | |
|                                 if Config::set(config) {
 | |
|                                     log::info!("config synced");
 | |
|                                 }
 | |
|                                 if Config2::set(config2) {
 | |
|                                     log::info!("config2 synced");
 | |
|                                 }
 | |
|                             }
 | |
|                             _ => {}
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             Err(err) => {
 | |
|                 log::info!("server not started (will try to start): {}", err);
 | |
|                 std::thread::spawn(|| start_server(true));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[cfg(target_os = "macos")]
 | |
| #[tokio::main(flavor = "current_thread")]
 | |
| pub async fn start_ipc_url_server() {
 | |
|     log::debug!("Start an ipc server for listening to url schemes");
 | |
|     match crate::ipc::new_listener("_url").await {
 | |
|         Ok(mut incoming) => {
 | |
|             while let Some(Ok(conn)) = incoming.next().await {
 | |
|                 let mut conn = crate::ipc::Connection::new(conn);
 | |
|                 match conn.next_timeout(1000).await {
 | |
|                     Ok(Some(data)) => match data {
 | |
|                         #[cfg(feature = "flutter")]
 | |
|                         Data::UrlLink(url) => {
 | |
|                             let mut m = HashMap::new();
 | |
|                             m.insert("name", "on_url_scheme_received");
 | |
|                             m.insert("url", url.as_str());
 | |
|                             let event = serde_json::to_string(&m).unwrap_or("".to_owned());
 | |
|                             match crate::flutter::push_global_event(
 | |
|                                 crate::flutter::APP_TYPE_MAIN,
 | |
|                                 event,
 | |
|                             ) {
 | |
|                                 None => log::warn!("No main window app found!"),
 | |
|                                 Some(..) => {}
 | |
|                             }
 | |
|                         }
 | |
|                         _ => {
 | |
|                             log::warn!("An unexpected data was sent to the ipc url server.")
 | |
|                         }
 | |
|                     },
 | |
|                     Err(err) => {
 | |
|                         log::error!("{}", err);
 | |
|                     }
 | |
|                     _ => {}
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         Err(err) => {
 | |
|             log::error!("{}", err);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[cfg(any(target_os = "macos", target_os = "linux"))]
 | |
| async fn sync_and_watch_config_dir() {
 | |
|     if crate::platform::is_root() {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     let mut cfg0 = (Config::get(), Config2::get());
 | |
|     let mut synced = false;
 | |
|     let tries = if crate::is_server() { 30 } else { 3 };
 | |
|     log::debug!("#tries of ipc service connection: {}", tries);
 | |
|     use hbb_common::sleep;
 | |
|     for i in 1..=tries {
 | |
|         sleep(i as f32 * CONFIG_SYNC_INTERVAL_SECS).await;
 | |
|         match crate::ipc::connect(1000, "_service").await {
 | |
|             Ok(mut conn) => {
 | |
|                 if !synced {
 | |
|                     if conn.send(&Data::SyncConfig(None)).await.is_ok() {
 | |
|                         if let Ok(Some(data)) = conn.next_timeout(1000).await {
 | |
|                             match data {
 | |
|                                 Data::SyncConfig(Some(configs)) => {
 | |
|                                     let (config, config2) = *configs;
 | |
|                                     let _chk = crate::ipc::CheckIfRestart::new();
 | |
|                                     if !config.is_empty() {
 | |
|                                         if cfg0.0 != config {
 | |
|                                             cfg0.0 = config.clone();
 | |
|                                             Config::set(config);
 | |
|                                             log::info!("sync config from root");
 | |
|                                         }
 | |
|                                         if cfg0.1 != config2 {
 | |
|                                             cfg0.1 = config2.clone();
 | |
|                                             Config2::set(config2);
 | |
|                                             log::info!("sync config2 from root");
 | |
|                                         }
 | |
|                                     }
 | |
|                                     synced = true;
 | |
|                                 }
 | |
|                                 _ => {}
 | |
|                             };
 | |
|                         };
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 loop {
 | |
|                     sleep(CONFIG_SYNC_INTERVAL_SECS).await;
 | |
|                     let cfg = (Config::get(), Config2::get());
 | |
|                     if cfg != cfg0 {
 | |
|                         log::info!("config updated, sync to root");
 | |
|                         match conn.send(&Data::SyncConfig(Some(cfg.clone().into()))).await {
 | |
|                             Err(e) => {
 | |
|                                 log::error!("sync config to root failed: {}", e);
 | |
|                                 match crate::ipc::connect(1000, "_service").await {
 | |
|                                     Ok(mut _conn) => {
 | |
|                                         conn = _conn;
 | |
|                                         log::info!("reconnected to ipc_service");
 | |
|                                         break;
 | |
|                                     }
 | |
|                                     _ => {}
 | |
|                                 }
 | |
|                             }
 | |
|                             _ => {
 | |
|                                 cfg0 = cfg;
 | |
|                                 conn.next_timeout(1000).await.ok();
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             Err(_) => {
 | |
|                 log::info!("#{} try: failed to connect to ipc_service", i);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     log::warn!("skipped config sync");
 | |
| }
 |