1 //! [![github]](https://github.com/dtolnay/paste) [![crates-io]](https://crates.io/crates/paste) [![docs-rs]](https://docs.rs/paste)
2 //!
3 //! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
4 //! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
5 //! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
6 //!
7 //! <br>
8 //!
9 //! The nightly-only [`concat_idents!`] macro in the Rust standard library is
10 //! notoriously underpowered in that its concatenated identifiers can only refer to
11 //! existing items, they can never be used to define something new.
12 //!
13 //! [`concat_idents!`]: https://doc.rust-lang.org/std/macro.concat_idents.html
14 //!
15 //! This crate provides a flexible way to paste together identifiers in a macro,
16 //! including using pasted identifiers to define new items.
17 //!
18 //! This approach works with any Rust compiler 1.31+.
19 //!
20 //! <br>
21 //!
22 //! # Pasting identifiers
23 //!
24 //! Within the `paste!` macro, identifiers inside `[<`...`>]` are pasted
25 //! together to form a single identifier.
26 //!
27 //! ```
28 //! use paste::paste;
29 //!
30 //! paste! {
31 //!     // Defines a const called `QRST`.
32 //!     const [<Q R S T>]: &str = "success!";
33 //! }
34 //!
35 //! fn main() {
36 //!     assert_eq!(
37 //!         paste! { [<Q R S T>].len() },
38 //!         8,
39 //!     );
40 //! }
41 //! ```
42 //!
43 //! <br><br>
44 //!
45 //! # More elaborate example
46 //!
47 //! The next example shows a macro that generates accessor methods for some
48 //! struct fields. It demonstrates how you might find it useful to bundle a
49 //! paste invocation inside of a macro\_rules macro.
50 //!
51 //! ```
52 //! use paste::paste;
53 //!
54 //! macro_rules! make_a_struct_and_getters {
55 //!     ($name:ident { $($field:ident),* }) => {
56 //!         // Define a struct. This expands to:
57 //!         //
58 //!         //     pub struct S {
59 //!         //         a: String,
60 //!         //         b: String,
61 //!         //         c: String,
62 //!         //     }
63 //!         pub struct $name {
64 //!             $(
65 //!                 $field: String,
66 //!             )*
67 //!         }
68 //!
69 //!         // Build an impl block with getters. This expands to:
70 //!         //
71 //!         //     impl S {
72 //!         //         pub fn get_a(&self) -> &str { &self.a }
73 //!         //         pub fn get_b(&self) -> &str { &self.b }
74 //!         //         pub fn get_c(&self) -> &str { &self.c }
75 //!         //     }
76 //!         paste! {
77 //!             impl $name {
78 //!                 $(
79 //!                     pub fn [<get_ $field>](&self) -> &str {
80 //!                         &self.$field
81 //!                     }
82 //!                 )*
83 //!             }
84 //!         }
85 //!     }
86 //! }
87 //!
88 //! make_a_struct_and_getters!(S { a, b, c });
89 //!
90 //! fn call_some_getters(s: &S) -> bool {
91 //!     s.get_a() == s.get_b() && s.get_c().is_empty()
92 //! }
93 //! #
94 //! # fn main() {}
95 //! ```
96 //!
97 //! <br><br>
98 //!
99 //! # Case conversion
100 //!
101 //! Use `$var:lower` or `$var:upper` in the segment list to convert an
102 //! interpolated segment to lower- or uppercase as part of the paste. For
103 //! example, `[<ld_ $reg:lower _expr>]` would paste to `ld_bc_expr` if invoked
104 //! with $reg=`Bc`.
105 //!
106 //! Use `$var:snake` to convert CamelCase input to snake\_case.
107 //! Use `$var:camel` to convert snake\_case to CamelCase.
108 //! These compose, so for example `$var:snake:upper` would give you SCREAMING\_CASE.
109 //!
110 //! The precise Unicode conversions are as defined by [`str::to_lowercase`] and
111 //! [`str::to_uppercase`].
112 //!
113 //! [`str::to_lowercase`]: https://doc.rust-lang.org/std/primitive.str.html#method.to_lowercase
114 //! [`str::to_uppercase`]: https://doc.rust-lang.org/std/primitive.str.html#method.to_uppercase
115 //!
116 //! <br>
117 //!
118 //! # Pasting documentation strings
119 //!
120 //! Within the `paste!` macro, arguments to a #\[doc ...\] attribute are
121 //! implicitly concatenated together to form a coherent documentation string.
122 //!
123 //! ```
124 //! use paste::paste;
125 //!
126 //! macro_rules! method_new {
127 //!     ($ret:ident) => {
128 //!         paste! {
129 //!             #[doc = "Create a new `" $ret "` object."]
130 //!             pub fn new() -> $ret { todo!() }
131 //!         }
132 //!     };
133 //! }
134 //!
135 //! pub struct Paste {}
136 //!
137 //! method_new!(Paste);  // expands to #[doc = "Create a new `Paste` object"]
138 //! ```
139 
140 #![allow(
141     clippy::doc_markdown,
142     clippy::module_name_repetitions,
143     clippy::needless_doctest_main,
144     clippy::too_many_lines
145 )]
146 
147 extern crate proc_macro;
148 
149 mod attr;
150 mod error;
151 mod segment;
152 
153 use crate::attr::expand_attr;
154 use crate::error::{Error, Result};
155 use crate::segment::Segment;
156 use proc_macro::{Delimiter, Group, Ident, Punct, Spacing, Span, TokenStream, TokenTree};
157 use std::iter;
158 use std::panic;
159 
160 #[proc_macro]
paste(input: TokenStream) -> TokenStream161 pub fn paste(input: TokenStream) -> TokenStream {
162     let mut contains_paste = false;
163     let flatten_single_interpolation = true;
164     match expand(input, &mut contains_paste, flatten_single_interpolation) {
165         Ok(expanded) => expanded,
166         Err(err) => err.to_compile_error(),
167     }
168 }
169 
170 #[doc(hidden)]
171 #[proc_macro]
item(input: TokenStream) -> TokenStream172 pub fn item(input: TokenStream) -> TokenStream {
173     paste(input)
174 }
175 
176 #[doc(hidden)]
177 #[proc_macro]
expr(input: TokenStream) -> TokenStream178 pub fn expr(input: TokenStream) -> TokenStream {
179     paste(input)
180 }
181 
expand( input: TokenStream, contains_paste: &mut bool, flatten_single_interpolation: bool, ) -> Result<TokenStream>182 fn expand(
183     input: TokenStream,
184     contains_paste: &mut bool,
185     flatten_single_interpolation: bool,
186 ) -> Result<TokenStream> {
187     let mut expanded = TokenStream::new();
188     let mut lookbehind = Lookbehind::Other;
189     let mut prev_none_group = None::<Group>;
190     let mut tokens = input.into_iter().peekable();
191     loop {
192         let token = tokens.next();
193         if let Some(group) = prev_none_group.take() {
194             if match (&token, tokens.peek()) {
195                 (Some(TokenTree::Punct(fst)), Some(TokenTree::Punct(snd))) => {
196                     fst.as_char() == ':' && snd.as_char() == ':' && fst.spacing() == Spacing::Joint
197                 }
198                 _ => false,
199             } {
200                 expanded.extend(group.stream());
201                 *contains_paste = true;
202             } else {
203                 expanded.extend(iter::once(TokenTree::Group(group)));
204             }
205         }
206         match token {
207             Some(TokenTree::Group(group)) => {
208                 let delimiter = group.delimiter();
209                 let content = group.stream();
210                 let span = group.span();
211                 if delimiter == Delimiter::Bracket && is_paste_operation(&content) {
212                     let segments = parse_bracket_as_segments(content, span)?;
213                     let pasted = segment::paste(&segments)?;
214                     let tokens = pasted_to_tokens(pasted, span)?;
215                     expanded.extend(tokens);
216                     *contains_paste = true;
217                 } else if flatten_single_interpolation
218                     && delimiter == Delimiter::None
219                     && is_single_interpolation_group(&content)
220                 {
221                     expanded.extend(content);
222                     *contains_paste = true;
223                 } else {
224                     let mut group_contains_paste = false;
225                     let is_attribute = delimiter == Delimiter::Bracket
226                         && (lookbehind == Lookbehind::Pound || lookbehind == Lookbehind::PoundBang);
227                     let mut nested = expand(
228                         content,
229                         &mut group_contains_paste,
230                         flatten_single_interpolation && !is_attribute,
231                     )?;
232                     if is_attribute {
233                         nested = expand_attr(nested, span, &mut group_contains_paste)?
234                     }
235                     let group = if group_contains_paste {
236                         let mut group = Group::new(delimiter, nested);
237                         group.set_span(span);
238                         *contains_paste = true;
239                         group
240                     } else {
241                         group.clone()
242                     };
243                     if delimiter != Delimiter::None {
244                         expanded.extend(iter::once(TokenTree::Group(group)));
245                     } else if lookbehind == Lookbehind::DoubleColon {
246                         expanded.extend(group.stream());
247                         *contains_paste = true;
248                     } else {
249                         prev_none_group = Some(group);
250                     }
251                 }
252                 lookbehind = Lookbehind::Other;
253             }
254             Some(TokenTree::Punct(punct)) => {
255                 lookbehind = match punct.as_char() {
256                     ':' if lookbehind == Lookbehind::JointColon => Lookbehind::DoubleColon,
257                     ':' if punct.spacing() == Spacing::Joint => Lookbehind::JointColon,
258                     '#' => Lookbehind::Pound,
259                     '!' if lookbehind == Lookbehind::Pound => Lookbehind::PoundBang,
260                     _ => Lookbehind::Other,
261                 };
262                 expanded.extend(iter::once(TokenTree::Punct(punct)));
263             }
264             Some(other) => {
265                 lookbehind = Lookbehind::Other;
266                 expanded.extend(iter::once(other));
267             }
268             None => return Ok(expanded),
269         }
270     }
271 }
272 
273 #[derive(PartialEq)]
274 enum Lookbehind {
275     JointColon,
276     DoubleColon,
277     Pound,
278     PoundBang,
279     Other,
280 }
281 
282 // https://github.com/dtolnay/paste/issues/26
is_single_interpolation_group(input: &TokenStream) -> bool283 fn is_single_interpolation_group(input: &TokenStream) -> bool {
284     #[derive(PartialEq)]
285     enum State {
286         Init,
287         Ident,
288         Literal,
289         Apostrophe,
290         Lifetime,
291         Colon1,
292         Colon2,
293     }
294 
295     let mut state = State::Init;
296     for tt in input.clone() {
297         state = match (state, &tt) {
298             (State::Init, TokenTree::Ident(_)) => State::Ident,
299             (State::Init, TokenTree::Literal(_)) => State::Literal,
300             (State::Init, TokenTree::Punct(punct)) if punct.as_char() == '\'' => State::Apostrophe,
301             (State::Apostrophe, TokenTree::Ident(_)) => State::Lifetime,
302             (State::Ident, TokenTree::Punct(punct))
303                 if punct.as_char() == ':' && punct.spacing() == Spacing::Joint =>
304             {
305                 State::Colon1
306             }
307             (State::Colon1, TokenTree::Punct(punct))
308                 if punct.as_char() == ':' && punct.spacing() == Spacing::Alone =>
309             {
310                 State::Colon2
311             }
312             (State::Colon2, TokenTree::Ident(_)) => State::Ident,
313             _ => return false,
314         };
315     }
316 
317     state == State::Ident || state == State::Literal || state == State::Lifetime
318 }
319 
is_paste_operation(input: &TokenStream) -> bool320 fn is_paste_operation(input: &TokenStream) -> bool {
321     let mut tokens = input.clone().into_iter();
322 
323     match &tokens.next() {
324         Some(TokenTree::Punct(punct)) if punct.as_char() == '<' => {}
325         _ => return false,
326     }
327 
328     let mut has_token = false;
329     loop {
330         match &tokens.next() {
331             Some(TokenTree::Punct(punct)) if punct.as_char() == '>' => {
332                 return has_token && tokens.next().is_none();
333             }
334             Some(_) => has_token = true,
335             None => return false,
336         }
337     }
338 }
339 
parse_bracket_as_segments(input: TokenStream, scope: Span) -> Result<Vec<Segment>>340 fn parse_bracket_as_segments(input: TokenStream, scope: Span) -> Result<Vec<Segment>> {
341     let mut tokens = input.into_iter().peekable();
342 
343     match &tokens.next() {
344         Some(TokenTree::Punct(punct)) if punct.as_char() == '<' => {}
345         Some(wrong) => return Err(Error::new(wrong.span(), "expected `<`")),
346         None => return Err(Error::new(scope, "expected `[< ... >]`")),
347     }
348 
349     let mut segments = segment::parse(&mut tokens)?;
350 
351     match &tokens.next() {
352         Some(TokenTree::Punct(punct)) if punct.as_char() == '>' => {}
353         Some(wrong) => return Err(Error::new(wrong.span(), "expected `>`")),
354         None => return Err(Error::new(scope, "expected `[< ... >]`")),
355     }
356 
357     if let Some(unexpected) = tokens.next() {
358         return Err(Error::new(
359             unexpected.span(),
360             "unexpected input, expected `[< ... >]`",
361         ));
362     }
363 
364     for segment in &mut segments {
365         if let Segment::String(string) = segment {
366             if string.value.contains(&['#', '\\', '.', '+'][..]) {
367                 return Err(Error::new(string.span, "unsupported literal"));
368             }
369             string.value = string
370                 .value
371                 .replace('"', "")
372                 .replace('\'', "")
373                 .replace('-', "_");
374         }
375     }
376 
377     Ok(segments)
378 }
379 
pasted_to_tokens(mut pasted: String, span: Span) -> Result<TokenStream>380 fn pasted_to_tokens(mut pasted: String, span: Span) -> Result<TokenStream> {
381     let mut tokens = TokenStream::new();
382 
383     if pasted.starts_with('\'') {
384         let mut apostrophe = TokenTree::Punct(Punct::new('\'', Spacing::Joint));
385         apostrophe.set_span(span);
386         tokens.extend(iter::once(apostrophe));
387         pasted.remove(0);
388     }
389 
390     let ident = match panic::catch_unwind(|| Ident::new(&pasted, span)) {
391         Ok(ident) => TokenTree::Ident(ident),
392         Err(_) => {
393             return Err(Error::new(
394                 span,
395                 &format!("`{:?}` is not a valid identifier", pasted),
396             ));
397         }
398     };
399 
400     tokens.extend(iter::once(ident));
401     Ok(tokens)
402 }
403