Construct commands separately to support flatpak-spawn and update async_process for flatpak-spawn

This commit is contained in:
Charlie Le 2024-11-20 11:14:12 -05:00
commit dd4db97215
15 changed files with 80 additions and 61 deletions

View file

@ -2,6 +2,7 @@ use std::ffi::OsStr;
use std::process::Stdio; use std::process::Stdio;
use std::{collections::HashMap, os::unix::process::ExitStatusExt}; use std::{collections::HashMap, os::unix::process::ExitStatusExt};
use tokio::process::Command; use tokio::process::Command;
use crate::is_flatpak::IS_FLATPAK;
pub struct AsyncProcessOut { pub struct AsyncProcessOut {
pub exit_code: i32, pub exit_code: i32,
@ -13,10 +14,25 @@ pub async fn async_process<S: AsRef<OsStr>, T: AsRef<OsStr>>(
cmd: S, cmd: S,
args: Option<&[T]>, args: Option<&[T]>,
env: Option<HashMap<String, String>>, env: Option<HashMap<String, String>>,
host_spawn: Option<bool>,
) -> anyhow::Result<AsyncProcessOut> { ) -> anyhow::Result<AsyncProcessOut> {
let cmd = Command::new(cmd) let mut command: Command;
.args(args.unwrap_or_default()) if *IS_FLATPAK && host_spawn.unwrap_or_default() {
.envs(env.unwrap_or_default()) let mut flatpak_env = vec![];
for (key, value) in env.unwrap_or_default() {
flatpak_env.push(format!("--env={}={}", key, value));
}
command = Command::new("flatpak-spawn");
command.arg("--host")
.args(flatpak_env)
.arg(cmd)
.args(args.unwrap_or_default());
} else {
command = Command::new(cmd);
command.args(args.unwrap_or_default()).envs(env.unwrap_or_default());
}
let cmd = command
.stdin(Stdio::piped()) .stdin(Stdio::piped())
.stdout(Stdio::piped()) .stdout(Stdio::piped())
.stderr(Stdio::piped()) .stderr(Stdio::piped())

View file

@ -26,7 +26,7 @@ impl Cmake {
} }
} }
args.push(self.source_dir.to_string_lossy().to_string()); args.push(self.source_dir.to_string_lossy().to_string());
WorkerJob::new_cmd(self.env.clone(), "cmake".into(), Some(args)) WorkerJob::new_cmd(self.env.clone(), "cmake".into(), Some(args), Some(false))
} }
pub fn get_build_job(&self) -> WorkerJob { pub fn get_build_job(&self) -> WorkerJob {
@ -37,6 +37,7 @@ impl Cmake {
"--build".into(), "--build".into(),
self.build_dir.to_string_lossy().to_string(), self.build_dir.to_string_lossy().to_string(),
]), ]),
Some(false),
) )
} }
@ -48,6 +49,7 @@ impl Cmake {
"--install".into(), "--install".into(),
self.build_dir.to_string_lossy().to_string(), self.build_dir.to_string_lossy().to_string(),
]), ]),
Some(false),
) )
} }
} }

View file

@ -13,7 +13,7 @@ impl Git {
fn cmd(&self, args: Vec<String>) -> WorkerJob { fn cmd(&self, args: Vec<String>) -> WorkerJob {
let mut nargs = vec!["-C".into(), self.dir.to_string_lossy().to_string()]; let mut nargs = vec!["-C".into(), self.dir.to_string_lossy().to_string()];
nargs.extend(args); nargs.extend(args);
WorkerJob::new_cmd(None, "git".into(), Some(nargs)) WorkerJob::new_cmd(None, "git".into(), Some(nargs), Some(false))
} }
fn get_repo(&self) -> String { fn get_repo(&self) -> String {
@ -84,6 +84,7 @@ impl Git {
self.dir.to_string_lossy().to_string(), self.dir.to_string_lossy().to_string(),
"--recurse-submodules".into(), "--recurse-submodules".into(),
]), ]),
Some(false),
) )
} }

View file

@ -79,6 +79,7 @@ pub fn get_build_basalt_jobs(profile: &Profile, clean_build: bool) -> VecDeque<W
.to_string_lossy() .to_string_lossy()
.to_string(), .to_string(),
]), ]),
Some(false),
)); ));
jobs.push_back(WorkerJob::new_cmd( jobs.push_back(WorkerJob::new_cmd(
None, None,
@ -100,6 +101,7 @@ pub fn get_build_basalt_jobs(profile: &Profile, clean_build: bool) -> VecDeque<W
.to_string_lossy() .to_string_lossy()
.to_string(), .to_string(),
]), ]),
Some(false),
)); ));
jobs jobs

