Skip to content

Commit a30a2e4

Browse files
committed
Support for running with specific user id and group id
Also some improvements related to typesafety. Note that with this change, it makes it feature wise complete with the Haskell's pid1.
1 parent 16677ac commit a30a2e4

File tree

2 files changed

+40
-8
lines changed

2 files changed

+40
-8
lines changed

README.md

+20
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,26 @@ RUN chmod +x /usr/bin/pid1
6262
ENTRYPOINT [ "pid1" ]
6363
```
6464

65+
Various options supported by the binary:
66+
67+
``` shellsession
68+
pid1 --help
69+
Usage:
70+
71+
Arguments:
72+
<COMMAND> Process to run
73+
[ARGS]... Arguments to the process
74+
75+
Options:
76+
-w, --workdir <DIR> Specify working direcory
77+
-t, --timeout <TIMEOUT> Timeout (in seconds) to wait for child proess to exit [default: 2]
78+
-v, --verbose Turn on verbose output
79+
-e, --env <ENV> Override environment variables. Can specify multiple times
80+
-u, --user-id <USER_ID> Run command with user ID
81+
-g, --group-id <GROUP_ID> Run command with group ID
82+
-h, --help Print help
83+
```
84+
6585
## Development
6686

6787
The testing steps are documented in [Development.md](./Development.md). We only have

pid1-exe/src/cli.rs

+20-8
Original file line numberDiff line numberDiff line change
@@ -26,19 +26,34 @@ pub(crate) struct Pid1App {
2626
/// Override environment variables. Can specify multiple times.
2727
#[arg(short, long, value_parser=parse_key_val::<OsString, OsString>)]
2828
pub(crate) env: Vec<(OsString, OsString)>,
29-
/// Process arguments
29+
/// Run command with user ID
30+
#[arg(short, long, value_name = "USER_ID")]
31+
user_id: Option<u32>,
32+
/// Run command with group ID
33+
#[arg(short, long, value_name = "GROUP_ID")]
34+
group_id: Option<u32>,
35+
/// Process to run
3036
#[arg(required = true)]
31-
child_process: Vec<String>,
37+
pub(crate) command: String,
38+
/// Arguments to the process
39+
#[arg(required = false)]
40+
pub(crate) args: Vec<String>,
3241
}
3342

3443
impl Pid1App {
3544
#[cfg(target_family = "unix")]
3645
pub(crate) fn run(self) -> ! {
37-
let mut child = std::process::Command::new(&self.child_process[0]);
38-
let child = child.args(&self.child_process[1..]);
46+
let mut child = std::process::Command::new(&self.command);
47+
let child = child.args(&self.args[..]);
3948
if let Some(workdir) = &self.workdir {
4049
child.current_dir(workdir);
4150
}
51+
if let Some(user_id) = &self.user_id {
52+
child.uid(*user_id);
53+
}
54+
if let Some(group_id) = &self.group_id {
55+
child.gid(*group_id);
56+
}
4257
for (key, value) in &self.env {
4358
child.env(key, value);
4459
}
@@ -55,10 +70,7 @@ impl Pid1App {
5570
let child = match child {
5671
Ok(child) => child,
5772
Err(err) => {
58-
eprintln!(
59-
"pid1: {} spawn failed. Got error: {err}",
60-
self.child_process[0]
61-
);
73+
eprintln!("pid1: {} spawn failed. Got error: {err}", self.command);
6274
std::process::exit(1);
6375
}
6476
};

0 commit comments

Comments
 (0)