Skip to content

Commit a57fa08

Browse files
committed
Auto merge of rust-lang#110444 - ferrocene:comptest-lines, r=Mark-Simulacrum
Add compare-output-lines-by-subset flag to compiletest For [ferrocene](github.com/ferrocene/) we have some compiletests that check the output of the cli arguments to the compiler, including printing things like the target list (`--print target-list`). Unfortunately those tend to change quite often so when we sync we end up with some outputs we have to re-bless constantly, even though the exact output doesn't really matter. We added a new compiletest flag to aid writing these kinds of tests: `compare-output-lines-by-subset`. It checks whether the lines of the expected output are a subset (or equal) to the lines of the actual output. If the expected output is empty it will fail unless the actual output is also empty. We opened this PR hoping the flag might be helpful for other tests in the future (especially if CLI-related tests are added in the future in the rust-lang/rust repo itself).
2 parents 791a7f2 + a30171c commit a57fa08

File tree

2 files changed

+68
-8
lines changed

2 files changed

+68
-8
lines changed

src/tools/compiletest/src/header.rs

+10
Original file line numberDiff line numberDiff line change
@@ -102,6 +102,9 @@ pub struct TestProps {
102102
pub dont_check_compiler_stdout: bool,
103103
// For UI tests, allows compiler to generate arbitrary output to stderr
104104
pub dont_check_compiler_stderr: bool,
105+
// When checking the output of stdout or stderr check
106+
// that the lines of expected output are a subset of the actual output.
107+
pub compare_output_lines_by_subset: bool,
105108
// Don't force a --crate-type=dylib flag on the command line
106109
//
107110
// Set this for example if you have an auxiliary test file that contains
@@ -209,6 +212,7 @@ mod directives {
209212
pub const KNOWN_BUG: &'static str = "known-bug";
210213
pub const MIR_UNIT_TEST: &'static str = "unit-test";
211214
pub const REMAP_SRC_BASE: &'static str = "remap-src-base";
215+
pub const COMPARE_OUTPUT_LINES_BY_SUBSET: &'static str = "compare-output-lines-by-subset";
212216
// This isn't a real directive, just one that is probably mistyped often
213217
pub const INCORRECT_COMPILER_FLAGS: &'static str = "compiler-flags";
214218
}
@@ -233,6 +237,7 @@ impl TestProps {
233237
check_run_results: false,
234238
dont_check_compiler_stdout: false,
235239
dont_check_compiler_stderr: false,
240+
compare_output_lines_by_subset: false,
236241
no_prefer_dynamic: false,
237242
pretty_expanded: false,
238243
pretty_mode: "normal".to_string(),
@@ -467,6 +472,11 @@ impl TestProps {
467472
s.trim().to_string()
468473
});
469474
config.set_name_directive(ln, REMAP_SRC_BASE, &mut self.remap_src_base);
475+
config.set_name_directive(
476+
ln,
477+
COMPARE_OUTPUT_LINES_BY_SUBSET,
478+
&mut self.compare_output_lines_by_subset,
479+
);
470480
});
471481
}
472482

src/tools/compiletest/src/runtest.rs

+58-8
Original file line numberDiff line numberDiff line change
@@ -3244,17 +3244,35 @@ impl<'test> TestCx<'test> {
32443244
match output_kind {
32453245
TestOutput::Compile => {
32463246
if !self.props.dont_check_compiler_stdout {
3247-
errors +=
3248-
self.compare_output(stdout_kind, &normalized_stdout, &expected_stdout);
3247+
errors += self.compare_output(
3248+
stdout_kind,
3249+
&normalized_stdout,
3250+
&expected_stdout,
3251+
self.props.compare_output_lines_by_subset,
3252+
);
32493253
}
32503254
if !self.props.dont_check_compiler_stderr {
3251-
errors +=
3252-
self.compare_output(stderr_kind, &normalized_stderr, &expected_stderr);
3255+
errors += self.compare_output(
3256+
stderr_kind,
3257+
&normalized_stderr,
3258+
&expected_stderr,
3259+
self.props.compare_output_lines_by_subset,
3260+
);
32533261
}
32543262
}
32553263
TestOutput::Run => {
3256-
errors += self.compare_output(stdout_kind, &normalized_stdout, &expected_stdout);
3257-
errors += self.compare_output(stderr_kind, &normalized_stderr, &expected_stderr);
3264+
errors += self.compare_output(
3265+
stdout_kind,
3266+
&normalized_stdout,
3267+
&expected_stdout,
3268+
self.props.compare_output_lines_by_subset,
3269+
);
3270+
errors += self.compare_output(
3271+
stderr_kind,
3272+
&normalized_stderr,
3273+
&expected_stderr,
3274+
self.props.compare_output_lines_by_subset,
3275+
);
32583276
}
32593277
}
32603278
errors
@@ -3338,7 +3356,12 @@ impl<'test> TestCx<'test> {
33383356
)
33393357
});
33403358

3341-
errors += self.compare_output("fixed", &fixed_code, &expected_fixed);
3359+
errors += self.compare_output(
3360+
"fixed",
3361+
&fixed_code,
3362+
&expected_fixed,
3363+
self.props.compare_output_lines_by_subset,
3364+
);
33423365
} else if !expected_fixed.is_empty() {
33433366
panic!(
33443367
"the `// run-rustfix` directive wasn't found but a `*.fixed` \
@@ -3796,11 +3819,38 @@ impl<'test> TestCx<'test> {
37963819
}
37973820
}
37983821

3799-
fn compare_output(&self, kind: &str, actual: &str, expected: &str) -> usize {
3822+
fn compare_output(
3823+
&self,
3824+
kind: &str,
3825+
actual: &str,
3826+
expected: &str,
3827+
compare_output_by_lines: bool,
3828+
) -> usize {
38003829
if actual == expected {
38013830
return 0;
38023831
}
38033832

3833+
let tmp;
3834+
let (expected, actual): (&str, &str) = if compare_output_by_lines {
3835+
let actual_lines: HashSet<_> = actual.lines().collect();
3836+
let expected_lines: Vec<_> = expected.lines().collect();
3837+
let mut used = expected_lines.clone();
3838+
used.retain(|line| actual_lines.contains(line));
3839+
// check if `expected` contains a subset of the lines of `actual`
3840+
if used.len() == expected_lines.len() && (expected.is_empty() == actual.is_empty()) {
3841+
return 0;
3842+
}
3843+
if expected_lines.is_empty() {
3844+
// if we have no lines to check, force a full overwite
3845+
("", actual)
3846+
} else {
3847+
tmp = (expected_lines.join("\n"), used.join("\n"));
3848+
(&tmp.0, &tmp.1)
3849+
}
3850+
} else {
3851+
(expected, actual)
3852+
};
3853+
38043854
if !self.config.bless {
38053855
if expected.is_empty() {
38063856
println!("normalized {}:\n{}\n", kind, actual);

0 commit comments

Comments
 (0)