View file

@ -21,6 +21,7 @@ pub fn get_build_mercury_jobs(profile: &Profile) -> VecDeque<WorkerJob> {
profile.prefix.to_string_lossy().to_string(), profile.prefix.to_string_lossy().to_string(),
get_cache_dir().to_string_lossy().to_string(), get_cache_dir().to_string_lossy().to_string(),
]), ]),
Some(false),
)); ));
jobs jobs

View file

@ -62,6 +62,7 @@ pub fn get_build_openhmd_jobs(profile: &Profile, clean_build: bool) -> VecDeque<
.to_string_lossy() .to_string_lossy()
.to_string(), .to_string(),
]), ]),
Some(false),
)); ));
} }
// build job // build job
@ -69,6 +70,7 @@ pub fn get_build_openhmd_jobs(profile: &Profile, clean_build: bool) -> VecDeque<
None, None,
"ninja".into(), "ninja".into(),
Some(vec!["-C".into(), build_dir.to_string_lossy().to_string()]), Some(vec!["-C".into(), build_dir.to_string_lossy().to_string()]),
Some(false),
)); ));
// install job // install job
jobs.push_back(WorkerJob::new_cmd( jobs.push_back(WorkerJob::new_cmd(
@ -79,6 +81,7 @@ pub fn get_build_openhmd_jobs(profile: &Profile, clean_build: bool) -> VecDeque<
build_dir.to_string_lossy().to_string(), build_dir.to_string_lossy().to_string(),
"install".into(), "install".into(),
]), ]),
Some(false),
)); ));
jobs jobs

View file

