Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 20ba13c

Browse files
committedJun 11, 2024
Auto merge of #125752 - jieyouxu:kaboom, r=Kobzol
run-make: arm command wrappers with drop bombs This PR is one in a series of cleanups to run-make tests and the run-make-support library. ### Summary It's easy to forget to actually executed constructed command wrappers, e.g. `rustc().input("foo.rs")` but forget the `run()`, so to help catch these mistakes, we arm command wrappers with drop bombs on construction to force them to be executed by test code. This PR also removes the `Deref`/`DerefMut` impl for our custom `Command` which derefs to `std::process::Command` because it can cause issues when trying to use a custom command: ```rs htmldocck().arg().run() ``` fails to compile because the `arg()` is resolved to `std::process::Command::arg`, which returns `&mut std::process::Command` that doesn't have a `run()` command. This PR also: - Removes `env_var` on the `impl_common_helper` macro that was wrongly named and is a footgun (no users). - Bumps the run-make-support library to version `0.1.0`. - Adds a changelog to the support library. ### Details Especially for command wrappers like `Rustc`, it's very easy to build up a command invocation but forget to actually execute it, e.g. by using `run()`. This commit adds "drop bombs" to command wrappers, which are armed on command wrapper construction, and only defused if the command is executed (through `run`, `run_fail`). If the test writer forgets to execute the command, the drop bomb will "explode" and panic with an error message. This is so that tests don't silently pass with constructed-but-not-executed command wrappers. This PR is best reviewed commit-by-commit. try-job: x86_64-msvc
2 parents 6a207f4 + 5ec3eef commit 20ba13c

File tree

22 files changed

+280
-80
lines changed

22 files changed

+280
-80
lines changed
 

‎Cargo.lock

