mirror of
https://github.com/dtolnay/thiserror.git
synced 2025-04-05 05:47:39 +03:00
Merge pull request #340 from dtolnay/fallbackscan
Add infallible expr scanner fallback for scanning invalid code
This commit is contained in:
commit
b3bc3e722d
1 changed files with 35 additions and 1 deletions
|
@ -1,10 +1,11 @@
|
||||||
use crate::ast::Field;
|
use crate::ast::Field;
|
||||||
use crate::attr::{Display, Trait};
|
use crate::attr::{Display, Trait};
|
||||||
use crate::scan_expr;
|
use crate::scan_expr;
|
||||||
use proc_macro2::TokenTree;
|
use proc_macro2::{TokenStream, TokenTree};
|
||||||
use quote::{format_ident, quote, quote_spanned};
|
use quote::{format_ident, quote, quote_spanned};
|
||||||
use std::collections::{BTreeSet as Set, HashMap as Map};
|
use std::collections::{BTreeSet as Set, HashMap as Map};
|
||||||
use syn::ext::IdentExt;
|
use syn::ext::IdentExt;
|
||||||
|
use syn::parse::discouraged::Speculative;
|
||||||
use syn::parse::{ParseStream, Parser};
|
use syn::parse::{ParseStream, Parser};
|
||||||
use syn::{Expr, Ident, Index, LitStr, Member, Result, Token};
|
use syn::{Expr, Ident, Index, LitStr, Member, Result, Token};
|
||||||
|
|
||||||
|
@ -118,7 +119,25 @@ impl Display<'_> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::unnecessary_wraps)]
|
||||||
fn explicit_named_args(input: ParseStream) -> Result<Set<Ident>> {
|
fn explicit_named_args(input: ParseStream) -> Result<Set<Ident>> {
|
||||||
|
let ahead = input.fork();
|
||||||
|
if let Ok(set) = try_explicit_named_args(&ahead) {
|
||||||
|
input.advance_to(&ahead);
|
||||||
|
return Ok(set);
|
||||||
|
}
|
||||||
|
|
||||||
|
let ahead = input.fork();
|
||||||
|
if let Ok(set) = fallback_explicit_named_args(&ahead) {
|
||||||
|
input.advance_to(&ahead);
|
||||||
|
return Ok(set);
|
||||||
|
}
|
||||||
|
|
||||||
|
input.parse::<TokenStream>().unwrap();
|
||||||
|
Ok(Set::new())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn try_explicit_named_args(input: ParseStream) -> Result<Set<Ident>> {
|
||||||
let scan_expr = if is_syn_full() {
|
let scan_expr = if is_syn_full() {
|
||||||
|input: ParseStream| input.parse::<Expr>().map(drop)
|
|input: ParseStream| input.parse::<Expr>().map(drop)
|
||||||
} else {
|
} else {
|
||||||
|
@ -143,6 +162,21 @@ fn explicit_named_args(input: ParseStream) -> Result<Set<Ident>> {
|
||||||
Ok(named_args)
|
Ok(named_args)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn fallback_explicit_named_args(input: ParseStream) -> Result<Set<Ident>> {
|
||||||
|
let mut named_args = Set::new();
|
||||||
|
|
||||||
|
while !input.is_empty() {
|
||||||
|
if input.peek(Token![,]) && input.peek2(Ident::peek_any) && input.peek3(Token![=]) {
|
||||||
|
input.parse::<Token![,]>()?;
|
||||||
|
let ident = input.call(Ident::parse_any)?;
|
||||||
|
input.parse::<Token![=]>()?;
|
||||||
|
named_args.insert(ident);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(named_args)
|
||||||
|
}
|
||||||
|
|
||||||
fn is_syn_full() -> bool {
|
fn is_syn_full() -> bool {
|
||||||
// Expr::Block contains syn::Block which contains Vec<syn::Stmt>. In the
|
// Expr::Block contains syn::Block which contains Vec<syn::Stmt>. In the
|
||||||
// current version of Syn, syn::Stmt is exhaustive and could only plausibly
|
// current version of Syn, syn::Stmt is exhaustive and could only plausibly
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue