use proc_macro2::{Ident, Span};
macro_rules! tokens {
(
punct: {
$($punct:tt pub struct $punct_name:ident/$len:tt #[$punct_doc:meta])*
}
delimiter: {
$($delimiter:tt pub struct $delimiter_name:ident #[$delimiter_doc:meta])*
}
keyword: {
$($keyword:tt pub struct $keyword_name:ident #[$keyword_doc:meta])*
}
) => (
$(token_punct_def! { #[$punct_doc] pub struct $punct_name/$len })*
$(token_punct_parser! { $punct pub struct $punct_name })*
$(token_delimiter! { #[$delimiter_doc] $delimiter pub struct $delimiter_name })*
$(token_keyword! { #[$keyword_doc] $keyword pub struct $keyword_name })*
)
}
macro_rules! token_punct_def {
(#[$doc:meta]pub struct $name:ident / $len:tt) => {
#[cfg_attr(feature = "clone-impls", derive(Copy, Clone))]
#[$doc]
pub struct $name(pub [Span; $len]);
impl $name {
pub fn new(span: Span) -> Self {
$name([span; $len])
}
}
impl ::std::default::Default for $name {
fn default() -> Self {
$name([Span::call_site(); $len])
}
}
#[cfg(feature = "extra-traits")]
impl ::std::fmt::Debug for $name {
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
f.write_str(stringify!($name))
}
}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::Eq for $name {}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::PartialEq for $name {
fn eq(&self, _other: &$name) -> bool {
true
}
}
#[cfg(feature = "extra-traits")]
impl ::std::hash::Hash for $name {
fn hash<H>(&self, _state: &mut H)
where
H: ::std::hash::Hasher,
{
}
}
impl From<Span> for $name {
fn from(span: Span) -> Self {
$name([span; $len])
}
}
};
}
macro_rules! token_punct_parser {
($s:tt pub struct $name:ident) => {
#[cfg(feature = "printing")]
impl ::quote::ToTokens for $name {
fn to_tokens(&self, tokens: &mut ::proc_macro2::TokenStream) {
printing::punct($s, &self.0, tokens);
}
}
#[cfg(feature = "parsing")]
impl ::Synom for $name {
fn parse(tokens: $crate::buffer::Cursor) -> $crate::synom::PResult<$name> {
parsing::punct($s, tokens, $name)
}
fn description() -> Option<&'static str> {
Some(concat!("`", $s, "`"))
}
}
};
}
macro_rules! token_keyword {
(#[$doc:meta] $s:tt pub struct $name:ident) => {
#[cfg_attr(feature = "clone-impls", derive(Copy, Clone))]
#[$doc]
pub struct $name(pub Span);
impl ::std::default::Default for $name {
fn default() -> Self {
$name(Span::call_site())
}
}
#[cfg(feature = "extra-traits")]
impl ::std::fmt::Debug for $name {
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
f.write_str(stringify!($name))
}
}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::Eq for $name {}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::PartialEq for $name {
fn eq(&self, _other: &$name) -> bool {
true
}
}
#[cfg(feature = "extra-traits")]
impl ::std::hash::Hash for $name {
fn hash<H>(&self, _state: &mut H)
where
H: ::std::hash::Hasher,
{
}
}
#[cfg(feature = "printing")]
impl ::quote::ToTokens for $name {
fn to_tokens(&self, tokens: &mut ::proc_macro2::TokenStream) {
printing::keyword($s, &self.0, tokens);
}
}
#[cfg(feature = "parsing")]
impl ::Synom for $name {
fn parse(tokens: $crate::buffer::Cursor) -> $crate::synom::PResult<$name> {
parsing::keyword($s, tokens, $name)
}
fn description() -> Option<&'static str> {
Some(concat!("`", $s, "`"))
}
}
impl From<Span> for $name {
fn from(span: Span) -> Self {
$name(span)
}
}
};
}
macro_rules! token_delimiter {
(#[$doc:meta] $s:tt pub struct $name:ident) => {
#[cfg_attr(feature = "clone-impls", derive(Copy, Clone))]
#[$doc]
pub struct $name(pub Span);
impl ::std::default::Default for $name {
fn default() -> Self {
$name(Span::call_site())
}
}
#[cfg(feature = "extra-traits")]
impl ::std::fmt::Debug for $name {
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
f.write_str(stringify!($name))
}
}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::Eq for $name {}
#[cfg(feature = "extra-traits")]
impl ::std::cmp::PartialEq for $name {
fn eq(&self, _other: &$name) -> bool {
true
}
}
#[cfg(feature = "extra-traits")]
impl ::std::hash::Hash for $name {
fn hash<H>(&self, _state: &mut H)
where
H: ::std::hash::Hasher,
{
}
}
impl $name {
#[cfg(feature = "printing")]
pub fn surround<F>(&self, tokens: &mut ::proc_macro2::TokenStream, f: F)
where
F: FnOnce(&mut ::proc_macro2::TokenStream),
{
printing::delim($s, &self.0, tokens, f);
}
#[cfg(feature = "parsing")]
pub fn parse<F, R>(
tokens: $crate::buffer::Cursor,
f: F,
) -> $crate::synom::PResult<($name, R)>
where
F: FnOnce($crate::buffer::Cursor) -> $crate::synom::PResult<R>,
{
parsing::delim($s, tokens, $name, f)
}
}
impl From<Span> for $name {
fn from(span: Span) -> Self {
$name(span)
}
}
};
}
token_punct_def! {
pub struct Underscore/1
}
#[cfg(feature = "printing")]
impl ::quote::ToTokens for Underscore {
fn to_tokens(&self, tokens: &mut ::proc_macro2::TokenStream) {
use quote::TokenStreamExt;
tokens.append(::proc_macro2::Ident::new("_", self.0[0]));
}
}
#[cfg(feature = "parsing")]
impl ::Synom for Underscore {
fn parse(input: ::buffer::Cursor) -> ::synom::PResult<Underscore> {
match input.ident() {
Some((ident, rest)) => {
if ident == "_" {
Ok((Underscore([ident.span()]), rest))
} else {
::parse_error()
}
}
None => parsing::punct("_", input, Underscore),
}
}
fn description() -> Option<&'static str> {
Some("`_`")
}
}
token_punct_def! {
pub struct Apostrophe/1
}
#[cfg(not(feature = "clone-impls"))]
impl Clone for Apostrophe {
fn clone(&self) -> Self {
Apostrophe(self.0)
}
}
#[cfg(feature = "printing")]
impl ::quote::ToTokens for Apostrophe {
fn to_tokens(&self, tokens: &mut ::proc_macro2::TokenStream) {
use quote::TokenStreamExt;
let mut token = ::proc_macro2::Punct::new('\'', ::proc_macro2::Spacing::Joint);
token.set_span(self.0[0]);
tokens.append(token);
}
}
#[cfg(feature = "parsing")]
impl ::Synom for Apostrophe {
fn parse(input: ::buffer::Cursor) -> ::synom::PResult<Apostrophe> {
match input.punct() {
Some((op, rest)) => {
if op.as_char() == '\'' && op.spacing() == ::proc_macro2::Spacing::Joint {
Ok((Apostrophe([op.span()]), rest))
} else {
::parse_error()
}
}
None => ::parse_error(),
}
}
fn description() -> Option<&'static str> {
Some("`'`")
}
}
tokens! {
punct: {
"+" pub struct Add/1
"+=" pub struct AddEq/2
"&" pub struct And/1
"&&" pub struct AndAnd/2
"&=" pub struct AndEq/2
"@" pub struct At/1
"!" pub struct Bang/1
"^" pub struct Caret/1
"^=" pub struct CaretEq/2
":" pub struct Colon/1
"::" pub struct Colon2/2
"," pub struct Comma/1
"/" pub struct Div/1
"/=" pub struct DivEq/2
"$" pub struct Dollar/1
"." pub struct Dot/1
".." pub struct Dot2/2
"..." pub struct Dot3/3
"..=" pub struct DotDotEq/3
"=" pub struct Eq/1
"==" pub struct EqEq/2
">=" pub struct Ge/2
">" pub struct Gt/1
"<=" pub struct Le/2
"<" pub struct Lt/1
"*=" pub struct MulEq/2
"!=" pub struct Ne/2
"|" pub struct Or/1
"|=" pub struct OrEq/2
"||" pub struct OrOr/2
"#" pub struct Pound/1
"?" pub struct Question/1
"->" pub struct RArrow/2
"<-" pub struct LArrow/2
"%" pub struct Rem/1
"%=" pub struct RemEq/2
"=>" pub struct FatArrow/2
";" pub struct Semi/1
"<<" pub struct Shl/2
"<<=" pub struct ShlEq/3
">>" pub struct Shr/2
">>=" pub struct ShrEq/3
"*" pub struct Star/1
"-" pub struct Sub/1
"-=" pub struct SubEq/2
}
delimiter: {
"{" pub struct Brace
"[" pub struct Bracket
"(" pub struct Paren
" " pub struct Group
}
keyword: {
"as" pub struct As
"async" pub struct Async
"auto" pub struct Auto
"box" pub struct Box
"break" pub struct Break
"Self" pub struct CapSelf
"catch" pub struct Catch
"const" pub struct Const
"continue" pub struct Continue
"crate" pub struct Crate
"default" pub struct Default
"do" pub struct Do
"dyn" pub struct Dyn
"else" pub struct Else
"enum" pub struct Enum
"extern" pub struct Extern
"fn" pub struct Fn
"for" pub struct For
"if" pub struct If
"impl" pub struct Impl
"in" pub struct In
"let" pub struct Let
"loop" pub struct Loop
"macro" pub struct Macro
"match" pub struct Match
"mod" pub struct Mod
"move" pub struct Move
"mut" pub struct Mut
"pub" pub struct Pub
"ref" pub struct Ref
"return" pub struct Return
"self" pub struct Self_
"static" pub struct Static
"struct" pub struct Struct
"super" pub struct Super
"trait" pub struct Trait
"type" pub struct Type
"union" pub struct Union
"unsafe" pub struct Unsafe
"use" pub struct Use
"where" pub struct Where
"while" pub struct While
"yield" pub struct Yield
}
}
#[macro_export]
#[cfg_attr(rustfmt, rustfmt_skip)]
macro_rules! Token {
(+) => { $crate::token::Add };
(+=) => { $crate::token::AddEq };
(&) => { $crate::token::And };
(&&) => { $crate::token::AndAnd };
(&=) => { $crate::token::AndEq };
(@) => { $crate::token::At };
(!) => { $crate::token::Bang };
(^) => { $crate::token::Caret };
(^=) => { $crate::token::CaretEq };
(:) => { $crate::token::Colon };
(::) => { $crate::token::Colon2 };
(,) => { $crate::token::Comma };
(/) => { $crate::token::Div };
(/=) => { $crate::token::DivEq };
(.) => { $crate::token::Dot };
(..) => { $crate::token::Dot2 };
(...) => { $crate::token::Dot3 };
(..=) => { $crate::token::DotDotEq };
(=) => { $crate::token::Eq };
(==) => { $crate::token::EqEq };
(>=) => { $crate::token::Ge };
(>) => { $crate::token::Gt };
(<=) => { $crate::token::Le };
(<) => { $crate::token::Lt };
(*=) => { $crate::token::MulEq };
(!=) => { $crate::token::Ne };
(|) => { $crate::token::Or };
(|=) => { $crate::token::OrEq };
(||) => { $crate::token::OrOr };
(#) => { $crate::token::Pound };
(?) => { $crate::token::Question };
(->) => { $crate::token::RArrow };
(<-) => { $crate::token::LArrow };
(%) => { $crate::token::Rem };
(%=) => { $crate::token::RemEq };
(=>) => { $crate::token::FatArrow };
(;) => { $crate::token::Semi };
(<<) => { $crate::token::Shl };
(<<=) => { $crate::token::ShlEq };
(>>) => { $crate::token::Shr };
(>>=) => { $crate::token::ShrEq };
(*) => { $crate::token::Star };
(-) => { $crate::token::Sub };
(-=) => { $crate::token::SubEq };
(_) => { $crate::token::Underscore };
(as) => { $crate::token::As };
(async) => { $crate::token::Async };
(auto) => { $crate::token::Auto };
(box) => { $crate::token::Box };
(break) => { $crate::token::Break };
(Self) => { $crate::token::CapSelf };
(catch) => { $crate::token::Catch };
(const) => { $crate::token::Const };
(continue) => { $crate::token::Continue };
(crate) => { $crate::token::Crate };
(default) => { $crate::token::Default };
(do) => { $crate::token::Do };
(dyn) => { $crate::token::Dyn };
(else) => { $crate::token::Else };
(enum) => { $crate::token::Enum };
(extern) => { $crate::token::Extern };
(fn) => { $crate::token::Fn };
(for) => { $crate::token::For };
(if) => { $crate::token::If };
(impl) => { $crate::token::Impl };
(in) => { $crate::token::In };
(let) => { $crate::token::Let };
(loop) => { $crate::token::Loop };
(macro) => { $crate::token::Macro };
(match) => { $crate::token::Match };
(mod) => { $crate::token::Mod };
(move) => { $crate::token::Move };
(mut) => { $crate::token::Mut };
(pub) => { $crate::token::Pub };
(ref) => { $crate::token::Ref };
(return) => { $crate::token::Return };
(self) => { $crate::token::Self_ };
(static) => { $crate::token::Static };
(struct) => { $crate::token::Struct };
(super) => { $crate::token::Super };
(trait) => { $crate::token::Trait };
(type) => { $crate::token::Type };
(union) => { $crate::token::Union };
(unsafe) => { $crate::token::Unsafe };
(use) => { $crate::token::Use };
(where) => { $crate::token::Where };
(while) => { $crate::token::While };
(yield) => { $crate::token::Yield };
}
#[cfg(feature = "parsing")]
#[macro_export]
#[cfg_attr(rustfmt, rustfmt_skip)]
macro_rules! punct {
($i:expr, +) => { call!($i, <$crate::token::Add as $crate::synom::Synom>::parse) };
($i:expr, +=) => { call!($i, <$crate::token::AddEq as $crate::synom::Synom>::parse) };
($i:expr, &) => { call!($i, <$crate::token::And as $crate::synom::Synom>::parse) };
($i:expr, &&) => { call!($i, <$crate::token::AndAnd as $crate::synom::Synom>::parse) };
($i:expr, &=) => { call!($i, <$crate::token::AndEq as $crate::synom::Synom>::parse) };
($i:expr, @) => { call!($i, <$crate::token::At as $crate::synom::Synom>::parse) };
($i:expr, !) => { call!($i, <$crate::token::Bang as $crate::synom::Synom>::parse) };
($i:expr, ^) => { call!($i, <$crate::token::Caret as $crate::synom::Synom>::parse) };
($i:expr, ^=) => { call!($i, <$crate::token::CaretEq as $crate::synom::Synom>::parse) };
($i:expr, :) => { call!($i, <$crate::token::Colon as $crate::synom::Synom>::parse) };
($i:expr, ::) => { call!($i, <$crate::token::Colon2 as $crate::synom::Synom>::parse) };
($i:expr, ,) => { call!($i, <$crate::token::Comma as $crate::synom::Synom>::parse) };
($i:expr, /) => { call!($i, <$crate::token::Div as $crate::synom::Synom>::parse) };
($i:expr, /=) => { call!($i, <$crate::token::DivEq as $crate::synom::Synom>::parse) };
($i:expr, .) => { call!($i, <$crate::token::Dot as $crate::synom::Synom>::parse) };
($i:expr, ..) => { call!($i, <$crate::token::Dot2 as $crate::synom::Synom>::parse) };
($i:expr, ...) => { call!($i, <$crate::token::Dot3 as $crate::synom::Synom>::parse) };
($i:expr, ..=) => { call!($i, <$crate::token::DotDotEq as $crate::synom::Synom>::parse) };
($i:expr, =) => { call!($i, <$crate::token::Eq as $crate::synom::Synom>::parse) };
($i:expr, ==) => { call!($i, <$crate::token::EqEq as $crate::synom::Synom>::parse) };
($i:expr, >=) => { call!($i, <$crate::token::Ge as $crate::synom::Synom>::parse) };
($i:expr, >) => { call!($i, <$crate::token::Gt as $crate::synom::Synom>::parse) };
($i:expr, <=) => { call!($i, <$crate::token::Le as $crate::synom::Synom>::parse) };
($i:expr, <) => { call!($i, <$crate::token::Lt as $crate::synom::Synom>::parse) };
($i:expr, *=) => { call!($i, <$crate::token::MulEq as $crate::synom::Synom>::parse) };
($i:expr, !=) => { call!($i, <$crate::token::Ne as $crate::synom::Synom>::parse) };
($i:expr, |) => { call!($i, <$crate::token::Or as $crate::synom::Synom>::parse) };
($i:expr, |=) => { call!($i, <$crate::token::OrEq as $crate::synom::Synom>::parse) };
($i:expr, ||) => { call!($i, <$crate::token::OrOr as $crate::synom::Synom>::parse) };
($i:expr, #) => { call!($i, <$crate::token::Pound as $crate::synom::Synom>::parse) };
($i:expr, ?) => { call!($i, <$crate::token::Question as $crate::synom::Synom>::parse) };
($i:expr, ->) => { call!($i, <$crate::token::RArrow as $crate::synom::Synom>::parse) };
($i:expr, <-) => { call!($i, <$crate::token::LArrow as $crate::synom::Synom>::parse) };
($i:expr, %) => { call!($i, <$crate::token::Rem as $crate::synom::Synom>::parse) };
($i:expr, %=) => { call!($i, <$crate::token::RemEq as $crate::synom::Synom>::parse) };
($i:expr, =>) => { call!($i, <$crate::token::FatArrow as $crate::synom::Synom>::parse) };
($i:expr, ;) => { call!($i, <$crate::token::Semi as $crate::synom::Synom>::parse) };
($i:expr, <<) => { call!($i, <$crate::token::Shl as $crate::synom::Synom>::parse) };
($i:expr, <<=) => { call!($i, <$crate::token::ShlEq as $crate::synom::Synom>::parse) };
($i:expr, >>) => { call!($i, <$crate::token::Shr as $crate::synom::Synom>::parse) };
($i:expr, >>=) => { call!($i, <$crate::token::ShrEq as $crate::synom::Synom>::parse) };
($i:expr, *) => { call!($i, <$crate::token::Star as $crate::synom::Synom>::parse) };
($i:expr, -) => { call!($i, <$crate::token::Sub as $crate::synom::Synom>::parse) };
($i:expr, -=) => { call!($i, <$crate::token::SubEq as $crate::synom::Synom>::parse) };
($i:expr, _) => { call!($i, <$crate::token::Underscore as $crate::synom::Synom>::parse) };
}
#[cfg(feature = "parsing")]
#[macro_export]
#[cfg_attr(rustfmt, rustfmt_skip)]
macro_rules! keyword {
($i:expr, as) => { call!($i, <$crate::token::As as $crate::synom::Synom>::parse) };
($i:expr, async) => { call!($i, <$crate::token::Async as $crate::synom::Synom>::parse) };
($i:expr, auto) => { call!($i, <$crate::token::Auto as $crate::synom::Synom>::parse) };
($i:expr, box) => { call!($i, <$crate::token::Box as $crate::synom::Synom>::parse) };
($i:expr, break) => { call!($i, <$crate::token::Break as $crate::synom::Synom>::parse) };
($i:expr, Self) => { call!($i, <$crate::token::CapSelf as $crate::synom::Synom>::parse) };
($i:expr, catch) => { call!($i, <$crate::token::Catch as $crate::synom::Synom>::parse) };
($i:expr, const) => { call!($i, <$crate::token::Const as $crate::synom::Synom>::parse) };
($i:expr, continue) => { call!($i, <$crate::token::Continue as $crate::synom::Synom>::parse) };
($i:expr, crate) => { call!($i, <$crate::token::Crate as $crate::synom::Synom>::parse) };
($i:expr, default) => { call!($i, <$crate::token::Default as $crate::synom::Synom>::parse) };
($i:expr, do) => { call!($i, <$crate::token::Do as $crate::synom::Synom>::parse) };
($i:expr, dyn) => { call!($i, <$crate::token::Dyn as $crate::synom::Synom>::parse) };
($i:expr, else) => { call!($i, <$crate::token::Else as $crate::synom::Synom>::parse) };
($i:expr, enum) => { call!($i, <$crate::token::Enum as $crate::synom::Synom>::parse) };
($i:expr, extern) => { call!($i, <$crate::token::Extern as $crate::synom::Synom>::parse) };
($i:expr, fn) => { call!($i, <$crate::token::Fn as $crate::synom::Synom>::parse) };
($i:expr, for) => { call!($i, <$crate::token::For as $crate::synom::Synom>::parse) };
($i:expr, if) => { call!($i, <$crate::token::If as $crate::synom::Synom>::parse) };
($i:expr, impl) => { call!($i, <$crate::token::Impl as $crate::synom::Synom>::parse) };
($i:expr, in) => { call!($i, <$crate::token::In as $crate::synom::Synom>::parse) };
($i:expr, let) => { call!($i, <$crate::token::Let as $crate::synom::Synom>::parse) };
($i:expr, loop) => { call!($i, <$crate::token::Loop as $crate::synom::Synom>::parse) };
($i:expr, macro) => { call!($i, <$crate::token::Macro as $crate::synom::Synom>::parse) };
($i:expr, match) => { call!($i, <$crate::token::Match as $crate::synom::Synom>::parse) };
($i:expr, mod) => { call!($i, <$crate::token::Mod as $crate::synom::Synom>::parse) };
($i:expr, move) => { call!($i, <$crate::token::Move as $crate::synom::Synom>::parse) };
($i:expr, mut) => { call!($i, <$crate::token::Mut as $crate::synom::Synom>::parse) };
($i:expr, pub) => { call!($i, <$crate::token::Pub as $crate::synom::Synom>::parse) };
($i:expr, ref) => { call!($i, <$crate::token::Ref as $crate::synom::Synom>::parse) };
($i:expr, return) => { call!($i, <$crate::token::Return as $crate::synom::Synom>::parse) };
($i:expr, self) => { call!($i, <$crate::token::Self_ as $crate::synom::Synom>::parse) };
($i:expr, static) => { call!($i, <$crate::token::Static as $crate::synom::Synom>::parse) };
($i:expr, struct) => { call!($i, <$crate::token::Struct as $crate::synom::Synom>::parse) };
($i:expr, super) => { call!($i, <$crate::token::Super as $crate::synom::Synom>::parse) };
($i:expr, trait) => { call!($i, <$crate::token::Trait as $crate::synom::Synom>::parse) };
($i:expr, type) => { call!($i, <$crate::token::Type as $crate::synom::Synom>::parse) };
($i:expr, union) => { call!($i, <$crate::token::Union as $crate::synom::Synom>::parse) };
($i:expr, unsafe) => { call!($i, <$crate::token::Unsafe as $crate::synom::Synom>::parse) };
($i:expr, use) => { call!($i, <$crate::token::Use as $crate::synom::Synom>::parse) };
($i:expr, where) => { call!($i, <$crate::token::Where as $crate::synom::Synom>::parse) };
($i:expr, while) => { call!($i, <$crate::token::While as $crate::synom::Synom>::parse) };
($i:expr, yield) => { call!($i, <$crate::token::Yield as $crate::synom::Synom>::parse) };
}
macro_rules! ident_from_token {
($token:ident) => {
impl From<Token![$token]> for Ident {
fn from(token: Token![$token]) -> Ident {
Ident::new(stringify!($token), token.0)
}
}
};
}
ident_from_token!(self);
ident_from_token!(Self);
ident_from_token!(super);
ident_from_token!(crate);
ident_from_token!(extern);
#[cfg(feature = "parsing")]
mod parsing {
use proc_macro2::{Delimiter, Spacing, Span};
use buffer::Cursor;
use parse_error;
use synom::PResult;
pub trait FromSpans: Sized {
fn from_spans(spans: &[Span]) -> Self;
}
impl FromSpans for [Span; 1] {
fn from_spans(spans: &[Span]) -> Self {
[spans[0]]
}
}
impl FromSpans for [Span; 2] {
fn from_spans(spans: &[Span]) -> Self {
[spans[0], spans[1]]
}
}
impl FromSpans for [Span; 3] {
fn from_spans(spans: &[Span]) -> Self {
[spans[0], spans[1], spans[2]]
}
}
pub fn punct<'a, T, R>(s: &str, mut tokens: Cursor<'a>, new: fn(T) -> R) -> PResult<'a, R>
where
T: FromSpans,
{
let mut spans = [Span::call_site(); 3];
assert!(s.len() <= spans.len());
let chars = s.chars();
for (i, (ch, slot)) in chars.zip(&mut spans).enumerate() {
match tokens.punct() {
Some((op, rest)) => {
if op.as_char() == ch {
if i != s.len() - 1 {
match op.spacing() {
Spacing::Joint => {}
_ => return parse_error(),
}
}
*slot = op.span();
tokens = rest;
} else {
return parse_error();
}
}
_ => return parse_error(),
}
}
Ok((new(T::from_spans(&spans)), tokens))
}
pub fn keyword<'a, T>(keyword: &str, tokens: Cursor<'a>, new: fn(Span) -> T) -> PResult<'a, T> {
if let Some((ident, rest)) = tokens.ident() {
if ident == keyword {
return Ok((new(ident.span()), rest));
}
}
parse_error()
}
pub fn delim<'a, F, R, T>(
delim: &str,
tokens: Cursor<'a>,
new: fn(Span) -> T,
f: F,
) -> PResult<'a, (T, R)>
where
F: FnOnce(Cursor) -> PResult<R>,
{
let delim = match delim {
"(" => Delimiter::Parenthesis,
"{" => Delimiter::Brace,
"[" => Delimiter::Bracket,
" " => Delimiter::None,
_ => panic!("unknown delimiter: {}", delim),
};
if let Some((inside, span, rest)) = tokens.group(delim) {
match f(inside) {
Ok((ret, remaining)) => {
if remaining.eof() {
return Ok(((new(span), ret), rest));
}
}
Err(err) => return Err(err),
}
}
parse_error()
}
}
#[cfg(feature = "printing")]
mod printing {
use proc_macro2::{Delimiter, Group, Ident, Punct, Spacing, Span, TokenStream};
use quote::TokenStreamExt;
pub fn punct(s: &str, spans: &[Span], tokens: &mut TokenStream) {
assert_eq!(s.len(), spans.len());
let mut chars = s.chars();
let mut spans = spans.iter();
let ch = chars.next_back().unwrap();
let span = spans.next_back().unwrap();
for (ch, span) in chars.zip(spans) {
let mut op = Punct::new(ch, Spacing::Joint);
op.set_span(*span);
tokens.append(op);
}
let mut op = Punct::new(ch, Spacing::Alone);
op.set_span(*span);
tokens.append(op);
}
pub fn keyword(s: &str, span: &Span, tokens: &mut TokenStream) {
tokens.append(Ident::new(s, *span));
}
pub fn delim<F>(s: &str, span: &Span, tokens: &mut TokenStream, f: F)
where
F: FnOnce(&mut TokenStream),
{
let delim = match s {
"(" => Delimiter::Parenthesis,
"[" => Delimiter::Bracket,
"{" => Delimiter::Brace,
" " => Delimiter::None,
_ => panic!("unknown delimiter: {}", s),
};
let mut inner = TokenStream::new();
f(&mut inner);
let mut g = Group::new(delim, inner);
g.set_span(*span);
tokens.append(g);
}
}