plugin_framework, mid commit
Signed-off-by: fufesou <shuanglongchen@yeah.net>
This commit is contained in:
parent
095ac46255
commit
8e117b6dde
@ -248,9 +248,6 @@ pub fn core_main() -> Option<Vec<String>> {
|
|||||||
crate::plugin::install_plugin_with_url(&args[1], &args[2]);
|
crate::plugin::install_plugin_with_url(&args[1], &args[2]);
|
||||||
}
|
}
|
||||||
return None;
|
return None;
|
||||||
} else if args[0] == "--plugin-uninstall" {
|
|
||||||
// Do nothing
|
|
||||||
return None;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,10 +9,10 @@ use hbb_common::{
|
|||||||
regex::{Captures, Regex},
|
regex::{Captures, Regex},
|
||||||
};
|
};
|
||||||
use std::{
|
use std::{
|
||||||
string::String,
|
|
||||||
cell::RefCell,
|
cell::RefCell,
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
process::{Child, Command},
|
process::{Child, Command},
|
||||||
|
string::String,
|
||||||
sync::{
|
sync::{
|
||||||
atomic::{AtomicBool, Ordering},
|
atomic::{AtomicBool, Ordering},
|
||||||
Arc,
|
Arc,
|
||||||
@ -453,9 +453,7 @@ pub fn get_active_username() -> String {
|
|||||||
|
|
||||||
pub fn get_user_home_by_name(username: &str) -> Option<PathBuf> {
|
pub fn get_user_home_by_name(username: &str) -> Option<PathBuf> {
|
||||||
return match get_user_by_name(username) {
|
return match get_user_by_name(username) {
|
||||||
None => {
|
None => None,
|
||||||
None
|
|
||||||
}
|
|
||||||
Some(user) => {
|
Some(user) => {
|
||||||
let home = user.home_dir();
|
let home = user.home_dir();
|
||||||
if Path::is_dir(home) {
|
if Path::is_dir(home) {
|
||||||
@ -464,7 +462,7 @@ pub fn get_user_home_by_name(username: &str) -> Option<PathBuf> {
|
|||||||
None
|
None
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_active_user_home() -> Option<PathBuf> {
|
pub fn get_active_user_home() -> Option<PathBuf> {
|
||||||
@ -835,12 +833,7 @@ mod desktop {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn get_display(&mut self) {
|
fn get_display(&mut self) {
|
||||||
let display_envs = vec![
|
let display_envs = vec![GNOME_SESSION_BINARY, XFCE4_PANEL, SDDM_GREETER, PLASMA_X11];
|
||||||
GNOME_SESSION_BINARY,
|
|
||||||
XFCE4_PANEL,
|
|
||||||
SDDM_GREETER,
|
|
||||||
PLASMA_X11,
|
|
||||||
];
|
|
||||||
for diplay_env in display_envs {
|
for diplay_env in display_envs {
|
||||||
self.display = get_env_tries("DISPLAY", &self.uid, diplay_env, 10);
|
self.display = get_env_tries("DISPLAY", &self.uid, diplay_env, 10);
|
||||||
if !self.display.is_empty() {
|
if !self.display.is_empty() {
|
||||||
@ -873,8 +866,8 @@ mod desktop {
|
|||||||
auth_found = true;
|
auth_found = true;
|
||||||
} else if auth_found {
|
} else if auth_found {
|
||||||
if std::path::Path::new(v).is_absolute()
|
if std::path::Path::new(v).is_absolute()
|
||||||
&& std::path::Path::new(v).exists() {
|
&& std::path::Path::new(v).exists()
|
||||||
|
{
|
||||||
self.xauth = v.to_string();
|
self.xauth = v.to_string();
|
||||||
} else {
|
} else {
|
||||||
if let Some(pid) = line.split_whitespace().nth(1) {
|
if let Some(pid) = line.split_whitespace().nth(1) {
|
||||||
@ -903,12 +896,7 @@ mod desktop {
|
|||||||
|
|
||||||
fn get_xauth(&mut self) {
|
fn get_xauth(&mut self) {
|
||||||
// try by direct access to window manager process by name
|
// try by direct access to window manager process by name
|
||||||
let display_envs = vec![
|
let display_envs = vec![GNOME_SESSION_BINARY, XFCE4_PANEL, SDDM_GREETER, PLASMA_X11];
|
||||||
GNOME_SESSION_BINARY,
|
|
||||||
XFCE4_PANEL,
|
|
||||||
SDDM_GREETER,
|
|
||||||
PLASMA_X11,
|
|
||||||
];
|
|
||||||
for diplay_env in display_envs {
|
for diplay_env in display_envs {
|
||||||
self.xauth = get_env_tries("XAUTHORITY", &self.uid, diplay_env, 10);
|
self.xauth = get_env_tries("XAUTHORITY", &self.uid, diplay_env, 10);
|
||||||
if !self.xauth.is_empty() {
|
if !self.xauth.is_empty() {
|
||||||
@ -928,7 +916,7 @@ mod desktop {
|
|||||||
gdm
|
gdm
|
||||||
} else {
|
} else {
|
||||||
let username = &self.username;
|
let username = &self.username;
|
||||||
match get_user_home_by_name(username) {
|
match get_user_home_by_name(username) {
|
||||||
None => {
|
None => {
|
||||||
if username == "root" {
|
if username == "root" {
|
||||||
format!("/{}/.Xauthority", username)
|
format!("/{}/.Xauthority", username)
|
||||||
@ -942,7 +930,10 @@ mod desktop {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some(home) => {
|
Some(home) => {
|
||||||
format!("{}/.Xauthority", home.as_path().to_string_lossy().to_string())
|
format!(
|
||||||
|
"{}/.Xauthority",
|
||||||
|
home.as_path().to_string_lossy().to_string()
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -54,6 +54,27 @@ pub fn get_active_username() -> String {
|
|||||||
#[cfg(target_os = "android")]
|
#[cfg(target_os = "android")]
|
||||||
pub const PA_SAMPLE_RATE: u32 = 48000;
|
pub const PA_SAMPLE_RATE: u32 = 48000;
|
||||||
|
|
||||||
|
#[cfg(any(target_os = "linux", target_os = "macos"))]
|
||||||
|
pub fn run_as_root(arg: Vec<&str>) -> ResultType<Option<Child>> {
|
||||||
|
let cmd = std::env::current_exe()?;
|
||||||
|
match cmd.to_str() {
|
||||||
|
Some(cmd) => {
|
||||||
|
let mut args = vec![cmd];
|
||||||
|
args.append(&mut arg.clone());
|
||||||
|
// -E required for opensuse
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
if is_opensuse() {
|
||||||
|
args.insert(0, "-E");
|
||||||
|
}
|
||||||
|
let task = Command::new("sudo").args(args).spawn()?;
|
||||||
|
Ok(Some(task))
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
bail!("Failed to get current exe as str");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
@ -58,6 +58,8 @@ pub struct Meta {
|
|||||||
pub name: String,
|
pub name: String,
|
||||||
pub version: String,
|
pub version: String,
|
||||||
pub description: String,
|
pub description: String,
|
||||||
|
#[serde(default)]
|
||||||
|
pub platforms: String,
|
||||||
pub author: String,
|
pub author: String,
|
||||||
pub home: String,
|
pub home: String,
|
||||||
pub license: String,
|
pub license: String,
|
||||||
|
@ -19,6 +19,13 @@ const MSG_TO_UI_PLUGIN_MANAGER_UNINSTALL: &str = "plugin_uninstall";
|
|||||||
|
|
||||||
const IPC_PLUGIN_POSTFIX: &str = "_plugin";
|
const IPC_PLUGIN_POSTFIX: &str = "_plugin";
|
||||||
|
|
||||||
|
#[cfg(target_os = "windows")]
|
||||||
|
const PLUGIN_PLATFORM: &str = "Windows";
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
const PLUGIN_PLATFORM: &str = "Linux";
|
||||||
|
#[cfg(target_os = "macos")]
|
||||||
|
const PLUGIN_PLATFORM: &str = "MacOS";
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
static ref PLUGIN_INFO: Arc<Mutex<HashMap<String, PluginInfo>>> = Arc::new(Mutex::new(HashMap::new()));
|
static ref PLUGIN_INFO: Arc<Mutex<HashMap<String, PluginInfo>>> = Arc::new(Mutex::new(HashMap::new()));
|
||||||
}
|
}
|
||||||
@ -73,6 +80,9 @@ fn get_source_plugins() -> HashMap<String, PluginInfo> {
|
|||||||
match toml::from_str::<ManagerMeta>(&text) {
|
match toml::from_str::<ManagerMeta>(&text) {
|
||||||
Ok(manager_meta) => {
|
Ok(manager_meta) => {
|
||||||
for meta in manager_meta.plugins.iter() {
|
for meta in manager_meta.plugins.iter() {
|
||||||
|
if !meta.platforms.contains(PLUGIN_PLATFORM) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
plugins.insert(
|
plugins.insert(
|
||||||
meta.id.clone(),
|
meta.id.clone(),
|
||||||
PluginInfo {
|
PluginInfo {
|
||||||
@ -147,33 +157,59 @@ pub fn load_plugin_list() {
|
|||||||
pub fn install_plugin(id: &str) -> ResultType<()> {
|
pub fn install_plugin(id: &str) -> ResultType<()> {
|
||||||
match PLUGIN_INFO.lock().unwrap().get(id) {
|
match PLUGIN_INFO.lock().unwrap().get(id) {
|
||||||
Some(plugin) => {
|
Some(plugin) => {
|
||||||
|
let mut same_plugin_exists = false;
|
||||||
|
if let Some(version) = super::plugins::get_version(id) {
|
||||||
|
if version == plugin.meta.version {
|
||||||
|
same_plugin_exists = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let plugin_url = format!(
|
||||||
|
"{}/plugins/{}/{}/{}_{}.zip",
|
||||||
|
plugin.source.url,
|
||||||
|
plugin.meta.id,
|
||||||
|
PLUGIN_PLATFORM,
|
||||||
|
plugin.meta.id,
|
||||||
|
plugin.meta.version
|
||||||
|
);
|
||||||
|
|
||||||
|
let allowed_install;
|
||||||
#[cfg(windows)]
|
#[cfg(windows)]
|
||||||
{
|
{
|
||||||
let mut same_plugin_exists = false;
|
|
||||||
if let Some(version) = super::plugins::get_version(id) {
|
|
||||||
if version == plugin.meta.version {
|
|
||||||
same_plugin_exists = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// to-do: Support args with space in quotes. 'arg 1' and "arg 2"
|
// to-do: Support args with space in quotes. 'arg 1' and "arg 2"
|
||||||
let plugin_url = format!(
|
|
||||||
"{}/plugins/{}/{}_{}.zip",
|
|
||||||
plugin.source.url, plugin.meta.id, plugin.meta.id, plugin.meta.version
|
|
||||||
);
|
|
||||||
let args = if same_plugin_exists {
|
let args = if same_plugin_exists {
|
||||||
format!("--plugin-install {}", id)
|
format!("--plugin-install {}", id)
|
||||||
} else {
|
} else {
|
||||||
format!("--plugin-install {} {}", id, plugin_url)
|
format!("--plugin-install {} {}", id, plugin_url)
|
||||||
};
|
};
|
||||||
let allowed = crate::platform::elevate(&args)?;
|
allowed_install = crate::platform::elevate(&args)?;
|
||||||
|
}
|
||||||
if allowed && same_plugin_exists {
|
#[cfg(target_os = "linux")]
|
||||||
super::ipc::load_plugin(id)?;
|
{
|
||||||
super::plugins::load_plugin(id)?;
|
let mut args = vec!["--plugin-install", id];
|
||||||
super::plugins::mark_uninstalled(id, false);
|
if !same_plugin_exists {
|
||||||
push_install_event(id, "finished");
|
args.push(&plugin_url);
|
||||||
}
|
}
|
||||||
|
allowed_install = match crate::platform::run_as_root(&args) {
|
||||||
|
Ok(child) => match child.wait() {
|
||||||
|
Ok(0) => true,
|
||||||
|
Ok(code) => {
|
||||||
|
log::error!("Failed to wait install process, process code: {}", code);
|
||||||
|
true
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
log::error!("Failed to wait install process, error: {}", e);
|
||||||
|
false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if allowed_install && same_plugin_exists {
|
||||||
|
super::ipc::load_plugin(id)?;
|
||||||
|
super::plugins::load_plugin(id)?;
|
||||||
|
super::plugins::mark_uninstalled(id, false);
|
||||||
|
push_install_event(id, "finished");
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -220,7 +256,8 @@ pub(super) fn remove_plugins() -> ResultType<()> {
|
|||||||
|
|
||||||
pub fn uninstall_plugin(id: &str, called_by_ui: bool) {
|
pub fn uninstall_plugin(id: &str, called_by_ui: bool) {
|
||||||
if called_by_ui {
|
if called_by_ui {
|
||||||
match crate::platform::elevate(&format!("--plugin-uninstall {}", id)) {
|
#[cfg(target_os = "windows")]
|
||||||
|
match crate::platform::check_super_user_permission() {
|
||||||
Ok(true) => {
|
Ok(true) => {
|
||||||
if let Err(e) = super::ipc::uninstall_plugin(id) {
|
if let Err(e) = super::ipc::uninstall_plugin(id) {
|
||||||
log::error!("Failed to uninstall plugin '{}': {}", id, e);
|
log::error!("Failed to uninstall plugin '{}': {}", id, e);
|
||||||
@ -236,7 +273,11 @@ pub fn uninstall_plugin(id: &str, called_by_ui: bool) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Failed to uninstall plugin '{}': {}", id, e);
|
log::error!(
|
||||||
|
"Failed to uninstall plugin '{}', check permission error: {}",
|
||||||
|
id,
|
||||||
|
e
|
||||||
|
);
|
||||||
push_uninstall_event(id, "failed");
|
push_uninstall_event(id, "failed");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -100,10 +100,27 @@ pub fn init() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
#[cfg(target_os = "windows")]
|
||||||
|
fn get_share_dir() -> ResultType<PathBuf> {
|
||||||
|
Ok(PathBuf::from(env::var("ProgramData")?))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
fn get_share_dir() -> ResultType<PathBuf> {
|
||||||
|
Ok(PathBuf::from("/usr/share"))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
#[cfg(target_os = "macos")]
|
||||||
|
fn get_share_dir() -> ResultType<PathBuf> {
|
||||||
|
Ok(PathBuf::from("/Library/Application Support"))
|
||||||
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn get_plugins_dir() -> ResultType<PathBuf> {
|
fn get_plugins_dir() -> ResultType<PathBuf> {
|
||||||
// to-do: linux and macos
|
Ok(get_share_dir()?
|
||||||
Ok(PathBuf::from(env::var("ProgramData")?)
|
|
||||||
.join("RustDesk")
|
.join("RustDesk")
|
||||||
.join(PLUGIN_SOURCE_LOCAL_DIR))
|
.join(PLUGIN_SOURCE_LOCAL_DIR))
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user