303 lines
8.9 KiB
Rust
303 lines
8.9 KiB
Rust
|
|
/// Define a type that supports parsing and printing a multi-character symbol
|
||
|
|
/// as if it were a punctuation token.
|
||
|
|
///
|
||
|
|
/// # Usage
|
||
|
|
///
|
||
|
|
/// ```
|
||
|
|
/// syn::custom_punctuation!(LeftRightArrow, <=>);
|
||
|
|
/// ```
|
||
|
|
///
|
||
|
|
/// The generated syntax tree node supports the following operations just like
|
||
|
|
/// any built-in punctuation token.
|
||
|
|
///
|
||
|
|
/// - [Peeking] — `input.peek(LeftRightArrow)`
|
||
|
|
///
|
||
|
|
/// - [Parsing] — `input.parse::<LeftRightArrow>()?`
|
||
|
|
///
|
||
|
|
/// - [Printing] — `quote!( ... #lrarrow ... )`
|
||
|
|
///
|
||
|
|
/// - Construction from a [`Span`] — `let lrarrow = LeftRightArrow(sp)`
|
||
|
|
///
|
||
|
|
/// - Construction from multiple [`Span`] — `let lrarrow = LeftRightArrow([sp, sp, sp])`
|
||
|
|
///
|
||
|
|
/// - Field access to its spans — `let spans = lrarrow.spans`
|
||
|
|
///
|
||
|
|
/// [Peeking]: crate::parse::ParseBuffer::peek
|
||
|
|
/// [Parsing]: crate::parse::ParseBuffer::parse
|
||
|
|
/// [Printing]: quote::ToTokens
|
||
|
|
/// [`Span`]: proc_macro2::Span
|
||
|
|
///
|
||
|
|
/// # Example
|
||
|
|
///
|
||
|
|
/// ```
|
||
|
|
/// use proc_macro2::{TokenStream, TokenTree};
|
||
|
|
/// use syn::parse::{Parse, ParseStream, Peek, Result};
|
||
|
|
/// use syn::punctuated::Punctuated;
|
||
|
|
/// use syn::Expr;
|
||
|
|
///
|
||
|
|
/// syn::custom_punctuation!(PathSeparator, </>);
|
||
|
|
///
|
||
|
|
/// // expr </> expr </> expr ...
|
||
|
|
/// struct PathSegments {
|
||
|
|
/// segments: Punctuated<Expr, PathSeparator>,
|
||
|
|
/// }
|
||
|
|
///
|
||
|
|
/// impl Parse for PathSegments {
|
||
|
|
/// fn parse(input: ParseStream) -> Result<Self> {
|
||
|
|
/// let mut segments = Punctuated::new();
|
||
|
|
///
|
||
|
|
/// let first = parse_until(input, PathSeparator)?;
|
||
|
|
/// segments.push_value(syn::parse2(first)?);
|
||
|
|
///
|
||
|
|
/// while input.peek(PathSeparator) {
|
||
|
|
/// segments.push_punct(input.parse()?);
|
||
|
|
///
|
||
|
|
/// let next = parse_until(input, PathSeparator)?;
|
||
|
|
/// segments.push_value(syn::parse2(next)?);
|
||
|
|
/// }
|
||
|
|
///
|
||
|
|
/// Ok(PathSegments { segments })
|
||
|
|
/// }
|
||
|
|
/// }
|
||
|
|
///
|
||
|
|
/// fn parse_until<E: Peek>(input: ParseStream, end: E) -> Result<TokenStream> {
|
||
|
|
/// let mut tokens = TokenStream::new();
|
||
|
|
/// while !input.is_empty() && !input.peek(end) {
|
||
|
|
/// let next: TokenTree = input.parse()?;
|
||
|
|
/// tokens.extend(Some(next));
|
||
|
|
/// }
|
||
|
|
/// Ok(tokens)
|
||
|
|
/// }
|
||
|
|
///
|
||
|
|
/// fn main() {
|
||
|
|
/// let input = r#" a::b </> c::d::e "#;
|
||
|
|
/// let _: PathSegments = syn::parse_str(input).unwrap();
|
||
|
|
/// }
|
||
|
|
/// ```
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {
|
||
|
|
pub struct $ident {
|
||
|
|
pub spans: $crate::custom_punctuation_repr!($($tt)+),
|
||
|
|
}
|
||
|
|
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[allow(dead_code, non_snake_case)]
|
||
|
|
pub fn $ident<__S: $crate::__private::IntoSpans<$crate::custom_punctuation_repr!($($tt)+)>>(
|
||
|
|
spans: __S,
|
||
|
|
) -> $ident {
|
||
|
|
let _validate_len = 0 $(+ $crate::custom_punctuation_len!(strict, $tt))*;
|
||
|
|
$ident {
|
||
|
|
spans: $crate::__private::IntoSpans::into_spans(spans)
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
const _: () = {
|
||
|
|
impl $crate::__private::Default for $ident {
|
||
|
|
fn default() -> Self {
|
||
|
|
$ident($crate::__private::Span::call_site())
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
$crate::impl_parse_for_custom_punctuation!($ident, $($tt)+);
|
||
|
|
$crate::impl_to_tokens_for_custom_punctuation!($ident, $($tt)+);
|
||
|
|
$crate::impl_clone_for_custom_punctuation!($ident, $($tt)+);
|
||
|
|
$crate::impl_extra_traits_for_custom_punctuation!($ident, $($tt)+);
|
||
|
|
};
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(feature = "parsing")]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_parse_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {
|
||
|
|
impl $crate::__private::CustomToken for $ident {
|
||
|
|
fn peek(cursor: $crate::buffer::Cursor) -> $crate::__private::bool {
|
||
|
|
$crate::__private::peek_punct(cursor, $crate::stringify_punct!($($tt)+))
|
||
|
|
}
|
||
|
|
|
||
|
|
fn display() -> &'static $crate::__private::str {
|
||
|
|
$crate::__private::concat!("`", $crate::stringify_punct!($($tt)+), "`")
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl $crate::parse::Parse for $ident {
|
||
|
|
fn parse(input: $crate::parse::ParseStream) -> $crate::parse::Result<$ident> {
|
||
|
|
let spans: $crate::custom_punctuation_repr!($($tt)+) =
|
||
|
|
$crate::__private::parse_punct(input, $crate::stringify_punct!($($tt)+))?;
|
||
|
|
Ok($ident(spans))
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(not(feature = "parsing"))]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_parse_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(feature = "printing")]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_to_tokens_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {
|
||
|
|
impl $crate::__private::ToTokens for $ident {
|
||
|
|
fn to_tokens(&self, tokens: &mut $crate::__private::TokenStream2) {
|
||
|
|
$crate::__private::print_punct($crate::stringify_punct!($($tt)+), &self.spans, tokens)
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(not(feature = "printing"))]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_to_tokens_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(feature = "clone-impls")]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_clone_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {
|
||
|
|
impl $crate::__private::Copy for $ident {}
|
||
|
|
|
||
|
|
#[allow(clippy::expl_impl_clone_on_copy)]
|
||
|
|
impl $crate::__private::Clone for $ident {
|
||
|
|
fn clone(&self) -> Self {
|
||
|
|
*self
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(not(feature = "clone-impls"))]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_clone_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(feature = "extra-traits")]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_extra_traits_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {
|
||
|
|
impl $crate::__private::Debug for $ident {
|
||
|
|
fn fmt(&self, f: &mut $crate::__private::Formatter) -> $crate::__private::FmtResult {
|
||
|
|
$crate::__private::Formatter::write_str(f, $crate::__private::stringify!($ident))
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl $crate::__private::Eq for $ident {}
|
||
|
|
|
||
|
|
impl $crate::__private::PartialEq for $ident {
|
||
|
|
fn eq(&self, _other: &Self) -> $crate::__private::bool {
|
||
|
|
true
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
impl $crate::__private::Hash for $ident {
|
||
|
|
fn hash<__H: $crate::__private::Hasher>(&self, _state: &mut __H) {}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[cfg(not(feature = "extra-traits"))]
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! impl_extra_traits_for_custom_punctuation {
|
||
|
|
($ident:ident, $($tt:tt)+) => {};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! custom_punctuation_repr {
|
||
|
|
($($tt:tt)+) => {
|
||
|
|
[$crate::__private::Span; 0 $(+ $crate::custom_punctuation_len!(lenient, $tt))+]
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
#[rustfmt::skip]
|
||
|
|
macro_rules! custom_punctuation_len {
|
||
|
|
($mode:ident, +) => { 1 };
|
||
|
|
($mode:ident, +=) => { 2 };
|
||
|
|
($mode:ident, &) => { 1 };
|
||
|
|
($mode:ident, &&) => { 2 };
|
||
|
|
($mode:ident, &=) => { 2 };
|
||
|
|
($mode:ident, @) => { 1 };
|
||
|
|
($mode:ident, !) => { 1 };
|
||
|
|
($mode:ident, ^) => { 1 };
|
||
|
|
($mode:ident, ^=) => { 2 };
|
||
|
|
($mode:ident, :) => { 1 };
|
||
|
|
($mode:ident, ::) => { 2 };
|
||
|
|
($mode:ident, ,) => { 1 };
|
||
|
|
($mode:ident, /) => { 1 };
|
||
|
|
($mode:ident, /=) => { 2 };
|
||
|
|
($mode:ident, .) => { 1 };
|
||
|
|
($mode:ident, ..) => { 2 };
|
||
|
|
($mode:ident, ...) => { 3 };
|
||
|
|
($mode:ident, ..=) => { 3 };
|
||
|
|
($mode:ident, =) => { 1 };
|
||
|
|
($mode:ident, ==) => { 2 };
|
||
|
|
($mode:ident, >=) => { 2 };
|
||
|
|
($mode:ident, >) => { 1 };
|
||
|
|
($mode:ident, <=) => { 2 };
|
||
|
|
($mode:ident, <) => { 1 };
|
||
|
|
($mode:ident, *=) => { 2 };
|
||
|
|
($mode:ident, !=) => { 2 };
|
||
|
|
($mode:ident, |) => { 1 };
|
||
|
|
($mode:ident, |=) => { 2 };
|
||
|
|
($mode:ident, ||) => { 2 };
|
||
|
|
($mode:ident, #) => { 1 };
|
||
|
|
($mode:ident, ?) => { 1 };
|
||
|
|
($mode:ident, ->) => { 2 };
|
||
|
|
($mode:ident, <-) => { 2 };
|
||
|
|
($mode:ident, %) => { 1 };
|
||
|
|
($mode:ident, %=) => { 2 };
|
||
|
|
($mode:ident, =>) => { 2 };
|
||
|
|
($mode:ident, ;) => { 1 };
|
||
|
|
($mode:ident, <<) => { 2 };
|
||
|
|
($mode:ident, <<=) => { 3 };
|
||
|
|
($mode:ident, >>) => { 2 };
|
||
|
|
($mode:ident, >>=) => { 3 };
|
||
|
|
($mode:ident, *) => { 1 };
|
||
|
|
($mode:ident, -) => { 1 };
|
||
|
|
($mode:ident, -=) => { 2 };
|
||
|
|
($mode:ident, ~) => { 1 };
|
||
|
|
(lenient, $tt:tt) => { 0 };
|
||
|
|
(strict, $tt:tt) => {{ $crate::custom_punctuation_unexpected!($tt); 0 }};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! custom_punctuation_unexpected {
|
||
|
|
() => {};
|
||
|
|
}
|
||
|
|
|
||
|
|
// Not public API.
|
||
|
|
#[doc(hidden)]
|
||
|
|
#[macro_export]
|
||
|
|
macro_rules! stringify_punct {
|
||
|
|
($($tt:tt)+) => {
|
||
|
|
$crate::__private::concat!($($crate::__private::stringify!($tt)),+)
|
||
|
|
};
|
||
|
|
}
|