mirror of
https://github.com/dtolnay/cargo-expand.git
synced 2025-04-04 13:27:37 +03:00
Format rest of items even if some caused panic
This commit is contained in:
parent
65699f084a
commit
54f0d33b13
4 changed files with 284 additions and 5 deletions
4
Cargo.lock
generated
4
Cargo.lock
generated
|
@ -696,9 +696,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "prettyplease"
|
name = "prettyplease"
|
||||||
version = "0.2.7"
|
version = "0.2.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "43ded2b5b204571f065ab8540367d738dfe1b3606ab9eb669dcfb5e7a3a07501"
|
checksum = "9825a04601d60621feed79c4e6b56d65db77cdca55cef43b46b0de1096d1c282"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"syn",
|
"syn",
|
||||||
|
|
|
@ -20,11 +20,11 @@ bat = { version = "0.23", default-features = false, features = ["paging", "regex
|
||||||
cargo-subcommand-metadata = "0.1"
|
cargo-subcommand-metadata = "0.1"
|
||||||
clap = { version = "4", features = ["deprecated", "derive"] }
|
clap = { version = "4", features = ["deprecated", "derive"] }
|
||||||
is-terminal = "0.4"
|
is-terminal = "0.4"
|
||||||
prettyplease = { version = "0.2.7", features = ["verbatim"] }
|
prettyplease = { version = "0.2.9", features = ["verbatim"] }
|
||||||
proc-macro2 = "1.0"
|
proc-macro2 = "1.0"
|
||||||
quote = { version = "1.0", default-features = false }
|
quote = { version = "1.0", default-features = false }
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
syn = { version = "2.0.16", default-features = false, features = ["full", "parsing", "printing", "visit-mut"] }
|
syn = { version = "2.0.16", default-features = false, features = ["full", "parsing", "printing", "clone-impls", "fold", "visit-mut"] }
|
||||||
syn-select = "0.3"
|
syn-select = "0.3"
|
||||||
tempfile = "3.0"
|
tempfile = "3.0"
|
||||||
termcolor = "1.0"
|
termcolor = "1.0"
|
||||||
|
|
|
@ -24,12 +24,14 @@ mod edit;
|
||||||
mod error;
|
mod error;
|
||||||
mod fmt;
|
mod fmt;
|
||||||
mod opts;
|
mod opts;
|
||||||
|
mod unparse;
|
||||||
|
|
||||||
use crate::cmd::Line;
|
use crate::cmd::Line;
|
||||||
use crate::config::Config;
|
use crate::config::Config;
|
||||||
use crate::error::Result;
|
use crate::error::Result;
|
||||||
use crate::opts::Coloring::*;
|
use crate::opts::Coloring::*;
|
||||||
use crate::opts::{Coloring, Expand, Subcommand};
|
use crate::opts::{Coloring, Expand, Subcommand};
|
||||||
|
use crate::unparse::unparse_maximal;
|
||||||
use bat::{PagingMode, PrettyPrinter};
|
use bat::{PagingMode, PrettyPrinter};
|
||||||
use clap::{Parser, ValueEnum};
|
use clap::{Parser, ValueEnum};
|
||||||
use is_terminal::IsTerminal;
|
use is_terminal::IsTerminal;
|
||||||
|
@ -154,7 +156,7 @@ fn cargo_expand() -> Result<i32> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !config.rustfmt {
|
if !config.rustfmt {
|
||||||
if let Ok(formatted) = ignore_panic(|| prettyplease::unparse(&syntax_tree)) {
|
if let Ok(formatted) = ignore_panic(|| unparse_maximal(&syntax_tree)) {
|
||||||
stage = Stage::Formatted(formatted);
|
stage = Stage::Formatted(formatted);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
277
src/unparse.rs
Normal file
277
src/unparse.rs
Normal file
|
@ -0,0 +1,277 @@
|
||||||
|
use proc_macro2::{Ident, Span};
|
||||||
|
use quote::quote;
|
||||||
|
use std::panic;
|
||||||
|
use syn::fold::{self, Fold};
|
||||||
|
use syn::punctuated::Punctuated;
|
||||||
|
use syn::{
|
||||||
|
token, Abi, Block, Expr, File, ForeignItem, Generics, ImplItem, Item, ItemConst, ItemFn,
|
||||||
|
ItemForeignMod, ItemImpl, ItemTrait, ReturnType, Signature, Stmt, Token, TraitItem, Type,
|
||||||
|
TypeInfer, Visibility,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub(crate) fn unparse_maximal(syntax_tree: &File) -> String {
|
||||||
|
if let Ok(formatted) = panic::catch_unwind(|| prettyplease::unparse(syntax_tree)) {
|
||||||
|
return formatted;
|
||||||
|
}
|
||||||
|
|
||||||
|
let redacted = UnparseMaximal.fold_file(syntax_tree.clone());
|
||||||
|
prettyplease::unparse(&redacted)
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UnparseMaximal;
|
||||||
|
|
||||||
|
impl Fold for UnparseMaximal {
|
||||||
|
fn fold_item(&mut self, item: Item) -> Item {
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec![item],
|
||||||
|
};
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let item = file.items.pop().unwrap();
|
||||||
|
if ok {
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
|
file.items.push(fold::fold_item(self, item));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
return file.items.pop().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
Item::Verbatim(quote!(...))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fold_stmt(&mut self, stmt: Stmt) -> Stmt {
|
||||||
|
// `fn main() { $stmt }`
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec![Item::Fn(ItemFn {
|
||||||
|
attrs: Vec::new(),
|
||||||
|
vis: Visibility::Inherited,
|
||||||
|
sig: Signature {
|
||||||
|
constness: None,
|
||||||
|
asyncness: None,
|
||||||
|
unsafety: None,
|
||||||
|
abi: None,
|
||||||
|
fn_token: token::Fn(Span::call_site()),
|
||||||
|
ident: Ident::new("main", Span::call_site()),
|
||||||
|
generics: Generics::default(),
|
||||||
|
paren_token: token::Paren(Span::call_site()),
|
||||||
|
inputs: Punctuated::new(),
|
||||||
|
variadic: None,
|
||||||
|
output: ReturnType::Default,
|
||||||
|
},
|
||||||
|
block: Box::new(Block {
|
||||||
|
brace_token: token::Brace(Span::call_site()),
|
||||||
|
stmts: vec![stmt],
|
||||||
|
}),
|
||||||
|
})],
|
||||||
|
};
|
||||||
|
|
||||||
|
fn unwrap_item_fn(item: &mut Item) -> &mut ItemFn {
|
||||||
|
match item {
|
||||||
|
Item::Fn(item) => item,
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let item_fn = unwrap_item_fn(&mut file.items[0]);
|
||||||
|
let stmt = item_fn.block.stmts.pop().unwrap();
|
||||||
|
if ok {
|
||||||
|
return stmt;
|
||||||
|
}
|
||||||
|
|
||||||
|
item_fn.block.stmts.push(fold::fold_stmt(self, stmt));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
let item_fn = unwrap_item_fn(&mut file.items[0]);
|
||||||
|
return item_fn.block.stmts.pop().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
Stmt::Item(Item::Verbatim(quote!(...)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fold_expr(&mut self, expr: Expr) -> Expr {
|
||||||
|
// `const _: _ = $expr;`
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec),
|
||||||
|
ident: Ident::from(Token)),
|
||||||
|
generics: Generics::default(),
|
||||||
|
colon_token: Token),
|
||||||
|
ty: Box::new(Type::Infer(TypeInfer {
|
||||||
|
underscore_token: Token),
|
||||||
|
})),
|
||||||
|
eq_token: Token),
|
||||||
|
expr: Box::new(expr),
|
||||||
|
semi_token: Token),
|
||||||
|
})],
|
||||||
|
};
|
||||||
|
|
||||||
|
fn unwrap_item_const(item: Item) -> ItemConst {
|
||||||
|
match item {
|
||||||
|
Item::Const(item) => item,
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let mut item_const = unwrap_item_const(file.items.pop().unwrap());
|
||||||
|
let expr = *item_const.expr;
|
||||||
|
if ok {
|
||||||
|
return expr;
|
||||||
|
}
|
||||||
|
|
||||||
|
item_const.expr = Box::new(fold::fold_expr(self, expr));
|
||||||
|
file.items.push(Item::Const(item_const));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
let item_const = unwrap_item_const(file.items.pop().unwrap());
|
||||||
|
return *item_const.expr;
|
||||||
|
}
|
||||||
|
|
||||||
|
Expr::Verbatim(quote!(...))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fold_foreign_item(&mut self, foreign_item: ForeignItem) -> ForeignItem {
|
||||||
|
// `extern { $foreign_item }`
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec),
|
||||||
|
name: None,
|
||||||
|
},
|
||||||
|
brace_token: token::Brace(Span::call_site()),
|
||||||
|
items: vec![foreign_item],
|
||||||
|
})],
|
||||||
|
};
|
||||||
|
|
||||||
|
fn unwrap_item_foreign_mod(item: &mut Item) -> &mut ItemForeignMod {
|
||||||
|
match item {
|
||||||
|
Item::ForeignMod(item) => item,
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let item_foreign_mod = unwrap_item_foreign_mod(&mut file.items[0]);
|
||||||
|
let foreign_item = item_foreign_mod.items.pop().unwrap();
|
||||||
|
if ok {
|
||||||
|
return foreign_item;
|
||||||
|
}
|
||||||
|
|
||||||
|
item_foreign_mod
|
||||||
|
.items
|
||||||
|
.push(fold::fold_foreign_item(self, foreign_item));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
let item_foreign_mod = unwrap_item_foreign_mod(&mut file.items[0]);
|
||||||
|
return item_foreign_mod.items.pop().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
ForeignItem::Verbatim(quote!(...))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fold_trait_item(&mut self, trait_item: TraitItem) -> TraitItem {
|
||||||
|
// `trait Trait { $trait_item }`
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec),
|
||||||
|
ident: Ident::new("Trait", Span::call_site()),
|
||||||
|
generics: Generics::default(),
|
||||||
|
colon_token: None,
|
||||||
|
supertraits: Punctuated::new(),
|
||||||
|
brace_token: token::Brace(Span::call_site()),
|
||||||
|
items: vec![trait_item],
|
||||||
|
})],
|
||||||
|
};
|
||||||
|
|
||||||
|
fn unwrap_item_trait(item: &mut Item) -> &mut ItemTrait {
|
||||||
|
match item {
|
||||||
|
Item::Trait(item) => item,
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let item_trait = unwrap_item_trait(&mut file.items[0]);
|
||||||
|
let trait_item = item_trait.items.pop().unwrap();
|
||||||
|
if ok {
|
||||||
|
return trait_item;
|
||||||
|
}
|
||||||
|
|
||||||
|
item_trait
|
||||||
|
.items
|
||||||
|
.push(fold::fold_trait_item(self, trait_item));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
let item_trait = unwrap_item_trait(&mut file.items[0]);
|
||||||
|
return item_trait.items.pop().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
TraitItem::Verbatim(quote!(...))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fold_impl_item(&mut self, impl_item: ImplItem) -> ImplItem {
|
||||||
|
// `impl _ { $impl_item }`
|
||||||
|
let mut file = File {
|
||||||
|
shebang: None,
|
||||||
|
attrs: Vec::new(),
|
||||||
|
items: vec),
|
||||||
|
generics: Generics::default(),
|
||||||
|
trait_: None,
|
||||||
|
self_ty: Box::new(Type::Infer(TypeInfer {
|
||||||
|
underscore_token: Token),
|
||||||
|
})),
|
||||||
|
brace_token: token::Brace(Span::call_site()),
|
||||||
|
items: vec![impl_item],
|
||||||
|
})],
|
||||||
|
};
|
||||||
|
|
||||||
|
fn unwrap_item_impl(item: &mut Item) -> &mut ItemImpl {
|
||||||
|
match item {
|
||||||
|
Item::Impl(item) => item,
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
let item_impl = unwrap_item_impl(&mut file.items[0]);
|
||||||
|
let impl_item = item_impl.items.pop().unwrap();
|
||||||
|
if ok {
|
||||||
|
return impl_item;
|
||||||
|
}
|
||||||
|
|
||||||
|
item_impl.items.push(fold::fold_impl_item(self, impl_item));
|
||||||
|
let ok = panic::catch_unwind(|| prettyplease::unparse(&file)).is_ok();
|
||||||
|
if ok {
|
||||||
|
let item_impl = unwrap_item_impl(&mut file.items[0]);
|
||||||
|
return item_impl.items.pop().unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImplItem::Verbatim(quote!(...))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue