1/// Define a type that supports parsing and printing a given identifier as if it
2/// were a keyword.
3///
4/// # Usage
5///
6/// As a convention, it is recommended that this macro be invoked within a
7/// module called `kw` or `keyword` and that the resulting parser be invoked
8/// with a `kw::` or `keyword::` prefix.
9///
10/// ```
11/// mod kw {
12/// syn::custom_keyword!(whatever);
13/// }
14/// ```
15///
16/// The generated syntax tree node supports the following operations just like
17/// any built-in keyword token.
18///
19/// - [Peeking] — `input.peek(kw::whatever)`
20///
21/// - [Parsing] — `input.parse::<kw::whatever>()?`
22///
23/// - [Printing] — `quote!( ... #whatever_token ... )`
24///
25/// - Construction from a [`Span`] — `let whatever_token = kw::whatever(sp)`
26///
27/// - Field access to its span — `let sp = whatever_token.span`
28///
29/// [Peeking]: crate::parse::ParseBuffer::peek
30/// [Parsing]: crate::parse::ParseBuffer::parse
31/// [Printing]: quote::ToTokens
32/// [`Span`]: proc_macro2::Span
33///
34/// # Example
35///
36/// This example parses input that looks like `bool = true` or `str = "value"`.
37/// The key must be either the identifier `bool` or the identifier `str`. If
38/// `bool`, the value may be either `true` or `false`. If `str`, the value may
39/// be any string literal.
40///
41/// The symbols `bool` and `str` are not reserved keywords in Rust so these are
42/// not considered keywords in the `syn::token` module. Like any other
43/// identifier that is not a keyword, these can be declared as custom keywords
44/// by crates that need to use them as such.
45///
46/// ```
47/// use syn::{LitBool, LitStr, Result, Token};
48/// use syn::parse::{Parse, ParseStream};
49///
50/// mod kw {
51/// syn::custom_keyword!(bool);
52/// syn::custom_keyword!(str);
53/// }
54///
55/// enum Argument {
56/// Bool {
57/// bool_token: kw::bool,
58/// eq_token: Token![=],
59/// value: LitBool,
60/// },
61/// Str {
62/// str_token: kw::str,
63/// eq_token: Token![=],
64/// value: LitStr,
65/// },
66/// }
67///
68/// impl Parse for Argument {
69/// fn parse(input: ParseStream) -> Result<Self> {
70/// let lookahead = input.lookahead1();
71/// if lookahead.peek(kw::bool) {
72/// Ok(Argument::Bool {
73/// bool_token: input.parse::<kw::bool>()?,
74/// eq_token: input.parse()?,
75/// value: input.parse()?,
76/// })
77/// } else if lookahead.peek(kw::str) {
78/// Ok(Argument::Str {
79/// str_token: input.parse::<kw::str>()?,
80/// eq_token: input.parse()?,
81/// value: input.parse()?,
82/// })
83/// } else {
84/// Err(lookahead.error())
85/// }
86/// }
87/// }
88/// ```
89#[macro_export]
90macro_rules! custom_keyword {
91 ($ident:ident) => {
92 #[allow(non_camel_case_types)]
93 pub struct $ident {
94 pub span: $crate::__private::Span,
95 }
96
97 #[doc(hidden)]
98 #[allow(dead_code, non_snake_case)]
99 pub fn $ident<__S: $crate::__private::IntoSpans<$crate::__private::Span>>(
100 span: __S,
101 ) -> $ident {
102 $ident {
103 span: $crate::__private::IntoSpans::into_spans(span),
104 }
105 }
106
107 const _: () = {
108 impl $crate::__private::Default for $ident {
109 fn default() -> Self {
110 $ident {
111 span: $crate::__private::Span::call_site(),
112 }
113 }
114 }
115
116 $crate::impl_parse_for_custom_keyword!($ident);
117 $crate::impl_to_tokens_for_custom_keyword!($ident);
118 $crate::impl_clone_for_custom_keyword!($ident);
119 $crate::impl_extra_traits_for_custom_keyword!($ident);
120 };
121 };
122}
123
124// Not public API.
125#[cfg(feature = "parsing")]
126#[doc(hidden)]
127#[macro_export]
128macro_rules! impl_parse_for_custom_keyword {
129 ($ident:ident) => {
130 // For peek.
131 impl $crate::token::CustomToken for $ident {
132 fn peek(cursor: $crate::buffer::Cursor) -> $crate::__private::bool {
133 if let $crate::__private::Some((ident, _rest)) = cursor.ident() {
134 ident == $crate::__private::stringify!($ident)
135 } else {
136 false
137 }
138 }
139
140 fn display() -> &'static $crate::__private::str {
141 $crate::__private::concat!("`", $crate::__private::stringify!($ident), "`")
142 }
143 }
144
145 impl $crate::parse::Parse for $ident {
146 fn parse(input: $crate::parse::ParseStream) -> $crate::parse::Result<$ident> {
147 input.step(|cursor| {
148 if let $crate::__private::Some((ident, rest)) = cursor.ident() {
149 if ident == $crate::__private::stringify!($ident) {
150 return $crate::__private::Ok(($ident { span: ident.span() }, rest));
151 }
152 }
153 $crate::__private::Err(cursor.error($crate::__private::concat!(
154 "expected `",
155 $crate::__private::stringify!($ident),
156 "`",
157 )))
158 })
159 }
160 }
161 };
162}
163
164// Not public API.
165#[cfg(not(feature = "parsing"))]
166#[doc(hidden)]
167#[macro_export]
168macro_rules! impl_parse_for_custom_keyword {
169 ($ident:ident) => {};
170}
171
172// Not public API.
173#[cfg(feature = "printing")]
174#[doc(hidden)]
175#[macro_export]
176macro_rules! impl_to_tokens_for_custom_keyword {
177 ($ident:ident) => {
178 impl $crate::__private::ToTokens for $ident {
179 fn to_tokens(&self, tokens: &mut $crate::__private::TokenStream2) {
180 let ident = $crate::Ident::new($crate::__private::stringify!($ident), self.span);
181 $crate::__private::TokenStreamExt::append(tokens, ident);
182 }
183 }
184 };
185}
186
187// Not public API.
188#[cfg(not(feature = "printing"))]
189#[doc(hidden)]
190#[macro_export]
191macro_rules! impl_to_tokens_for_custom_keyword {
192 ($ident:ident) => {};
193}
194
195// Not public API.
196#[cfg(feature = "clone-impls")]
197#[doc(hidden)]
198#[macro_export]
199macro_rules! impl_clone_for_custom_keyword {
200 ($ident:ident) => {
201 impl $crate::__private::Copy for $ident {}
202
203 #[allow(clippy::expl_impl_clone_on_copy)]
204 impl $crate::__private::Clone for $ident {
205 fn clone(&self) -> Self {
206 *self
207 }
208 }
209 };
210}
211
212// Not public API.
213#[cfg(not(feature = "clone-impls"))]
214#[doc(hidden)]
215#[macro_export]
216macro_rules! impl_clone_for_custom_keyword {
217 ($ident:ident) => {};
218}
219
220// Not public API.
221#[cfg(feature = "extra-traits")]
222#[doc(hidden)]
223#[macro_export]
224macro_rules! impl_extra_traits_for_custom_keyword {
225 ($ident:ident) => {
226 impl $crate::__private::Debug for $ident {
227 fn fmt(&self, f: &mut $crate::__private::Formatter) -> $crate::__private::FmtResult {
228 $crate::__private::Formatter::write_str(
229 f,
230 $crate::__private::concat!(
231 "Keyword [",
232 $crate::__private::stringify!($ident),
233 "]",
234 ),
235 )
236 }
237 }
238
239 impl $crate::__private::Eq for $ident {}
240
241 impl $crate::__private::PartialEq for $ident {
242 fn eq(&self, _other: &Self) -> $crate::__private::bool {
243 true
244 }
245 }
246
247 impl $crate::__private::Hash for $ident {
248 fn hash<__H: $crate::__private::Hasher>(&self, _state: &mut __H) {}
249 }
250 };
251}
252
253// Not public API.
254#[cfg(not(feature = "extra-traits"))]
255#[doc(hidden)]
256#[macro_export]
257macro_rules! impl_extra_traits_for_custom_keyword {
258 ($ident:ident) => {};
259}
260