+1-1
Original file line numberDiff line numberDiff line change
@@ -3452,7 +3452,7 @@ dependencies = [
34523452

34533453
[[package]]
34543454
name = "run_make_support"
3455-
version = "0.0.0"
3455+
version = "0.1.0"
34563456
dependencies = [
34573457
"gimli 0.28.1",
34583458
"object 0.34.0",
+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
# Changelog
2+
3+
All notable changes to the `run_make_support` library should be documented in this file.
4+
5+
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/) and the support
6+
library should adhere to [Semantic Versioning](https://semver.org/spec/v2.0.0.html) even if it's
7+
not intended for public consumption (it's moreso to help internally, to help test writers track
8+
changes to the support library).
9+
10+
This support library will probably never reach 1.0. Please bump the minor version in `Cargo.toml` if
11+
you make any breaking changes or other significant changes, or bump the patch version for bug fixes.
12+
13+
## [0.1.0] - 2024-06-09
14+
15+
### Changed
16+
17+
- Use *drop bombs* to enforce that commands are executed; a command invocation will panic if it is
18+
constructed but never executed. Execution methods `Command::{run, run_fail}` will defuse the drop
19+
bomb.
20+
- Added `Command` helpers that forward to `std::process::Command` counterparts.
21+
22+
### Removed
23+
24+
- The `env_var` method which was incorrectly named and is `env_clear` underneath and is a footgun
25+
from `impl_common_helpers`. For example, removing `TMPDIR` on Unix and `TMP`/`TEMP` breaks
26+
`std::env::temp_dir` and wrecks anything using that, such as rustc's codgen.
27+
- Removed `Deref`/`DerefMut` for `run_make_support::Command` -> `std::process::Command` because it
28+
causes a method chain like `htmldocck().arg().run()` to fail, because `arg()` resolves to
29+
`std::process::Command` which also returns a `&mut std::process::Command`, causing the `run()` to
30+
be not found.
31+
32+
## [0.0.0] - 2024-06-09
33+
34+
Consider this version to contain all changes made to the support library before we started to track
35+
changes in this changelog.
36+
37+
### Added
38+
39+
- Custom command wrappers around `std::process::Command` (`run_make_support::Command`) and custom
40+
wrapper around `std::process::Output` (`CompletedProcess`) to make it more convenient to work with
41+
commands and their output, and help avoid forgetting to check for exit status.
42+
- `Command`: `set_stdin`, `run`, `run_fail`.
43+
- `CompletedProcess`: `std{err,out}_utf8`, `status`, `assert_std{err,out}_{equals, contains,
44+
not_contains}`, `assert_exit_code`.
45+
- `impl_common_helpers` macro to avoid repeating adding common convenience methods, including:
46+
- Environment manipulation methods: `env`, `env_remove`
47+
- Command argument providers: `arg`, `args`
48+
- Common invocation inspection (of the command invocation up until `inspect` is called):
49+
`inspect`
50+
- Execution methods: `run` (for commands expected to succeed execution, exit status `0`) and
51+
`run_fail` (for commands expected to fail execution, exit status non-zero).
52+
- Command wrappers around: `rustc`, `clang`, `cc`, `rustc`, `rustdoc`, `llvm-readobj`.
53+
- Thin helpers to construct `python` and `htmldocck` commands.
54+
- `run` and `run_fail` (like `Command::{run, run_fail}`) for running binaries, which sets suitable
55+
env vars (like `LD_LIB_PATH` or equivalent, `TARGET_RPATH_ENV`, `PATH` on Windows).
56+
- Pseudo command `diff` which has similar functionality as the cli util but not the same API.
57+
- Convenience panic-on-fail helpers `env_var`, `env_var_os`, `cwd` for their `std::env` conterparts.
58+
- Convenience panic-on-fail helpers for reading respective env vars: `target`, `source_root`.
59+
- Platform check helpers: `is_windows`, `is_msvc`, `cygpath_windows`, `uname`.
60+
- fs helpers: `copy_dir_all`.
61+
- `recursive_diff` helper.
62+
- Generic `assert_not_contains` helper.
63+
- Scoped run-with-teardown helper `run_in_tmpdir` which is designed to run commands in a temporary
64+
directory that is cleared when closure returns.
65+
- Helpers for constructing the name of binaries and libraries: `rust_lib_name`, `static_lib_name`,
66+
`bin_name`, `dynamic_lib_name`.
67+
- Re-export libraries: `gimli`, `object`, `regex`, `wasmparsmer`.

‎src/tools/run-make-support/Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[package]
22
name = "run_make_support"
3-
version = "0.0.0"
3+
version = "0.1.0"
44
edition = "2021"
55

66
[dependencies]

‎src/tools/run-make-support/src/cc.rs

+2-5
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ use crate::{bin_name, cygpath_windows, env_var, is_msvc, is_windows, uname};
77
///
88
/// WARNING: This means that what flags are accepted by the underlying C compiler is
99
/// platform- AND compiler-specific. Consult the relevant docs for `gcc`, `clang` and `mvsc`.
10+
#[track_caller]
1011
pub fn cc() -> Cc {
1112
Cc::new()
1213
}
@@ -25,6 +26,7 @@ impl Cc {
2526
///
2627
/// WARNING: This means that what flags are accepted by the underlying C compile is
2728
/// platform- AND compiler-specific. Consult the relevant docs for `gcc`, `clang` and `mvsc`.
29+
#[track_caller]
2830
pub fn new() -> Self {
2931
let compiler = env_var("CC");
3032

@@ -84,11 +86,6 @@ impl Cc {
8486
self.cmd.arg(path.as_ref());
8587
self
8688
}
87-
88-
/// Get the [`Output`][::std::process::Output] of the finished process.
89-
pub fn command_output(&mut self) -> ::std::process::Output {
90-
self.cmd.output().expect("failed to get output of finished process")
91-
}
9289
}
9390

9491
/// `EXTRACFLAGS`

‎src/tools/run-make-support/src/clang.rs

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ use crate::command::Command;
44
use crate::{bin_name, env_var};
55

66
/// Construct a new `clang` invocation. `clang` is not always available for all targets.
7+
#[track_caller]
78
pub fn clang() -> Clang {
89
Clang::new()
910
}
@@ -18,6 +19,7 @@ crate::impl_common_helpers!(Clang);
1819

1920
impl Clang {
2021
/// Construct a new `clang` invocation. `clang` is not always available for all targets.
22+
#[track_caller]
2123
pub fn new() -> Self {
2224
let clang = env_var("CLANG");
2325
let cmd = Command::new(clang);

‎src/tools/run-make-support/src/command.rs

+85-30
Original file line numberDiff line numberDiff line change
@@ -1,55 +1,124 @@
1-
use crate::{assert_not_contains, handle_failed_output};
1+
use std::ffi;
22
use std::ffi::OsStr;
33
use std::io::Write;
4-
use std::ops::{Deref, DerefMut};
4+
use std::panic;
5+
use std::path::Path;
56
use std::process::{Command as StdCommand, ExitStatus, Output, Stdio};
67

7-
/// This is a custom command wrapper that simplifies working with commands
8-
/// and makes it easier to ensure that we check the exit status of executed
9-
/// processes.
8+
use crate::drop_bomb::DropBomb;
9+
use crate::{assert_not_contains, handle_failed_output};
10+
11+
/// This is a custom command wrapper that simplifies working with commands and makes it easier to
12+
/// ensure that we check the exit status of executed processes.
13+
///
14+
/// # A [`Command`] must be executed
15+
///
16+
/// A [`Command`] is armed by a [`DropBomb`] on construction to enforce that it will be executed. If
17+
/// a [`Command`] is constructed but never executed, the drop bomb will explode and cause the test
18+
/// to panic. Execution methods [`run`] and [`run_fail`] will defuse the drop bomb. A test
19+
/// containing constructed but never executed commands is dangerous because it can give a false
20+
/// sense of confidence.
21+
///
22+
/// [`run`]: Self::run
23+
/// [`run_fail`]: Self::run_fail
1024
#[derive(Debug)]
1125
pub struct Command {
1226
cmd: StdCommand,
1327
stdin: Option<Box<[u8]>>,
28+
drop_bomb: DropBomb,
1429
}
1530

1631
impl Command {
17-
pub fn new<S: AsRef<OsStr>>(program: S) -> Self {
18-
Self { cmd: StdCommand::new(program), stdin: None }
32+
#[track_caller]
33+
pub fn new<P: AsRef<OsStr>>(program: P) -> Self {
34+
let program = program.as_ref();
35+
Self { cmd: StdCommand::new(program), stdin: None, drop_bomb: DropBomb::arm(program) }
1936
}
2037

2138
pub fn set_stdin(&mut self, stdin: Box<[u8]>) {
2239
self.stdin = Some(stdin);
2340
}
2441

42+
/// Specify an environment variable.
43+
pub fn env<K, V>(&mut self, key: K, value: V) -> &mut Self
44+
where
45+
K: AsRef<ffi::OsStr>,
46+
V: AsRef<ffi::OsStr>,
47+
{
48+
self.cmd.env(key, value);
49+
self
50+
}
51+
52+
/// Remove an environmental variable.
53+
pub fn env_remove<K>(&mut self, key: K) -> &mut Self
54+
where
55+
K: AsRef<ffi::OsStr>,
56+
{
57+
self.cmd.env_remove(key);
58+
self
59+
}
60+
61+
/// Generic command argument provider. Prefer specific helper methods if possible.
62+
/// Note that for some executables, arguments might be platform specific. For C/C++
63+
/// compilers, arguments might be platform *and* compiler specific.
64+
pub fn arg<S>(&mut self, arg: S) -> &mut Self
65+
where
66+
S: AsRef<ffi::OsStr>,
67+
{
68+
self.cmd.arg(arg);
69+
self
70+
}
71+
72+
/// Generic command arguments provider. Prefer specific helper methods if possible.
73+
/// Note that for some executables, arguments might be platform specific. For C/C++
74+
/// compilers, arguments might be platform *and* compiler specific.
75+
pub fn args<S>(&mut self, args: &[S]) -> &mut Self
76+
where
77+
S: AsRef<ffi::OsStr>,
78+
{
79+
self.cmd.args(args);
80+
self
81+
}
82+
83+
/// Inspect what the underlying [`std::process::Command`] is up to the
84+
/// current construction.
85+
pub fn inspect<I>(&mut self, inspector: I) -> &mut Self
86+
where
87+
I: FnOnce(&StdCommand),
88+
{
89+
inspector(&self.cmd);
90+
self
91+
}
92+
93+
/// Set the path where the command will be run.
94+
pub fn current_dir<P: AsRef<Path>>(&mut self, path: P) -> &mut Self {
95+
self.cmd.current_dir(path);
96+
self
97+
}
98+
2599
/// Run the constructed command and assert that it is successfully run.
26100
#[track_caller]
27101
pub fn run(&mut self) -> CompletedProcess {
28-
let caller_location = std::panic::Location::caller();
29-
let caller_line_number = caller_location.line();
30-
31102
let output = self.command_output();
32103
if !output.status().success() {
33-
handle_failed_output(&self, output, caller_line_number);
104+
handle_failed_output(&self, output, panic::Location::caller().line());
34105
}
35106
output
36107
}
37108

38109
/// Run the constructed command and assert that it does not successfully run.
39110
#[track_caller]
40111
pub fn run_fail(&mut self) -> CompletedProcess {
41-
let caller_location = std::panic::Location::caller();
42-
let caller_line_number = caller_location.line();
43-
44112
let output = self.command_output();
45113
if output.status().success() {
46-
handle_failed_output(&self, output, caller_line_number);
114+
handle_failed_output(&self, output, panic::Location::caller().line());
47115
}
48116
output
49117
}
50118

51119
#[track_caller]
52-
pub(crate) fn command_output(&mut self) -> CompletedProcess {
120+
fn command_output(&mut self) -> CompletedProcess {
121+
self.drop_bomb.defuse();
53122
// let's make sure we piped all the input and outputs
54123
self.cmd.stdin(Stdio::piped());
55124
self.cmd.stdout(Stdio::piped());
@@ -71,20 +140,6 @@ impl Command {
71140
}
72141
}
73142

74-
impl Deref for Command {
75-
type Target = StdCommand;
76-
77-
fn deref(&self) -> &Self::Target {
78-
&self.cmd
79-
}
80-
}
81-
82-
impl DerefMut for Command {
83-
fn deref_mut(&mut self) -> &mut Self::Target {
84-
&mut self.cmd
85-
}
86-
}
87-
88143
/// Represents the result of an executed process.
89144
/// The various `assert_` helper methods should preferably be used for
90145
/// checking the contents of stdout/stderr.

‎src/tools/run-make-support/src/diff/mod.rs

+8-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,12 @@ use regex::Regex;
22
use similar::TextDiff;
33
use std::path::Path;
44

5+
use crate::drop_bomb::DropBomb;
6+
57
#[cfg(test)]
68
mod tests;
79

10+
#[track_caller]
811
pub fn diff() -> Diff {
912
Diff::new()
1013
}
@@ -16,17 +19,20 @@ pub struct Diff {
1619
actual: Option<String>,
1720
actual_name: Option<String>,
1821
normalizers: Vec<(String, String)>,
22+
drop_bomb: DropBomb,
1923
}
2024

2125
impl Diff {
2226
/// Construct a bare `diff` invocation.
27+
#[track_caller]
2328
pub fn new() -> Self {
2429
Self {
2530
expected: None,
2631
expected_name: None,
2732
actual: None,
2833
actual_name: None,
2934
normalizers: Vec::new(),
35+
drop_bomb: DropBomb::arm("diff"),
3036
}
3137
}
3238

@@ -79,9 +85,9 @@ impl Diff {
7985
self
8086
}
8187

82-
/// Executes the diff process, prints any differences to the standard error.
8388
#[track_caller]
84-
pub fn run(&self) {
89+
pub fn run(&mut self) {
90+
self.drop_bomb.defuse();
8591
let expected = self.expected.as_ref().expect("expected text not set");
8692
let mut actual = self.actual.as_ref().expect("actual text not set").to_string();
8793
let expected_name = self.expected_name.as_ref().unwrap();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
//! This module implements "drop bombs" intended for use by command wrappers to ensure that the
2+
//! constructed commands are *eventually* executed. This is exactly like `rustc_errors::Diag` where
3+
//! we force every `Diag` to be consumed or we emit a bug, but we panic instead.
4+
//!
5+
//! This is adapted from <https://docs.rs/drop_bomb/latest/drop_bomb/> and simplified for our
6+
//! purposes.
7+
8+
use std::ffi::{OsStr, OsString};
9+
use std::panic;
10+
11+
#[cfg(test)]
12+
mod tests;
13+
14+
#[derive(Debug)]
15+
pub(crate) struct DropBomb {
16+
command: OsString,
17+
defused: bool,
18+
armed_line: u32,
19+
}
20+
21+
impl DropBomb {
22+
/// Arm a [`DropBomb`]. If the value is dropped without being [`defused`][Self::defused], then
23+
/// it will panic. It is expected that the command wrapper uses `#[track_caller]` to help
24+
/// propagate the caller info from rmake.rs.
25+
#[track_caller]
26+
pub(crate) fn arm<S: AsRef<OsStr>>(command: S) -> DropBomb {
27+
DropBomb {
28+
command: command.as_ref().into(),
29+
defused: false,
30+
armed_line: panic::Location::caller().line(),
31+
}
32+
}
33+
34+
/// Defuse the [`DropBomb`]. This will prevent the drop bomb from panicking when dropped.
35+
pub(crate) fn defuse(&mut self) {
36+
self.defused = true;
37+
}
38+
}
39+
40+
impl Drop for DropBomb {
41+
fn drop(&mut self) {
42+
if !self.defused && !std::thread::panicking() {
43+
panic!(
44+
"command constructed but not executed at line {}: `{}`",
45+
self.armed_line,
46+
self.command.to_string_lossy()
47+
)
48+
}
49+
}
50+
}
There was a problem loading the remainder of the diff.

0 commit comments

Comments
 (0)
Failed to load comments.