-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathcommandline.rs
120 lines (109 loc) · 4.12 KB
/
commandline.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use std::env;
use std::fs::read_link;
use std::io::{stdout, stderr};
use std::path::Path;
use std::str::FromStr;
use libc::pid_t;
use argparse::{ArgumentParser};
use crate::config::command::{CommandInfo, Run, WriteMode};
use crate::config::volumes::{PersistentInfo, Volume};
use crate::process_util::{convert_status, copy_env_vars, run_and_wait, set_fake_uidmap};
use crate::wrapper::init_persistent::{Guard, PersistentVolumeGuard};
use super::{setup, Wrapper};
use super::util::{gen_command, warn_if_data_container};
pub fn commandline_cmd(cmd_name: &str, command: &CommandInfo,
wrapper: &Wrapper, mut cmdline: Vec<String>)
-> Result<i32, String>
{
// TODO(tailhook) detect other shells too
let has_args = command.accepts_arguments
.unwrap_or(!matches!(command.run, Run::Shell(..)));
let mut args = Vec::new();
if !has_args {
let mut ap = ArgumentParser::new();
ap.set_description(command.description.as_ref()
.map(|x| &x[..]).unwrap_or(""));
ap.stop_on_first_argument(true);
match ap.parse(cmdline, &mut stdout(), &mut stderr()) {
Ok(()) => {}
Err(0) => return Ok(0),
Err(_) => {
return Ok(122);
}
}
} else {
cmdline.remove(0);
args.extend(cmdline.into_iter());
}
let pid: pid_t = read_link(&Path::new("/proc/self"))
.map_err(|e| format!("Can't read /proc/self: {}", e))
.and_then(|v| v.to_str().and_then(|x| FromStr::from_str(x).ok())
.ok_or(format!("Can't parse pid: {:?}", v)))?;
setup::setup_base_filesystem(
wrapper.project_root, wrapper.ext_settings)?;
let cconfig = wrapper.config.containers.get(&command.container)
.ok_or(format!("Container {} not found", command.container))?;
let write_mode = match command.write_mode {
WriteMode::read_only => setup::WriteMode::ReadOnly,
WriteMode::transient_hard_link_copy
=> setup::WriteMode::TransientHardlinkCopy(pid),
};
let cont_ver = wrapper.root.as_ref().unwrap();
let mut setup_info = setup::SetupInfo::from_container(&cconfig);
setup_info
.volumes(&command.volumes)
.write_mode(write_mode);
let mut guards = Vec::<Box<dyn Guard>>::new();
for (_, &volume) in &setup_info.volumes {
match volume {
&Volume::Persistent(ref info @ PersistentInfo {
init_command: Some(_), .. })
if info.init_command.as_ref().unwrap() == cmd_name => {
match PersistentVolumeGuard::new(&info) {
Ok(Some(guard)) => {
guards.push(Box::new(guard));
setup_info.tmp_volumes.insert(&info.name);
}
Ok(None) => {}
Err(e) => {
return Err(format!("Persistent volume {:?} error: {}",
info.name, e));
}
}
}
_ => {}
}
}
warn_if_data_container(&cconfig);
setup::setup_filesystem(&setup_info, &cont_ver)?;
let env = setup::get_environment(&wrapper.settings, cconfig,
Some(&command))?;
let mut cmd = gen_command(&cconfig.default_shell, &command.run, &env)?;
cmd.args(&args);
cmd.env_clear();
copy_env_vars(&mut cmd, &wrapper.settings);
if let Some(euid) = command.external_user_id {
set_fake_uidmap(&mut cmd, command.user_id, euid)?;
}
cmd.uid(command.user_id);
cmd.gid(command.group_id);
cmd.groups(command.supplementary_gids.clone());
if let Some(ref wd) = command.work_dir {
cmd.current_dir(Path::new("/work").join(&wd));
} else {
cmd.current_dir(env::var("_VAGGA_WORKDIR")
.unwrap_or("/work".to_string()));
}
for (ref k, ref v) in env.iter() {
cmd.env(k, v);
}
let result = run_and_wait(&mut cmd)
.map(convert_status);
if result == Ok(0) {
for guard in guards {
guard.commit()
.map_err(|e| format!("Error commiting guard: {}", e))?;
}
}
return result;
}