| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | use super::{linux::*, ResultType};
 | 
					
						
							| 
									
										
										
										
											2023-04-17 20:49:37 +08:00
										 |  |  | use crate::client::{
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |     LOGIN_MSG_DESKTOP_NO_DESKTOP, LOGIN_MSG_DESKTOP_SESSION_ANOTHER_USER,
 | 
					
						
							|  |  |  |     LOGIN_MSG_DESKTOP_SESSION_NOT_READY, LOGIN_MSG_DESKTOP_XORG_NOT_FOUND,
 | 
					
						
							|  |  |  |     LOGIN_MSG_DESKTOP_XSESSION_FAILED,
 | 
					
						
							|  |  |  | };
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | use hbb_common::{allow_err, bail, log, rand::prelude::*, tokio::time};
 | 
					
						
							|  |  |  | use pam;
 | 
					
						
							|  |  |  | use std::{
 | 
					
						
							|  |  |  |     collections::HashMap,
 | 
					
						
							|  |  |  |     os::unix::process::CommandExt,
 | 
					
						
							|  |  |  |     path::Path,
 | 
					
						
							|  |  |  |     process::{Child, Command},
 | 
					
						
							|  |  |  |     sync::{
 | 
					
						
							|  |  |  |         atomic::{AtomicBool, Ordering},
 | 
					
						
							|  |  |  |         mpsc::{sync_channel, SyncSender},
 | 
					
						
							|  |  |  |         Arc, Mutex,
 | 
					
						
							|  |  |  |     },
 | 
					
						
							|  |  |  |     time::{Duration, Instant},
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | use users::{get_user_by_name, os::unix::UserExt, User};
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | lazy_static::lazy_static! {
 | 
					
						
							|  |  |  |     static ref DESKTOP_RUNNING: Arc<AtomicBool> = Arc::new(AtomicBool::new(false));
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     static ref DESKTOP_MANAGER: Arc<Mutex<Option<DesktopManager>>> = Arc::new(Mutex::new(None));
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[derive(Debug)]
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  | struct DesktopManager {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 14:11:56 +08:00
										 |  |  |     seat0_username: String,
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |     seat0_display_server: String,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     child_username: String,
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     child_exit: Arc<AtomicBool>,
 | 
					
						
							|  |  |  |     is_child_running: Arc<AtomicBool>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  | fn check_desktop_manager() {
 | 
					
						
							|  |  |  |     let mut desktop_manager = DESKTOP_MANAGER.lock().unwrap();
 | 
					
						
							|  |  |  |     if let Some(desktop_manager) = &mut (*desktop_manager) {
 | 
					
						
							|  |  |  |         if desktop_manager.is_child_running.load(Ordering::SeqCst) {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 20:20:45 +08:00
										 |  |  |             return;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         desktop_manager.child_exit.store(true, Ordering::SeqCst);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub fn start_xdesktop() {
 | 
					
						
							| 
									
										
										
										
											2024-05-02 13:54:09 +08:00
										 |  |  |     debug_assert!(crate::is_server());
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     std::thread::spawn(|| {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         *DESKTOP_MANAGER.lock().unwrap() = Some(DesktopManager::new());
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let interval = time::Duration::from_millis(super::SERVICE_INTERVAL);
 | 
					
						
							|  |  |  |         DESKTOP_RUNNING.store(true, Ordering::SeqCst);
 | 
					
						
							|  |  |  |         while DESKTOP_RUNNING.load(Ordering::SeqCst) {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             check_desktop_manager();
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             std::thread::sleep(interval);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         log::info!("xdesktop child thread exit");
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     });
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pub fn stop_xdesktop() {
 | 
					
						
							|  |  |  |     DESKTOP_RUNNING.store(false, Ordering::SeqCst);
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     *DESKTOP_MANAGER.lock().unwrap() = None;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 13:14:20 +08:00
										 |  |  | fn detect_headless() -> Option<&'static str> {
 | 
					
						
							|  |  |  |     match run_cmds(&format!("which {}", DesktopManager::get_xorg())) {
 | 
					
						
							|  |  |  |         Ok(output) => {
 | 
					
						
							|  |  |  |             if output.trim().is_empty() {
 | 
					
						
							|  |  |  |                 return Some(LOGIN_MSG_DESKTOP_XORG_NOT_FOUND);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         _ => {
 | 
					
						
							|  |  |  |             return Some(LOGIN_MSG_DESKTOP_XORG_NOT_FOUND);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     match run_cmds("ls /usr/share/xsessions/") {
 | 
					
						
							|  |  |  |         Ok(output) => {
 | 
					
						
							|  |  |  |             if output.trim().is_empty() {
 | 
					
						
							|  |  |  |                 return Some(LOGIN_MSG_DESKTOP_NO_DESKTOP);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         _ => {
 | 
					
						
							|  |  |  |             return Some(LOGIN_MSG_DESKTOP_NO_DESKTOP);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     None
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  | pub fn try_start_desktop(_username: &str, _passsword: &str) -> String {
 | 
					
						
							| 
									
										
										
										
											2024-05-02 13:54:09 +08:00
										 |  |  |     debug_assert!(crate::is_server());
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |     if _username.is_empty() {
 | 
					
						
							|  |  |  |         let username = get_username();
 | 
					
						
							|  |  |  |         if username.is_empty() {
 | 
					
						
							| 
									
										
										
										
											2023-04-03 13:14:20 +08:00
										 |  |  |             if let Some(msg) = detect_headless() {
 | 
					
						
							|  |  |  |                 msg
 | 
					
						
							|  |  |  |             } else {
 | 
					
						
							|  |  |  |                 LOGIN_MSG_DESKTOP_SESSION_NOT_READY
 | 
					
						
							|  |  |  |             }
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |         } else {
 | 
					
						
							|  |  |  |             ""
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         .to_owned()
 | 
					
						
							|  |  |  |     } else {
 | 
					
						
							|  |  |  |         let username = get_username();
 | 
					
						
							|  |  |  |         if username == _username {
 | 
					
						
							|  |  |  |             // No need to verify password here.
 | 
					
						
							|  |  |  |             return "".to_owned();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-07-29 15:22:25 +08:00
										 |  |  |         if !username.is_empty() {
 | 
					
						
							| 
									
										
										
										
											2023-07-29 13:06:18 +08:00
										 |  |  |             // Another user is logged in. No need to start a new xsession.
 | 
					
						
							|  |  |  |             return "".to_owned();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 13:14:20 +08:00
										 |  |  |         if let Some(msg) = detect_headless() {
 | 
					
						
							|  |  |  |             return msg.to_owned();
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         match try_start_x_session(_username, _passsword) {
 | 
					
						
							|  |  |  |             Ok((username, x11_ready)) => {
 | 
					
						
							|  |  |  |                 if x11_ready {
 | 
					
						
							|  |  |  |                     if _username != username {
 | 
					
						
							|  |  |  |                         LOGIN_MSG_DESKTOP_SESSION_ANOTHER_USER.to_owned()
 | 
					
						
							|  |  |  |                     } else {
 | 
					
						
							|  |  |  |                         "".to_owned()
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     LOGIN_MSG_DESKTOP_SESSION_NOT_READY.to_owned()
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             Err(e) => {
 | 
					
						
							|  |  |  |                 log::error!("Failed to start xsession {}", e);
 | 
					
						
							|  |  |  |                 LOGIN_MSG_DESKTOP_XSESSION_FAILED.to_owned()
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn try_start_x_session(username: &str, password: &str) -> ResultType<(String, bool)> {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     let mut desktop_manager = DESKTOP_MANAGER.lock().unwrap();
 | 
					
						
							|  |  |  |     if let Some(desktop_manager) = &mut (*desktop_manager) {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         if let Some(seat0_username) = desktop_manager.get_supported_display_seat0_username() {
 | 
					
						
							|  |  |  |             return Ok((seat0_username, true));
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let _ = desktop_manager.try_start_x_session(username, password)?;
 | 
					
						
							| 
									
										
										
										
											2023-03-28 20:20:45 +08:00
										 |  |  |         log::debug!(
 | 
					
						
							|  |  |  |             "try_start_x_session, username: {}, {:?}",
 | 
					
						
							|  |  |  |             &username,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             &desktop_manager
 | 
					
						
							| 
									
										
										
										
											2023-03-28 20:20:45 +08:00
										 |  |  |         );
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         Ok((
 | 
					
						
							|  |  |  |             desktop_manager.child_username.clone(),
 | 
					
						
							|  |  |  |             desktop_manager.is_running(),
 | 
					
						
							|  |  |  |         ))
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     } else {
 | 
					
						
							| 
									
										
										
										
											2023-04-17 20:49:37 +08:00
										 |  |  |         bail!(crate::client::LOGIN_MSG_DESKTOP_NOT_INITED);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-30 10:48:18 +08:00
										 |  |  | #[inline]
 | 
					
						
							|  |  |  | pub fn is_headless() -> bool {
 | 
					
						
							|  |  |  |     DESKTOP_MANAGER
 | 
					
						
							|  |  |  |         .lock()
 | 
					
						
							|  |  |  |         .unwrap()
 | 
					
						
							|  |  |  |         .as_ref()
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         .map_or(false, |manager| {
 | 
					
						
							|  |  |  |             manager.get_supported_display_seat0_username().is_none()
 | 
					
						
							|  |  |  |         })
 | 
					
						
							| 
									
										
										
										
											2023-03-30 10:48:18 +08:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  | pub fn get_username() -> String {
 | 
					
						
							|  |  |  |     match &*DESKTOP_MANAGER.lock().unwrap() {
 | 
					
						
							|  |  |  |         Some(manager) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |             if let Some(seat0_username) = manager.get_supported_display_seat0_username() {
 | 
					
						
							|  |  |  |                 seat0_username
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             } else {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                 if manager.is_running() && !manager.child_username.is_empty() {
 | 
					
						
							|  |  |  |                     manager.child_username.clone()
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 } else {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                     "".to_owned()
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         None => "".to_owned(),
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  | impl Drop for DesktopManager {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     fn drop(&mut self) {
 | 
					
						
							|  |  |  |         self.stop_children();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  | impl DesktopManager {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     fn fatal_exit() {
 | 
					
						
							|  |  |  |         std::process::exit(0);
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn new() -> Self {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 14:11:56 +08:00
										 |  |  |         let mut seat0_username = "".to_owned();
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         let mut seat0_display_server = "".to_owned();
 | 
					
						
							| 
									
										
										
										
											2023-03-30 18:16:48 +08:00
										 |  |  |         let seat0_values = get_values_of_seat0(&[0, 2]);
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         if !seat0_values[0].is_empty() {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 18:16:48 +08:00
										 |  |  |             seat0_username = seat0_values[1].clone();
 | 
					
						
							|  |  |  |             seat0_display_server = get_display_server_of_session(&seat0_values[0]);
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         Self {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 14:11:56 +08:00
										 |  |  |             seat0_username,
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |             seat0_display_server,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             child_username: "".to_owned(),
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             child_exit: Arc::new(AtomicBool::new(true)),
 | 
					
						
							|  |  |  |             is_child_running: Arc::new(AtomicBool::new(false)),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |     fn get_supported_display_seat0_username(&self) -> Option<String> {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 21:35:47 +08:00
										 |  |  |         if is_gdm_user(&self.seat0_username) && self.seat0_display_server == DISPLAY_SERVER_WAYLAND
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         {
 | 
					
						
							|  |  |  |             None
 | 
					
						
							| 
									
										
										
										
											2023-03-30 21:35:47 +08:00
										 |  |  |         } else if self.seat0_username.is_empty() {
 | 
					
						
							|  |  |  |             None
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         } else {
 | 
					
						
							| 
									
										
										
										
											2023-03-30 21:35:47 +08:00
										 |  |  |             Some(self.seat0_username.clone())
 | 
					
						
							| 
									
										
										
										
											2023-03-30 16:17:24 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn get_xauth() -> String {
 | 
					
						
							|  |  |  |         let xauth = get_env_var("XAUTHORITY");
 | 
					
						
							|  |  |  |         if xauth.is_empty() {
 | 
					
						
							|  |  |  |             "/tmp/.Xauthority".to_owned()
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             xauth
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn is_running(&self) -> bool {
 | 
					
						
							|  |  |  |         self.is_child_running.load(Ordering::SeqCst)
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn try_start_x_session(&mut self, username: &str, password: &str) -> ResultType<()> {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         match get_user_by_name(username) {
 | 
					
						
							|  |  |  |             Some(userinfo) => {
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |                 let mut client = pam::Client::with_password(&pam_get_service_name())?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 client
 | 
					
						
							|  |  |  |                     .conversation_mut()
 | 
					
						
							|  |  |  |                     .set_credentials(username, password);
 | 
					
						
							|  |  |  |                 match client.authenticate() {
 | 
					
						
							|  |  |  |                     Ok(_) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                         if self.is_running() {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                             return Ok(());
 | 
					
						
							|  |  |  |                         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                         match self.start_x_session(&userinfo, username, password) {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                             Ok(_) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                                 log::info!("Succeeded to start x11");
 | 
					
						
							|  |  |  |                                 self.child_username = username.to_string();
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                                 Ok(())
 | 
					
						
							|  |  |  |                             }
 | 
					
						
							|  |  |  |                             Err(e) => {
 | 
					
						
							|  |  |  |                                 bail!("failed to start x session, {}", e);
 | 
					
						
							|  |  |  |                             }
 | 
					
						
							|  |  |  |                         }
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     Err(e) => {
 | 
					
						
							|  |  |  |                         bail!("failed to check user pass for {}, {}", username, e);
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             None => {
 | 
					
						
							|  |  |  |                 bail!("failed to get userinfo of {}", username);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 14:39:38 +08:00
										 |  |  |     // The logic mainly from https://github.com/neutrinolabs/xrdp/blob/34fe9b60ebaea59e8814bbc3ca5383cabaa1b869/sesman/session.c#L334.
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     fn get_avail_display() -> ResultType<u32> {
 | 
					
						
							|  |  |  |         let display_range = 0..51;
 | 
					
						
							|  |  |  |         for i in display_range.clone() {
 | 
					
						
							|  |  |  |             if Self::is_x_server_running(i) {
 | 
					
						
							|  |  |  |                 continue;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             return Ok(i);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2024-03-27 01:08:56 +08:00
										 |  |  |         bail!("No available display found in range {:?}", display_range)
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn is_x_server_running(display: u32) -> bool {
 | 
					
						
							|  |  |  |         Path::new(&format!("/tmp/.X11-unix/X{}", display)).exists()
 | 
					
						
							|  |  |  |             || Path::new(&format!("/tmp/.X{}-lock", display)).exists()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 20:20:45 +08:00
										 |  |  |     fn start_x_session(
 | 
					
						
							|  |  |  |         &mut self,
 | 
					
						
							|  |  |  |         userinfo: &User,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         username: &str,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 20:20:45 +08:00
										 |  |  |         password: &str,
 | 
					
						
							|  |  |  |     ) -> ResultType<()> {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         self.stop_children();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let display_num = Self::get_avail_display()?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         // "xServer_ip:display_num.screen_num"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let uid = userinfo.uid();
 | 
					
						
							|  |  |  |         let gid = userinfo.primary_group_id();
 | 
					
						
							|  |  |  |         let envs = HashMap::from([
 | 
					
						
							|  |  |  |             ("SHELL", userinfo.shell().to_string_lossy().to_string()),
 | 
					
						
							|  |  |  |             ("PATH", "/sbin:/bin:/usr/bin:/usr/local/bin".to_owned()),
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             ("USER", username.to_string()),
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             ("UID", userinfo.uid().to_string()),
 | 
					
						
							|  |  |  |             ("HOME", userinfo.home_dir().to_string_lossy().to_string()),
 | 
					
						
							|  |  |  |             (
 | 
					
						
							|  |  |  |                 "XDG_RUNTIME_DIR",
 | 
					
						
							|  |  |  |                 format!("/run/user/{}", userinfo.uid().to_string()),
 | 
					
						
							|  |  |  |             ),
 | 
					
						
							|  |  |  |             // ("DISPLAY", self.display.clone()),
 | 
					
						
							|  |  |  |             // ("XAUTHORITY", self.xauth.clone()),
 | 
					
						
							|  |  |  |             // (ENV_DESKTOP_PROTOCAL, XProtocal::X11.to_string()),
 | 
					
						
							|  |  |  |         ]);
 | 
					
						
							|  |  |  |         self.child_exit.store(false, Ordering::SeqCst);
 | 
					
						
							|  |  |  |         let is_child_running = self.is_child_running.clone();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let (tx_res, rx_res) = sync_channel(1);
 | 
					
						
							|  |  |  |         let password = password.to_string();
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let username = username.to_string();
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         // start x11
 | 
					
						
							|  |  |  |         std::thread::spawn(move || {
 | 
					
						
							|  |  |  |             match Self::start_x_session_thread(
 | 
					
						
							|  |  |  |                 tx_res.clone(),
 | 
					
						
							|  |  |  |                 is_child_running,
 | 
					
						
							|  |  |  |                 uid,
 | 
					
						
							|  |  |  |                 gid,
 | 
					
						
							|  |  |  |                 display_num,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                 username,
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 password,
 | 
					
						
							|  |  |  |                 envs,
 | 
					
						
							|  |  |  |             ) {
 | 
					
						
							|  |  |  |                 Ok(_) => {}
 | 
					
						
							|  |  |  |                 Err(e) => {
 | 
					
						
							|  |  |  |                     log::error!("Failed to start x session thread");
 | 
					
						
							|  |  |  |                     allow_err!(tx_res.send(format!("Failed to start x session thread, {}", e)));
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // wait x11
 | 
					
						
							|  |  |  |         match rx_res.recv_timeout(Duration::from_millis(10_000)) {
 | 
					
						
							|  |  |  |             Ok(res) => {
 | 
					
						
							|  |  |  |                 if res == "" {
 | 
					
						
							|  |  |  |                     Ok(())
 | 
					
						
							|  |  |  |                 } else {
 | 
					
						
							|  |  |  |                     bail!(res)
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             Err(e) => {
 | 
					
						
							|  |  |  |                 bail!("Failed to recv x11 result {}", e)
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     #[inline]
 | 
					
						
							|  |  |  |     fn display_from_num(num: u32) -> String {
 | 
					
						
							|  |  |  |         format!(":{num}")
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     fn start_x_session_thread(
 | 
					
						
							|  |  |  |         tx_res: SyncSender<String>,
 | 
					
						
							|  |  |  |         is_child_running: Arc<AtomicBool>,
 | 
					
						
							|  |  |  |         uid: u32,
 | 
					
						
							|  |  |  |         gid: u32,
 | 
					
						
							|  |  |  |         display_num: u32,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         username: String,
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         password: String,
 | 
					
						
							|  |  |  |         envs: HashMap<&str, String>,
 | 
					
						
							|  |  |  |     ) -> ResultType<()> {
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |         let mut client = pam::Client::with_password(&pam_get_service_name())?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         client
 | 
					
						
							|  |  |  |             .conversation_mut()
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             .set_credentials(&username, &password);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         client.authenticate()?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         client.set_item(pam::PamItemType::TTY, &Self::display_from_num(display_num))?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         client.open_session()?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // fixme: FreeBSD kernel needs to login here.
 | 
					
						
							|  |  |  |         // see: https://github.com/neutrinolabs/xrdp/blob/a64573b596b5fb07ca3a51590c5308d621f7214e/sesman/session.c#L556
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let (child_xorg, child_wm) = Self::start_x11(uid, gid, username, display_num, &envs)?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         is_child_running.store(true, Ordering::SeqCst);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         log::info!("Start xorg and wm done, notify and wait xtop x11");
 | 
					
						
							|  |  |  |         allow_err!(tx_res.send("".to_owned()));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Self::wait_stop_x11(child_xorg, child_wm);
 | 
					
						
							|  |  |  |         log::info!("Wait x11 stop done");
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn wait_xorg_exit(child_xorg: &mut Child) -> ResultType<String> {
 | 
					
						
							|  |  |  |         if let Ok(_) = child_xorg.kill() {
 | 
					
						
							|  |  |  |             for _ in 0..3 {
 | 
					
						
							|  |  |  |                 match child_xorg.try_wait() {
 | 
					
						
							|  |  |  |                     Ok(Some(status)) => return Ok(format!("Xorg exit with {}", status)),
 | 
					
						
							|  |  |  |                     Ok(None) => {}
 | 
					
						
							|  |  |  |                     Err(e) => {
 | 
					
						
							|  |  |  |                         // fatal error
 | 
					
						
							|  |  |  |                         log::error!("Failed to wait xorg process, {}", e);
 | 
					
						
							|  |  |  |                         bail!("Failed to wait xorg process, {}", e)
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 std::thread::sleep(std::time::Duration::from_millis(1_000));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             log::error!("Failed to wait xorg process, not exit");
 | 
					
						
							|  |  |  |             bail!("Failed to wait xorg process, not exit")
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             Ok("Xorg is already exited".to_owned())
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |     fn add_xauth_cookie(
 | 
					
						
							|  |  |  |         file: &str,
 | 
					
						
							|  |  |  |         display: &str,
 | 
					
						
							|  |  |  |         uid: u32,
 | 
					
						
							|  |  |  |         gid: u32,
 | 
					
						
							|  |  |  |         envs: &HashMap<&str, String>,
 | 
					
						
							|  |  |  |     ) -> ResultType<()> {
 | 
					
						
							|  |  |  |         let randstr = (0..16)
 | 
					
						
							|  |  |  |             .map(|_| format!("{:02x}", random::<u8>()))
 | 
					
						
							|  |  |  |             .collect::<String>();
 | 
					
						
							|  |  |  |         let output = Command::new("xauth")
 | 
					
						
							|  |  |  |             .uid(uid)
 | 
					
						
							|  |  |  |             .gid(gid)
 | 
					
						
							|  |  |  |             .envs(envs)
 | 
					
						
							|  |  |  |             .args(vec!["-q", "-f", file, "add", display, ".", &randstr])
 | 
					
						
							|  |  |  |             .output()?;
 | 
					
						
							|  |  |  |         // xauth run success, even the following error occurs.
 | 
					
						
							|  |  |  |         // Ok(Output { status: ExitStatus(unix_wait_status(0)), stdout: "", stderr: "xauth:  file .Xauthority does not exist\n" })
 | 
					
						
							|  |  |  |         let errmsg = String::from_utf8_lossy(&output.stderr).to_string();
 | 
					
						
							|  |  |  |         if !errmsg.is_empty() {
 | 
					
						
							|  |  |  |             if !errmsg.contains("does not exist") {
 | 
					
						
							|  |  |  |                 bail!("Failed to launch xauth, {}", errmsg)
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         Ok(())
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn wait_x_server_running(pid: u32, display_num: u32, max_wait_secs: u64) -> ResultType<()> {
 | 
					
						
							|  |  |  |         let wait_begin = Instant::now();
 | 
					
						
							|  |  |  |         loop {
 | 
					
						
							| 
									
										
										
										
											2023-03-31 17:49:35 +08:00
										 |  |  |             if run_cmds(&format!("ls /proc/{}", pid))?.is_empty() {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                 bail!("X server exit");
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if Self::is_x_server_running(display_num) {
 | 
					
						
							|  |  |  |                 return Ok(());
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             if wait_begin.elapsed().as_secs() > max_wait_secs {
 | 
					
						
							|  |  |  |                 bail!("Failed to wait xserver after {} seconds", max_wait_secs);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             std::thread::sleep(Duration::from_millis(300));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     fn start_x11(
 | 
					
						
							|  |  |  |         uid: u32,
 | 
					
						
							|  |  |  |         gid: u32,
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         username: String,
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         display_num: u32,
 | 
					
						
							|  |  |  |         envs: &HashMap<&str, String>,
 | 
					
						
							|  |  |  |     ) -> ResultType<(Child, Child)> {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         log::debug!("envs of user {}: {:?}", &username, &envs);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let xauth = Self::get_xauth();
 | 
					
						
							|  |  |  |         let display = Self::display_from_num(display_num);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Self::add_xauth_cookie(&xauth, &display, uid, gid, &envs)?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Start Xorg
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let mut child_xorg = Self::start_x_server(&xauth, &display, uid, gid, &envs)?;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         log::info!("xorg started, wait 10 secs to ensuer x server is running");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let max_wait_secs = 10;
 | 
					
						
							|  |  |  |         // wait x server running
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         if let Err(e) = Self::wait_x_server_running(child_xorg.id(), display_num, max_wait_secs) {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             match Self::wait_xorg_exit(&mut child_xorg) {
 | 
					
						
							|  |  |  |                 Ok(msg) => log::info!("{}", msg),
 | 
					
						
							|  |  |  |                 Err(e) => {
 | 
					
						
							|  |  |  |                     log::error!("{}", e);
 | 
					
						
							|  |  |  |                     Self::fatal_exit();
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             bail!(e)
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         log::info!(
 | 
					
						
							|  |  |  |             "xorg is running, start x window manager with DISPLAY: {}, XAUTHORITY: {}",
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |             &display,
 | 
					
						
							|  |  |  |             &xauth
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         std::env::set_var("DISPLAY", &display);
 | 
					
						
							|  |  |  |         std::env::set_var("XAUTHORITY", &xauth);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         // start window manager (startwm.sh)
 | 
					
						
							|  |  |  |         let child_wm = match Self::start_x_window_manager(uid, gid, &envs) {
 | 
					
						
							|  |  |  |             Ok(c) => c,
 | 
					
						
							|  |  |  |             Err(e) => {
 | 
					
						
							|  |  |  |                 match Self::wait_xorg_exit(&mut child_xorg) {
 | 
					
						
							|  |  |  |                     Ok(msg) => log::info!("{}", msg),
 | 
					
						
							|  |  |  |                     Err(e) => {
 | 
					
						
							|  |  |  |                         log::error!("{}", e);
 | 
					
						
							|  |  |  |                         Self::fatal_exit();
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 bail!(e)
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         };
 | 
					
						
							|  |  |  |         log::info!("x window manager is started");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Ok((child_xorg, child_wm))
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn try_wait_x11_child_exit(child_xorg: &mut Child, child_wm: &mut Child) -> bool {
 | 
					
						
							|  |  |  |         match child_xorg.try_wait() {
 | 
					
						
							|  |  |  |             Ok(Some(status)) => {
 | 
					
						
							|  |  |  |                 log::info!("Xorg exit with {}", status);
 | 
					
						
							|  |  |  |                 return true;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             Ok(None) => {}
 | 
					
						
							|  |  |  |             Err(e) => log::error!("Failed to wait xorg process, {}", e),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         match child_wm.try_wait() {
 | 
					
						
							|  |  |  |             Ok(Some(status)) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-29 18:31:42 +08:00
										 |  |  |                 // Logout may result "wm exit with signal: 11 (SIGSEGV) (core dumped)"
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 log::info!("wm exit with {}", status);
 | 
					
						
							|  |  |  |                 return true;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             Ok(None) => {}
 | 
					
						
							|  |  |  |             Err(e) => log::error!("Failed to wait xorg process, {}", e),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         false
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn wait_x11_children_exit(child_xorg: &mut Child, child_wm: &mut Child) {
 | 
					
						
							|  |  |  |         log::debug!("Try kill child process xorg");
 | 
					
						
							|  |  |  |         if let Ok(_) = child_xorg.kill() {
 | 
					
						
							|  |  |  |             let mut exited = false;
 | 
					
						
							|  |  |  |             for _ in 0..2 {
 | 
					
						
							|  |  |  |                 match child_xorg.try_wait() {
 | 
					
						
							|  |  |  |                     Ok(Some(status)) => {
 | 
					
						
							|  |  |  |                         log::info!("Xorg exit with {}", status);
 | 
					
						
							|  |  |  |                         exited = true;
 | 
					
						
							|  |  |  |                         break;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     Ok(None) => {}
 | 
					
						
							|  |  |  |                     Err(e) => {
 | 
					
						
							|  |  |  |                         log::error!("Failed to wait xorg process, {}", e);
 | 
					
						
							|  |  |  |                         Self::fatal_exit();
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 std::thread::sleep(std::time::Duration::from_millis(1_000));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             if !exited {
 | 
					
						
							|  |  |  |                 log::error!("Failed to wait child xorg, after kill()");
 | 
					
						
							|  |  |  |                 // try kill -9?
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         log::debug!("Try kill child process wm");
 | 
					
						
							|  |  |  |         if let Ok(_) = child_wm.kill() {
 | 
					
						
							|  |  |  |             let mut exited = false;
 | 
					
						
							|  |  |  |             for _ in 0..2 {
 | 
					
						
							|  |  |  |                 match child_wm.try_wait() {
 | 
					
						
							|  |  |  |                     Ok(Some(status)) => {
 | 
					
						
							| 
									
										
										
										
											2023-03-29 18:31:42 +08:00
										 |  |  |                         // Logout may result "wm exit with signal: 11 (SIGSEGV) (core dumped)"
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                         log::info!("wm exit with {}", status);
 | 
					
						
							|  |  |  |                         exited = true;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     Ok(None) => {}
 | 
					
						
							|  |  |  |                     Err(e) => {
 | 
					
						
							|  |  |  |                         log::error!("Failed to wait wm process, {}", e);
 | 
					
						
							|  |  |  |                         Self::fatal_exit();
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 std::thread::sleep(std::time::Duration::from_millis(1_000));
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             if !exited {
 | 
					
						
							|  |  |  |                 log::error!("Failed to wait child xorg, after kill()");
 | 
					
						
							|  |  |  |                 // try kill -9?
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn try_wait_stop_x11(child_xorg: &mut Child, child_wm: &mut Child) -> bool {
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         let mut desktop_manager = DESKTOP_MANAGER.lock().unwrap();
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         let mut exited = true;
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |         if let Some(desktop_manager) = &mut (*desktop_manager) {
 | 
					
						
							|  |  |  |             if desktop_manager.child_exit.load(Ordering::SeqCst) {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 exited = true;
 | 
					
						
							|  |  |  |             } else {
 | 
					
						
							|  |  |  |                 exited = Self::try_wait_x11_child_exit(child_xorg, child_wm);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             if exited {
 | 
					
						
							| 
									
										
										
										
											2023-03-29 18:31:42 +08:00
										 |  |  |                 log::debug!("Wait x11 children exiting");
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 Self::wait_x11_children_exit(child_xorg, child_wm);
 | 
					
						
							| 
									
										
										
										
											2023-03-28 22:20:15 +08:00
										 |  |  |                 desktop_manager
 | 
					
						
							|  |  |  |                     .is_child_running
 | 
					
						
							|  |  |  |                     .store(false, Ordering::SeqCst);
 | 
					
						
							|  |  |  |                 desktop_manager.child_exit.store(true, Ordering::SeqCst);
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         exited
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn wait_stop_x11(mut child_xorg: Child, mut child_wm: Child) {
 | 
					
						
							|  |  |  |         loop {
 | 
					
						
							|  |  |  |             if Self::try_wait_stop_x11(&mut child_xorg, &mut child_wm) {
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             std::thread::sleep(Duration::from_millis(super::SERVICE_INTERVAL));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |     fn get_xorg() -> &'static str {
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         // Fedora 26 or later
 | 
					
						
							|  |  |  |         let xorg = "/usr/libexec/Xorg";
 | 
					
						
							|  |  |  |         if Path::new(xorg).is_file() {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |             return xorg;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         // Debian 9 or later
 | 
					
						
							|  |  |  |         let xorg = "/usr/lib/xorg/Xorg";
 | 
					
						
							|  |  |  |         if Path::new(xorg).is_file() {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |             return xorg;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         // Ubuntu 16.04 or later
 | 
					
						
							|  |  |  |         let xorg = "/usr/lib/xorg/Xorg";
 | 
					
						
							|  |  |  |         if Path::new(xorg).is_file() {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |             return xorg;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         // Arch Linux
 | 
					
						
							|  |  |  |         let xorg = "/usr/lib/xorg-server/Xorg";
 | 
					
						
							|  |  |  |         if Path::new(xorg).is_file() {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |             return xorg;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         // Arch Linux
 | 
					
						
							|  |  |  |         let xorg = "/usr/lib/Xorg";
 | 
					
						
							|  |  |  |         if Path::new(xorg).is_file() {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |             return xorg;
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         // CentOS 7 /usr/bin/Xorg or param=Xorg
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         log::warn!("Failed to find xorg, use default Xorg.\n Please add \"allowed_users=anybody\" to \"/etc/X11/Xwrapper.config\".");
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |         "Xorg"
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn start_x_server(
 | 
					
						
							|  |  |  |         xauth: &str,
 | 
					
						
							|  |  |  |         display: &str,
 | 
					
						
							|  |  |  |         uid: u32,
 | 
					
						
							|  |  |  |         gid: u32,
 | 
					
						
							|  |  |  |         envs: &HashMap<&str, String>,
 | 
					
						
							|  |  |  |     ) -> ResultType<Child> {
 | 
					
						
							| 
									
										
										
										
											2023-04-01 00:28:56 +08:00
										 |  |  |         let xorg = Self::get_xorg();
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         log::info!("Use xorg: {}", &xorg);
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |         let app_name = crate::get_app_name().to_lowercase();
 | 
					
						
							|  |  |  |         let conf = format!("/etc/{app_name}/xorg.conf");
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |         match Command::new(xorg)
 | 
					
						
							|  |  |  |             .envs(envs)
 | 
					
						
							|  |  |  |             .uid(uid)
 | 
					
						
							|  |  |  |             .gid(gid)
 | 
					
						
							|  |  |  |             .args(vec![
 | 
					
						
							|  |  |  |                 "-noreset",
 | 
					
						
							|  |  |  |                 "+extension",
 | 
					
						
							|  |  |  |                 "GLX",
 | 
					
						
							|  |  |  |                 "+extension",
 | 
					
						
							|  |  |  |                 "RANDR",
 | 
					
						
							|  |  |  |                 "+extension",
 | 
					
						
							|  |  |  |                 "RENDER",
 | 
					
						
							|  |  |  |                 "-config",
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |                 conf.as_ref(),
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |                 "-auth",
 | 
					
						
							|  |  |  |                 xauth,
 | 
					
						
							|  |  |  |                 display,
 | 
					
						
							|  |  |  |             ])
 | 
					
						
							|  |  |  |             .spawn()
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             Ok(c) => Ok(c),
 | 
					
						
							|  |  |  |             Err(e) => {
 | 
					
						
							|  |  |  |                 bail!("Failed to start Xorg with display {}, {}", display, e);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn start_x_window_manager(
 | 
					
						
							|  |  |  |         uid: u32,
 | 
					
						
							|  |  |  |         gid: u32,
 | 
					
						
							|  |  |  |         envs: &HashMap<&str, String>,
 | 
					
						
							|  |  |  |     ) -> ResultType<Child> {
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |         let app_name = crate::get_app_name().to_lowercase();
 | 
					
						
							|  |  |  |         match Command::new(&format!("/etc/{app_name}/startwm.sh"))
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |             .envs(envs)
 | 
					
						
							|  |  |  |             .uid(uid)
 | 
					
						
							|  |  |  |             .gid(gid)
 | 
					
						
							|  |  |  |             .spawn()
 | 
					
						
							|  |  |  |         {
 | 
					
						
							|  |  |  |             Ok(c) => Ok(c),
 | 
					
						
							|  |  |  |             Err(e) => {
 | 
					
						
							|  |  |  |                 bail!("Failed to start window manager, {}", e);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn stop_children(&mut self) {
 | 
					
						
							|  |  |  |         self.child_exit.store(true, Ordering::SeqCst);
 | 
					
						
							|  |  |  |         for _i in 1..10 {
 | 
					
						
							|  |  |  |             if !self.is_child_running.load(Ordering::SeqCst) {
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             std::thread::sleep(Duration::from_millis(super::SERVICE_INTERVAL));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         if self.is_child_running.load(Ordering::SeqCst) {
 | 
					
						
							|  |  |  |             log::warn!("xdesktop child is still running!");
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  | fn pam_get_service_name() -> String {
 | 
					
						
							|  |  |  |     let app_name = crate::get_app_name().to_lowercase();
 | 
					
						
							|  |  |  |     if Path::new(&format!("/etc/pam.d/{app_name}")).is_file() {
 | 
					
						
							|  |  |  |         app_name
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     } else {
 | 
					
						
							| 
									
										
										
										
											2024-05-06 22:02:13 +08:00
										 |  |  |         "gdm".to_owned()
 | 
					
						
							| 
									
										
										
										
											2023-03-22 17:01:11 +08:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 |