mirror of
https://github.com/dtolnay/cargo-expand.git
synced 2025-04-03 21:07:37 +03:00
Format output using prettyplease
This commit is contained in:
parent
2a14aaa9d4
commit
3176152239
4 changed files with 88 additions and 34 deletions
11
.github/workflows/ci.yml
vendored
11
.github/workflows/ci.yml
vendored
|
@ -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
11
Cargo.lock
generated
|
@ -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"
|
||||
|
|
|
@ -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"] }
|
||||
|
|
99
src/main.rs
99
src/main.rs
|
@ -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
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue