Move flag_value to an extension trait

This commit is contained in:
David Tolnay 2024-04-06 10:02:56 -07:00
parent 5c0f97ff03
commit 2a5e85c131
No known key found for this signature in database
GPG key ID: F9BA143B95FF6D82
2 changed files with 51 additions and 89 deletions

View file

@ -1,24 +1,15 @@
use std::ffi::{OsStr, OsString}; use std::ffi::OsStr;
use std::slice; use std::process::Command;
use std::vec;
pub struct CommandArgs { pub trait CommandExt {
args: Vec<OsString>, fn flag_value<K, V>(&mut self, k: K, v: V)
}
impl CommandArgs {
pub fn new() -> Self {
CommandArgs { args: Vec::new() }
}
pub fn arg<S>(&mut self, arg: S)
where where
S: AsRef<OsStr>, K: AsRef<OsStr>,
{ V: AsRef<OsStr>;
self.args.push(arg.as_ref().to_owned());
} }
pub fn flag_value<K, V>(&mut self, k: K, v: V) impl CommandExt for Command {
fn flag_value<K, V>(&mut self, k: K, v: V)
where where
K: AsRef<OsStr>, K: AsRef<OsStr>,
V: AsRef<OsStr>, V: AsRef<OsStr>,
@ -34,31 +25,4 @@ impl CommandArgs {
self.arg(k); self.arg(k);
self.arg(v); self.arg(v);
} }
pub fn args<I, S>(&mut self, args: I)
where
I: IntoIterator<Item = S>,
S: AsRef<OsStr>,
{
self.args
.extend(args.into_iter().map(|arg| arg.as_ref().to_owned()));
}
}
impl IntoIterator for CommandArgs {
type Item = OsString;
type IntoIter = vec::IntoIter<OsString>;
fn into_iter(self) -> Self::IntoIter {
self.args.into_iter()
}
}
impl<'a> IntoIterator for &'a CommandArgs {
type Item = &'a OsString;
type IntoIter = slice::Iter<'a, OsString>;
fn into_iter(self) -> Self::IntoIter {
self.args.iter()
}
} }

View file

@ -27,7 +27,7 @@ mod opts;
mod unparse; mod unparse;
mod version; mod version;
use crate::cmd::CommandArgs; use crate::cmd::CommandExt as _;
use crate::config::Config; use crate::config::Config;
use crate::error::Result; use crate::error::Result;
use crate::opts::{Coloring, Expand, Subcommand}; use crate::opts::{Coloring, Expand, Subcommand};
@ -320,78 +320,77 @@ fn which_rustfmt() -> Option<PathBuf> {
} }
fn apply_args(cmd: &mut Command, args: &Expand, color: &Coloring, outfile: &Path) -> Result<()> { fn apply_args(cmd: &mut Command, args: &Expand, color: &Coloring, outfile: &Path) -> Result<()> {
let mut line = CommandArgs::new(); cmd.arg("rustc");
line.arg("rustc");
if args.verbose { if args.verbose {
line.arg("--verbose"); cmd.arg("--verbose");
} }
match color { match color {
Coloring::Auto => { Coloring::Auto => {
if cfg!(not(windows)) && io::stderr().is_terminal() { if cfg!(not(windows)) && io::stderr().is_terminal() {
line.flag_value("--color", "always"); cmd.flag_value("--color", "always");
} else { } else {
line.flag_value("--color", "never"); cmd.flag_value("--color", "never");
} }
} }
color => line.flag_value("--color", color.to_possible_value().unwrap().get_name()), color => cmd.flag_value("--color", color.to_possible_value().unwrap().get_name()),
} }
for kv in &args.config { for kv in &args.config {
line.flag_value("--config", kv); cmd.flag_value("--config", kv);
} }
for unstable_flag in &args.unstable_flags { for unstable_flag in &args.unstable_flags {
line.arg(format!("-Z{}", unstable_flag)); cmd.arg(format!("-Z{}", unstable_flag));
} }
if let Some(opt_package) = &args.package { if let Some(opt_package) = &args.package {
if let Some(package) = opt_package { if let Some(package) = opt_package {
line.flag_value("--package", package); cmd.flag_value("--package", package);
} else { } else {
line.arg("--package"); cmd.arg("--package");
} }
} }
let mut has_explicit_build_target = false; let mut has_explicit_build_target = false;
if args.lib { if args.lib {
line.arg("--lib"); cmd.arg("--lib");
has_explicit_build_target = true; has_explicit_build_target = true;
} }
if let Some(opt_bin) = &args.bin { if let Some(opt_bin) = &args.bin {
if let Some(bin) = opt_bin { if let Some(bin) = opt_bin {
line.flag_value("--bin", bin); cmd.flag_value("--bin", bin);
} else { } else {
line.arg("--bin"); cmd.arg("--bin");
} }
has_explicit_build_target = true; has_explicit_build_target = true;
} }
if let Some(opt_example) = &args.example { if let Some(opt_example) = &args.example {
if let Some(example) = opt_example { if let Some(example) = opt_example {
line.flag_value("--example", example); cmd.flag_value("--example", example);
} else { } else {
line.arg("--example"); cmd.arg("--example");
} }
has_explicit_build_target = true; has_explicit_build_target = true;
} }
if let Some(opt_test) = &args.test { if let Some(opt_test) = &args.test {
if let Some(test) = opt_test { if let Some(test) = opt_test {
line.flag_value("--test", test); cmd.flag_value("--test", test);
} else { } else {
line.arg("--test"); cmd.arg("--test");
} }
has_explicit_build_target = true; has_explicit_build_target = true;
} }
if let Some(opt_bench) = &args.bench { if let Some(opt_bench) = &args.bench {
if let Some(bench) = opt_bench { if let Some(bench) = opt_bench {
line.flag_value("--bench", bench); cmd.flag_value("--bench", bench);
} else { } else {
line.arg("--bench"); cmd.arg("--bench");
} }
has_explicit_build_target = true; has_explicit_build_target = true;
} }
@ -400,77 +399,76 @@ fn apply_args(cmd: &mut Command, args: &Expand, color: &Coloring, outfile: &Path
if let Ok(cargo_manifest) = manifest::parse(args.manifest_path.as_deref()) { if let Ok(cargo_manifest) = manifest::parse(args.manifest_path.as_deref()) {
if let Some(root_package) = cargo_manifest.package { if let Some(root_package) = cargo_manifest.package {
if let Some(default_run) = &root_package.default_run { if let Some(default_run) = &root_package.default_run {
line.flag_value("--bin", default_run); cmd.flag_value("--bin", default_run);
} }
} }
} }
} }
if let Some(features) = &args.features { if let Some(features) = &args.features {
line.flag_value("--features", features); cmd.flag_value("--features", features);
} }
if args.all_features { if args.all_features {
line.arg("--all-features"); cmd.arg("--all-features");
} }
if args.no_default_features { if args.no_default_features {
line.arg("--no-default-features"); cmd.arg("--no-default-features");
} }
if let Some(jobs) = args.jobs { if let Some(jobs) = args.jobs {
line.flag_value("--jobs", jobs.to_string()); cmd.flag_value("--jobs", jobs.to_string());
} }
if let Some(profile) = &args.profile { if let Some(profile) = &args.profile {
line.flag_value("--profile", profile); cmd.flag_value("--profile", profile);
} else if args.tests && args.test.is_none() { } else if args.tests && args.test.is_none() {
if args.release { if args.release {
line.flag_value("--profile", "bench"); cmd.flag_value("--profile", "bench");
} else { } else {
line.flag_value("--profile", "test"); cmd.flag_value("--profile", "test");
} }
} else if args.release { } else if args.release {
line.flag_value("--profile", "release"); cmd.flag_value("--profile", "release");
} else { } else {
line.flag_value("--profile", "check"); cmd.flag_value("--profile", "check");
} }
if let Some(target) = &args.target { if let Some(target) = &args.target {
line.flag_value("--target", target); cmd.flag_value("--target", target);
} }
if let Some(target_dir) = &args.target_dir { if let Some(target_dir) = &args.target_dir {
line.flag_value("--target-dir", target_dir); cmd.flag_value("--target-dir", target_dir);
} }
if let Some(manifest_path) = &args.manifest_path { if let Some(manifest_path) = &args.manifest_path {
line.flag_value("--manifest-path", manifest_path); cmd.flag_value("--manifest-path", manifest_path);
} }
if args.frozen { if args.frozen {
line.arg("--frozen"); cmd.arg("--frozen");
} }
if args.locked { if args.locked {
line.arg("--locked"); cmd.arg("--locked");
} }
if args.offline { if args.offline {
line.arg("--offline"); cmd.arg("--offline");
} }
line.arg("--"); cmd.arg("--");
line.arg("-o"); cmd.arg("-o");
line.arg(outfile); cmd.arg(outfile);
line.arg(ARG_Z_UNPRETTY_EXPANDED); cmd.arg(ARG_Z_UNPRETTY_EXPANDED);
if args.verbose { if args.verbose {
print_command(&line, color)?; print_command(cmd, color)?;
} }
cmd.args(line);
Ok(()) Ok(())
} }
@ -520,10 +518,10 @@ fn needs_rustc_bootstrap() -> bool {
!status.success() !status.success()
} }
fn print_command(args: &CommandArgs, color: &Coloring) -> Result<()> { fn print_command(cmd: &Command, color: &Coloring) -> Result<()> {
let mut shell_words = String::new(); let mut shell_words = String::new();
let quoter = shlex::Quoter::new().allow_nul(true); let quoter = shlex::Quoter::new().allow_nul(true);
for arg in args { for arg in cmd.get_args() {
let arg_lossy = arg.to_string_lossy(); let arg_lossy = arg.to_string_lossy();
shell_words.push(' '); shell_words.push(' ');
match arg_lossy.split_once('=') { match arg_lossy.split_once('=') {