From 88f22dcb17c7b772a6229c417c870b2e172ffa52 Mon Sep 17 00:00:00 2001 From: Ross Smyth Date: Mon, 26 Feb 2024 17:19:30 -0500 Subject: [PATCH 1/2] Fix .\miri fmt on Windows --- miri-script/src/commands.rs | 46 ++++++++++++++---------------------- miri-script/src/util.rs | 47 +++++++++++++++++++++++++++++++++++-- 2 files changed, 63 insertions(+), 30 deletions(-) diff --git a/miri-script/src/commands.rs b/miri-script/src/commands.rs index 88ee62fc18..dc6ef58c52 100644 --- a/miri-script/src/commands.rs +++ b/miri-script/src/commands.rs @@ -526,37 +526,27 @@ impl Command { } fn fmt(flags: Vec) -> Result<()> { + use itertools::Itertools; + let e = MiriEnv::new()?; - let toolchain = &e.toolchain; let config_path = path!(e.miri_dir / "rustfmt.toml"); - let mut cmd = cmd!( - e.sh, - "rustfmt +{toolchain} --edition=2021 --config-path {config_path} --unstable-features --skip-children {flags...}" - ); - eprintln!("$ {cmd} ..."); - - // Add all the filenames to the command. - // FIXME: `rustfmt` will follow the `mod` statements in these files, so we get a bunch of - // duplicate diffs. - for item in WalkDir::new(&e.miri_dir).into_iter().filter_entry(|entry| { - let name = entry.file_name().to_string_lossy(); - let ty = entry.file_type(); - if ty.is_file() { - name.ends_with(".rs") - } else { - // dir or symlink. skip `target` and `.git`. - &name != "target" && &name != ".git" - } - }) { - let item = item?; - if item.file_type().is_file() { - cmd = cmd.arg(item.into_path()); - } - } + // Collect each rust file in the miri repo. + let files = WalkDir::new(&e.miri_dir) + .into_iter() + .filter_entry(|entry| { + let name = entry.file_name().to_string_lossy(); + let ty = entry.file_type(); + if ty.is_file() { + name.ends_with(".rs") + } else { + // dir or symlink. skip `target` and `.git`. + &name != "target" && &name != ".git" + } + }) + .filter_ok(|item| item.file_type().is_file()) + .map_ok(|item| item.into_path()); - // We want our own error message, repeating the command is too much. - cmd.quiet().run().map_err(|_| anyhow!("`rustfmt` failed"))?; - Ok(()) + e.format_files(files, &e.toolchain[..], &config_path, &flags[..]) } } diff --git a/miri-script/src/util.rs b/miri-script/src/util.rs index 420ecdab63..5c02024324 100644 --- a/miri-script/src/util.rs +++ b/miri-script/src/util.rs @@ -1,7 +1,7 @@ use std::ffi::{OsStr, OsString}; -use std::path::PathBuf; +use std::path::{Path, PathBuf}; -use anyhow::{Context, Result}; +use anyhow::{anyhow, Context, Result}; use dunce::canonicalize; use path_macro::path; use xshell::{cmd, Shell}; @@ -145,4 +145,47 @@ impl MiriEnv { .run()?; Ok(()) } + + /// Receives an iterator of files. + /// Will format each file with the miri rustfmt config. + /// Does not follow module relationships. + pub fn format_files( + &self, + files: impl Iterator>, + toolchain: &str, + config_path: &Path, + flags: &[OsString], + ) -> anyhow::Result<()> { + use itertools::Itertools; + + let mut first = true; + + // Format in batches as not all out files fit into Windows' command argument limit. + for batch in &files.chunks(256) { + // Build base command. + let mut cmd = cmd!( + self.sh, + "rustfmt +{toolchain} --edition=2021 --config-path {config_path} --unstable-features --skip-children {flags...}" + ); + if first { + eprintln!("$ {cmd} ..."); + first = false; + } + // Add files. + for file in batch { + // Make it a relative path so that on platforms with extremely tight argument + // limits (like Windows), we become immune to someone cloning the repo + // 50 directories deep. + let file = file?; + let file = file.strip_prefix(&self.miri_dir)?; + cmd = cmd.arg(file); + } + + // Run commands. + // We want our own error message, repeating the command is too much. + cmd.quiet().run().map_err(|_| anyhow!("`rustfmt` failed"))?; + } + + Ok(()) + } } From f2ec3214c90f4a84554fffb5e002eff8ee814a6b Mon Sep 17 00:00:00 2001 From: Ralf Jung Date: Sun, 3 Mar 2024 15:00:47 +0100 Subject: [PATCH 2/2] nits and typos --- miri-script/src/util.rs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/miri-script/src/util.rs b/miri-script/src/util.rs index 5c02024324..361a4ca0cf 100644 --- a/miri-script/src/util.rs +++ b/miri-script/src/util.rs @@ -148,7 +148,7 @@ impl MiriEnv { /// Receives an iterator of files. /// Will format each file with the miri rustfmt config. - /// Does not follow module relationships. + /// Does not recursively format modules. pub fn format_files( &self, files: impl Iterator>, @@ -160,7 +160,7 @@ impl MiriEnv { let mut first = true; - // Format in batches as not all out files fit into Windows' command argument limit. + // Format in batches as not all our files fit into Windows' command argument limit. for batch in &files.chunks(256) { // Build base command. let mut cmd = cmd!( @@ -168,6 +168,7 @@ impl MiriEnv { "rustfmt +{toolchain} --edition=2021 --config-path {config_path} --unstable-features --skip-children {flags...}" ); if first { + // Log an abbreviating command, and only once. eprintln!("$ {cmd} ..."); first = false; } @@ -181,7 +182,7 @@ impl MiriEnv { cmd = cmd.arg(file); } - // Run commands. + // Run rustfmt. // We want our own error message, repeating the command is too much. cmd.quiet().run().map_err(|_| anyhow!("`rustfmt` failed"))?; }