tokenstream2_tmpl/lib.rs
1//! [![github]](https://github.com/chesedo/tokenstream2-tmpl) [![crates-io]](https://crates.io/crates/tokenstream2-tmpl) [![docs-rs]](https://docs.rs/tokenstream2-tmpl) [![workflow]](https://github.com/chesedo/tokenstream2-tmpl/actions?query=workflow%3ARust)
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//! [workflow]: https://img.shields.io/github/workflow/status/chesedo/tokenstream2-tmpl/Rust?color=green&label=&labelColor=555555&logo=github%20actions&logoColor=white&style=for-the-badge
7//!
8//! This crate is meant to be a complement to [quote]. Where as [quote] does quasi-quote interpolations at
9//! compile-time, this crate does them at run-time. This is handy for macros receiving templates from client code with
10//! markers to be replaced when the macro is run.
11//!
12//! [quote]: https://github.com/dtolnay/quote
13//!
14//! # Examples
15//! ```
16//! use proc_macro2::TokenStream;
17//! use tokenstream2_tmpl::interpolate;
18//! use quote::ToTokens;
19//! use std::collections::HashMap;
20//! use syn::{Ident, parse_str};
21//!
22//! let input: TokenStream = parse_str("let NAME: int = 5;")?;
23//! let expected: TokenStream = parse_str("let age: int = 5;")?;
24//!
25//! let mut replacements: HashMap<&str, &dyn ToTokens> = HashMap::new();
26//! let ident = parse_str::<Ident>("age")?;
27//! replacements.insert("NAME", &ident);
28//!
29//! let output = interpolate(input, &replacements);
30//! assert_eq!(
31//! format!("{}", output),
32//! format!("{}", expected)
33//! );
34//!
35//! # Ok::<(), Box<dyn std::error::Error>>(())
36//! ```
37//!
38//! Here `input` might be some input to a macro that functions as a template. [quote] would have tried to expand `NAME`
39//! at the macro's compile-time. [tokenstream2-tmpl] will expand it at the macro's run-time.
40//!
41//! [tokenstream2-tmpl]: https://gitlab.com/chesedo/tokenstream2-tmpl
42//!
43//! ```
44//! extern crate proc_macro;
45//! use proc_macro2::TokenStream;
46//! use std::collections::HashMap;
47//! use syn::{Ident, parse::{Parse, ParseStream, Result}, parse_macro_input, punctuated::Punctuated, Token};
48//! use tokenstream2_tmpl::{Interpolate, interpolate};
49//! use quote::ToTokens;
50//!
51//! /// Create a token for macro using [syn](syn)
52//! /// Type that holds a key and the value it maps to.
53//! /// An acceptable stream will have the following form:
54//! /// ```text
55//! /// key => value
56//! /// ```
57//! struct KeyValue {
58//! pub key: Ident,
59//! pub arrow_token: Token![=>],
60//! pub value: Ident,
61//! }
62//!
63//! /// Make KeyValue parsable from a token stream
64//! impl Parse for KeyValue {
65//! fn parse(input: ParseStream) -> Result<Self> {
66//! Ok(KeyValue {
67//! key: input.parse()?,
68//! arrow_token: input.parse()?,
69//! value: input.parse()?,
70//! })
71//! }
72//! }
73//!
74//! /// Make KeyValue interpolatible
75//! impl Interpolate for KeyValue {
76//! fn interpolate(&self, stream: TokenStream) -> TokenStream {
77//! let mut replacements: HashMap<_, &dyn ToTokens> = HashMap::new();
78//!
79//! // Replace each "KEY" with the key
80//! replacements.insert("KEY", &self.key);
81//!
82//! // Replace each "VALUE" with the value
83//! replacements.insert("VALUE", &self.value);
84//!
85//! interpolate(stream, &replacements)
86//! }
87//! }
88//!
89//! /// Macro to take a list of key-values with a template to expand each key-value
90//! # const IGNORE: &str = stringify! {
91//! #[proc_macro_attribute]
92//! # };
93//! pub fn map(tokens: proc_macro::TokenStream, template: proc_macro::TokenStream) -> proc_macro::TokenStream {
94//! // Parse a comma separated list of key-values
95//! let maps =
96//! parse_macro_input!(tokens with Punctuated::<KeyValue, Token![,]>::parse_terminated);
97//!
98//! maps.interpolate(template.into()).into()
99//! }
100//!
101//! pub fn main() {
102//! # const IGNORE: &str = stringify! {
103//! #[map(
104//! usize => 10,
105//! isize => -2,
106//! bool => false,
107//! )]
108//! let _: KEY = VALUE;
109//! # };
110//! // Output:
111//! // let _: usize = 10;
112//! // let _: isize = -2;
113//! // let _: bool = false;
114//! }
115//! ```
116
117use proc_macro2::{Group, TokenStream, TokenTree};
118use quote::{ToTokens, TokenStreamExt};
119use std::collections::HashMap;
120use syn::punctuated::Punctuated;
121
122/// Trait for tokens that can replace interpolation markers
123pub trait Interpolate {
124 /// Take a token stream and replace interpolation markers with their actual values into a new stream
125 /// using [interpolate](interpolate)
126 fn interpolate(&self, stream: TokenStream) -> TokenStream;
127}
128
129/// Make a Punctuated list interpolatible if it holds interpolatible types
130impl<T: Interpolate, P> Interpolate for Punctuated<T, P> {
131 fn interpolate(&self, stream: TokenStream) -> TokenStream {
132 self.iter()
133 .fold(TokenStream::new(), |mut implementations, t| {
134 implementations.extend(t.interpolate(stream.clone()));
135 implementations
136 })
137 }
138}
139
140/// Replace the interpolation markers in a token stream with a specific text.
141/// See this [crate's](crate) documentation for an example on how to use this.
142pub fn interpolate(
143 stream: TokenStream,
144 replacements: &HashMap<&str, &dyn ToTokens>,
145) -> TokenStream {
146 let mut new = TokenStream::new();
147
148 // Loop over each token in the stream
149 // `Literal`, `Punct`, and `Group` are kept as is
150 for token in stream.into_iter() {
151 match token {
152 TokenTree::Literal(literal) => new.append(literal),
153 TokenTree::Punct(punct) => new.append(punct),
154 TokenTree::Group(group) => {
155 // Recursively interpolate the stream in group
156 let mut new_group =
157 Group::new(group.delimiter(), interpolate(group.stream(), replacements));
158 new_group.set_span(group.span());
159
160 new.append(new_group);
161 }
162 TokenTree::Ident(ident) => {
163 let ident_str: &str = &ident.to_string();
164
165 // Check if identifier is in the replacement set
166 if let Some(value) = replacements.get(ident_str) {
167 // Replace with replacement value
168 value.to_tokens(&mut new);
169
170 continue;
171 }
172
173 // Identifier did not match, so copy as is
174 new.append(ident);
175 }
176 }
177 }
178
179 new
180}
181
182#[cfg(test)]
183mod tests {
184 use super::*;
185 use pretty_assertions::assert_eq;
186 use quote::quote;
187 use syn::{parse_str, Ident, Token, Type};
188
189 type Result = std::result::Result<(), Box<dyn std::error::Error>>;
190
191 #[test]
192 fn complete_replacements() -> Result {
193 let input = quote! {
194 let VAR: TRAIT = if true {
195 CONCRETE{}
196 } else {
197 Alternative{}
198 }
199 };
200
201 let expected = quote! {
202 let var: abstract_type = if true {
203 concrete{}
204 } else {
205 Alternative{}
206 }
207 };
208
209 let mut r: HashMap<&str, &dyn ToTokens> = HashMap::new();
210 let v: Ident = parse_str("var")?;
211 let a: Type = parse_str("abstract_type")?;
212 let c: Type = parse_str("concrete")?;
213
214 r.insert("VAR", &v);
215 r.insert("TRAIT", &a);
216 r.insert("CONCRETE", &c);
217
218 assert_eq!(
219 format!("{}", &interpolate(input, &r)),
220 format!("{}", expected)
221 );
222
223 Ok(())
224 }
225
226 /// Partial replacements should preverse the uninterpolated identifiers
227 #[test]
228 fn partial_replacements() -> Result {
229 let input: TokenStream = parse_str("let a: TRAIT = OTHER;")?;
230 let expected: TokenStream = parse_str("let a: Display = OTHER;")?;
231
232 let mut r: HashMap<&str, &dyn ToTokens> = HashMap::new();
233 let t: Type = parse_str("Display")?;
234 r.insert("TRAIT", &t);
235
236 assert_eq!(
237 format!("{}", interpolate(input, &r)),
238 format!("{}", expected)
239 );
240
241 Ok(())
242 }
243
244 /// Test the interpolation of Punctuated items
245 #[test]
246 fn interpolate_on_punctuated() -> Result {
247 #[allow(dead_code)]
248 pub struct TraitSpecifier {
249 pub abstract_trait: Type,
250 pub arrow_token: Token![=>],
251 pub concrete: Type,
252 }
253
254 /// Make TraitSpecifier interpolatible
255 impl Interpolate for TraitSpecifier {
256 fn interpolate(&self, stream: TokenStream) -> TokenStream {
257 let mut replacements: HashMap<_, &dyn ToTokens> = HashMap::new();
258
259 // Replace each "TRAIT" with the absract trait
260 replacements.insert("TRAIT", &self.abstract_trait);
261
262 // Replace each "CONCRETE" with the concrete type
263 replacements.insert("CONCRETE", &self.concrete);
264
265 interpolate(stream, &replacements)
266 }
267 }
268 let mut traits: Punctuated<TraitSpecifier, Token![,]> = Punctuated::new();
269
270 traits.push(TraitSpecifier {
271 abstract_trait: parse_str("IButton")?,
272 arrow_token: Default::default(),
273 concrete: parse_str("BigButton")?,
274 });
275 traits.push(TraitSpecifier {
276 abstract_trait: parse_str("IWindow")?,
277 arrow_token: Default::default(),
278 concrete: parse_str("MinimalWindow")?,
279 });
280
281 let input = quote! {
282 let _: TRAIT = CONCRETE{};
283 };
284 let expected = quote! {
285 let _: IButton = BigButton{};
286 let _: IWindow = MinimalWindow{};
287 };
288
289 assert_eq!(
290 format!("{}", traits.interpolate(input)),
291 format!("{}", expected)
292 );
293
294 Ok(())
295 }
296}