@ -34,7 +34,7 @@ use crate::{
}, },
}, is_flatpak::IS_FLATPAK, linux_distro::LinuxDistro, openxr_prober::is_openxr_ready, paths::get_data_dir, profile::{Profile, XRServiceType}, stateless_action, steam_linux_runtime_injector::{ }, is_flatpak::IS_FLATPAK, linux_distro::LinuxDistro, openxr_prober::is_openxr_ready, paths::get_data_dir, profile::{Profile, XRServiceType}, stateless_action, steam_linux_runtime_injector::{
restore_runtime_entrypoint, set_runtime_entrypoint_launch_opts_from_profile, restore_runtime_entrypoint, set_runtime_entrypoint_launch_opts_from_profile,
}, util::file_utils::{setcap_cap_sys_nice_eip, setcap_cap_sys_nice_eip_cmd}, vulkaninfo::VulkanInfo, xr_devices::XRDevice }, util::{cmd_utils::make_command, file_utils::{setcap_cap_sys_nice_eip, setcap_cap_sys_nice_eip_cmd}}, vulkaninfo::VulkanInfo, xr_devices::XRDevice
}; };
use adw::{prelude::*, ResponseAppearance}; use adw::{prelude::*, ResponseAppearance};
use gtk::glib::{self, clone}; use gtk::glib::{self, clone};
@ -43,7 +43,7 @@ use relm4::{
new_action_group, new_stateful_action, new_stateless_action, new_action_group, new_stateful_action, new_stateless_action,
prelude::*, prelude::*,
}; };
use std::{collections::{HashMap, VecDeque}, fs::remove_file, process::Command, time::Duration}; use std::{collections::VecDeque, fs::remove_file, time::Duration};
pub struct App { pub struct App {
application: adw::Application, application: adw::Application,
@ -163,12 +163,7 @@ impl App {
self.debug_view.sender().emit(DebugViewMsg::ClearLog); self.debug_view.sender().emit(DebugViewMsg::ClearLog);
self.xr_devices = vec![]; self.xr_devices = vec![];
if *IS_FLATPAK { if *IS_FLATPAK {
Command::new("flatpak-spawn") make_command(String::from("rm"), Some(&[prof.xrservice_type.ipc_file_path().to_str().unwrap()]), None, Some(true))
.args(vec![
"--host",
"rm",
prof.xrservice_type.ipc_file_path().to_str().unwrap()
])
.output() .output()
.expect("Failed to remove xrservice IPC file"); .expect("Failed to remove xrservice IPC file");
} else { } else {
@ -216,28 +211,12 @@ impl App {
let prof = self.get_selected_profile(); let prof = self.get_selected_profile();
if let Some(autostart_cmd) = &prof.autostart_command { if let Some(autostart_cmd) = &prof.autostart_command {
let mut jobs = VecDeque::new(); let mut jobs = VecDeque::new();
jobs.push_back(WorkerJob::new_cmd(
if *IS_FLATPAK { Some(prof.environment.clone()),
let mut args = vec![String::from("--host")]; "sh".into(),
for (key, value) in &prof.environment { Some(vec!["-c".into(), autostart_cmd.clone()]),
args.push(format!("--env={}={}", key, value)); Some(true)
} ));
args.push(String::from("sh"));
args.push(String::from("-c"));
args.push(autostart_cmd.clone());
jobs.push_back(WorkerJob::new_cmd(
Some(HashMap::new()),
"flatpak-spawn".into(),
Some(args),
));
} else {
jobs.push_back(WorkerJob::new_cmd(
Some(prof.environment.clone()),
"sh".into(),
Some(vec!["-c".into(), autostart_cmd.clone()]),
));
}
let autostart_worker = JobWorker::new(jobs, sender.input_sender(), |msg| match msg { let autostart_worker = JobWorker::new(jobs, sender.input_sender(), |msg| match msg {
JobWorkerOut::Log(rows) => Msg::OnServiceLog(rows), JobWorkerOut::Log(rows) => Msg::OnServiceLog(rows),
JobWorkerOut::Exit(code) => Msg::OnAutostartExit(code), JobWorkerOut::Exit(code) => Msg::OnAutostartExit(code),

View file

@ -218,6 +218,7 @@ impl AsyncComponent for InstallWivrnBox {
"adb", "adb",
Some(&["install", "-r", "-g", apk_path.to_string_lossy().to_string().as_str()]), Some(&["install", "-r", "-g", apk_path.to_string_lossy().to_string().as_str()]),
None, None,
None
) )
.await .await
{ {

View file

@ -3,12 +3,12 @@ use super::{
state::JobWorkerState, state::JobWorkerState,
}; };
use crate::{ use crate::{
is_flatpak::IS_FLATPAK, profile::{LighthouseDriver, Profile}, ui::SENDER_IO_ERR_MSG profile::{LighthouseDriver, Profile}, ui::SENDER_IO_ERR_MSG, util::cmd_utils::make_command
}; };
use nix::unistd::Pid; use nix::unistd::Pid;
use relm4::{prelude::*, Worker}; use relm4::{prelude::*, Worker};
use std::{ use std::{
collections::{HashMap, VecDeque}, io::{BufRead, BufReader}, mem, os::unix::process::ExitStatusExt, process::{Command, Stdio}, sync::{Arc, Mutex}, thread collections::VecDeque, io::{BufRead, BufReader}, mem, os::unix::process::ExitStatusExt, process::Stdio, sync::{Arc, Mutex}, thread
}; };
macro_rules! logger_thread { macro_rules! logger_thread {
@ -85,9 +85,7 @@ impl Worker for InternalJobWorker {
match &mut job { match &mut job {
WorkerJob::Cmd(data) => { WorkerJob::Cmd(data) => {
let data = data.clone(); let data = data.clone();
if let Ok(mut cmd) = Command::new(data.command) if let Ok(mut cmd) = make_command(data.command, Some(&data.args), Some(data.environment), Some(data.host_spawn))
.args(data.args)
.envs(data.environment)
.stdin(Stdio::piped()) .stdin(Stdio::piped())
.stderr(Stdio::piped()) .stderr(Stdio::piped())
.stdout(Stdio::piped()) .stdout(Stdio::piped())
@ -210,21 +208,12 @@ impl InternalJobWorker {
vec![], vec![],
), ),
}; };
let mut data = CmdWorkerData { let data = CmdWorkerData {
environment: env.clone(), environment: env.clone(),
command: command.clone(), command: command.clone(),
args: args.clone(), args: args.clone(),
host_spawn: true,
}; };
if *IS_FLATPAK {
data.environment = HashMap::new();
data.command = String::from("flatpak-spawn");
data.args = vec![String::from("--host")];
for (key, value) in env {
data.args.push(format!("--env={}={}", key, value));
}
data.args.push(command);
data.args.extend(args);
}
let mut jobs = VecDeque::new(); let mut jobs = VecDeque::new();
jobs.push_back(WorkerJob::Cmd(data)); jobs.push_back(WorkerJob::Cmd(data));
Self::builder().detach_worker(JobWorkerInit { jobs, state }) Self::builder().detach_worker(JobWorkerInit { jobs, state })

View file

@ -7,6 +7,7 @@ pub struct CmdWorkerData {
pub environment: HashMap<String, String>, pub environment: HashMap<String, String>,
pub command: String, pub command: String,
pub args: Vec<String>, pub args: Vec<String>,
pub host_spawn: bool,
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -38,11 +39,13 @@ impl WorkerJob {
env: Option<HashMap<String, String>>, env: Option<HashMap<String, String>>,
cmd: String, cmd: String,
args: Option<Vec<String>>, args: Option<Vec<String>>,
host_spawn: Option<bool>
) -> Self { ) -> Self {
Self::Cmd(CmdWorkerData { Self::Cmd(CmdWorkerData {
environment: env.unwrap_or_default(), environment: env.unwrap_or_default(),
command: cmd, command: cmd,
args: args.unwrap_or_default(), args: args.unwrap_or_default(),
host_spawn: host_spawn.unwrap_or_default(),
}) })
} }

View file

@ -159,6 +159,7 @@ impl SimpleComponent for SteamVrCalibrationBox {
Some(env.clone()), Some(env.clone()),
vrcmd.clone(), vrcmd.clone(),
Some(vec!["--pollposes".into()]), Some(vec!["--pollposes".into()]),
Some(false),
)); ));
JobWorker::new(jobs, sender.input_sender(), |msg| match msg { JobWorker::new(jobs, sender.input_sender(), |msg| match msg {
JobWorkerOut::Log(_) => Self::Input::NoOp, JobWorkerOut::Log(_) => Self::Input::NoOp,
@ -178,6 +179,7 @@ impl SimpleComponent for SteamVrCalibrationBox {
Some(env), Some(env),
vrcmd, vrcmd,
Some(vec!["--resetroomsetup".into()]), Some(vec!["--resetroomsetup".into()]),
Some(false),
)); ));
JobWorker::new(jobs, sender.input_sender(), |msg| match msg { JobWorker::new(jobs, sender.input_sender(), |msg| match msg {
JobWorkerOut::Log(_) => Self::Input::NoOp, JobWorkerOut::Log(_) => Self::Input::NoOp,

View file

@ -136,6 +136,7 @@ impl AsyncComponent for WivrnWiredStartBox {
"adb shell am start -a android.intent.action.VIEW -d \"wivrn+tcp://127.0.0.1\" $PACKAGE" "adb shell am start -a android.intent.action.VIEW -d \"wivrn+tcp://127.0.0.1\" $PACKAGE"
) )
]), ]),
None,
None None
).await { ).await {
Ok(out) if out.exit_code == 0 => Ok(out) if out.exit_code == 0 =>

26
src/util/cmd_utils.rs Normal file
View file

@ -0,0 +1,26 @@
use std::{collections::HashMap, ffi::OsStr, process::Command};
use crate::is_flatpak::IS_FLATPAK;
pub fn make_command<S: AsRef<OsStr>, T: AsRef<OsStr>>(
cmd: S,
args: Option<&[T]>,
env: Option<HashMap<String, String>>,
host_spawn: Option<bool>,
) -> Command {
if *IS_FLATPAK && host_spawn.unwrap_or_default() {
let mut flatpak_env = vec![];
for (key, value) in env.unwrap_or_default() {
flatpak_env.push(format!("--env={}={}", key, value));
}
let mut command = Command::new("flatpak-spawn");
command.arg("--host")
.args(flatpak_env)
.arg(cmd)
.args(args.unwrap_or_default());
return command;
}
let mut command = Command::new(cmd);
command.args(args.unwrap_or_default()).envs(env.unwrap_or_default());
command
}

View file

@ -1,4 +1,4 @@
use crate::{async_process::async_process, is_flatpak::IS_FLATPAK, profile::Profile}; use crate::{async_process::async_process, profile::Profile};
use anyhow::bail; use anyhow::bail;
use nix::{ use nix::{
errno::Errno, errno::Errno,
@ -81,15 +81,7 @@ pub fn setcap_cap_sys_nice_eip_cmd(profile: &Profile) -> Vec<String> {
} }
pub async fn setcap_cap_sys_nice_eip(profile: &Profile) { pub async fn setcap_cap_sys_nice_eip(profile: &Profile) {
let setcap_cmd = setcap_cap_sys_nice_eip_cmd(profile); if let Err(e) = async_process("pkexec", Some(&setcap_cap_sys_nice_eip_cmd(profile)), None, Some(true)).await
if *IS_FLATPAK {
let mut flatpak_cmd = vec!["--host".into(), "pkexec".into()];
flatpak_cmd.extend(setcap_cmd);
if let Err(e) = async_process("flatpak-spawn", Some(&flatpak_cmd), None).await
{
eprintln!("Error: failed running setcap: {e}");
}
} else if let Err(e) = async_process("pkexec", Some(&setcap_cmd), None).await
{ {
eprintln!("Error: failed running setcap: {e}"); eprintln!("Error: failed running setcap: {e}");
} }

View file

@ -1,3 +1,4 @@
pub mod file_utils; pub mod file_utils;
pub mod hash; pub mod hash;
pub mod steamvr_utils; pub mod steamvr_utils;
pub mod cmd_utils;