mirror of
https://github.com/rustdesk/rustdesk.git
synced 2026-04-13 03:11:29 +03:00
feat/virtual_display_privacy_mode
Signed-off-by: fufesou <shuanglongchen@yeah.net>
This commit is contained in:
297
src/privacy_mode.rs
Normal file
297
src/privacy_mode.rs
Normal file
@@ -0,0 +1,297 @@
|
||||
#[cfg(windows)]
|
||||
use crate::ipc::{connect, Data};
|
||||
#[cfg(all(windows, feature = "virtual_display_driver"))]
|
||||
use crate::platform::is_installed;
|
||||
use crate::{display_service, ipc::PrivacyModeState, ui_interface::get_option};
|
||||
#[cfg(windows)]
|
||||
use hbb_common::tokio;
|
||||
use hbb_common::{anyhow::anyhow, bail, lazy_static, ResultType};
|
||||
use std::{
|
||||
collections::HashMap,
|
||||
sync::{Arc, Mutex},
|
||||
};
|
||||
|
||||
#[cfg(windows)]
|
||||
mod win_input;
|
||||
|
||||
#[cfg(windows)]
|
||||
pub mod win_mag;
|
||||
|
||||
#[cfg(all(windows, feature = "virtual_display_driver"))]
|
||||
mod win_virtual_display;
|
||||
#[cfg(all(windows, feature = "virtual_display_driver"))]
|
||||
pub use win_virtual_display::restore_reg_connectivity;
|
||||
|
||||
pub const INVALID_PRIVACY_MODE_CONN_ID: i32 = 0;
|
||||
pub const OCCUPIED: &'static str = "Privacy occupied by another one";
|
||||
pub const TURN_OFF_OTHER_ID: &'static str =
|
||||
"Failed to turn off privacy mode that belongs to someone else";
|
||||
pub const NO_DISPLAYS: &'static str = "No displays";
|
||||
|
||||
#[cfg(windows)]
|
||||
pub const PRIVACY_MODE_IMPL_WIN_MAG: &str = win_mag::PRIVACY_MODE_IMPL;
|
||||
|
||||
#[cfg(all(windows, feature = "virtual_display_driver"))]
|
||||
pub const PRIVACY_MODE_IMPL_WIN_VIRTUAL_DISPLAY: &str = win_virtual_display::PRIVACY_MODE_IMPL;
|
||||
|
||||
pub trait PrivacyMode: Sync + Send {
|
||||
fn init(&self) -> ResultType<()>;
|
||||
fn clear(&mut self);
|
||||
fn turn_on_privacy(&mut self, conn_id: i32) -> ResultType<bool>;
|
||||
fn turn_off_privacy(&mut self, conn_id: i32, state: Option<PrivacyModeState>)
|
||||
-> ResultType<()>;
|
||||
|
||||
fn pre_conn_id(&self) -> i32;
|
||||
|
||||
#[inline]
|
||||
fn check_on_conn_id(&self, conn_id: i32) -> ResultType<bool> {
|
||||
let pre_conn_id = self.pre_conn_id();
|
||||
if pre_conn_id == conn_id {
|
||||
return Ok(true);
|
||||
}
|
||||
if pre_conn_id != INVALID_PRIVACY_MODE_CONN_ID {
|
||||
bail!(OCCUPIED);
|
||||
}
|
||||
Ok(false)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn check_off_conn_id(&self, conn_id: i32) -> ResultType<()> {
|
||||
let pre_conn_id = self.pre_conn_id();
|
||||
if pre_conn_id != INVALID_PRIVACY_MODE_CONN_ID
|
||||
&& conn_id != INVALID_PRIVACY_MODE_CONN_ID
|
||||
&& pre_conn_id != conn_id
|
||||
{
|
||||
bail!(TURN_OFF_OTHER_ID)
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
lazy_static::lazy_static! {
|
||||
pub static ref DEFAULT_PRIVACY_MODE_IMPL: String = {
|
||||
#[cfg(windows)]
|
||||
{
|
||||
if display_service::is_privacy_mode_mag_supported() {
|
||||
PRIVACY_MODE_IMPL_WIN_MAG
|
||||
} else {
|
||||
#[cfg(feature = "virtual_display_driver")]
|
||||
{
|
||||
if is_installed() {
|
||||
PRIVACY_MODE_IMPL_WIN_VIRTUAL_DISPLAY
|
||||
} else {
|
||||
""
|
||||
}
|
||||
}
|
||||
#[cfg(not(feature = "virtual_display_driver"))]
|
||||
{
|
||||
""
|
||||
}
|
||||
}.to_owned()
|
||||
}
|
||||
#[cfg(not(windows))]
|
||||
{
|
||||
"".to_owned()
|
||||
}
|
||||
};
|
||||
|
||||
static ref CUR_PRIVACY_MODE_IMPL: Arc<Mutex<String>> = {
|
||||
let mut cur_impl = get_option("privacy-mode-impl-key".to_owned());
|
||||
if !get_supported_privacy_mode_impl().iter().any(|(k, _)| k == &cur_impl) {
|
||||
cur_impl = DEFAULT_PRIVACY_MODE_IMPL.to_owned();
|
||||
}
|
||||
Arc::new(Mutex::new(cur_impl))
|
||||
};
|
||||
static ref PRIVACY_MODE: Arc<Mutex<Option<Box<dyn PrivacyMode>>>> = {
|
||||
let cur_impl = (*CUR_PRIVACY_MODE_IMPL.lock().unwrap()).clone();
|
||||
let privacy_mode = match PRIVACY_MODE_CREATOR.lock().unwrap().get(&(&cur_impl as &str)) {
|
||||
Some(creator) => Some(creator()),
|
||||
None => None,
|
||||
};
|
||||
Arc::new(Mutex::new(privacy_mode))
|
||||
};
|
||||
}
|
||||
|
||||
pub type PrivacyModeCreator = fn() -> Box<dyn PrivacyMode>;
|
||||
lazy_static::lazy_static! {
|
||||
static ref PRIVACY_MODE_CREATOR: Arc<Mutex<HashMap<&'static str, PrivacyModeCreator>>> = {
|
||||
#[cfg(not(windows))]
|
||||
let map: HashMap<&'static str, PrivacyModeCreator> = HashMap::new();
|
||||
#[cfg(windows)]
|
||||
let mut map: HashMap<&'static str, PrivacyModeCreator> = HashMap::new();
|
||||
#[cfg(windows)]
|
||||
{
|
||||
map.insert(win_mag::PRIVACY_MODE_IMPL, || {
|
||||
Box::new(win_mag::PrivacyModeImpl::default())
|
||||
});
|
||||
|
||||
#[cfg(feature = "virtual_display_driver")]
|
||||
map.insert(win_virtual_display::PRIVACY_MODE_IMPL, || {
|
||||
Box::new(win_virtual_display::PrivacyModeImpl::default())
|
||||
});
|
||||
}
|
||||
Arc::new(Mutex::new(map))
|
||||
};
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn init() -> Option<ResultType<()>> {
|
||||
Some(PRIVACY_MODE.lock().unwrap().as_ref()?.init())
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn clear() -> Option<()> {
|
||||
Some(PRIVACY_MODE.lock().unwrap().as_mut()?.clear())
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn switch(impl_key: &str) {
|
||||
let mut cur_impl_lock = CUR_PRIVACY_MODE_IMPL.lock().unwrap();
|
||||
if *cur_impl_lock == impl_key {
|
||||
return;
|
||||
}
|
||||
if let Some(creator) = PRIVACY_MODE_CREATOR.lock().unwrap().get(impl_key) {
|
||||
*PRIVACY_MODE.lock().unwrap() = Some(creator());
|
||||
*cur_impl_lock = impl_key.to_owned();
|
||||
}
|
||||
}
|
||||
|
||||
fn get_supported_impl(impl_key: &str) -> String {
|
||||
let supported_impls = get_supported_privacy_mode_impl();
|
||||
if supported_impls.iter().any(|(k, _)| k == &impl_key) {
|
||||
return impl_key.to_owned();
|
||||
};
|
||||
// fallback
|
||||
let mut cur_impl = get_option("privacy-mode-impl-key".to_owned());
|
||||
if !get_supported_privacy_mode_impl()
|
||||
.iter()
|
||||
.any(|(k, _)| k == &cur_impl)
|
||||
{
|
||||
// fallback
|
||||
cur_impl = DEFAULT_PRIVACY_MODE_IMPL.to_owned();
|
||||
}
|
||||
cur_impl
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn turn_on_privacy(impl_key: &str, conn_id: i32) -> Option<ResultType<bool>> {
|
||||
// Check if privacy mode is already on or occupied by another one
|
||||
let mut privacy_mode_lock = PRIVACY_MODE.lock().unwrap();
|
||||
if let Some(privacy_mode) = privacy_mode_lock.as_ref() {
|
||||
let check_on_conn_id = privacy_mode.check_on_conn_id(conn_id);
|
||||
match check_on_conn_id.as_ref() {
|
||||
Ok(true) | Err(_) => return Some(check_on_conn_id),
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
// Check or switch privacy mode implementation
|
||||
let impl_key = get_supported_impl(impl_key);
|
||||
let mut cur_impl_lock = CUR_PRIVACY_MODE_IMPL.lock().unwrap();
|
||||
if *cur_impl_lock != impl_key {
|
||||
if let Some(creator) = PRIVACY_MODE_CREATOR
|
||||
.lock()
|
||||
.unwrap()
|
||||
.get(&(&impl_key as &str))
|
||||
{
|
||||
*privacy_mode_lock = Some(creator());
|
||||
*cur_impl_lock = impl_key.to_owned();
|
||||
} else {
|
||||
return Some(Err(anyhow!("Unsupported privacy mode: {}", impl_key)));
|
||||
}
|
||||
}
|
||||
|
||||
// turn on privacy mode
|
||||
Some(privacy_mode_lock.as_mut()?.turn_on_privacy(conn_id))
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn turn_off_privacy(conn_id: i32, state: Option<PrivacyModeState>) -> Option<ResultType<()>> {
|
||||
Some(
|
||||
PRIVACY_MODE
|
||||
.lock()
|
||||
.unwrap()
|
||||
.as_mut()?
|
||||
.turn_off_privacy(conn_id, state),
|
||||
)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn check_on_conn_id(conn_id: i32) -> Option<ResultType<bool>> {
|
||||
Some(
|
||||
PRIVACY_MODE
|
||||
.lock()
|
||||
.unwrap()
|
||||
.as_ref()?
|
||||
.check_on_conn_id(conn_id),
|
||||
)
|
||||
}
|
||||
|
||||
#[cfg(windows)]
|
||||
#[tokio::main(flavor = "current_thread")]
|
||||
async fn set_privacy_mode_state(
|
||||
conn_id: i32,
|
||||
state: PrivacyModeState,
|
||||
impl_key: String,
|
||||
ms_timeout: u64,
|
||||
) -> ResultType<()> {
|
||||
let mut c = connect(ms_timeout, "_cm").await?;
|
||||
c.send(&Data::PrivacyModeState((conn_id, state, impl_key)))
|
||||
.await
|
||||
}
|
||||
|
||||
pub fn get_supported_privacy_mode_impl() -> Vec<(&'static str, &'static str)> {
|
||||
#[cfg(target_os = "windows")]
|
||||
{
|
||||
let mut vec_impls = Vec::new();
|
||||
if display_service::is_privacy_mode_mag_supported() {
|
||||
vec_impls.push((PRIVACY_MODE_IMPL_WIN_MAG, "privacy_mode_impl_mag_tip"));
|
||||
}
|
||||
#[cfg(feature = "virtual_display_driver")]
|
||||
if is_installed() {
|
||||
vec_impls.push((
|
||||
PRIVACY_MODE_IMPL_WIN_VIRTUAL_DISPLAY,
|
||||
"privacy_mode_impl_virtual_display_tip",
|
||||
));
|
||||
}
|
||||
vec_impls
|
||||
}
|
||||
#[cfg(not(target_os = "windows"))]
|
||||
{
|
||||
Vec::new()
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn is_current_privacy_mode_impl(impl_key: &str) -> bool {
|
||||
*CUR_PRIVACY_MODE_IMPL.lock().unwrap() == impl_key
|
||||
}
|
||||
|
||||
#[inline]
|
||||
#[cfg(not(windows))]
|
||||
pub fn check_privacy_mode_err(
|
||||
_privacy_mode_id: i32,
|
||||
_display_idx: usize,
|
||||
_timeout_millis: u64,
|
||||
) -> String {
|
||||
"".to_owned()
|
||||
}
|
||||
|
||||
#[inline]
|
||||
#[cfg(windows)]
|
||||
pub fn check_privacy_mode_err(
|
||||
privacy_mode_id: i32,
|
||||
display_idx: usize,
|
||||
timeout_millis: u64,
|
||||
) -> String {
|
||||
if is_current_privacy_mode_impl(PRIVACY_MODE_IMPL_WIN_MAG) {
|
||||
crate::video_service::test_create_capturer(privacy_mode_id, display_idx, timeout_millis)
|
||||
} else {
|
||||
"".to_owned()
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn is_privacy_mode_supported() -> bool {
|
||||
!DEFAULT_PRIVACY_MODE_IMPL.is_empty()
|
||||
}
|
||||
Reference in New Issue
Block a user