summaryrefslogtreecommitdiffstats
path: root/vendor/ui_test/src
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-30 18:31:44 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-30 18:31:44 +0000
commitc23a457e72abe608715ac76f076f47dc42af07a5 (patch)
tree2772049aaf84b5c9d0ed12ec8d86812f7a7904b6 /vendor/ui_test/src
parentReleasing progress-linux version 1.73.0+dfsg1-1~progress7.99u1. (diff)
downloadrustc-c23a457e72abe608715ac76f076f47dc42af07a5.tar.xz
rustc-c23a457e72abe608715ac76f076f47dc42af07a5.zip
Merging upstream version 1.74.1+dfsg1.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'vendor/ui_test/src')
-rw-r--r--vendor/ui_test/src/cmd.rs120
-rw-r--r--vendor/ui_test/src/config.rs201
-rw-r--r--vendor/ui_test/src/dependencies.rs183
-rw-r--r--vendor/ui_test/src/diff.rs175
-rw-r--r--vendor/ui_test/src/error.rs72
-rw-r--r--vendor/ui_test/src/github_actions.rs94
-rw-r--r--vendor/ui_test/src/lib.rs1066
-rw-r--r--vendor/ui_test/src/mode.rs84
-rw-r--r--vendor/ui_test/src/parser.rs679
-rw-r--r--vendor/ui_test/src/parser/tests.rs137
-rw-r--r--vendor/ui_test/src/rustc_stderr.rs160
-rw-r--r--vendor/ui_test/src/status_emitter.rs577
-rw-r--r--vendor/ui_test/src/tests.rs338
13 files changed, 0 insertions, 3886 deletions
diff --git a/vendor/ui_test/src/cmd.rs b/vendor/ui_test/src/cmd.rs
deleted file mode 100644
index 63d055e28..000000000
--- a/vendor/ui_test/src/cmd.rs
+++ /dev/null
@@ -1,120 +0,0 @@
-use std::{
- ffi::OsString,
- path::{Path, PathBuf},
- process::Command,
-};
-
-#[derive(Debug, Clone)]
-/// A command, its args and its environment. Used for
-/// the main command, the dependency builder and the cfg-reader.
-pub struct CommandBuilder {
- /// Path to the binary.
- pub program: PathBuf,
- /// Arguments to the binary.
- pub args: Vec<OsString>,
- /// A flag to prefix before the path to where output files should be written.
- pub out_dir_flag: Option<OsString>,
- /// A flag to set as the last flag in the command, so the `build` caller can
- /// append the filename themselves.
- pub input_file_flag: Option<OsString>,
- /// Environment variables passed to the binary that is executed.
- /// The environment variable is removed if the second tuple field is `None`
- pub envs: Vec<(OsString, Option<OsString>)>,
-}
-
-impl CommandBuilder {
- /// Uses the `CARGO` env var or just a program named `cargo` and the argument `build`.
- pub fn cargo() -> Self {
- Self {
- program: PathBuf::from(std::env::var_os("CARGO").unwrap_or_else(|| "cargo".into())),
- args: vec!["build".into()],
- out_dir_flag: Some("--target-dir".into()),
- input_file_flag: Some("--manifest-path".into()),
- envs: vec![],
- }
- }
-
- /// Uses the `RUSTC` env var or just a program named `rustc` and the argument `--error-format=json`.
- ///
- /// Take care to only append unless you actually meant to overwrite the defaults.
- /// Overwriting the defaults may make `//~ ERROR` style comments stop working.
- pub fn rustc() -> Self {
- Self {
- program: PathBuf::from(std::env::var_os("RUSTC").unwrap_or_else(|| "rustc".into())),
- args: vec!["--error-format=json".into()],
- out_dir_flag: Some("--out-dir".into()),
- input_file_flag: None,
- envs: vec![],
- }
- }
-
- /// Same as [`rustc`], but with arguments for obtaining the cfgs.
- pub fn cfgs() -> Self {
- Self {
- args: vec!["--print".into(), "cfg".into()],
- ..Self::rustc()
- }
- }
-
- /// Build a `CommandBuilder` for a command without any argumemnts.
- /// You can still add arguments later.
- pub fn cmd(cmd: impl Into<PathBuf>) -> Self {
- Self {
- program: cmd.into(),
- args: vec![],
- out_dir_flag: None,
- input_file_flag: None,
- envs: vec![],
- }
- }
-
- /// Render the command like you'd use it on a command line.
- pub fn display(&self) -> impl std::fmt::Display + '_ {
- struct Display<'a>(&'a CommandBuilder);
- impl std::fmt::Display for Display<'_> {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- for (var, val) in &self.0.envs {
- if let Some(val) = val {
- write!(f, "{var:?}={val:?} ")?;
- }
- }
- self.0.program.display().fmt(f)?;
- for arg in &self.0.args {
- write!(f, " {arg:?}")?;
- }
- if let Some(flag) = &self.0.out_dir_flag {
- write!(f, " {flag:?} OUT_DIR")?;
- }
- if let Some(flag) = &self.0.input_file_flag {
- write!(f, " {flag:?}")?;
- }
- Ok(())
- }
- }
- Display(self)
- }
-
- /// Create a command with the given settings.
- pub fn build(&self, out_dir: &Path) -> Command {
- let mut cmd = Command::new(&self.program);
- cmd.args(self.args.iter());
- if let Some(flag) = &self.out_dir_flag {
- cmd.arg(flag).arg(out_dir);
- }
- if let Some(flag) = &self.input_file_flag {
- cmd.arg(flag);
- }
- self.apply_env(&mut cmd);
- cmd
- }
-
- pub(crate) fn apply_env(&self, cmd: &mut Command) {
- for (var, val) in self.envs.iter() {
- if let Some(val) = val {
- cmd.env(var, val);
- } else {
- cmd.env_remove(var);
- }
- }
- }
-}
diff --git a/vendor/ui_test/src/config.rs b/vendor/ui_test/src/config.rs
deleted file mode 100644
index 8d92fe8d8..000000000
--- a/vendor/ui_test/src/config.rs
+++ /dev/null
@@ -1,201 +0,0 @@
-use regex::bytes::Regex;
-
-use crate::{dependencies::build_dependencies, CommandBuilder, Filter, Match, Mode};
-pub use color_eyre;
-use color_eyre::eyre::Result;
-use std::{
- ffi::OsString,
- num::NonZeroUsize,
- path::{Path, PathBuf},
-};
-
-#[derive(Debug, Clone)]
-/// Central datastructure containing all information to run the tests.
-pub struct Config {
- /// Arguments passed to the binary that is executed.
- /// These arguments are passed *after* the args inserted via `//@compile-flags:`.
- pub trailing_args: Vec<OsString>,
- /// Host triple; usually will be auto-detected.
- pub host: Option<String>,
- /// `None` to run on the host, otherwise a target triple
- pub target: Option<String>,
- /// Filters applied to stderr output before processing it.
- /// By default contains a filter for replacing backslashes with regular slashes.
- /// On windows, contains a filter to replace `\n` with `\r\n`.
- pub stderr_filters: Filter,
- /// Filters applied to stdout output before processing it.
- /// On windows, contains a filter to replace `\n` with `\r\n`.
- pub stdout_filters: Filter,
- /// The folder in which to start searching for .rs files
- pub root_dir: PathBuf,
- /// The mode in which to run the tests.
- pub mode: Mode,
- /// The binary to actually execute.
- pub program: CommandBuilder,
- /// The command to run to obtain the cfgs that the output is supposed to
- pub cfgs: CommandBuilder,
- /// What to do in case the stdout/stderr output differs from the expected one.
- /// By default, errors in case of conflict, but emits a message informing the user
- /// that running `cargo test -- -- --bless` will automatically overwrite the
- /// `.stdout` and `.stderr` files with the latest output.
- pub output_conflict_handling: OutputConflictHandling,
- /// Path to a `Cargo.toml` that describes which dependencies the tests can access.
- pub dependencies_crate_manifest_path: Option<PathBuf>,
- /// The command to run can be changed from `cargo` to any custom command to build the
- /// dependencies in `dependencies_crate_manifest_path`
- pub dependency_builder: CommandBuilder,
- /// How many threads to use for running tests. Defaults to number of cores
- pub num_test_threads: NonZeroUsize,
- /// Where to dump files like the binaries compiled from tests.
- /// Defaults to `target/ui` in the current directory.
- pub out_dir: PathBuf,
- /// The default edition to use on all tests
- pub edition: Option<String>,
-}
-
-impl Config {
- /// Create a configuration for testing the output of running
- /// `rustc` on the test files.
- pub fn rustc(root_dir: PathBuf) -> Self {
- Self {
- trailing_args: vec![],
- host: None,
- target: None,
- stderr_filters: vec![
- (Match::Exact(vec![b'\\']), b"/"),
- #[cfg(windows)]
- (Match::Exact(vec![b'\r']), b""),
- ],
- stdout_filters: vec![
- #[cfg(windows)]
- (Match::Exact(vec![b'\r']), b""),
- ],
- root_dir,
- mode: Mode::Fail {
- require_patterns: true,
- },
- program: CommandBuilder::rustc(),
- cfgs: CommandBuilder::cfgs(),
- output_conflict_handling: OutputConflictHandling::Error(
- "cargo test -- -- --bless".into(),
- ),
- dependencies_crate_manifest_path: None,
- dependency_builder: CommandBuilder::cargo(),
- num_test_threads: std::thread::available_parallelism().unwrap(),
- out_dir: std::env::var_os("CARGO_TARGET_DIR")
- .map(PathBuf::from)
- .unwrap_or_else(|| std::env::current_dir().unwrap().join("target"))
- .join("ui"),
- edition: Some("2021".into()),
- }
- }
-
- /// Create a configuration for testing the output of running
- /// `cargo` on the test `Cargo.toml` files.
- pub fn cargo(root_dir: PathBuf) -> Self {
- Self {
- program: CommandBuilder::cargo(),
- edition: None,
- ..Self::rustc(root_dir)
- }
- }
-
- /// Replace all occurrences of a path in stderr with a byte string.
- pub fn path_stderr_filter(
- &mut self,
- path: &Path,
- replacement: &'static (impl AsRef<[u8]> + ?Sized),
- ) {
- let pattern = path.canonicalize().unwrap();
- self.stderr_filters
- .push((pattern.parent().unwrap().into(), replacement.as_ref()));
- }
-
- /// Replace all occurrences of a regex pattern in stderr with a byte string.
- pub fn stderr_filter(
- &mut self,
- pattern: &str,
- replacement: &'static (impl AsRef<[u8]> + ?Sized),
- ) {
- self.stderr_filters
- .push((Regex::new(pattern).unwrap().into(), replacement.as_ref()));
- }
-
- /// Replace all occurrences of a regex pattern in stdout with a byte string.
- pub fn stdout_filter(
- &mut self,
- pattern: &str,
- replacement: &'static (impl AsRef<[u8]> + ?Sized),
- ) {
- self.stdout_filters
- .push((Regex::new(pattern).unwrap().into(), replacement.as_ref()));
- }
-
- /// Compile dependencies and make sure `Config::program` contains the right flags
- /// to find the dependencies.
- pub fn build_dependencies_and_link_them(&mut self) -> Result<()> {
- let dependencies = build_dependencies(self)?;
- for (name, artifacts) in dependencies.dependencies {
- for dependency in artifacts {
- self.program.args.push("--extern".into());
- let mut dep = OsString::from(&name);
- dep.push("=");
- dep.push(dependency);
- self.program.args.push(dep);
- }
- }
- for import_path in dependencies.import_paths {
- self.program.args.push("-L".into());
- self.program.args.push(import_path.into());
- }
- Ok(())
- }
-
- /// Make sure we have the host and target triples.
- pub fn fill_host_and_target(&mut self) -> Result<()> {
- if self.host.is_none() {
- self.host = Some(
- rustc_version::VersionMeta::for_command(std::process::Command::new(
- &self.program.program,
- ))
- .map_err(|err| {
- color_eyre::eyre::Report::new(err).wrap_err(format!(
- "failed to parse rustc version info: {}",
- self.program.display()
- ))
- })?
- .host,
- );
- }
- if self.target.is_none() {
- self.target = Some(self.host.clone().unwrap());
- }
- Ok(())
- }
-
- pub(crate) fn has_asm_support(&self) -> bool {
- static ASM_SUPPORTED_ARCHS: &[&str] = &[
- "x86", "x86_64", "arm", "aarch64", "riscv32",
- "riscv64",
- // These targets require an additional asm_experimental_arch feature.
- // "nvptx64", "hexagon", "mips", "mips64", "spirv", "wasm32",
- ];
- ASM_SUPPORTED_ARCHS
- .iter()
- .any(|arch| self.target.as_ref().unwrap().contains(arch))
- }
-}
-
-#[derive(Debug, Clone)]
-/// The different options for what to do when stdout/stderr files differ from the actual output.
-pub enum OutputConflictHandling {
- /// The default: emit a diff of the expected/actual output.
- ///
- /// The string should be a command that can be executed to bless all tests.
- Error(String),
- /// Ignore mismatches in the stderr/stdout files.
- Ignore,
- /// Instead of erroring if the stderr/stdout differs from the expected
- /// automatically replace it with the found output (after applying filters).
- Bless,
-}
diff --git a/vendor/ui_test/src/dependencies.rs b/vendor/ui_test/src/dependencies.rs
deleted file mode 100644
index 800a3a447..000000000
--- a/vendor/ui_test/src/dependencies.rs
+++ /dev/null
@@ -1,183 +0,0 @@
-use cargo_metadata::{camino::Utf8PathBuf, DependencyKind};
-use cargo_platform::Cfg;
-use color_eyre::eyre::{bail, Result};
-use std::{
- collections::{HashMap, HashSet},
- path::PathBuf,
- process::Command,
- str::FromStr,
-};
-
-use crate::{Config, Mode, OutputConflictHandling};
-
-#[derive(Default, Debug)]
-pub struct Dependencies {
- /// All paths that must be imported with `-L dependency=`. This is for
- /// finding proc macros run on the host and dependencies for the target.
- pub import_paths: Vec<PathBuf>,
- /// The name as chosen in the `Cargo.toml` and its corresponding rmeta file.
- pub dependencies: Vec<(String, Vec<Utf8PathBuf>)>,
-}
-
-fn cfgs(config: &Config) -> Result<Vec<Cfg>> {
- let mut cmd = config.cfgs.build(&config.out_dir);
- cmd.arg("--target").arg(config.target.as_ref().unwrap());
- let output = cmd.output()?;
- let stdout = String::from_utf8(output.stdout)?;
-
- if !output.status.success() {
- let stderr = String::from_utf8(output.stderr)?;
- bail!(
- "failed to obtain `cfg` information from {cmd:?}:\nstderr:\n{stderr}\n\nstdout:{stdout}"
- );
- }
- let mut cfgs = vec![];
-
- for line in stdout.lines() {
- cfgs.push(Cfg::from_str(line)?);
- }
-
- Ok(cfgs)
-}
-
-/// Compiles dependencies and returns the crate names and corresponding rmeta files.
-pub fn build_dependencies(config: &mut Config) -> Result<Dependencies> {
- let manifest_path = match &config.dependencies_crate_manifest_path {
- Some(path) => path.to_owned(),
- None => return Ok(Default::default()),
- };
- let manifest_path = &manifest_path;
- config.fill_host_and_target()?;
- eprintln!(" Building test dependencies...");
- let mut build = config.dependency_builder.build(&config.out_dir);
- build.arg(manifest_path);
-
- if let Some(target) = &config.target {
- build.arg(format!("--target={target}"));
- }
-
- // Reusable closure for setting up the environment both for artifact generation and `cargo_metadata`
- let set_locking = |cmd: &mut Command| match (&config.output_conflict_handling, &config.mode) {
- (_, Mode::Yolo) => {}
- (OutputConflictHandling::Error(_), _) => {
- cmd.arg("--locked");
- }
- _ => {}
- };
-
- set_locking(&mut build);
- build.arg("--message-format=json");
-
- let output = build.output()?;
-
- if !output.status.success() {
- let stdout = String::from_utf8(output.stdout)?;
- let stderr = String::from_utf8(output.stderr)?;
- bail!("failed to compile dependencies:\ncommand: {build:?}\nstderr:\n{stderr}\n\nstdout:{stdout}");
- }
-
- // Collect all artifacts generated
- let artifact_output = output.stdout;
- let artifact_output = String::from_utf8(artifact_output)?;
- let mut import_paths: HashSet<PathBuf> = HashSet::new();
- let mut artifacts: HashMap<_, _> = artifact_output
- .lines()
- .filter_map(|line| {
- let message = serde_json::from_str::<cargo_metadata::Message>(line).ok()?;
- if let cargo_metadata::Message::CompilerArtifact(artifact) = message {
- for filename in &artifact.filenames {
- import_paths.insert(filename.parent().unwrap().into());
- }
- Some((artifact.package_id, artifact.filenames))
- } else {
- None
- }
- })
- .collect();
-
- // Check which crates are mentioned in the crate itself
- let mut metadata = cargo_metadata::MetadataCommand::new().cargo_command();
- metadata.arg("--manifest-path").arg(manifest_path);
- config.dependency_builder.apply_env(&mut metadata);
- set_locking(&mut metadata);
- let output = metadata.output()?;
-
- if !output.status.success() {
- let stdout = String::from_utf8(output.stdout)?;
- let stderr = String::from_utf8(output.stderr)?;
- bail!("failed to run cargo-metadata:\nstderr:\n{stderr}\n\nstdout:{stdout}");
- }
-
- let output = output.stdout;
- let output = String::from_utf8(output)?;
-
- let cfg = cfgs(config)?;
-
- for line in output.lines() {
- if !line.starts_with('{') {
- continue;
- }
- let metadata: cargo_metadata::Metadata = serde_json::from_str(line)?;
- // Only take artifacts that are defined in the Cargo.toml
-
- // First, find the root artifact
- let root = metadata
- .packages
- .iter()
- .find(|package| {
- package.manifest_path.as_std_path().canonicalize().unwrap()
- == manifest_path.canonicalize().unwrap()
- })
- .unwrap();
-
- // Then go over all of its dependencies
- let dependencies = root
- .dependencies
- .iter()
- .filter(|dep| matches!(dep.kind, DependencyKind::Normal))
- // Only consider dependencies that are enabled on the current target
- .filter(|dep| match &dep.target {
- Some(platform) => platform.matches(config.target.as_ref().unwrap(), &cfg),
- None => true,
- })
- .map(|dep| {
- let package = metadata
- .packages
- .iter()
- .find(|&p| p.name == dep.name && dep.req.matches(&p.version))
- .expect("dependency does not exist");
- (
- package,
- dep.rename.clone().unwrap_or_else(|| package.name.clone()),
- )
- })
- // Also expose the root crate
- .chain(std::iter::once((root, root.name.clone())))
- .filter_map(|(package, name)| {
- // Get the id for the package matching the version requirement of the dep
- let id = &package.id;
- // Return the name chosen in `Cargo.toml` and the path to the corresponding artifact
- match artifacts.remove(id) {
- Some(artifacts) => Some((name.replace('-', "_"), artifacts)),
- None => {
- if name == root.name {
- // If there are no artifacts, this is the root crate and it is being built as a binary/test
- // instead of a library. We simply add no artifacts, meaning you can't depend on functions
- // and types declared in the root crate.
- None
- } else {
- panic!("no artifact found for `{name}`(`{id}`):`\n{artifact_output}")
- }
- }
- }
- })
- .collect();
- let import_paths = import_paths.into_iter().collect();
- return Ok(Dependencies {
- dependencies,
- import_paths,
- });
- }
-
- bail!("no json found in cargo-metadata output")
-}
diff --git a/vendor/ui_test/src/diff.rs b/vendor/ui_test/src/diff.rs
deleted file mode 100644
index 916645dd1..000000000
--- a/vendor/ui_test/src/diff.rs
+++ /dev/null
@@ -1,175 +0,0 @@
-use colored::*;
-use diff::{chars, lines, Result, Result::*};
-
-#[derive(Default)]
-struct DiffState<'a> {
- /// Whether we've already printed something, so we should print starting context, too.
- print_start_context: bool,
- /// When we skip lines, remember the last `CONTEXT` ones to
- /// display after the "skipped N lines" message
- skipped_lines: Vec<&'a str>,
- /// When we see a removed line, we don't print it, we
- /// keep it around to compare it with the next added line.
- prev_left: Option<&'a str>,
-}
-
-/// How many lines of context are displayed around the actual diffs
-const CONTEXT: usize = 2;
-
-impl<'a> DiffState<'a> {
- /// Print `... n lines skipped ...` followed by the last `CONTEXT` lines.
- fn print_end_skip(&self, skipped: usize) {
- self.print_skipped_msg(skipped);
- for line in self.skipped_lines.iter().rev().take(CONTEXT).rev() {
- eprintln!(" {line}");
- }
- }
-
- fn print_skipped_msg(&self, skipped: usize) {
- match skipped {
- // When the amount of skipped lines is exactly `CONTEXT * 2`, we already
- // print all the context and don't actually skip anything.
- 0 => {}
- // Instead of writing a line saying we skipped one line, print that one line
- 1 => eprintln!(" {}", self.skipped_lines[CONTEXT]),
- _ => eprintln!("... {skipped} lines skipped ..."),
- }
- }
-
- /// Print an initial `CONTEXT` amount of lines.
- fn print_start_skip(&self) {
- for line in self.skipped_lines.iter().take(CONTEXT) {
- eprintln!(" {line}");
- }
- }
-
- fn print_skip(&mut self) {
- let half = self.skipped_lines.len() / 2;
- if !self.print_start_context {
- self.print_start_context = true;
- self.print_end_skip(self.skipped_lines.len().saturating_sub(CONTEXT));
- } else if half < CONTEXT {
- // Print all the skipped lines if the amount of context desired is less than the amount of lines
- for line in self.skipped_lines.drain(..) {
- eprintln!(" {line}");
- }
- } else {
- self.print_start_skip();
- let skipped = self.skipped_lines.len() - CONTEXT * 2;
- self.print_end_skip(skipped);
- }
- self.skipped_lines.clear();
- }
-
- fn skip(&mut self, line: &'a str) {
- self.skipped_lines.push(line);
- }
-
- fn print_prev(&mut self) {
- if let Some(l) = self.prev_left.take() {
- self.print_left(l);
- }
- }
-
- fn print_left(&self, l: &str) {
- eprintln!("{}{}", "-".red(), l.red());
- }
-
- fn print_right(&self, r: &str) {
- eprintln!("{}{}", "+".green(), r.green());
- }
-
- fn row(&mut self, row: Result<&'a str>) {
- match row {
- Left(l) => {
- self.print_skip();
- self.print_prev();
- self.prev_left = Some(l);
- }
- Both(l, _) => {
- self.print_prev();
- self.skip(l);
- }
- Right(r) => {
- // When there's an added line after a removed line, we'll want to special case some print cases.
- // FIXME(oli-obk): also do special printing modes when there are multiple lines that only have minor changes.
- if let Some(l) = self.prev_left.take() {
- let diff = chars(l, r);
- let mut seen_l = false;
- let mut seen_r = false;
- for char in &diff {
- match char {
- Left(l) if !l.is_whitespace() => seen_l = true,
- Right(r) if !r.is_whitespace() => seen_r = true,
- _ => {}
- }
- }
- if seen_l && seen_r {
- // The line both adds and removes chars, print both lines, but highlight their differences instead of
- // drawing the entire line in red/green.
- eprint!("{}", "-".red());
- for char in &diff {
- match *char {
- Left(l) => eprint!("{}", l.to_string().red()),
- Right(_) => {}
- Both(l, _) => eprint!("{l}"),
- }
- }
- eprintln!();
- eprint!("{}", "+".green());
- for char in diff {
- match char {
- Left(_) => {}
- Right(r) => eprint!("{}", r.to_string().green()),
- Both(l, _) => eprint!("{l}"),
- }
- }
- eprintln!();
- } else {
- // The line only adds or only removes chars, print a single line highlighting their differences.
- eprint!("{}", "~".yellow());
- for char in diff {
- match char {
- Left(l) => eprint!("{}", l.to_string().red()),
- Both(l, _) => eprint!("{l}"),
- Right(r) => eprint!("{}", r.to_string().green()),
- }
- }
- eprintln!();
- }
- } else {
- self.print_skip();
- self.print_right(r);
- }
- }
- }
- }
-
- fn finish(self) {
- self.print_start_skip();
- self.print_skipped_msg(self.skipped_lines.len().saturating_sub(CONTEXT));
- eprintln!()
- }
-}
-
-pub fn print_diff(expected: &[u8], actual: &[u8]) {
- let expected_str = String::from_utf8_lossy(expected);
- let actual_str = String::from_utf8_lossy(actual);
-
- if expected_str.as_bytes() != expected || actual_str.as_bytes() != actual {
- eprintln!(
- "{}",
- "Non-UTF8 characters in output, diff may be imprecise.".red()
- );
- }
-
- let pat = |c: char| c.is_whitespace() && c != ' ' && c != '\n' && c != '\r';
- let expected_str = expected_str.replace(pat, "░");
- let actual_str = actual_str.replace(pat, "░");
-
- let mut state = DiffState::default();
- for row in lines(&expected_str, &actual_str) {
- state.row(row);
- }
- state.finish();
-}
diff --git a/vendor/ui_test/src/error.rs b/vendor/ui_test/src/error.rs
deleted file mode 100644
index 18ce52ecc..000000000
--- a/vendor/ui_test/src/error.rs
+++ /dev/null
@@ -1,72 +0,0 @@
-use crate::{parser::Pattern, rustc_stderr::Message, Mode};
-use std::{path::PathBuf, process::ExitStatus};
-
-/// All the ways in which a test can fail.
-#[derive(Debug)]
-pub enum Error {
- /// Got an invalid exit status for the given mode.
- ExitStatus {
- /// The expected mode.
- mode: Mode,
- /// The exit status of the command.
- status: ExitStatus,
- /// The expected exit status as set in the file or derived from the mode.
- expected: i32,
- },
- /// A pattern was declared but had no matching error.
- PatternNotFound {
- /// The pattern that was missing an error
- pattern: Pattern,
- /// The line in which the pattern was defined.
- definition_line: usize,
- },
- /// A ui test checking for failure does not have any failure patterns
- NoPatternsFound,
- /// A ui test checking for success has failure patterns
- PatternFoundInPassTest,
- /// Stderr/Stdout differed from the `.stderr`/`.stdout` file present.
- OutputDiffers {
- /// The file containing the expected output that differs from the actual output.
- path: PathBuf,
- /// The output from the command.
- actual: Vec<u8>,
- /// The contents of the file.
- expected: Vec<u8>,
- /// A command, that when run, causes the output to get blessed instead of erroring.
- bless_command: String,
- },
- /// There were errors that don't have a pattern.
- ErrorsWithoutPattern {
- /// The main message of the error.
- msgs: Vec<Message>,
- /// File and line information of the error.
- path: Option<(PathBuf, usize)>,
- },
- /// A comment failed to parse.
- InvalidComment {
- /// The comment
- msg: String,
- /// THe line in which it was defined.
- line: usize,
- },
- /// A subcommand (e.g. rustfix) of a test failed.
- Command {
- /// The name of the subcommand (e.g. "rustfix").
- kind: String,
- /// The exit status of the command.
- status: ExitStatus,
- },
- /// This catches crashes of ui tests and reports them along the failed test.
- Bug(String),
- /// An auxiliary build failed with its own set of errors.
- Aux {
- /// Path to the aux file.
- path: PathBuf,
- /// The errors that occurred during the build of the aux file.
- errors: Vec<Error>,
- /// The line in which the aux file was requested to be built.
- line: usize,
- },
-}
-
-pub(crate) type Errors = Vec<Error>;
diff --git a/vendor/ui_test/src/github_actions.rs b/vendor/ui_test/src/github_actions.rs
deleted file mode 100644
index cad9bf569..000000000
--- a/vendor/ui_test/src/github_actions.rs
+++ /dev/null
@@ -1,94 +0,0 @@
-//! An interface to github actions workflow commands.
-
-use std::fmt::{Debug, Write};
-
-/// Shows an error message directly in a github diff view on drop.
-pub struct Error {
- file: String,
- line: usize,
- title: String,
- message: String,
-}
-impl Error {
- /// Set a line for this error. By default the message is shown at the top of the file.
- pub fn line(mut self, line: usize) -> Self {
- self.line = line;
- self
- }
-}
-
-/// Create an error to be shown for the given file and with the given title.
-pub fn error(file: impl std::fmt::Display, title: impl Into<String>) -> Error {
- Error {
- file: file.to_string(),
- line: 0,
- title: title.into(),
- message: String::new(),
- }
-}
-
-impl Write for Error {
- fn write_str(&mut self, s: &str) -> std::fmt::Result {
- self.message.write_str(s)
- }
-}
-
-impl Drop for Error {
- fn drop(&mut self) {
- if std::env::var_os("GITHUB_ACTION").is_some() {
- let Error {
- file,
- line,
- title,
- message,
- } = self;
- let message = message.trim();
- let message = if message.is_empty() {
- "::no message".into()
- } else {
- format!("::{}", github_action_multiline_escape(message))
- };
- eprintln!("::error file={file},line={line},title={title}{message}");
- eprintln!("error file={file},line={line},title={title}{message}");
- }
- }
-}
-
-/// Append to the summary file that will be shown for the entire CI run.
-pub fn summary() -> Option<impl std::io::Write> {
- let path = std::env::var_os("GITHUB_STEP_SUMMARY")?;
- Some(std::fs::OpenOptions::new().append(true).open(path).unwrap())
-}
-
-fn github_action_multiline_escape(s: &str) -> String {
- s.replace('%', "%25")
- .replace('\n', "%0A")
- .replace('\r', "%0D")
-}
-
-/// All github actions log messages from this call to the Drop of the return value
-/// will be grouped and hidden by default in logs. Note that nesting these does
-/// not really work.
-pub fn group(name: impl std::fmt::Display) -> Group {
- if std::env::var_os("GITHUB_ACTION").is_some() {
- eprintln!("::group::{name}");
- }
- Group(())
-}
-
-/// A guard that closes the current github actions log group on drop.
-pub struct Group(());
-
-impl Debug for Group {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- f.write_str("a handle that will close the github action group on drop")
- }
-}
-
-impl Drop for Group {
- fn drop(&mut self) {
- if std::env::var_os("GITHUB_ACTION").is_some() {
- eprintln!("::endgroup::");
- }
- }
-}
diff --git a/vendor/ui_test/src/lib.rs b/vendor/ui_test/src/lib.rs
deleted file mode 100644
index 1420e2881..000000000
--- a/vendor/ui_test/src/lib.rs
+++ /dev/null
@@ -1,1066 +0,0 @@
-#![allow(
- clippy::enum_variant_names,
- clippy::useless_format,
- clippy::too_many_arguments,
- rustc::internal
-)]
-#![deny(missing_docs)]
-
-//! A crate to run the Rust compiler (or other binaries) and test their command line output.
-
-use bstr::ByteSlice;
-pub use color_eyre;
-use color_eyre::eyre::{eyre, Result};
-use crossbeam_channel::{unbounded, Receiver, Sender};
-use parser::{ErrorMatch, Revisioned};
-use regex::bytes::Regex;
-use rustc_stderr::{Diagnostics, Level, Message};
-use status_emitter::StatusEmitter;
-use std::borrow::Cow;
-use std::collections::{HashSet, VecDeque};
-use std::path::{Path, PathBuf};
-use std::process::Command;
-use std::thread;
-
-use crate::parser::{Comments, Condition};
-
-mod cmd;
-mod config;
-mod dependencies;
-mod diff;
-mod error;
-pub mod github_actions;
-mod mode;
-mod parser;
-mod rustc_stderr;
-pub mod status_emitter;
-#[cfg(test)]
-mod tests;
-
-pub use cmd::*;
-pub use config::*;
-pub use error::*;
-pub use mode::*;
-
-/// A filter's match rule.
-#[derive(Clone, Debug)]
-pub enum Match {
- /// If the regex matches, the filter applies
- Regex(Regex),
- /// If the exact byte sequence is found, the filter applies
- Exact(Vec<u8>),
-}
-impl Match {
- fn replace_all<'a>(&self, text: &'a [u8], replacement: &[u8]) -> Cow<'a, [u8]> {
- match self {
- Match::Regex(regex) => regex.replace_all(text, replacement),
- Match::Exact(needle) => text.replace(needle, replacement).into(),
- }
- }
-}
-
-impl From<&'_ Path> for Match {
- fn from(v: &Path) -> Self {
- let mut v = v.display().to_string();
- // Normalize away windows canonicalized paths.
- if v.starts_with(r#"\\?\"#) {
- v.drain(0..4);
- }
- let mut v = v.into_bytes();
- // Normalize paths on windows to use slashes instead of backslashes,
- // So that paths are rendered the same on all systems.
- for c in &mut v {
- if *c == b'\\' {
- *c = b'/';
- }
- }
- Self::Exact(v)
- }
-}
-
-impl From<Regex> for Match {
- fn from(v: Regex) -> Self {
- Self::Regex(v)
- }
-}
-
-/// Replacements to apply to output files.
-pub type Filter = Vec<(Match, &'static [u8])>;
-
-/// Run all tests as described in the config argument.
-pub fn run_tests(config: Config) -> Result<()> {
- eprintln!(" Compiler: {}", config.program.display());
-
- let name = config.root_dir.display().to_string();
-
- run_tests_generic(
- config,
- default_file_filter,
- default_per_file_config,
- (status_emitter::Text, status_emitter::Gha::<true> { name }),
- )
-}
-
-/// The filter used by `run_tests` to only run on `.rs` files.
-pub fn default_file_filter(path: &Path) -> bool {
- path.extension().map(|ext| ext == "rs").unwrap_or(false)
-}
-
-/// The default per-file config used by `run_tests`.
-pub fn default_per_file_config(config: &Config, path: &Path) -> Option<Config> {
- let mut config = config.clone();
- // Heuristic:
- // * if the file contains `#[test]`, automatically pass `--cfg test`.
- // * if the file does not contain `fn main()` or `#[start]`, automatically pass `--crate-type=lib`.
- // This avoids having to spam `fn main() {}` in almost every test.
- let file_contents = std::fs::read(path).unwrap();
- if file_contents.find(b"#[proc_macro]").is_some()
- || file_contents.find(b"#[proc_macro_attribute]").is_some()
- || file_contents.find(b"#[proc_macro_derive]").is_some()
- {
- config.program.args.push("--crate-type=proc-macro".into())
- } else if file_contents.find(b"#[test]").is_some() {
- config.program.args.push("--test".into());
- } else if file_contents.find(b"fn main()").is_none()
- && file_contents.find(b"#[start]").is_none()
- {
- config.program.args.push("--crate-type=lib".into());
- }
- Some(config)
-}
-
-/// Create a command for running a single file, with the settings from the `config` argument.
-/// Ignores various settings from `Config` that relate to finding test files.
-pub fn test_command(mut config: Config, path: &Path) -> Result<Command> {
- config.build_dependencies_and_link_them()?;
-
- let comments =
- Comments::parse_file(path)?.map_err(|errors| color_eyre::eyre::eyre!("{errors:#?}"))?;
- let mut errors = vec![];
- let result = build_command(path, &config, "", &comments, &mut errors);
- assert!(errors.is_empty(), "{errors:#?}");
- Ok(result)
-}
-
-#[allow(clippy::large_enum_variant)]
-/// The possible results a single test can have.
-pub enum TestResult {
- /// The test passed
- Ok,
- /// The test was ignored due to a rule (`//@only-*` or `//@ignore-*`)
- Ignored,
- /// The test was filtered with the `file_filter` argument.
- Filtered,
- /// The test failed.
- Errored {
- /// Command that failed
- command: Command,
- /// The errors that were encountered.
- errors: Vec<Error>,
- /// The full stderr of the test run.
- stderr: Vec<u8>,
- },
-}
-
-struct TestRun {
- result: TestResult,
- path: PathBuf,
- revision: String,
-}
-
-/// A version of `run_tests` that allows more fine-grained control over running tests.
-pub fn run_tests_generic(
- mut config: Config,
- file_filter: impl Fn(&Path) -> bool + Sync,
- per_file_config: impl Fn(&Config, &Path) -> Option<Config> + Sync,
- mut status_emitter: impl StatusEmitter + Send,
-) -> Result<()> {
- config.fill_host_and_target()?;
-
- config.build_dependencies_and_link_them()?;
-
- let mut results = vec![];
-
- run_and_collect(
- config.num_test_threads.get(),
- |submit| {
- let mut todo = VecDeque::new();
- todo.push_back(config.root_dir.clone());
- while let Some(path) = todo.pop_front() {
- if path.is_dir() {
- if path.file_name().unwrap() == "auxiliary" {
- continue;
- }
- // Enqueue everything inside this directory.
- // We want it sorted, to have some control over scheduling of slow tests.
- let mut entries = std::fs::read_dir(path)
- .unwrap()
- .collect::<Result<Vec<_>, _>>()
- .unwrap();
- entries.sort_by_key(|e| e.file_name());
- for entry in entries {
- todo.push_back(entry.path());
- }
- } else if file_filter(&path) {
- // Forward .rs files to the test workers.
- submit.send(path).unwrap();
- }
- }
- },
- |receive, finished_files_sender| -> Result<()> {
- for path in receive {
- let maybe_config;
- let config = match per_file_config(&config, &path) {
- None => &config,
- Some(config) => {
- maybe_config = config;
- &maybe_config
- }
- };
- let result = match std::panic::catch_unwind(|| parse_and_test_file(&path, config)) {
- Ok(res) => res,
- Err(err) => {
- finished_files_sender.send(TestRun {
- result: TestResult::Errored {
- command: Command::new("<unknown>"),
- errors: vec![Error::Bug(
- *Box::<dyn std::any::Any + Send + 'static>::downcast::<String>(
- err,
- )
- .unwrap(),
- )],
- stderr: vec![],
- },
- path,
- revision: String::new(),
- })?;
- continue;
- }
- };
- for result in result {
- finished_files_sender.send(result)?;
- }
- }
- Ok(())
- },
- |finished_files_recv| {
- for run in finished_files_recv {
- status_emitter.test_result(&run.path, &run.revision, &run.result);
-
- results.push(run);
- }
- },
- )?;
-
- let mut failures = vec![];
- let mut succeeded = 0;
- let mut ignored = 0;
- let mut filtered = 0;
-
- for run in results {
- match run.result {
- TestResult::Ok => succeeded += 1,
- TestResult::Ignored => ignored += 1,
- TestResult::Filtered => filtered += 1,
- TestResult::Errored {
- command,
- errors,
- stderr,
- } => failures.push((run.path, command, run.revision, errors, stderr)),
- }
- }
-
- let mut failure_emitter = status_emitter.finalize(failures.len(), succeeded, ignored, filtered);
- for (path, command, revision, errors, stderr) in &failures {
- let _guard = status_emitter.failed_test(revision, path, command, stderr);
- failure_emitter.test_failure(path, revision, errors);
- }
-
- if failures.is_empty() {
- Ok(())
- } else {
- Err(eyre!("tests failed"))
- }
-}
-
-/// A generic multithreaded runner that has a thread for producing work,
-/// a thread for collecting work, and `num_threads` threads for doing the work.
-pub fn run_and_collect<SUBMISSION: Send, RESULT: Send>(
- num_threads: usize,
- submitter: impl FnOnce(Sender<SUBMISSION>) + Send,
- runner: impl Sync + Fn(&Receiver<SUBMISSION>, Sender<RESULT>) -> Result<()>,
- collector: impl FnOnce(Receiver<RESULT>) + Send,
-) -> Result<()> {
- // A channel for files to process
- let (submit, receive) = unbounded();
-
- thread::scope(|s| {
- // Create a thread that is in charge of walking the directory and submitting jobs.
- // It closes the channel when it is done.
- s.spawn(|| submitter(submit));
-
- // A channel for the messages emitted by the individual test threads.
- // Used to produce live updates while running the tests.
- let (finished_files_sender, finished_files_recv) = unbounded();
-
- s.spawn(|| collector(finished_files_recv));
-
- let mut threads = vec![];
-
- // Create N worker threads that receive files to test.
- for _ in 0..num_threads {
- let finished_files_sender = finished_files_sender.clone();
- threads.push(s.spawn(|| runner(&receive, finished_files_sender)));
- }
-
- for thread in threads {
- thread.join().unwrap()?;
- }
- Ok(())
- })
-}
-
-fn parse_and_test_file(path: &Path, config: &Config) -> Vec<TestRun> {
- let comments = match parse_comments_in_file(path) {
- Ok(comments) => comments,
- Err((stderr, errors)) => {
- return vec![TestRun {
- result: TestResult::Errored {
- command: Command::new("parse comments"),
- errors,
- stderr,
- },
- path: path.into(),
- revision: "".into(),
- }]
- }
- };
- // Run the test for all revisions
- comments
- .revisions
- .clone()
- .unwrap_or_else(|| vec![String::new()])
- .into_iter()
- .map(|revision| {
- // Ignore file if only/ignore rules do (not) apply
- if !test_file_conditions(&comments, config, &revision) {
- return TestRun {
- result: TestResult::Ignored,
- path: path.into(),
- revision,
- };
- }
- let (command, errors, stderr) = run_test(path, config, &revision, &comments);
- let result = if errors.is_empty() {
- TestResult::Ok
- } else {
- TestResult::Errored {
- command,
- errors,
- stderr,
- }
- };
- TestRun {
- result,
- revision,
- path: path.into(),
- }
- })
- .collect()
-}
-
-fn parse_comments_in_file(path: &Path) -> Result<Comments, (Vec<u8>, Vec<Error>)> {
- match Comments::parse_file(path) {
- Ok(Ok(comments)) => Ok(comments),
- Ok(Err(errors)) => Err((vec![], errors)),
- Err(err) => Err((format!("{err:?}").into(), vec![])),
- }
-}
-
-fn build_command(
- path: &Path,
- config: &Config,
- revision: &str,
- comments: &Comments,
- errors: &mut Vec<Error>,
-) -> Command {
- let mut cmd = config.program.build(&config.out_dir);
- cmd.arg(path);
- if !revision.is_empty() {
- cmd.arg(format!("--cfg={revision}"));
- }
- for arg in comments
- .for_revision(revision)
- .flat_map(|r| r.compile_flags.iter())
- {
- cmd.arg(arg);
- }
- let edition = comments.edition(errors, revision, config);
- if let Some((edition, _)) = edition {
- cmd.arg("--edition").arg(edition);
- }
- cmd.args(config.trailing_args.iter());
- cmd.envs(
- comments
- .for_revision(revision)
- .flat_map(|r| r.env_vars.iter())
- .map(|(k, v)| (k, v)),
- );
-
- cmd
-}
-
-fn build_aux(
- aux_file: &Path,
- path: &Path,
- config: &Config,
- revision: &str,
- comments: &Comments,
- kind: &str,
- aux: &Path,
- extra_args: &mut Vec<String>,
-) -> std::result::Result<(), (Command, Vec<Error>, Vec<u8>)> {
- let comments = match parse_comments_in_file(aux_file) {
- Ok(comments) => comments,
- Err((msg, mut errors)) => {
- return Err((
- build_command(path, config, revision, comments, &mut errors),
- errors,
- msg,
- ))
- }
- };
- assert_eq!(comments.revisions, None);
-
- let mut config = config.clone();
-
- // Strip any `crate-type` flags from the args, as we need to set our own,
- // and they may conflict (e.g. `lib` vs `proc-macro`);
- let mut prev_was_crate_type = false;
- config.program.args.retain(|arg| {
- if prev_was_crate_type {
- prev_was_crate_type = false;
- return false;
- }
- if arg == "--test" {
- false
- } else if arg == "--crate-type" {
- prev_was_crate_type = true;
- false
- } else if let Some(arg) = arg.to_str() {
- !arg.starts_with("--crate-type=")
- } else {
- true
- }
- });
-
- // Put aux builds into a separate directory per test so that
- // tests running in parallel but building the same aux build don't conflict.
- // FIXME: put aux builds into the regular build queue.
- config.out_dir = config.out_dir.join(path.with_extension(""));
-
- let mut errors = vec![];
-
- let mut aux_cmd = build_command(aux_file, &config, revision, &comments, &mut errors);
-
- if !errors.is_empty() {
- return Err((aux_cmd, errors, vec![]));
- }
-
- let current_extra_args =
- build_aux_files(aux_file, aux_file.parent().unwrap(), &comments, "", &config)?;
- // Make sure we see our dependencies
- aux_cmd.args(current_extra_args.iter());
- // Make sure our dependents also see our dependencies.
- extra_args.extend(current_extra_args);
-
- aux_cmd.arg("--crate-type").arg(kind);
- aux_cmd.arg("--emit=link");
- let filename = aux.file_stem().unwrap().to_str().unwrap();
- let output = aux_cmd.output().unwrap();
- if !output.status.success() {
- let error = Error::Command {
- kind: "compilation of aux build failed".to_string(),
- status: output.status,
- };
- return Err((
- aux_cmd,
- vec![error],
- rustc_stderr::process(path, &output.stderr).rendered,
- ));
- }
-
- // Now run the command again to fetch the output filenames
- aux_cmd.arg("--print").arg("file-names");
- let output = aux_cmd.output().unwrap();
- assert!(output.status.success());
-
- for file in output.stdout.lines() {
- let file = std::str::from_utf8(file).unwrap();
- let crate_name = filename.replace('-', "_");
- let path = config.out_dir.join(file);
- extra_args.push("--extern".into());
- extra_args.push(format!("{crate_name}={}", path.display()));
- // Help cargo find the crates added with `--extern`.
- extra_args.push("-L".into());
- extra_args.push(config.out_dir.display().to_string());
- }
- Ok(())
-}
-
-fn run_test(
- path: &Path,
- config: &Config,
- revision: &str,
- comments: &Comments,
-) -> (Command, Errors, Vec<u8>) {
- let extra_args = match build_aux_files(
- path,
- &path.parent().unwrap().join("auxiliary"),
- comments,
- revision,
- config,
- ) {
- Ok(value) => value,
- Err(value) => return value,
- };
-
- let mut errors = vec![];
-
- let mut cmd = build_command(path, config, revision, comments, &mut errors);
- cmd.args(&extra_args);
-
- let output = cmd
- .output()
- .unwrap_or_else(|err| panic!("could not execute {cmd:?}: {err}"));
- let mode = config.mode.maybe_override(comments, revision, &mut errors);
- let status_check = mode.ok(output.status);
- if status_check.is_empty() && matches!(mode, Mode::Run { .. }) {
- let cmd = run_test_binary(mode, path, revision, comments, cmd, config, &mut errors);
- return (cmd, errors, vec![]);
- }
- errors.extend(status_check);
- if output.status.code() == Some(101) && !matches!(config.mode, Mode::Panic | Mode::Yolo) {
- let stderr = String::from_utf8_lossy(&output.stderr);
- let stdout = String::from_utf8_lossy(&output.stdout);
- errors.push(Error::Bug(format!(
- "test panicked: stderr:\n{stderr}\nstdout:\n{stdout}",
- )));
- return (cmd, errors, vec![]);
- }
- // Always remove annotation comments from stderr.
- let diagnostics = rustc_stderr::process(path, &output.stderr);
- let rustfixed = matches!(mode, Mode::Fix).then(|| {
- run_rustfix(
- &output.stderr,
- path,
- comments,
- revision,
- config,
- extra_args,
- &mut errors,
- )
- });
- let stderr = check_test_result(
- path,
- config,
- revision,
- comments,
- &mut errors,
- &output.stdout,
- diagnostics,
- );
- if let Some((mut rustfix, rustfix_path)) = rustfixed {
- // picking the crate name from the file name is problematic when `.revision_name` is inserted
- rustfix.arg("--crate-name").arg(
- path.file_stem()
- .unwrap()
- .to_str()
- .unwrap()
- .replace('-', "_"),
- );
- let output = rustfix.output().unwrap();
- if !output.status.success() {
- errors.push(Error::Command {
- kind: "rustfix".into(),
- status: output.status,
- });
- return (
- rustfix,
- errors,
- rustc_stderr::process(&rustfix_path, &output.stderr).rendered,
- );
- }
- }
- (cmd, errors, stderr)
-}
-
-fn build_aux_files(
- path: &Path,
- aux_dir: &Path,
- comments: &Comments,
- revision: &str,
- config: &Config,
-) -> Result<Vec<String>, (Command, Vec<Error>, Vec<u8>)> {
- let mut extra_args = vec![];
- for rev in comments.for_revision(revision) {
- for (aux, kind, line) in &rev.aux_builds {
- let aux_file = if aux.starts_with("..") {
- aux_dir.parent().unwrap().join(aux)
- } else {
- aux_dir.join(aux)
- };
- if let Err((command, errors, msg)) = build_aux(
- &aux_file,
- path,
- config,
- revision,
- comments,
- kind,
- aux,
- &mut extra_args,
- ) {
- return Err((
- command,
- vec![Error::Aux {
- path: aux_file,
- errors,
- line: *line,
- }],
- msg,
- ));
- }
- }
- }
- Ok(extra_args)
-}
-
-fn run_test_binary(
- mode: Mode,
- path: &Path,
- revision: &str,
- comments: &Comments,
- mut cmd: Command,
- config: &Config,
- errors: &mut Vec<Error>,
-) -> Command {
- cmd.arg("--print").arg("file-names");
- let output = cmd.output().unwrap();
- assert!(output.status.success());
-
- let mut files = output.stdout.lines();
- let file = files.next().unwrap();
- assert_eq!(files.next(), None);
- let file = std::str::from_utf8(file).unwrap();
- let exe = config.out_dir.join(file);
- let mut exe = Command::new(exe);
- let output = exe.output().unwrap();
-
- check_test_output(
- path,
- errors,
- revision,
- config,
- comments,
- &output.stdout,
- &output.stderr,
- );
-
- errors.extend(mode.ok(output.status));
-
- exe
-}
-
-fn run_rustfix(
- stderr: &[u8],
- path: &Path,
- comments: &Comments,
- revision: &str,
- config: &Config,
- extra_args: Vec<String>,
- errors: &mut Vec<Error>,
-) -> (Command, PathBuf) {
- let input = std::str::from_utf8(stderr).unwrap();
- let suggestions = rustfix::get_suggestions_from_json(
- input,
- &HashSet::new(),
- if let Mode::Yolo = config.mode {
- rustfix::Filter::Everything
- } else {
- rustfix::Filter::MachineApplicableOnly
- },
- )
- .unwrap_or_else(|err| {
- panic!("could not deserialize diagnostics json for rustfix {err}:{input}")
- });
- let fixed_code =
- rustfix::apply_suggestions(&std::fs::read_to_string(path).unwrap(), &suggestions)
- .unwrap_or_else(|e| {
- panic!(
- "failed to apply suggestions for {:?} with rustfix: {e}",
- path.display()
- )
- });
- let edition = comments.edition(errors, revision, config);
- let rustfix_comments = Comments {
- revisions: None,
- revisioned: std::iter::once((
- vec![],
- Revisioned {
- line: 0,
- ignore: vec![],
- only: vec![],
- stderr_per_bitwidth: false,
- compile_flags: comments
- .for_revision(revision)
- .flat_map(|r| r.compile_flags.iter().cloned())
- .collect(),
- env_vars: comments
- .for_revision(revision)
- .flat_map(|r| r.env_vars.iter().cloned())
- .collect(),
- normalize_stderr: vec![],
- error_in_other_files: vec![],
- error_matches: vec![],
- require_annotations_for_level: None,
- aux_builds: comments
- .for_revision(revision)
- .flat_map(|r| r.aux_builds.iter().cloned())
- .collect(),
- edition,
- mode: Some((Mode::Pass, 0)),
- needs_asm_support: false,
- },
- ))
- .collect(),
- };
- let path = check_output(
- fixed_code.as_bytes(),
- path,
- errors,
- revised(revision, "fixed"),
- &Filter::default(),
- config,
- &rustfix_comments,
- revision,
- );
-
- let mut cmd = build_command(&path, config, revision, &rustfix_comments, errors);
- cmd.args(extra_args);
- (cmd, path)
-}
-
-fn revised(revision: &str, extension: &str) -> String {
- if revision.is_empty() {
- extension.to_string()
- } else {
- format!("{revision}.{extension}")
- }
-}
-
-fn check_test_result(
- path: &Path,
- config: &Config,
- revision: &str,
- comments: &Comments,
- errors: &mut Errors,
- stdout: &[u8],
- diagnostics: Diagnostics,
-) -> Vec<u8> {
- check_test_output(
- path,
- errors,
- revision,
- config,
- comments,
- stdout,
- &diagnostics.rendered,
- );
- // Check error annotations in the source against output
- check_annotations(
- diagnostics.messages,
- diagnostics.messages_from_unknown_file_or_line,
- path,
- errors,
- config,
- revision,
- comments,
- );
- diagnostics.rendered
-}
-
-fn check_test_output(
- path: &Path,
- errors: &mut Vec<Error>,
- revision: &str,
- config: &Config,
- comments: &Comments,
- stdout: &[u8],
- stderr: &[u8],
-) {
- // Check output files (if any)
- // Check output files against actual output
- check_output(
- stderr,
- path,
- errors,
- revised(revision, "stderr"),
- &config.stderr_filters,
- config,
- comments,
- revision,
- );
- check_output(
- stdout,
- path,
- errors,
- revised(revision, "stdout"),
- &config.stdout_filters,
- config,
- comments,
- revision,
- );
-}
-
-fn check_annotations(
- mut messages: Vec<Vec<Message>>,
- mut messages_from_unknown_file_or_line: Vec<Message>,
- path: &Path,
- errors: &mut Errors,
- config: &Config,
- revision: &str,
- comments: &Comments,
-) {
- let error_patterns = comments
- .for_revision(revision)
- .flat_map(|r| r.error_in_other_files.iter());
-
- let mut seen_error_match = false;
- for (error_pattern, definition_line) in error_patterns {
- seen_error_match = true;
- // first check the diagnostics messages outside of our file. We check this first, so that
- // you can mix in-file annotations with //@error-in-other-file annotations, even if there is overlap
- // in the messages.
- if let Some(i) = messages_from_unknown_file_or_line
- .iter()
- .position(|msg| error_pattern.matches(&msg.message))
- {
- messages_from_unknown_file_or_line.remove(i);
- } else {
- errors.push(Error::PatternNotFound {
- pattern: error_pattern.clone(),
- definition_line: *definition_line,
- });
- }
- }
-
- // The order on `Level` is such that `Error` is the highest level.
- // We will ensure that *all* diagnostics of level at least `lowest_annotation_level`
- // are matched.
- let mut lowest_annotation_level = Level::Error;
- for &ErrorMatch {
- ref pattern,
- definition_line,
- line,
- level,
- } in comments
- .for_revision(revision)
- .flat_map(|r| r.error_matches.iter())
- {
- seen_error_match = true;
- // If we found a diagnostic with a level annotation, make sure that all
- // diagnostics of that level have annotations, even if we don't end up finding a matching diagnostic
- // for this pattern.
- lowest_annotation_level = std::cmp::min(lowest_annotation_level, level);
-
- if let Some(msgs) = messages.get_mut(line) {
- let found = msgs
- .iter()
- .position(|msg| pattern.matches(&msg.message) && msg.level == level);
- if let Some(found) = found {
- msgs.remove(found);
- continue;
- }
- }
-
- errors.push(Error::PatternNotFound {
- pattern: pattern.clone(),
- definition_line,
- });
- }
-
- let required_annotation_level = comments
- .find_one_for_revision(
- revision,
- |r| r.require_annotations_for_level,
- |_| {
- errors.push(Error::InvalidComment {
- msg: "`require_annotations_for_level` specified twice for same revision".into(),
- line: 0,
- })
- },
- )
- .unwrap_or(lowest_annotation_level);
- let filter = |mut msgs: Vec<Message>| -> Vec<_> {
- msgs.retain(|msg| msg.level >= required_annotation_level);
- msgs
- };
-
- let mode = config.mode.maybe_override(comments, revision, errors);
-
- if !matches!(config.mode, Mode::Yolo) {
- let messages_from_unknown_file_or_line = filter(messages_from_unknown_file_or_line);
- if !messages_from_unknown_file_or_line.is_empty() {
- errors.push(Error::ErrorsWithoutPattern {
- path: None,
- msgs: messages_from_unknown_file_or_line,
- });
- }
-
- for (line, msgs) in messages.into_iter().enumerate() {
- let msgs = filter(msgs);
- if !msgs.is_empty() {
- errors.push(Error::ErrorsWithoutPattern {
- path: Some((path.to_path_buf(), line)),
- msgs,
- });
- }
- }
- }
-
- match (mode, seen_error_match) {
- (Mode::Pass, true) | (Mode::Panic, true) => errors.push(Error::PatternFoundInPassTest),
- (
- Mode::Fail {
- require_patterns: true,
- },
- false,
- ) => errors.push(Error::NoPatternsFound),
- _ => {}
- }
-}
-
-fn check_output(
- output: &[u8],
- path: &Path,
- errors: &mut Errors,
- kind: String,
- filters: &Filter,
- config: &Config,
- comments: &Comments,
- revision: &str,
-) -> PathBuf {
- let target = config.target.as_ref().unwrap();
- let output = normalize(path, output, filters, comments, revision);
- let path = output_path(path, comments, kind, target, revision);
- match &config.output_conflict_handling {
- OutputConflictHandling::Bless => {
- if output.is_empty() {
- let _ = std::fs::remove_file(&path);
- } else {
- std::fs::write(&path, &output).unwrap();
- }
- }
- OutputConflictHandling::Error(bless_command) => {
- let expected_output = std::fs::read(&path).unwrap_or_default();
- if output != expected_output {
- errors.push(Error::OutputDiffers {
- path: path.clone(),
- actual: output,
- expected: expected_output,
- bless_command: bless_command.clone(),
- });
- }
- }
- OutputConflictHandling::Ignore => {}
- }
- path
-}
-
-fn output_path(
- path: &Path,
- comments: &Comments,
- kind: String,
- target: &str,
- revision: &str,
-) -> PathBuf {
- if comments
- .for_revision(revision)
- .any(|r| r.stderr_per_bitwidth)
- {
- return path.with_extension(format!("{}bit.{kind}", get_pointer_width(target)));
- }
- path.with_extension(kind)
-}
-
-fn test_condition(condition: &Condition, config: &Config) -> bool {
- let target = config.target.as_ref().unwrap();
- match condition {
- Condition::Bitwidth(bits) => get_pointer_width(target) == *bits,
- Condition::Target(t) => target.contains(t),
- Condition::Host(t) => config.host.as_ref().unwrap().contains(t),
- Condition::OnHost => target == config.host.as_ref().unwrap(),
- }
-}
-
-/// Returns whether according to the in-file conditions, this file should be run.
-fn test_file_conditions(comments: &Comments, config: &Config, revision: &str) -> bool {
- if comments
- .for_revision(revision)
- .flat_map(|r| r.ignore.iter())
- .any(|c| test_condition(c, config))
- {
- return false;
- }
- if comments
- .for_revision(revision)
- .any(|r| r.needs_asm_support && !config.has_asm_support())
- {
- return false;
- }
- comments
- .for_revision(revision)
- .flat_map(|r| r.only.iter())
- .all(|c| test_condition(c, config))
-}
-
-// Taken 1:1 from compiletest-rs
-fn get_pointer_width(triple: &str) -> u8 {
- if (triple.contains("64") && !triple.ends_with("gnux32") && !triple.ends_with("gnu_ilp32"))
- || triple.starts_with("s390x")
- {
- 64
- } else if triple.starts_with("avr") {
- 16
- } else {
- 32
- }
-}
-
-fn normalize(
- path: &Path,
- text: &[u8],
- filters: &Filter,
- comments: &Comments,
- revision: &str,
-) -> Vec<u8> {
- // Useless paths
- let path_filter = (Match::from(path.parent().unwrap()), b"$DIR" as &[u8]);
- let filters = filters.iter().chain(std::iter::once(&path_filter));
- let mut text = text.to_owned();
- if let Some(lib_path) = option_env!("RUSTC_LIB_PATH") {
- text = text.replace(lib_path, "RUSTLIB");
- }
-
- for (regex, replacement) in filters {
- text = regex.replace_all(&text, replacement).into_owned();
- }
-
- for (from, to) in comments
- .for_revision(revision)
- .flat_map(|r| r.normalize_stderr.iter())
- {
- text = from.replace_all(&text, to).into_owned();
- }
- text
-}
diff --git a/vendor/ui_test/src/mode.rs b/vendor/ui_test/src/mode.rs
deleted file mode 100644
index 9c4ce8c60..000000000
--- a/vendor/ui_test/src/mode.rs
+++ /dev/null
@@ -1,84 +0,0 @@
-use super::Error;
-use super::Errors;
-use crate::parser::Comments;
-use std::fmt::Display;
-use std::process::ExitStatus;
-
-#[derive(Copy, Clone, Debug)]
-/// Decides what is expected of each test's exit status.
-pub enum Mode {
- /// The test fails with an error, but passes after running rustfix
- Fix,
- /// The test passes a full execution of the rustc driver
- Pass,
- /// The test produces an executable binary that can get executed on the host
- Run {
- /// The expected exit code
- exit_code: i32,
- },
- /// The rustc driver panicked
- Panic,
- /// The rustc driver emitted an error
- Fail {
- /// Whether failing tests must have error patterns. Set to false if you just care about .stderr output.
- require_patterns: bool,
- },
- /// Run the tests, but always pass them as long as all annotations are satisfied and stderr files match.
- Yolo,
-}
-
-impl Mode {
- pub(crate) fn ok(self, status: ExitStatus) -> Errors {
- let expected = match self {
- Mode::Run { exit_code } => exit_code,
- Mode::Pass => 0,
- Mode::Panic => 101,
- Mode::Fail { .. } => 1,
- Mode::Fix | Mode::Yolo => return vec![],
- };
- if status.code() == Some(expected) {
- vec![]
- } else {
- vec![Error::ExitStatus {
- mode: self,
- status,
- expected,
- }]
- }
- }
- pub(crate) fn maybe_override(
- self,
- comments: &Comments,
- revision: &str,
- errors: &mut Vec<Error>,
- ) -> Self {
- comments
- .find_one_for_revision(
- revision,
- |r| r.mode.as_ref(),
- |&(_, line)| {
- errors.push(Error::InvalidComment {
- msg: "multiple mode changes found".into(),
- line,
- })
- },
- )
- .map(|&(mode, _)| mode)
- .unwrap_or(self)
- }
-}
-
-impl Display for Mode {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- match self {
- Mode::Run { exit_code } => write!(f, "run({exit_code})"),
- Mode::Pass => write!(f, "pass"),
- Mode::Panic => write!(f, "panic"),
- Mode::Fail {
- require_patterns: _,
- } => write!(f, "fail"),
- Mode::Yolo => write!(f, "yolo"),
- Mode::Fix => write!(f, "fix"),
- }
- }
-}
diff --git a/vendor/ui_test/src/parser.rs b/vendor/ui_test/src/parser.rs
deleted file mode 100644
index b041fb812..000000000
--- a/vendor/ui_test/src/parser.rs
+++ /dev/null
@@ -1,679 +0,0 @@
-use std::{
- collections::HashMap,
- path::{Path, PathBuf},
-};
-
-use bstr::{ByteSlice, Utf8Error};
-use regex::bytes::Regex;
-
-use crate::{rustc_stderr::Level, Error, Mode};
-
-use color_eyre::eyre::{Context, Result};
-
-#[cfg(test)]
-mod tests;
-
-/// This crate supports various magic comments that get parsed as file-specific
-/// configuration values. This struct parses them all in one go and then they
-/// get processed by their respective use sites.
-#[derive(Default, Debug)]
-pub(crate) struct Comments {
- /// List of revision names to execute. Can only be specified once
- pub revisions: Option<Vec<String>>,
- /// Comments that are only available under specific revisions.
- /// The defaults are in key `vec![]`
- pub revisioned: HashMap<Vec<String>, Revisioned>,
-}
-
-impl Comments {
- /// Check that a comment isn't specified twice across multiple differently revisioned statements.
- /// e.g. `//@[foo, bar] error-in-other-file: bop` and `//@[foo, baz] error-in-other-file boop` would end up
- /// specifying two error patterns that are available in revision `foo`.
- pub fn find_one_for_revision<'a, T: 'a>(
- &'a self,
- revision: &'a str,
- f: impl Fn(&'a Revisioned) -> Option<T>,
- error: impl FnOnce(T),
- ) -> Option<T> {
- let mut rev = self.for_revision(revision).filter_map(f);
- let result = rev.next();
- if let Some(next) = rev.next() {
- error(next);
- }
- result
- }
-
- /// Returns an iterator over all revisioned comments that match the revision.
- pub fn for_revision<'a>(&'a self, revision: &'a str) -> impl Iterator<Item = &'a Revisioned> {
- self.revisioned.iter().filter_map(move |(k, v)| {
- if k.is_empty() || k.iter().any(|rev| rev == revision) {
- Some(v)
- } else {
- None
- }
- })
- }
-
- pub(crate) fn edition(
- &self,
- errors: &mut Vec<Error>,
- revision: &str,
- config: &crate::Config,
- ) -> Option<(String, usize)> {
- self.find_one_for_revision(
- revision,
- |r| r.edition.as_ref(),
- |&(_, line)| {
- errors.push(Error::InvalidComment {
- msg: "`edition` specified twice".into(),
- line,
- })
- },
- )
- .cloned()
- .or(config.edition.clone().map(|e| (e, 0)))
- }
-}
-
-#[derive(Default, Debug)]
-/// Comments that can be filtered for specific revisions.
-pub(crate) struct Revisioned {
- /// The line in which this revisioned item was first added.
- /// Used for reporting errors on unknown revisions.
- pub line: usize,
- /// Don't run this test if any of these filters apply
- pub ignore: Vec<Condition>,
- /// Only run this test if all of these filters apply
- pub only: Vec<Condition>,
- /// Generate one .stderr file per bit width, by prepending with `.64bit` and similar
- pub stderr_per_bitwidth: bool,
- /// Additional flags to pass to the executable
- pub compile_flags: Vec<String>,
- /// Additional env vars to set for the executable
- pub env_vars: Vec<(String, String)>,
- /// Normalizations to apply to the stderr output before emitting it to disk
- pub normalize_stderr: Vec<(Regex, Vec<u8>)>,
- /// Arbitrary patterns to look for in the stderr.
- /// The error must be from another file, as errors from the current file must be
- /// checked via `error_matches`.
- pub error_in_other_files: Vec<(Pattern, usize)>,
- pub error_matches: Vec<ErrorMatch>,
- /// Ignore diagnostics below this level.
- /// `None` means pick the lowest level from the `error_pattern`s.
- pub require_annotations_for_level: Option<Level>,
- pub aux_builds: Vec<(PathBuf, String, usize)>,
- pub edition: Option<(String, usize)>,
- /// Overwrites the mode from `Config`.
- pub mode: Option<(Mode, usize)>,
- pub needs_asm_support: bool,
-}
-
-#[derive(Debug)]
-struct CommentParser<T> {
- /// The comments being built.
- comments: T,
- /// Any errors that ocurred during comment parsing.
- errors: Vec<Error>,
- /// The line currently being parsed.
- line: usize,
-}
-
-impl<T> std::ops::Deref for CommentParser<T> {
- type Target = T;
-
- fn deref(&self) -> &Self::Target {
- &self.comments
- }
-}
-
-impl<T> std::ops::DerefMut for CommentParser<T> {
- fn deref_mut(&mut self) -> &mut Self::Target {
- &mut self.comments
- }
-}
-
-/// The conditions used for "ignore" and "only" filters.
-#[derive(Debug)]
-pub(crate) enum Condition {
- /// The given string must appear in the host triple.
- Host(String),
- /// The given string must appear in the target triple.
- Target(String),
- /// Tests that the bitwidth is the given one.
- Bitwidth(u8),
- /// Tests that the target is the host.
- OnHost,
-}
-
-#[derive(Debug, Clone)]
-/// An error pattern parsed from a `//~` comment.
-pub enum Pattern {
- SubString(String),
- Regex(Regex),
-}
-
-#[derive(Debug)]
-pub(crate) struct ErrorMatch {
- pub pattern: Pattern,
- pub level: Level,
- /// The line where the message was defined, for reporting issues with it (e.g. in case it wasn't found).
- pub definition_line: usize,
- /// The line this pattern is expecting to find a message in.
- pub line: usize,
-}
-
-impl Condition {
- fn parse(c: &str) -> std::result::Result<Self, String> {
- if c == "on-host" {
- Ok(Condition::OnHost)
- } else if let Some(bits) = c.strip_suffix("bit") {
- let bits: u8 = bits.parse().map_err(|_err| {
- format!("invalid ignore/only filter ending in 'bit': {c:?} is not a valid bitwdith")
- })?;
- Ok(Condition::Bitwidth(bits))
- } else if let Some(triple_substr) = c.strip_prefix("target-") {
- Ok(Condition::Target(triple_substr.to_owned()))
- } else if let Some(triple_substr) = c.strip_prefix("host-") {
- Ok(Condition::Host(triple_substr.to_owned()))
- } else {
- Err(format!(
- "`{c}` is not a valid condition, expected `on-host`, /[0-9]+bit/, /host-.*/, or /target-.*/"
- ))
- }
- }
-}
-
-impl Comments {
- pub(crate) fn parse_file(path: &Path) -> Result<std::result::Result<Self, Vec<Error>>> {
- let content =
- std::fs::read(path).wrap_err_with(|| format!("failed to read {}", path.display()))?;
- Ok(Self::parse(&content))
- }
-
- /// Parse comments in `content`.
- /// `path` is only used to emit diagnostics if parsing fails.
- pub(crate) fn parse(
- content: &(impl AsRef<[u8]> + ?Sized),
- ) -> std::result::Result<Self, Vec<Error>> {
- let mut parser = CommentParser {
- comments: Comments::default(),
- errors: vec![],
- line: 0,
- };
-
- let mut fallthrough_to = None; // The line that a `|` will refer to.
- for (l, line) in content.as_ref().lines().enumerate() {
- let l = l + 1; // enumerate starts at 0, but line numbers start at 1
- parser.line = l;
- match parser.parse_checked_line(&mut fallthrough_to, line) {
- Ok(()) => {}
- Err(e) => parser.errors.push(Error::InvalidComment {
- msg: format!("Comment is not utf8: {e:?}"),
- line: l,
- }),
- }
- }
- if let Some(revisions) = &parser.comments.revisions {
- for (key, revisioned) in &parser.comments.revisioned {
- for rev in key {
- if !revisions.contains(rev) {
- parser.errors.push(Error::InvalidComment {
- msg: format!("the revision `{rev}` is not known"),
- line: revisioned.line,
- })
- }
- }
- }
- } else {
- for (key, revisioned) in &parser.comments.revisioned {
- if !key.is_empty() {
- parser.errors.push(Error::InvalidComment {
- msg: "there are no revisions in this test".into(),
- line: revisioned.line,
- })
- }
- }
- }
- if parser.errors.is_empty() {
- Ok(parser.comments)
- } else {
- Err(parser.errors)
- }
- }
-}
-
-impl CommentParser<Comments> {
- fn parse_checked_line(
- &mut self,
- fallthrough_to: &mut Option<usize>,
- line: &[u8],
- ) -> std::result::Result<(), Utf8Error> {
- if let Some(command) = line.strip_prefix(b"//@") {
- self.parse_command(command.trim().to_str()?)
- } else if let Some((_, pattern)) = line.split_once_str("//~") {
- let (revisions, pattern) = self.parse_revisions(pattern.to_str()?);
- self.revisioned(revisions, |this| {
- this.parse_pattern(pattern, fallthrough_to)
- })
- } else {
- *fallthrough_to = None;
- for pos in line.find_iter("//") {
- let rest = &line[pos + 2..];
- for rest in std::iter::once(rest).chain(rest.strip_prefix(b" ")) {
- if let Some('@' | '~' | '[' | ']' | '^' | '|') = rest.chars().next() {
- self.errors.push(Error::InvalidComment {
- msg: format!(
- "comment looks suspiciously like a test suite command: `{}`\n\
- All `//@` test suite commands must be at the start of the line.\n\
- The `//` must be directly followed by `@` or `~`.",
- rest.to_str()?,
- ),
- line: self.line,
- })
- } else {
- let mut parser = Self {
- line: 0,
- errors: vec![],
- comments: Comments::default(),
- };
- parser.parse_command(rest.to_str()?);
- if parser.errors.is_empty() {
- self.errors.push(Error::InvalidComment {
- msg: "a compiletest-rs style comment was detected.\n\
- Please use text that could not also be interpreted as a command,\n\
- and prefix all actual commands with `//@`"
- .into(),
- line: self.line,
- });
- }
- }
- }
- }
- }
- Ok(())
- }
-}
-
-impl<CommentsType> CommentParser<CommentsType> {
- fn error(&mut self, s: impl Into<String>) {
- self.errors.push(Error::InvalidComment {
- msg: s.into(),
- line: self.line,
- });
- }
-
- fn check(&mut self, cond: bool, s: impl Into<String>) {
- if !cond {
- self.error(s);
- }
- }
-
- fn check_some<T>(&mut self, opt: Option<T>, s: impl Into<String>) -> Option<T> {
- self.check(opt.is_some(), s);
- opt
- }
-}
-
-impl CommentParser<Comments> {
- fn parse_command(&mut self, command: &str) {
- let (revisions, command) = self.parse_revisions(command);
-
- // Commands are letters or dashes, grab everything until the first character that is neither of those.
- let (command, args) = match command
- .char_indices()
- .find_map(|(i, c)| (!c.is_alphanumeric() && c != '-' && c != '_').then_some(i))
- {
- None => (command, ""),
- Some(i) => {
- let (command, args) = command.split_at(i);
- let mut args = args.chars();
- // Commands are separated from their arguments by ':' or ' '
- let next = args
- .next()
- .expect("the `position` above guarantees that there is at least one char");
- self.check(
- next == ':',
- "test command must be followed by `:` (or end the line)",
- );
- (command, args.as_str().trim())
- }
- };
-
- if command == "revisions" {
- self.check(
- revisions.is_empty(),
- "revisions cannot be declared under a revision",
- );
- self.check(self.revisions.is_none(), "cannot specify `revisions` twice");
- self.revisions = Some(args.split_whitespace().map(|s| s.to_string()).collect());
- return;
- }
- self.revisioned(revisions, |this| this.parse_command(command, args));
- }
-
- fn revisioned(
- &mut self,
- revisions: Vec<String>,
- f: impl FnOnce(&mut CommentParser<&mut Revisioned>),
- ) {
- let line = self.line;
- let mut this = CommentParser {
- errors: std::mem::take(&mut self.errors),
- line,
- comments: self
- .revisioned
- .entry(revisions)
- .or_insert_with(|| Revisioned {
- line,
- ..Default::default()
- }),
- };
- f(&mut this);
- self.errors = this.errors;
- }
-}
-
-impl CommentParser<&mut Revisioned> {
- fn parse_command(&mut self, command: &str, args: &str) {
- match command {
- "compile-flags" => {
- self.compile_flags
- .extend(args.split_whitespace().map(|s| s.to_string()));
- }
- "rustc-env" => {
- for env in args.split_whitespace() {
- if let Some((k, v)) = self.check_some(
- env.split_once('='),
- "environment variables must be key/value pairs separated by a `=`",
- ) {
- self.env_vars.push((k.to_string(), v.to_string()));
- }
- }
- }
- "normalize-stderr-test" => {
- let (from, rest) = self.parse_str(args);
-
- let to = match rest.strip_prefix("->") {
- Some(v) => v,
- None => {
- self.error("normalize-stderr-test needs a pattern and replacement separated by `->`");
- return;
- },
- }.trim_start();
- let (to, rest) = self.parse_str(to);
-
- self.check(
- rest.is_empty(),
- format!("trailing text after pattern replacement: {rest}"),
- );
-
- if let Some(regex) = self.parse_regex(from) {
- self.normalize_stderr
- .push((regex, to.as_bytes().to_owned()))
- }
- }
- "error-pattern" => {
- self.error("`error-pattern` has been renamed to `error-in-other-file`");
- }
- "error-in-other-file" => {
- let pat = self.parse_error_pattern(args.trim());
- let line = self.line;
- self.error_in_other_files.push((pat, line));
- }
- "stderr-per-bitwidth" => {
- // args are ignored (can be used as comment)
- self.check(
- !self.stderr_per_bitwidth,
- "cannot specify `stderr-per-bitwidth` twice",
- );
- self.stderr_per_bitwidth = true;
- }
- "run-rustfix" => {
- // args are ignored (can be used as comment)
- self.check(
- self.mode.is_none(),
- "cannot specify test mode changes twice",
- );
- self.mode = Some((Mode::Fix, self.line))
- }
- "needs-asm-support" => {
- // args are ignored (can be used as comment)
- self.check(
- !self.needs_asm_support,
- "cannot specify `needs-asm-support` twice",
- );
- self.needs_asm_support = true;
- }
- "aux-build" => {
- let (name, kind) = args.split_once(':').unwrap_or((args, "lib"));
- let line = self.line;
- self.aux_builds.push((name.into(), kind.into(), line));
- }
- "edition" => {
- self.check(self.edition.is_none(), "cannot specify `edition` twice");
- self.edition = Some((args.into(), self.line))
- }
- "check-pass" => {
- // args are ignored (can be used as comment)
- self.check(
- self.mode.is_none(),
- "cannot specify test mode changes twice",
- );
- self.mode = Some((Mode::Pass, self.line))
- }
- "run" => {
- self.check(
- self.mode.is_none(),
- "cannot specify test mode changes twice",
- );
- let mut set = |exit_code| self.mode = Some((Mode::Run { exit_code }, self.line));
- if args.is_empty() {
- set(0);
- } else {
- match args.parse() {
- Ok(exit_code) => set(exit_code),
- Err(err) => self.error(err.to_string()),
- }
- }
- }
- "require-annotations-for-level" => {
- self.check(
- self.require_annotations_for_level.is_none(),
- "cannot specify `require-annotations-for-level` twice",
- );
- match args.trim().parse() {
- Ok(it) => self.require_annotations_for_level = Some(it),
- Err(msg) => self.error(msg),
- }
- }
- command => {
- if let Some(s) = command.strip_prefix("ignore-") {
- // args are ignored (can be used as comment)
- match Condition::parse(s) {
- Ok(cond) => self.ignore.push(cond),
- Err(msg) => self.error(msg),
- }
- return;
- }
-
- if let Some(s) = command.strip_prefix("only-") {
- // args are ignored (can be used as comment)
- match Condition::parse(s) {
- Ok(cond) => self.only.push(cond),
- Err(msg) => self.error(msg),
- }
- return;
- }
- self.error(format!("unknown command `{command}`"));
- }
- }
- }
-}
-
-impl<CommentsType> CommentParser<CommentsType> {
- fn parse_regex(&mut self, regex: &str) -> Option<Regex> {
- match Regex::new(regex) {
- Ok(regex) => Some(regex),
- Err(err) => {
- self.error(format!("invalid regex: {err:?}"));
- None
- }
- }
- }
-
- /// Parses a string literal. `s` has to start with `"`; everything until the next `"` is
- /// returned in the first component. `\` can be used to escape arbitrary character.
- /// Second return component is the rest of the string with leading whitespace removed.
- fn parse_str<'a>(&mut self, s: &'a str) -> (&'a str, &'a str) {
- let mut chars = s.char_indices();
- match chars.next() {
- Some((_, '"')) => {
- let s = chars.as_str();
- let mut escaped = false;
- for (i, c) in chars {
- if escaped {
- // Accept any character as literal after a `\`.
- escaped = false;
- } else if c == '"' {
- return (&s[..(i - 1)], s[i..].trim_start());
- } else {
- escaped = c == '\\';
- }
- }
- self.error(format!("no closing quotes found for {s}"));
- (s, "")
- }
- Some((_, c)) => {
- self.error(format!("expected `\"`, got `{c}`"));
- (s, "")
- }
- None => {
- self.error("expected quoted string, but found end of line");
- (s, "")
- }
- }
- }
-
- // parse something like \[[a-z]+(,[a-z]+)*\]
- fn parse_revisions<'a>(&mut self, pattern: &'a str) -> (Vec<String>, &'a str) {
- match pattern.chars().next() {
- Some('[') => {
- // revisions
- let s = &pattern[1..];
- let end = s.char_indices().find_map(|(i, c)| match c {
- ']' => Some(i),
- _ => None,
- });
- let Some(end) = end else {
- self.error("`[` without corresponding `]`");
- return (vec![], pattern);
- };
- let (revision, pattern) = s.split_at(end);
- (
- revision.split(',').map(|s| s.trim().to_string()).collect(),
- // 1.. because `split_at` includes the separator
- pattern[1..].trim_start(),
- )
- }
- _ => (vec![], pattern),
- }
- }
-}
-
-impl CommentParser<&mut Revisioned> {
- // parse something like (\[[a-z]+(,[a-z]+)*\])?(?P<offset>\||[\^]+)? *(?P<level>ERROR|HELP|WARN|NOTE): (?P<text>.*)
- fn parse_pattern(&mut self, pattern: &str, fallthrough_to: &mut Option<usize>) {
- let (match_line, pattern) = match pattern.chars().next() {
- Some('|') => (
- match fallthrough_to {
- Some(fallthrough) => *fallthrough,
- None => {
- self.error("`//~|` pattern without preceding line");
- return;
- }
- },
- &pattern[1..],
- ),
- Some('^') => {
- let offset = pattern.chars().take_while(|&c| c == '^').count();
- (self.line - offset, &pattern[offset..])
- }
- Some(_) => (self.line, pattern),
- None => {
- self.error("no pattern specified");
- return;
- }
- };
-
- let pattern = pattern.trim_start();
- let offset = match pattern.chars().position(|c| !c.is_ascii_alphabetic()) {
- Some(offset) => offset,
- None => {
- self.error("pattern without level");
- return;
- }
- };
-
- let level = match pattern[..offset].parse() {
- Ok(level) => level,
- Err(msg) => {
- self.error(msg);
- return;
- }
- };
- let pattern = &pattern[offset..];
- let pattern = match pattern.strip_prefix(':') {
- Some(offset) => offset,
- None => {
- self.error("no `:` after level found");
- return;
- }
- };
-
- let pattern = pattern.trim();
-
- self.check(!pattern.is_empty(), "no pattern specified");
-
- let pattern = self.parse_error_pattern(pattern);
-
- *fallthrough_to = Some(match_line);
-
- let definition_line = self.line;
- self.error_matches.push(ErrorMatch {
- pattern,
- level,
- definition_line,
- line: match_line,
- });
- }
-}
-
-impl Pattern {
- pub(crate) fn matches(&self, message: &str) -> bool {
- match self {
- Pattern::SubString(s) => message.contains(s),
- Pattern::Regex(r) => r.is_match(message.as_bytes()),
- }
- }
-}
-
-impl<CommentsType> CommentParser<CommentsType> {
- fn parse_error_pattern(&mut self, pattern: &str) -> Pattern {
- if let Some(regex) = pattern.strip_prefix('/') {
- match regex.strip_suffix('/') {
- Some(regex) => match self.parse_regex(regex) {
- Some(regex) => Pattern::Regex(regex),
- None => Pattern::SubString(pattern.to_string()),
- },
- None => {
- self.error(
- "expected regex pattern due to leading `/`, but found no closing `/`",
- );
- Pattern::SubString(pattern.to_string())
- }
- }
- } else {
- Pattern::SubString(pattern.to_string())
- }
- }
-}
diff --git a/vendor/ui_test/src/parser/tests.rs b/vendor/ui_test/src/parser/tests.rs
deleted file mode 100644
index 1263d28ce..000000000
--- a/vendor/ui_test/src/parser/tests.rs
+++ /dev/null
@@ -1,137 +0,0 @@
-use crate::{
- parser::{Condition, Pattern},
- Error,
-};
-
-use super::Comments;
-
-#[test]
-fn parse_simple_comment() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address $HEX is unallocated)
-}
- ";
- let comments = Comments::parse(s).unwrap();
- println!("parsed comments: {:#?}", comments);
- assert_eq!(comments.revisioned.len(), 1);
- let revisioned = &comments.revisioned[&vec![]];
- assert_eq!(revisioned.error_matches[0].definition_line, 5);
- match &revisioned.error_matches[0].pattern {
- Pattern::SubString(s) => {
- assert_eq!(
- s,
- "encountered a dangling reference (address $HEX is unallocated)"
- )
- }
- other => panic!("expected substring, got {other:?}"),
- }
-}
-
-#[test]
-fn parse_missing_level() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ encountered a dangling reference (address $HEX is unallocated)
-}
- ";
- let errors = Comments::parse(s).unwrap_err();
- println!("parsed comments: {:#?}", errors);
- assert_eq!(errors.len(), 1);
- match &errors[0] {
- Error::InvalidComment { msg, line: 5 } => assert_eq!(msg, "unknown level `encountered`"),
- _ => unreachable!(),
- }
-}
-
-#[test]
-fn parse_slash_slash_at() {
- let s = r"
-//@ error-in-other-file: foomp
-use std::mem;
-
- ";
- let comments = Comments::parse(s).unwrap();
- println!("parsed comments: {:#?}", comments);
- assert_eq!(comments.revisioned.len(), 1);
- let revisioned = &comments.revisioned[&vec![]];
- let pat = &revisioned.error_in_other_files[0];
- assert_eq!(format!("{:?}", pat.0), r#"SubString("foomp")"#);
- assert_eq!(pat.1, 2);
-}
-
-#[test]
-fn parse_regex_error_pattern() {
- let s = r"
-//@ error-in-other-file: /foomp/
-use std::mem;
-
- ";
- let comments = Comments::parse(s).unwrap();
- println!("parsed comments: {:#?}", comments);
- assert_eq!(comments.revisioned.len(), 1);
- let revisioned = &comments.revisioned[&vec![]];
- let pat = &revisioned.error_in_other_files[0];
- assert_eq!(format!("{:?}", pat.0), r#"Regex(foomp)"#);
- assert_eq!(pat.1, 2);
-}
-
-#[test]
-fn parse_slash_slash_at_fail() {
- let s = r"
-//@ error-patttern foomp
-use std::mem;
-
- ";
- let errors = Comments::parse(s).unwrap_err();
- println!("parsed comments: {:#?}", errors);
- assert_eq!(errors.len(), 2);
- match &errors[0] {
- Error::InvalidComment { msg, line: 2 } => {
- assert!(msg.contains("must be followed by `:`"))
- }
- _ => unreachable!(),
- }
- match &errors[1] {
- Error::InvalidComment { msg, line: 2 } => {
- assert_eq!(msg, "unknown command `error-patttern`");
- }
- _ => unreachable!(),
- }
-}
-
-#[test]
-fn missing_colon_fail() {
- let s = r"
-//@stderr-per-bitwidth hello
-use std::mem;
-
- ";
- let errors = Comments::parse(s).unwrap_err();
- println!("parsed comments: {:#?}", errors);
- assert_eq!(errors.len(), 1);
- match &errors[0] {
- Error::InvalidComment { msg, line: 2 } => {
- assert!(msg.contains("must be followed by `:`"))
- }
- _ => unreachable!(),
- }
-}
-
-#[test]
-fn parse_x86_64() {
- let s = r"//@ only-target-x86_64-unknown-linux";
- let comments = Comments::parse(s).unwrap();
- println!("parsed comments: {:#?}", comments);
- assert_eq!(comments.revisioned.len(), 1);
- let revisioned = &comments.revisioned[&vec![]];
- assert_eq!(revisioned.only.len(), 1);
- match &revisioned.only[0] {
- Condition::Target(t) => assert_eq!(t, "x86_64-unknown-linux"),
- _ => unreachable!(),
- }
-}
diff --git a/vendor/ui_test/src/rustc_stderr.rs b/vendor/ui_test/src/rustc_stderr.rs
deleted file mode 100644
index 64e5928c5..000000000
--- a/vendor/ui_test/src/rustc_stderr.rs
+++ /dev/null
@@ -1,160 +0,0 @@
-use std::path::{Path, PathBuf};
-
-use bstr::ByteSlice;
-use regex::Regex;
-
-#[derive(serde::Deserialize, Debug)]
-struct RustcMessage {
- rendered: Option<String>,
- spans: Vec<Span>,
- level: String,
- message: String,
- children: Vec<RustcMessage>,
-}
-
-#[derive(Copy, Clone, Debug, PartialOrd, Ord, PartialEq, Eq)]
-pub(crate) enum Level {
- Ice = 5,
- Error = 4,
- Warn = 3,
- Help = 2,
- Note = 1,
- /// Only used for "For more information about this error, try `rustc --explain EXXXX`".
- FailureNote = 0,
-}
-
-#[derive(Debug)]
-/// A diagnostic message.
-pub struct Message {
- pub(crate) level: Level,
- pub(crate) message: String,
-}
-
-/// Information about macro expansion.
-#[derive(serde::Deserialize, Debug)]
-struct Expansion {
- span: Span,
-}
-
-#[derive(serde::Deserialize, Debug)]
-struct Span {
- line_start: usize,
- file_name: PathBuf,
- is_primary: bool,
- expansion: Option<Box<Expansion>>,
-}
-
-impl std::str::FromStr for Level {
- type Err = String;
- fn from_str(s: &str) -> Result<Self, Self::Err> {
- match s {
- "ERROR" | "error" => Ok(Self::Error),
- "WARN" | "warning" => Ok(Self::Warn),
- "HELP" | "help" => Ok(Self::Help),
- "NOTE" | "note" => Ok(Self::Note),
- "failure-note" => Ok(Self::FailureNote),
- "error: internal compiler error" => Ok(Self::Ice),
- _ => Err(format!("unknown level `{s}`")),
- }
- }
-}
-
-#[derive(Debug)]
-pub(crate) struct Diagnostics {
- /// Rendered and concatenated version of all diagnostics.
- /// This is equivalent to non-json diagnostics.
- pub rendered: Vec<u8>,
- /// Per line, a list of messages for that line.
- pub messages: Vec<Vec<Message>>,
- /// Messages not on any line (usually because they are from libstd)
- pub messages_from_unknown_file_or_line: Vec<Message>,
-}
-
-impl RustcMessage {
- fn line(&self, file: &Path) -> Option<usize> {
- let span = |primary| self.spans.iter().find_map(|span| span.line(file, primary));
- span(true).or_else(|| span(false))
- }
-
- /// Put the message and its children into the line-indexed list.
- fn insert_recursive(
- self,
- file: &Path,
- messages: &mut Vec<Vec<Message>>,
- messages_from_unknown_file_or_line: &mut Vec<Message>,
- line: Option<usize>,
- ) {
- let line = self.line(file).or(line);
- let msg = Message {
- level: self.level.parse().unwrap(),
- message: self.message,
- };
- if let Some(line) = line {
- if messages.len() <= line {
- messages.resize_with(line + 1, Vec::new);
- }
- messages[line].push(msg);
- // All other messages go into the general bin, unless they are specifically of the
- // "aborting due to X previous errors" variety, as we never want to match those. They
- // only count the number of errors and provide no useful information about the tests.
- } else if !(msg.message.starts_with("aborting due to")
- && msg.message.contains("previous error"))
- {
- messages_from_unknown_file_or_line.push(msg);
- }
- for child in self.children {
- child.insert_recursive(file, messages, messages_from_unknown_file_or_line, line)
- }
- }
-}
-
-impl Span {
- /// Returns the most expanded line number *in the given file*, if possible.
- fn line(&self, file: &Path, primary: bool) -> Option<usize> {
- if let Some(exp) = &self.expansion {
- if let Some(line) = exp.span.line(file, primary && !self.is_primary) {
- return Some(line);
- }
- }
- ((!primary || self.is_primary) && self.file_name == file).then_some(self.line_start)
- }
-}
-
-pub(crate) fn filter_annotations_from_rendered(rendered: &str) -> std::borrow::Cow<'_, str> {
- let annotations = Regex::new(r" *//(\[[a-z,]+\])?~.*").unwrap();
- annotations.replace_all(rendered, "")
-}
-
-pub(crate) fn process(file: &Path, stderr: &[u8]) -> Diagnostics {
- let mut rendered = Vec::new();
- let mut messages = vec![];
- let mut messages_from_unknown_file_or_line = vec![];
- for (line_number, line) in stderr.lines_with_terminator().enumerate() {
- if line.starts_with_str(b"{") {
- match serde_json::from_slice::<RustcMessage>(line) {
- Ok(msg) => {
- rendered.extend(
- filter_annotations_from_rendered(msg.rendered.as_ref().unwrap()).as_bytes(),
- );
- msg.insert_recursive(
- file,
- &mut messages,
- &mut messages_from_unknown_file_or_line,
- None,
- );
- }
- Err(err) => {
- panic!("failed to parse rustc JSON output at line {line_number}: {err}")
- }
- }
- } else {
- // FIXME: do we want to throw interpreter stderr into a separate file?
- rendered.extend(line);
- }
- }
- Diagnostics {
- rendered,
- messages,
- messages_from_unknown_file_or_line,
- }
-}
diff --git a/vendor/ui_test/src/status_emitter.rs b/vendor/ui_test/src/status_emitter.rs
deleted file mode 100644
index a277a9808..000000000
--- a/vendor/ui_test/src/status_emitter.rs
+++ /dev/null
@@ -1,577 +0,0 @@
-//! Variaous schemes for reporting messages during testing or after testing is done.
-
-use bstr::ByteSlice;
-use colored::Colorize;
-
-use crate::{github_actions, parser::Pattern, rustc_stderr::Message, Error, Errors, TestResult};
-use std::{
- fmt::{Debug, Write as _},
- io::Write as _,
- path::Path,
- process::Command,
-};
-
-/// A generic way to handle the output of this crate.
-pub trait StatusEmitter: Sync {
- /// Invoked before each failed test prints its errors along with a drop guard that can
- /// gets invoked afterwards.
- fn failed_test<'a>(
- &'a self,
- revision: &'a str,
- path: &'a Path,
- cmd: &'a Command,
- stderr: &'a [u8],
- ) -> Box<dyn Debug + 'a>;
-
- /// A test has finished, handle the result immediately.
- fn test_result(&mut self, _path: &Path, _revision: &str, _result: &TestResult) {}
-
- /// Create a report about the entire test run at the end.
- #[allow(clippy::type_complexity)]
- fn finalize(
- &self,
- failed: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary>;
-}
-
-/// Report a summary at the end of a test run.
-pub trait Summary {
- /// A test has finished, handle the result.
- fn test_failure(&mut self, _path: &Path, _revision: &str, _errors: &Errors) {}
-}
-
-impl Summary for () {}
-
-/// A human readable output emitter.
-pub struct Text;
-impl StatusEmitter for Text {
- fn failed_test<'a>(
- &self,
- revision: &str,
- path: &Path,
- cmd: &Command,
- stderr: &'a [u8],
- ) -> Box<dyn Debug + 'a> {
- eprintln!();
- let path = path.display().to_string();
- eprint!("{}", path.underline().bold());
- let revision = if revision.is_empty() {
- String::new()
- } else {
- format!(" (revision `{revision}`)")
- };
- eprint!("{revision}");
- eprint!(" {}", "FAILED:".red().bold());
- eprintln!();
- eprintln!("command: {cmd:?}");
- eprintln!();
-
- #[derive(Debug)]
- struct Guard<'a>(&'a [u8]);
- impl<'a> Drop for Guard<'a> {
- fn drop(&mut self) {
- eprintln!("full stderr:");
- std::io::stderr().write_all(self.0).unwrap();
- eprintln!();
- eprintln!();
- }
- }
- Box::new(Guard(stderr))
- }
-
- fn test_result(&mut self, path: &Path, revision: &str, result: &TestResult) {
- let result = match result {
- TestResult::Ok => "ok".green(),
- TestResult::Errored { .. } => "FAILED".red().bold(),
- TestResult::Ignored => "ignored (in-test comment)".yellow(),
- TestResult::Filtered => return,
- };
- eprint!(
- "{}{} ... ",
- path.display(),
- if revision.is_empty() {
- "".into()
- } else {
- format!(" ({revision})")
- }
- );
- eprintln!("{result}");
- }
-
- fn finalize(
- &self,
- failures: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary> {
- // Print all errors in a single thread to show reliable output
- if failures == 0 {
- eprintln!();
- eprintln!(
- "test result: {}. {} tests passed, {} ignored, {} filtered out",
- "ok".green(),
- succeeded.to_string().green(),
- ignored.to_string().yellow(),
- filtered.to_string().yellow(),
- );
- eprintln!();
- Box::new(())
- } else {
- struct Summarizer {
- failures: Vec<String>,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- }
-
- impl Summary for Summarizer {
- fn test_failure(&mut self, path: &Path, revision: &str, errors: &Errors) {
- for error in errors {
- print_error(error, &path.display().to_string());
- }
-
- self.failures.push(if revision.is_empty() {
- format!(" {}", path.display())
- } else {
- format!(" {} (revision {revision})", path.display())
- });
- }
- }
-
- impl Drop for Summarizer {
- fn drop(&mut self) {
- eprintln!("{}", "FAILURES:".red().underline().bold());
- for line in &self.failures {
- eprintln!("{line}");
- }
- eprintln!();
- eprintln!(
- "test result: {}. {} tests failed, {} tests passed, {} ignored, {} filtered out",
- "FAIL".red(),
- self.failures.len().to_string().red().bold(),
- self.succeeded.to_string().green(),
- self.ignored.to_string().yellow(),
- self.filtered.to_string().yellow(),
- );
- }
- }
- Box::new(Summarizer {
- failures: vec![],
- succeeded,
- ignored,
- filtered,
- })
- }
- }
-}
-
-fn print_error(error: &Error, path: &str) {
- match error {
- Error::ExitStatus {
- mode,
- status,
- expected,
- } => {
- eprintln!("{mode} test got {status}, but expected {expected}")
- }
- Error::Command { kind, status } => {
- eprintln!("{kind} failed with {status}");
- }
- Error::PatternNotFound {
- pattern,
- definition_line,
- } => {
- match pattern {
- Pattern::SubString(s) => {
- eprintln!("substring `{s}` {} in stderr output", "not found".red())
- }
- Pattern::Regex(r) => {
- eprintln!("`/{r}/` does {} stderr output", "not match".red())
- }
- }
- eprintln!(
- "expected because of pattern here: {}",
- format!("{path}:{definition_line}").bold()
- );
- }
- Error::NoPatternsFound => {
- eprintln!("{}", "no error patterns found in fail test".red());
- }
- Error::PatternFoundInPassTest => {
- eprintln!("{}", "error pattern found in pass test".red())
- }
- Error::OutputDiffers {
- path: output_path,
- actual,
- expected,
- bless_command,
- } => {
- eprintln!("{}", "actual output differed from expected".underline());
- eprintln!(
- "Execute `{}` to update `{}` to the actual output",
- bless_command,
- output_path.display()
- );
- eprintln!("{}", format!("--- {}", output_path.display()).red());
- eprintln!("{}", "+++ <stderr output>".green());
- crate::diff::print_diff(expected, actual);
- }
- Error::ErrorsWithoutPattern { path: None, msgs } => {
- eprintln!(
- "There were {} unmatched diagnostics that occurred outside the testfile and had no pattern",
- msgs.len(),
- );
- for Message { level, message } in msgs {
- eprintln!(" {level:?}: {message}")
- }
- }
- Error::ErrorsWithoutPattern {
- path: Some((path, line)),
- msgs,
- } => {
- let path = path.display();
- eprintln!(
- "There were {} unmatched diagnostics at {path}:{line}",
- msgs.len(),
- );
- for Message { level, message } in msgs {
- eprintln!(" {level:?}: {message}")
- }
- }
- Error::InvalidComment { msg, line } => {
- eprintln!("Could not parse comment in {path}:{line} because\n{msg}",)
- }
- Error::Bug(msg) => {
- eprintln!("A bug in `ui_test` occurred: {msg}");
- }
- Error::Aux {
- path: aux_path,
- errors,
- line,
- } => {
- eprintln!("Aux build from {path}:{line} failed");
- for error in errors {
- print_error(error, &aux_path.display().to_string());
- }
- }
- }
- eprintln!();
-}
-
-fn gha_error(error: &Error, path: &str, revision: &str) {
- match error {
- Error::ExitStatus {
- mode,
- status,
- expected,
- } => {
- github_actions::error(
- path,
- format!("{mode} test{revision} got {status}, but expected {expected}"),
- );
- }
- Error::Command { kind, status } => {
- github_actions::error(path, format!("{kind}{revision} failed with {status}"));
- }
- Error::PatternNotFound {
- pattern: _,
- definition_line,
- } => {
- github_actions::error(path, format!("Pattern not found{revision}"))
- .line(*definition_line);
- }
- Error::NoPatternsFound => {
- github_actions::error(
- path,
- format!("no error patterns found in fail test{revision}"),
- );
- }
- Error::PatternFoundInPassTest => {
- github_actions::error(path, format!("error pattern found in pass test{revision}"));
- }
- Error::OutputDiffers {
- path: output_path,
- actual,
- expected,
- bless_command: _,
- } => {
- let mut err = github_actions::error(
- if expected.is_empty() {
- path.to_owned()
- } else {
- output_path.display().to_string()
- },
- "actual output differs from expected",
- );
- writeln!(err, "```diff").unwrap();
- let mut seen_diff_line = Some(0);
- for r in ::diff::lines(expected.to_str().unwrap(), actual.to_str().unwrap()) {
- if let Some(line) = &mut seen_diff_line {
- *line += 1;
- }
- let mut seen_diff = || {
- if let Some(line) = seen_diff_line.take() {
- writeln!(err, "{line} unchanged lines skipped").unwrap();
- }
- };
- match r {
- ::diff::Result::Both(l, r) => {
- if l != r {
- seen_diff();
- writeln!(err, "-{l}").unwrap();
- writeln!(err, "+{r}").unwrap();
- } else if seen_diff_line.is_none() {
- writeln!(err, " {l}").unwrap()
- }
- }
- ::diff::Result::Left(l) => {
- seen_diff();
- writeln!(err, "-{l}").unwrap();
- }
- ::diff::Result::Right(r) => {
- seen_diff();
- writeln!(err, "+{r}").unwrap();
- }
- }
- }
- writeln!(err, "```").unwrap();
- }
- Error::ErrorsWithoutPattern { path: None, msgs } => {
- let mut err = github_actions::error(
- path,
- format!("Unmatched diagnostics outside the testfile{revision}"),
- );
- for Message { level, message } in msgs {
- writeln!(err, "{level:?}: {message}").unwrap();
- }
- }
- Error::ErrorsWithoutPattern {
- path: Some((path, line)),
- msgs,
- } => {
- let path = path.display();
- let mut err = github_actions::error(&path, format!("Unmatched diagnostics{revision}"))
- .line(*line);
- for Message { level, message } in msgs {
- writeln!(err, "{level:?}: {message}").unwrap();
- }
- }
- Error::InvalidComment { msg, line } => {
- let mut err =
- github_actions::error(path, format!("Could not parse comment")).line(*line);
- writeln!(err, "{msg}").unwrap();
- }
- Error::Bug(_) => {}
- Error::Aux {
- path: aux_path,
- errors,
- line,
- } => {
- github_actions::error(path, format!("Aux build failed")).line(*line);
- for error in errors {
- gha_error(error, &aux_path.display().to_string(), "")
- }
- }
- }
- eprintln!();
-}
-
-/// Just print some dots instead of a whole line per run test.
-#[derive(Default)]
-pub struct Quiet {
- n: usize,
-}
-
-impl StatusEmitter for Quiet {
- fn test_result(&mut self, _path: &Path, _revision: &str, result: &TestResult) {
- // Humans start counting at 1
- self.n += 1;
- match result {
- TestResult::Ok => eprint!("{}", ".".green()),
- TestResult::Errored { .. } => eprint!("{}", "F".red().bold()),
- TestResult::Ignored => eprint!("{}", "i".yellow()),
- TestResult::Filtered => {}
- }
- if self.n % 100 == 0 {
- eprintln!(" {}", self.n);
- }
- }
-
- fn failed_test<'a>(
- &'a self,
- revision: &'a str,
- path: &'a Path,
- cmd: &'a Command,
- stderr: &'a [u8],
- ) -> Box<dyn Debug + 'a> {
- Text.failed_test(revision, path, cmd, stderr)
- }
-
- fn finalize(
- &self,
- failed: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary> {
- Text.finalize(failed, succeeded, ignored, filtered)
- }
-}
-
-/// Emits Github Actions Workspace commands to show the failures directly in the github diff view.
-/// If the const generic `GROUP` boolean is `true`, also emit `::group` commands.
-pub struct Gha<const GROUP: bool> {
- /// Show a specific name for the final summary.
- pub name: String,
-}
-
-impl<const GROUP: bool> StatusEmitter for Gha<GROUP> {
- fn failed_test(
- &self,
- revision: &str,
- path: &Path,
- _cmd: &Command,
- _stderr: &[u8],
- ) -> Box<dyn Debug> {
- if GROUP {
- Box::new(github_actions::group(format_args!(
- "{}:{revision}",
- path.display()
- )))
- } else {
- Box::new(())
- }
- }
-
- fn test_result(&mut self, _path: &Path, _revision: &str, _result: &TestResult) {}
-
- fn finalize(
- &self,
- _failures: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary> {
- struct Summarizer<const GROUP: bool> {
- failures: Vec<String>,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- name: String,
- }
-
- impl<const GROUP: bool> Summary for Summarizer<GROUP> {
- fn test_failure(&mut self, path: &Path, revision: &str, errors: &Errors) {
- let revision = if revision.is_empty() {
- "".to_string()
- } else {
- format!(" (revision: {revision})")
- };
- for error in errors {
- gha_error(error, &path.display().to_string(), &revision);
- }
- self.failures.push(format!("{}{revision}", path.display()));
- }
- }
- impl<const GROUP: bool> Drop for Summarizer<GROUP> {
- fn drop(&mut self) {
- if let Some(mut file) = github_actions::summary() {
- writeln!(file, "### {}", self.name).unwrap();
- for line in &self.failures {
- writeln!(file, "* {line}").unwrap();
- }
- writeln!(file).unwrap();
- writeln!(file, "| failed | passed | ignored | filtered out |").unwrap();
- writeln!(file, "| --- | --- | --- | --- |").unwrap();
- writeln!(
- file,
- "| {} | {} | {} | {} |",
- self.failures.len(),
- self.succeeded,
- self.ignored,
- self.filtered,
- )
- .unwrap();
- }
- }
- }
-
- Box::new(Summarizer::<GROUP> {
- failures: vec![],
- succeeded,
- ignored,
- filtered,
- name: self.name.clone(),
- })
- }
-}
-
-impl<T: StatusEmitter, U: StatusEmitter> StatusEmitter for (T, U) {
- fn failed_test<'a>(
- &'a self,
- revision: &'a str,
- path: &'a Path,
- cmd: &'a Command,
- stderr: &'a [u8],
- ) -> Box<dyn Debug + 'a> {
- Box::new((
- self.0.failed_test(revision, path, cmd, stderr),
- self.1.failed_test(revision, path, cmd, stderr),
- ))
- }
-
- fn test_result(&mut self, path: &Path, revision: &str, result: &TestResult) {
- self.0.test_result(path, revision, result);
- self.1.test_result(path, revision, result);
- }
-
- fn finalize(
- &self,
- failures: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary> {
- Box::new((
- self.1.finalize(failures, succeeded, ignored, filtered),
- self.0.finalize(failures, succeeded, ignored, filtered),
- ))
- }
-}
-
-impl<T: StatusEmitter + ?Sized> StatusEmitter for Box<T> {
- fn failed_test<'a>(
- &'a self,
- revision: &'a str,
- path: &'a Path,
- cmd: &'a Command,
- stderr: &'a [u8],
- ) -> Box<dyn Debug + 'a> {
- (**self).failed_test(revision, path, cmd, stderr)
- }
-
- fn test_result(&mut self, path: &Path, revision: &str, result: &TestResult) {
- (**self).test_result(path, revision, result);
- }
-
- fn finalize(
- &self,
- failures: usize,
- succeeded: usize,
- ignored: usize,
- filtered: usize,
- ) -> Box<dyn Summary> {
- (**self).finalize(failures, succeeded, ignored, filtered)
- }
-}
-
-impl Summary for (Box<dyn Summary>, Box<dyn Summary>) {
- fn test_failure(&mut self, path: &Path, revision: &str, errors: &Errors) {
- self.0.test_failure(path, revision, errors);
- self.1.test_failure(path, revision, errors);
- }
-}
diff --git a/vendor/ui_test/src/tests.rs b/vendor/ui_test/src/tests.rs
deleted file mode 100644
index 4aca1d2d0..000000000
--- a/vendor/ui_test/src/tests.rs
+++ /dev/null
@@ -1,338 +0,0 @@
-use std::path::{Path, PathBuf};
-
-use crate::rustc_stderr::Level;
-use crate::rustc_stderr::Message;
-
-use super::*;
-
-fn config() -> Config {
- Config {
- root_dir: PathBuf::from("$RUSTROOT"),
- program: CommandBuilder::cmd("cake"),
- ..Config::rustc(PathBuf::new())
- }
-}
-
-#[test]
-fn issue_2156() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address $HEX is unallocated)
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let mut errors = vec![];
- let config = config();
- let messages = vec![
- vec![], vec![], vec![], vec![], vec![],
- vec![
- Message {
- message:"Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- }
- ]
- ];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [Error::PatternNotFound {
- definition_line: 5, ..
- }, Error::ErrorsWithoutPattern {
- path: Some((_, 5)), ..
- }] => {}
- _ => panic!("{:#?}", errors),
- }
-}
-
-#[test]
-fn find_pattern() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address 0x10 is unallocated)
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let config = config();
- {
- let messages = vec![vec![], vec![], vec![], vec![], vec![], vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- }
- ]
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [] => {}
- _ => panic!("{:#?}", errors),
- }
- }
-
- // only difference to above is a wrong line number
- {
- let messages = vec![vec![], vec![], vec![], vec![], vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- }
- ]
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [Error::PatternNotFound {
- definition_line: 5, ..
- }, Error::ErrorsWithoutPattern {
- path: Some((_, 4)), ..
- }] => {}
- _ => panic!("not the expected error: {:#?}", errors),
- }
- }
-
- // only difference to first is a wrong level
- {
- let messages = vec![
- vec![], vec![], vec![], vec![], vec![],
- vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Note,
- }
- ]
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- // Note no `ErrorsWithoutPattern`, because there are no `//~NOTE` in the test file, so we ignore them
- [Error::PatternNotFound {
- definition_line: 5, ..
- }] => {}
- _ => panic!("not the expected error: {:#?}", errors),
- }
- }
-}
-
-#[test]
-fn duplicate_pattern() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address 0x10 is unallocated)
- //~^ ERROR: encountered a dangling reference (address 0x10 is unallocated)
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let config = config();
- let messages = vec![
- vec![], vec![], vec![], vec![], vec![],
- vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- }
- ]
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [Error::PatternNotFound {
- definition_line: 6, ..
- }] => {}
- _ => panic!("{:#?}", errors),
- }
-}
-
-#[test]
-fn missing_pattern() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address 0x10 is unallocated)
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let config = config();
- let messages = vec![
- vec![], vec![], vec![], vec![], vec![],
- vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- },
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- }
- ]
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [Error::ErrorsWithoutPattern {
- path: Some((_, 5)), ..
- }] => {}
- _ => panic!("{:#?}", errors),
- }
-}
-
-#[test]
-fn missing_warn_pattern() {
- let s = r"
-use std::mem;
-
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address 0x10 is unallocated)
- //~^ WARN: cake
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let config = config();
- let messages= vec![
- vec![],
- vec![],
- vec![],
- vec![],
- vec![],
- vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- },
- Message {
- message: "kaboom".to_string(),
- level: Level::Warn,
- },
- Message {
- message: "cake".to_string(),
- level: Level::Warn,
- },
- ],
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [Error::ErrorsWithoutPattern {
- path: Some((_, 5)),
- msgs,
- ..
- }] => match &msgs[..] {
- [Message {
- message,
- level: Level::Warn,
- }] if message == "kaboom" => {}
- _ => panic!("{:#?}", msgs),
- },
- _ => panic!("{:#?}", errors),
- }
-}
-
-#[test]
-fn missing_implicit_warn_pattern() {
- let s = r"
-use std::mem;
-//@require-annotations-for-level: ERROR
-fn main() {
- let _x: &i32 = unsafe { mem::transmute(16usize) }; //~ ERROR: encountered a dangling reference (address 0x10 is unallocated)
- //~^ WARN: cake
-}
- ";
- let comments = Comments::parse(s).unwrap();
- let config = config();
- let messages = vec![
- vec![],
- vec![],
- vec![],
- vec![],
- vec![],
- vec![
- Message {
- message: "Undefined Behavior: type validation failed: encountered a dangling reference (address 0x10 is unallocated)".to_string(),
- level: Level::Error,
- },
- Message {
- message: "kaboom".to_string(),
- level: Level::Warn,
- },
- Message {
- message: "cake".to_string(),
- level: Level::Warn,
- },
- ],
- ];
- let mut errors = vec![];
- check_annotations(
- messages,
- vec![],
- Path::new("moobar"),
- &mut errors,
- &config,
- "",
- &comments,
- );
- match &errors[..] {
- [] => {}
- _ => panic!("{:#?}", errors),
- }
-}