use std::error::Error;
use std::path::PathBuf;
use clap::Parser;
#[derive(Parser, Debug)]
pub struct Exec {
#[clap(long)]
pub console_socket: Option<PathBuf>,
#[clap(long)]
pub cwd: Option<PathBuf>,
#[clap(short, long, value_parser = parse_env::<String, String>, number_of_values = 1)]
pub env: Vec<(String, String)>,
#[clap(short, long)]
pub tty: bool,
#[clap(short, long, value_parser = parse_user::<u32, u32>)]
pub user: Option<(u32, Option<u32>)>,
#[clap(long, short = 'g', number_of_values = 1)]
pub additional_gids: Vec<u32>,
#[clap(short, long)]
pub process: Option<PathBuf>,
#[clap(short, long)]
pub detach: bool,
#[clap(long)]
pub pid_file: Option<PathBuf>,
#[clap(long)]
pub process_label: Option<String>,
#[clap(long)]
pub apparmor: Option<String>,
#[clap(long)]
pub no_new_privs: bool,
#[clap(long, number_of_values = 1)]
pub cap: Vec<String>,
#[clap(long, default_value = "0")]
pub preserve_fds: i32,
#[clap(long)]
pub ignore_paused: bool,
#[clap(long)]
pub cgroup: Option<String>,
#[clap(value_parser = clap::builder::NonEmptyStringValueParser::new(), required = true)]
pub container_id: String,
#[clap(required = false)]
pub command: Vec<String>,
}
fn parse_env<T, U>(s: &str) -> Result<(T, U), Box<dyn Error + Send + Sync + 'static>>
where
T: std::str::FromStr,
T::Err: Error + Send + Sync + 'static,
U: std::str::FromStr,
U::Err: Error + Send + Sync + 'static,
{
let pos = s
.find('=')
.ok_or_else(|| format!("invalid VAR=value: no `=` found in `{s}`"))?;
Ok((s[..pos].parse()?, s[pos + 1..].parse()?))
}
fn parse_user<T, U>(s: &str) -> Result<(T, Option<U>), Box<dyn Error + Send + Sync + 'static>>
where
T: std::str::FromStr,
T::Err: Error + Send + Sync + 'static,
U: std::str::FromStr,
U::Err: Error + Send + Sync + 'static,
{
if let Some(pos) = s.find(':') {
Ok((s[..pos].parse()?, Some(s[pos + 1..].parse()?)))
} else {
Ok((s.parse()?, None))
}
}