Format output using prettyplease

This commit is contained in:
David Tolnay 2022-01-12 13:24:43 -08:00
parent 2a14aaa9d4
commit 3176152239
No known key found for this signature in database
GPG key ID: F9BA143B95FF6D82
4 changed files with 88 additions and 34 deletions

View file

@ -19,6 +19,8 @@ jobs:
components: rustfmt
- run: cargo run -- expand --manifest-path tests/Cargo.toml > expand.rs
- run: diff tests/lib.expand.rs expand.rs
- run: cargo run --features prettyplease -- expand --manifest-path tests/Cargo.toml > expand.rs
- run: diff tests/lib.expand.rs expand.rs
build:
name: Rust ${{matrix.rust}}
@ -27,15 +29,18 @@ jobs:
fail-fast: false
matrix:
# beta broken: https://github.com/rust-lang/rust/issues/91872
#rust: [nightly, beta, stable, 1.54.0]
rust: [nightly, stable, 1.54.0]
#rust: [nightly, beta, stable, 1.56.0]
rust: [nightly, stable, 1.56.0]
steps:
- uses: actions/checkout@v2
- uses: dtolnay/rust-toolchain@master
with:
toolchain: ${{matrix.rust}}
- run: cargo check
- run: cargo update && cargo check
- run: cargo check --features prettyplease
- run: cargo update
- run: cargo check
- run: cargo check --features prettyplease
- run: cargo test
clippy:

11
Cargo.lock generated
View file

@ -133,6 +133,7 @@ dependencies = [
"atty",
"bat",
"clap",
"prettyplease",
"proc-macro2",
"quote",
"serde",
@ -537,6 +538,16 @@ dependencies = [
"xml-rs",
]
[[package]]
name = "prettyplease"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3f3ebd569945e276e2c8000677484f50f5c242b3495929ae1c7d42d89778939a"
dependencies = [
"proc-macro2",
"syn",
]
[[package]]
name = "proc-macro-error"
version = "1.0.4"

View file

@ -15,6 +15,7 @@ autotests = false
[dependencies]
atty = "0.2"
clap = { version = "3.0", default-features = false, features = ["derive", "std", "suggestions"] }
prettyplease = { version = "0.1", optional = true }
proc-macro2 = "1.0"
quote = { version = "1.0", default-features = false }
serde = { version = "1.0", features = ["derive"] }

View file

@ -1,5 +1,6 @@
#![allow(
clippy::enum_glob_use,
clippy::items_after_statements,
clippy::let_underscore_drop,
clippy::manual_strip,
clippy::match_like_matches_macro,
@ -126,9 +127,9 @@ fn cargo_expand() -> Result<i32> {
return Ok(0);
}
let rustfmt;
match (&args.item, args.ugly) {
(Some(item), true) => {
let mut rustfmt = None;
if let Some(item) = &args.item {
if args.ugly {
let _ = writeln!(
io::stderr(),
"ERROR: cannot expand single item ({}) in ugly mode.",
@ -136,7 +137,7 @@ fn cargo_expand() -> Result<i32> {
);
return Ok(1);
}
(Some(item), false) => {
if !cfg!(feature = "prettyplease") {
rustfmt = which_rustfmt();
if rustfmt.is_none() {
let _ = writeln!(
@ -151,8 +152,6 @@ fn cargo_expand() -> Result<i32> {
return Ok(1);
}
}
(None, true) => rustfmt = None,
(None, false) => rustfmt = which_rustfmt(),
}
let mut builder = tempfile::Builder::new();
@ -176,20 +175,30 @@ fn cargo_expand() -> Result<i32> {
return Ok(if code == 0 { 1 } else { code });
}
// Run rustfmt
if let Some(rustfmt) = rustfmt {
// Format the expanded code
if !args.ugly {
let questionably_formatted = content;
// Work around rustfmt not being able to parse paths containing $crate.
// This placeholder should be the same width as $crate to preserve
// alignments.
const DOLLAR_CRATE_PLACEHOLDER: &str = "Ξcrate";
content = content.replace("$crate", DOLLAR_CRATE_PLACEHOLDER);
let wip = questionably_formatted.replace("$crate", DOLLAR_CRATE_PLACEHOLDER);
// Support cargo-expand built with panic=abort, as otherwise proc-macro2
// ends up using a catch_unwind.
proc_macro2::fallback::force();
enum Stage {
Formatted(String),
Unformatted(String),
QuestionablyFormatted,
}
let mut stage = Stage::QuestionablyFormatted;
// Discard comments, which are misplaced by the compiler
if let Ok(mut syntax_tree) = syn::parse_file(&content) {
if let Ok(mut syntax_tree) = syn::parse_file(&wip) {
edit::sanitize(&mut syntax_tree);
if let Some(filter) = args.item {
syntax_tree.shebang = None;
@ -200,30 +209,58 @@ fn cargo_expand() -> Result<i32> {
return Ok(1);
}
}
content = quote!(#syntax_tree).to_string();
}
fs::write(&outfile_path, content)?;
fmt::write_rustfmt_config(&outdir)?;
let output = Command::new(&rustfmt)
.arg("--edition=2018")
.arg(&outfile_path)
.stderr(Stdio::null())
.output();
if let Ok(output) = output {
if !output.status.success() {
// Probably was the wrong edition.
let _status = Command::new(&rustfmt)
.arg("--edition=2015")
.arg(&outfile_path)
.stderr(Stdio::null())
.status();
#[cfg(feature = "prettyplease")]
// This is behind a feature because it's probably not mature enough
// to use in panic=abort mode yet. I'll remove the feature and do
// this by default when prettyplease is further along, or when
// cfg(panic = "unwind") is stabilized, whichever comes first.
// Tracking issue: https://github.com/rust-lang/rust/issues/77443
{
if let Ok(formatted) =
std::panic::catch_unwind(|| prettyplease::unparse(&syntax_tree))
{
stage = Stage::Formatted(formatted);
}
}
if let Stage::QuestionablyFormatted = stage {
let unformatted = quote!(#syntax_tree).to_string();
stage = Stage::Unformatted(unformatted);
}
}
content = fs::read_to_string(&outfile_path)?;
content = content.replace(DOLLAR_CRATE_PLACEHOLDER, "$crate");
let to_rustfmt = match &stage {
Stage::Formatted(_) => None,
Stage::Unformatted(unformatted) => Some(unformatted),
Stage::QuestionablyFormatted => Some(&wip),
};
if let Some(unformatted) = to_rustfmt {
if let Some(rustfmt) = rustfmt.or_else(which_rustfmt) {
fs::write(&outfile_path, unformatted)?;
fmt::write_rustfmt_config(&outdir)?;
for edition in &["2018", "2015"] {
let output = Command::new(&rustfmt)
.arg("--edition")
.arg(edition)
.arg(&outfile_path)
.stderr(Stdio::null())
.output();
if let Ok(output) = output {
if output.status.success() {
stage = Stage::Formatted(fs::read_to_string(&outfile_path)?);
break;
}
}
}
}
}
content = match stage {
Stage::Formatted(formatted) => formatted.replace(DOLLAR_CRATE_PLACEHOLDER, "$crate"),
Stage::Unformatted(_) | Stage::QuestionablyFormatted => questionably_formatted,
};
}
// Run pretty printer