mirror of
https://github.com/rustdesk/rustdesk.git
synced 2025-01-23 10:56:31 +08:00
821f7245b0
* h265 encoding: the second frame always fails, use repeat encoding to achieve real-time encoding * h264 encoding: Not supported because encoding fails too frequently, about 50%, with one failure followed by another success. * h264/h265 decoding: No issues found. * Does not support dynamically changing the bitrate and changing the quality by resetting the encoder. Signed-off-by: 21pages <sunboeasy@gmail.com>
657 lines
22 KiB
Rust
657 lines
22 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)
|
|
.map(|s| s.is_subed(conn_id))
|
|
.unwrap_or(false)
|
|
})
|
|
.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(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);
|
|
if crate::is_server() {
|
|
log::error!("ipc is occupied by another process, try kill it");
|
|
std::thread::spawn(stop_main_window_process).join().ok();
|
|
}
|
|
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();
|
|
#[cfg(feature = "hwcodec")]
|
|
#[cfg(not(any(target_os = "android", target_os = "ios")))]
|
|
scrap::hwcodec::start_check_process();
|
|
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");
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
#[cfg(feature = "hwcodec")]
|
|
#[cfg(any(target_os = "windows", target_os = "linux"))]
|
|
crate::ipc::client_get_hwcodec_config_thread(0);
|
|
}
|
|
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");
|
|
}
|
|
|
|
#[tokio::main(flavor = "current_thread")]
|
|
pub async fn stop_main_window_process() {
|
|
// this may also kill another --server process,
|
|
// but --server usually can be auto restarted by --service, so it is ok
|
|
if let Ok(mut conn) = crate::ipc::connect(1000, "").await {
|
|
conn.send(&crate::ipc::Data::Close).await.ok();
|
|
}
|
|
#[cfg(windows)]
|
|
{
|
|
// in case above failure, e.g. zombie process
|
|
if let Err(e) = crate::platform::try_kill_rustdesk_main_window_process() {
|
|
log::error!("kill failed: {}", e);
|
|
}
|
|
}
|
|
}
|