bon/__/
cfg_eval.rs

1/// This is all a big embarrassing workaround, please don't oversee 😳😳😳.
2///
3/// Anyway, if you are curious what the hell is going on here, then here is
4/// an explanation 😸. So... where to start 🤔. Ah! The problem!
5///
6/// ## The problem
7///
8/// Proc macro attributes (like `#[builder]`) see all the `#[cfg(...)]` and `#[cfg_attr(...)]`
9/// attributes unexpanded. For example, if you write smth like this:
10///
11/// ```
12/// #[bon::builder]
13/// fn func(
14///     #[cfg(windows)]
15///     windows_only_param: u32,
16/// ) {}
17///
18/// ```
19///
20/// then the `#[builder]` macro will see the full `#[cfg(...)]` attribute with
21/// the `windows_only_param` it is attached to verbatim. The `#[cfg(...)]` isn't
22/// removed by the time the `#[builder]`'s macro expansion is invoked.
23///
24/// It is a problem because the `#[builder]` macro needs to know the exact list
25/// of members it has to generate setters for. It doesn't know whether
26/// the `windows` predicate evaluates to `true` or `false`, especially if this was
27/// a more complex predicate. So it can't decide whether to generate a setter for
28/// the `windows_only_param` or not.
29///
30/// ## The solution
31///
32/// This macro allows us to evaluate the `cfg` predicates by using a variation of
33/// [the trick] shared by @recatek.
34///
35/// When the `#[builder]` macro finds any usage of `#[cfg(...)]` or `#[cfg_attr(...)]`
36/// it generates a call to this macro with all `cfg` predicates collected from the
37/// item it was placed on. The `#[builder]` macro deduplicates and sorts the `cfg`
38/// predicates and passes them as `$pred` to this macro.
39///
40/// This macro then dispatches to `__eval_cfg_callback_true` or `__eval_cfg_callback_false`
41/// by defining a conditional `use ...` statement for each predicate and collects the
42/// results of the evaluation in the `$results` list.
43///
44/// For the last call to this macro (when no more `$pred` are left) the macro calls back
45/// to the proc macro attribute that called it with the results of the evaluation and
46/// the original parameters and the item which are passed through via the `$rest` macro variable.
47///
48/// [the trick]: https://users.rust-lang.org/t/supporting-or-evaluating-cfg-in-proc-macro-parameters/93240/2
49#[macro_export]
50#[doc(hidden)]
51macro_rules! __eval_cfg_callback {
52    (
53        { $($results:tt)* }
54        ( $pred_id:ident: $($pred:tt)* )
55        $($rest:tt)*
56    ) => {
57        // The `pred_id` is required to be a unique identifier for the current
58        // predicate evaluation so that we can use it in a `use` statement to define
59        // a new unique name for the macro to call.
60        #[cfg($($pred)*)]
61        #[doc(hidden)]
62        #[allow(deprecated)]
63        use $crate::__eval_cfg_callback_true as $pred_id;
64
65        #[cfg(not($($pred)*))]
66        #[doc(hidden)]
67        #[allow(deprecated)]
68        use $crate::__eval_cfg_callback_false as $pred_id;
69
70        // The trick here is that `$pred_id` now resolves either to
71        // `__eval_cfg_callback_true` or `__eval_cfg_callback_false`
72        // depending on the evaluation of the cfg predicate, so by
73        // invoking it as a macro, that macro internally pushes either
74        // `true` or `false` to the `$results` list.
75        $pred_id! {
76            { $($results)* }
77            $($rest)*
78        }
79    };
80
81    // The terminal case for the recursion when there are no more predicates left.
82    // We have collected all the results of the cfg evaluations and now we can
83    // delegate them to the proc macro attribute that called this macro.
84    (
85        // The results of the cfg evaluation
86        { $($results:tt)* }
87
88        // The proc macro attribute to invoke with the results
89        $final_macro:path,
90
91        // The number of times this macro was called recursively from the proc macro
92        $recursion_counter:literal,
93
94        // Parameters to pass to the proc macro attribute after the cfg results
95        ( $($macro_params:tt)* )
96
97        // The item to attach the proc macro attribute to
98        $($item:tt)*
99    ) => {
100        // The special `__cfgs(...)` prefix is parsed by the proc macro attribute
101        // to get the results of the cfg evaluations.
102        #[$final_macro(__cfgs($recursion_counter, $($results)*) $($macro_params)*)]
103        $($item)*
104    };
105}
106
107/// The `cfg` predicate evaluated to `true`, now push that information into
108/// the `$results` list.
109#[macro_export]
110#[doc(hidden)]
111macro_rules! __eval_cfg_callback_true {
112    (
113        { $($results:tt)* }
114        $($tt:tt)*
115    ) => {
116        $crate::__eval_cfg_callback! {
117            { $($results)* true, }
118            $($tt)*
119        }
120    };
121}
122
123/// The `cfg` predicate evaluated to `false`, now push that information into
124/// the `$results` list.
125#[macro_export]
126#[doc(hidden)]
127macro_rules! __eval_cfg_callback_false {
128    (
129        { $($results:tt)* }
130        $($tt:tt)*
131    ) => {
132        $crate::__eval_cfg_callback! {
133            { $($results)* false, }
134            $($tt)*
135        }
136    };
137}