::defile

Helper proc-macro to "ungroup" a captured metavariable (thus potentially breaking their hygiene, hence the name).

Repository Latest version Documentation MSRV unsafe forbidden License CI no_std

This is useful when using helper macro_rules macros, that need to parse using some special rule (e.g. :expr, :path, :pat), but that later want to further inspect the captured variable.

This is not something a macro_rules! can do on its own, since such so-called metavariables are seen as an opaque single token (:tt) (the sequence of tokens captured in the metavariable have been grouped (≈ parenthesized) but using invisible parenthesis.

Example

``rust macro_rules! check_expr { ( 42 ) => ({ println!("Got42`!"); });

(
    $($tt:tt)*
) => ({
    println!("Did not get `42`. Instead, got the following tokens:\n[");
    $(
        println!("    `{}`,", stringify!($tt));
    )*
    println!("]");
});

}

macrorules! checkallexprs {( $( $expr:expr // use :expr to be able to use , as a delimiter ),* $(,)? ) => ( fn main () { $( println!("vvvvvvvvvvvvv"); checkexpr!($expr); println!("^^^^^^^^^^^^^\n"); )* } )}

checkallexprs!(42, 1 + 1); ```

outputs:

``text vvvvvvvvvvvvv Did not get42. Instead, got the following tokens: [ 42`, ] ^^^^^^^^^^^^^

vvvvvvvvvvvvv Did not get 42. Instead, got the following tokens: [ 1 + 1, ] ^^^^^^^^^^^^^ ```

But by doing:

macro_rules! check_all_exprs {(
    $(
        $expr:expr // use :expr to be able to use `,` as a delimiter
    ),* $(,)?
) => (::defile::defile! { // 👈
    fn main () {
        $(
            println!("vvvvvvvvvvvvv");
            check_expr!(@$expr);
//                      👆
            println!("^^^^^^^^^^^^^\n");
        )*
    }
})}

we do get:

``text vvvvvvvvvvvvv Got42`! ^^^^^^^^^^^^^

vvvvvvvvvvvvv Did not get 42. Instead, got the following tokens: [ 1, +, 1, ] ^^^^^^^^^^^^^ ```

Caveats

Currently (1.45.0), there are several bugs regarding the interaction between macro_rules! macros and procedural macros, that may lead to defile! and any other helper procedural macro to split groups that are not @-prefixed.

Hopefully those bugs are solved, making the actual implementation of defile! meaningful.