包阅导读总结
1.
关键词:Rust、Procedural Macros、Derive Macros、Attribute Macros、Functional Macros
2.
总结:本文是 Rust 程序语言中关于程序宏的初学者手册,介绍了宏的作用、类型(包括声明式和程序式,后者又分派生、属性和函数式宏),并通过示例阐述了如何编写不同类型的程序宏,还提到了宏的优缺点等内容。
3.
主要内容:
– Rust 程序宏
– 宏的概述
– 是 Rust 语言的重要组成部分,可在编译时执行代码生成等操作。
– 宏的类型
– 声明式宏:基于语法解析。
– 程序式宏
– 派生宏:用于扩展数据类型功能。
– 属性宏:通常用于函数、代码块等,可转换或添加信息。
– 函数式宏:类似函数的宏。
– 编写程序宏
– 简单和复杂的派生宏示例。
– 简单和复杂的属性宏示例。
– 简单和复杂的函数式宏示例。
– 宏的相关内容
– 前提条件和有用的依赖。
– 宏的优点和缺点。
– 有用的工具和库。
思维导图:
文章地址:https://www.freecodecamp.org/news/procedural-macros-in-rust/
文章来源:freecodecamp.org
作者:Anshul Sanghi
发布时间:2024/7/9 17:23
语言:英文
总字数:12140字
预计阅读时间:49分钟
评分:87分
标签:Rust,手册
以下为原文内容
本内容来源于用户推荐转载,旨在分享知识与观点,如有侵权请联系删除 联系邮箱 media@ilingban.com
In this handbook, you’ll learn about procedural macros in Rust, and what purposes they serve. You’ll also learn how to write your own procedural macros with both hypothetical and real-world examples.
This guide assumes that you’re familiar with Rust and its basic concepts, such as data-types, iterators, and traits. If you need to establish or review your Rust basics, check out this interactive course.
You don’t need any prior knowledge of macros, as this article covers everything from the ground up.
Table of Contents
- What are Macros in Rust?
- Types of Macros in Rust
- Types of Procedural Macros
- Prerequisites
- Helpful Dependencies
- How to Write a Simple Derive Macro
- The
IntoStringHashMap
Derive Macro - How to Declare a Derive Macro
- How to Parse Macro Input
- How to Ensure a Struct Target for the Macro
- How to Build the Output Code
- How to Use Your Derive Macro
- How to Improve Our Implementation
- The
- A More Elaborate Derive macro
- The
DeriveCustomModel
Macro - How to Separate Implementation From Declaration
- How to Parse Derive Macro Arguments
- How to Implement
DeriveCustomModel
- How to Generate Each Custom Model
- How to Use Your
DeriveCustomModal
Macro
- The
- A Simple Attribute Macro
- The
log_duration
Attribute - How to Declare an Attribute Macro
- How to Implement the
log_duration
Attribute Macro - How to Use Your
log_duration
Macro
- The
- A More Elaborate Attribute macro
- The
cached_fn
Attribute - How to Implement the
cached_fn
Attribute Macro cached_fn
Attribute Arguments- How to Use the
cached_fn
Macro
- The
- A Simple Function-like Macro
- The
constant_string
Macro - How to Declare a Function-like Macro
- How to Implement the
constant_string
Macro - How to Use the
constant_string
Macro
- The
- A More Elaborate Function-like Macro
- The
hash_mapify
Macro - How to Implement the
hash_mapify
Macro - How to Parse
hash_mapify
‘s Input - How to Generate Output Code
- How to Convert Custom Data Types To Output Tokens
- How to Use the
hash_mapify
Macro
- The
- Beyond Writing Macros
- Helpful Crates/Tools
- Downsides of Macros
- Debugging (or lack thereof)
- Compile Time Costs
- Lack of auto-complete and code checks
- Where do we draw the line?
- Wrapping Up
- Enjoying my work?
What are Macros in Rust?
Macros are an integral part of the Rust programming language. It doesn’t take long before you start encountering them when first learning the language.
In their simplest form, macros in Rust allow you to execute some code at compile-time. Rust pretty much allows you to do whatever you want when it comes to macros and what you can do with them. The most common use-case of this feature is writing code that generates other code.
Macros are a way to extend functionality of the compiler beyond what’s supported as standard. Whether you want to generate code based on existing code, or you want to transform existing code in some form, macros are your go-to tool.
Here’s how the official Rust book describes it:
The term macro refers to a family of features in Rust.
Fundamentally, macros are a way of writing code that writes other code, which is known as metaprogramming.
Metaprogramming is useful for reducing the amount of code you have to write and maintain, which is also one of the roles of functions. However, macros have some additional powers that functions don’t.
Using macros, you can also dynamically add things that are required to be added at compilation time, which is not possible using functions since they get called at runtime. One such feature, for example, is implementing Traits on types, which is required to be implemented at compilation time.
Another advantage of macros is that they can be very flexible, since they can take a dynamic amount of parameters or inputs unlike a function.
Macros do have their own syntax for both writing and using them, which we’ll explore in detail in the coming sections.
Some examples of how macros are being used really helps convey just how powerful they are:
- The SQLx project uses macros to verify all your SQL queries and statements (as long as you created them using the provided macro) at compile-time by actually executing them against a running instance of DB (yes, at compile time).
- typed_html implements a complete HTML parser with compile-time validation, all while using the familiar JSX syntax.
Types of Macros in Rust
In Rust, there are 2 different types of macros: declarative and procedural.
Declarative macros
Declarative macros work based on syntax parsing. While the official docs define them as allowing you to write syntax extensions, I believe it’s more intuitive to consider them as an advanced version of the match
keyword for the compiler.
You can define one or more patterns to match, and their body should return the output Rust code you’d like the macro to produce.
We’re not going to be talking about them in this article, but if you’d like to learn more, this is a good place to start.
Procedural macros
These macros, in their most basic use cases, execute any Rust code you want at compile time. The only requirement is that they should take Rust code as input, and return Rust code as output.
There’s no special syntax parsing involved for writing these macros (unless you want to do so), which is why they’re personally easier for me to understand and write.
Procedural macros are further divided into 3 categories: derive macros, attribute macros, and functional macros.
Types of Procedural Macros
Derive macros
Derive macros are, generally speaking, applied to data types in Rust. They are a way to extend the type declaration to also automatically “derive” functionality for it.
You can use them to generate “derived” types from a type, or as a way to implement methods on the target data type automatically. This should make sense once you look at the following example below.
Printing non-primitive data types, such as structs, enums or even errors (which are just structs, but let’s assume they’re not), for debugging purposes is a very common feature for any language, not just Rust. In Rust, only primitives implicitly have the ability to be printed in “debug” contexts.
If you think about how everything in Rust is just traits (even basic operations like add and equals), this makes sense. You want to be able to debug print your custom data types, but Rust has no way of saying “please apply this trait to every single data type in the code out there, ever”.
This is where the Debug
derive macro comes in. There’s a standard way of debug-printing each type of data structure in Rust that it uses for its internal types. The Debug
macro allows you to automatically implement the Debug
trait for your custom types, while following the same rules and style guide as the implementation for internal data types.
#[derive(Debug)]pub struct User { username: String, first_name: String, last_name: String,}
The Debug
derive macro will result in the following code (presentational, not exact):
impl core::fmt::Debug for User { fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result { f.debug_struct( "User" ) .field("username", &self.username) .field("first_name", &self.first_name) .field("last_name", &self.last_name) .finish() }}
As you might be able to tell, nobody wants to write this code for all of their custom structs and enums again and again. This simple macro gives you a sense of both the power of macros in Rust, as well as why they’re an essential part of the language itself.
During actual compilation, the same code would give the following as the result:
pub struct User { username: String, first_name: String, last_name: String,}impl core::fmt::Debug for User { fn fmt(&self, f: &mut core::fmt::Formatter) -> ::core::fmt::Result { f.debug_struct( "User" ) .field("username", &self.username) .field("first_name", &self.first_name) .field("last_name", &self.last_name) .finish() }}
Notice how the original type declaration is preserved in the output code. This is one of the main differences between derive macros vs others. Derive macros preserve the input type without modifications. They only add additional code to the output. On the other hand, all the other macros do not behave the same way. They only preserve the target when the output for macro itself includes the target as well.
Attribute macros
Attribute macros, in addition to data types, are usually applied to code blocks such as functions, impl blocks, inline blocks, and so on. They’re usually used to either transform the target code in some way, or annotate it with additional information.
The most common use case for these is to modify a function to add additional functionality or logic to it. For example, you can easily write an attribute macro that:
- Logs all input and output parameters
- Logs the total runtime of the function
- Counts the number of times that function is called
- Adds pre-determined additional fields to any struct
and so on.
All of the things I mentioned above, and much more, combined form the insanely popular and useful instrumentation
macro in Rust provided by the tracing
crate. Of course I’m massively simplifying here, but it’s good enough as an example.
If you’re used to using Clippy, it might have screamed at you a couple of times to add the #[must_use]
attribute to your function or method.
That is an example of macros used to annotate the function with additional information. It tells the compiler to warn the user if the return value from this function call isn’t used. The Result
type is already annotated with #[must_use]
by default, which is how you see the warning Unused Result<...> that must be used
when you don’t use a return value of Result
type.
Attribute macros are also what powers conditional compilation in Rust.
Functional macros
Functional macros are macros disguised as functions. These are the least restrictive type of procedural macros, as they can be used literally anywhere, as long as they output code that’s valid in the context that they’re used in.
These macros aren’t “applied” to anything unlike the 2 others, but rather called just like you’d call a function. As arguments, you can literally pass in anything you want, as long as your macro knows how to parse it. This includes everything all the way from no arguments to valid Rust code to random gibberish that only your macro can make sense of.
They’re in a sense the procedural version of declarative macros. If you need to execute Rust code and also be able to parse custom syntax, functional macros are your go-to tool. They’re also useful if you need macro-like functionality in places where other macros cannot be used.
After that lengthy description of the basic information regarding macros, it’s finally time to dive into actually writing procedural macros.
Prerequisites
There are certain rules around writing your own procedural-macros that you’ll need to follow. These rules apply to all 3 types of procedural macros. They are:
- Procedural macros can only be added to a project that is marked as
proc-macro
inCargo.toml
- Projects marked as such cannot export anything other than procedural macros.
- The macros themselves have to all be declared in the
lib.rs
file.
Let’s begin by setting up our project with this code:
cargo new --bin my-appcd my-appcargo new --lib my-app-macros;
This will create a root project, as well as a sub-project within it that will host our macros. You need some changes in the Cargo.toml
files for both these projects.
First, the Cargo.toml
file for my-app-macros
should have the following contents (notice that you need to declare a lib section that has the proc-macro
property):
[package]name = "my-app-macros"version = "0.1.0"edition = "2021"[lib]name = "my_app_macros"path = "src/lib.rs"proc-macro = true[dependencies]
Next, the Cargo.toml
file for my-app
should have the following contents:
workspace = { members = ["my-app-macros"] }[package]name = "my-app"version = "0.1.0"edition = "2021"resolver = "2"[dependencies]my-app-macros = { path = "./my-app-macros" }
You need to set the dependency resolver version to “2”, and add your macros project as a dependency of the my-app
project.
Helpful dependencies
From the compiler’s point of view, this is how macros work:
- They take a stream of tokens as input (and optionally a stream of tokens as arguments to the macro itself).
- They return a stream of tokens as output.
That’s all that the compiler knows! And as you’ll soon see, it’s enough for the compiler to know that.
This does create a problem though. You need to be able to make sense of this “stream of tokens” in a way where you correctly understand them, whether as Rust code or custom syntax, are able to modify them, and also output them. Doing so manually is no easy task, and for the purposes of this tutorial, it is out of scope.
We can, however, rely on great open source work done by many developers to ease this for us. You need to add a few dependencies to help with this problem:
syn
— A syntax parser for Rust. This helps you to parse the input token stream as Rust AST. AST is a concept that you mostly run into when trying to write your own interpreter or compiler, but a basic understanding is essential for working with macros. Macros, after all, are just extensions that you write for the compiler in a sense. If you’re interested in learning more about what ASTs are, check out this very helpful introduction.quote
— quote is, and this is a huge generalisation, a crate that helps us perform the reverse operation of whatsyn
does. It helps us convert Rust source code into a stream of tokens that we can output from our macro.proc-macro2
— The standard library provides aproc-macro
crate, but the types it provides cannot exist outside of procedural macros.proc-macro2
is a wrapper around the standard library that makes all of the internal types usable outside of the context of macros. This, for example, allows bothsyn
andquote
to not only be used for procedural macros, but in regular Rust code as well, should you ever have such a need. And we will indeed be using that extensively if we ever want to unit test our macros or their expansions.darling
–It facilitates parsing and working with macro arguments, which is otherwise a tedious process due to having to manually parse it from the syntax tree.darling
provides us withserde
-like ability to automatically parse input argument tree into our arguments struct. It also helps us in error handling around invalid arguments, required arguments, and so on.
While these projects are contributed to by many developers, I want to give special thanks to David Tolnay. He’s a legend in the Rust community and is the creator of most of these projects, and many many more open source crates in Rust.
Let’s quickly add these dependencies to our project and start writing our macro:
// my-app-macroscargo add syn quote proc-macro2 darling
How to Write a Simple Derive Macro
You are going to learn how to write a Derive
macro in this section. By this time, you should already be aware of the different types of macros and what they entail, as we talked about them in the above sections.
The IntoStringHashMap
Derive Macro
Let’s say you have an app where you need to be able to convert structs into hash maps, that uses the String
type for both keys and values. This means that it should work with any struct where all of the fields are convertible to String
type using the Into
trait.
How to Declare a Derive Macro
You declare macros by creating a function, and annotating that function using attribute macros that tell the compiler to consider that function as a macro declaration. Since your lib.rs
is empty right now, you also need to declare proc-macro2
as an extern crate:
extern crate proc_macro;use proc_macro::TokenStream;#[proc_macro_derive(IntoStringHashMap)]pub fn derive_into_hash_map(item: TokenStream) -> TokenStream { todo!()}
All we’re doing here is declaring our macro as a derive macro with the identifier IntoStringHashMap
. Note that the function name is not important here. What’s important is the identifier passed to the proc_macro_derive
attribute macro.
Let’s immediately see how you can use this – we’ll come back and finish the implementation later:
use my_app_macros::IntoStringHashMap;#[derive(IntoStringHashMap)]pub struct User { username: String, first_name: String, last_name: String, age: u32,}fn main() {}
You can just use your macro as any other derive macro, using the identifier you declared for it (in this case it was IntoStringHashMap
).
If you try and compile your code at this stage, you should see the following compilation error:
Compiling my-app v0.1.0 error: proc-macro derive panicked --> src/main.rs:3:10 |3 | #[derive(IntoHashMap)] | ^^^^^^^^^^^ | = help: message: not yet implementederror: could not compile `my-app` (bin "my-app") due to 1 previous error
This clearly proves that our macro was executed during the compilation stage, as, if you’re not familiar with the todo!()
macro, panics with help: message: not yet implemented
when executed.
This means that both our macro declaration and its usage works. We can move on to actually implementing this macro now.
How to Parse the Macro’s Input
First, you parse the input token stream as a DeriveInput
using syn
, which is a representation of any target that you can use a derive macro with:
let input = syn::parse_macro_input!(item as syn::DeriveInput);
syn
provides us with the parse_macro_input
macro that uses a somewhat custom syntax as its arguments. You provide it the name of your input variable, the as
keyword, and the data type in syn
that it should parse the input token stream as (in our case, a DeriveInput
).
If you jump into the source code for DeriveInput
, you’ll see that it gives us the following information:
attrs
: Attributes applied to this type, whether other attribute macros declared by us, or the built-in ones such asmust_use
.vis
: The visibility specifier for this type declaration.ident
: The identifier (name) of the type.generics
: Information about the generic parameters this type takes, including lifetimes.data
: An enum that describes whether the target is a struct, an enum, or a union, and also provides us with more information for each of these.
These field names and their types (apart from data field) are pretty standard across targets supported by syn
, such as functions, enums, and so on.
If you further jump into the declaration of the Data
enum, and into DataStruct
in particular, you’ll see that it provides you with a field called fields
. This is a collection of all the fields of this struct and you can use it to iterate over them. This is exactly what we need to build our hash map!
The complete implementation for this macro looks like this:
extern crate proc_macro2;use proc_macro::TokenStream;use quote::quote;use syn::Data;#[proc_macro_derive(IntoHashMap)]pub fn into_hash_map(item: TokenStream) -> TokenStream { let input = syn::parse_macro_input!(item as syn::DeriveInput); let struct_identifier = &input.ident; match &input.data { Data::Struct(syn::DataStruct { fields, .. }) => { let mut implementation = quote!{ let mut hash_map = std::collections::HashMap::<String, String>::new(); }; for field in fields { let identifier = field.ident.as_ref().unwrap(); implementation.extend(quote!{ hash_map.insert(stringify!(#identifier).to_string(), String::from(value.#identifier)); }); } quote! { #[automatically_derived] impl From<#struct_identifier> for std::collections::HashMap<String, String> { fn from(value: #struct_identifier) -> Self { #implementation hash_map } } } } _ => unimplemented!() }.into()}
There’s a lot going on here, so let’s break it down:
How to Ensure a struct
Target for the Macro
let struct_identifier = &input.ident;
: You store the struct identifier into a separate variable, so that you can easily use it later.
match &input.data { Data::struct(syn::DataStruct { fields, .. }) => { ... }, _ => unimplemented!()}
You match over the parsed data field from DeriveInput
. If it is of type DataStruct
(a Rust struct) then continue, else panic, as the macro isn’t implemented for other types.
How to Build the Output Code
Let’s take a look at the match arm implementation when you do have a DataStruct
:
let mut implementation = quote!{ let mut hash_map = std::collections::HashMap::<String, String>::new();};
Here you created a new TokenStream
using quote
. This TokenStream
is different than the one provided by the standard library, so don’t confuse it with that. This needs to be mutable, as we’ll be adding more code to this TokenStream
soon.
TokenStream
is basically the inverse representation of an AST. You provide actual Rust code to the quote
macro, and it gives us the “stream of tokens” as you’ve called it previously for that source code.
This TokenStream
can either be converted to the macro’s output type, or be manipulated using methods provided by quote
such as extend.
Moving on,
for field in fields { let identifier = field.ident.as_ref().unwrap(); implementation.extend(quote!{ hash_map.insert( stringify!(#identifier).to_string(), String::from(value.#identifier) ); });}
You loop over all of the fields. In each iteration, you first create a variable identifier
to hold the name of the field for later use. You then use the extend
method on our previously created TokenStream
to add additional code to it.
The extend
method just takes another TokenStream
, which can easily be generated using quote
macro. For the extension, you simply write code to insert a new entry into the hash_map
that will be created in the macro output.
Let’s have a closer look at that:
hash_map.insert( stringify!(#identifier).to_string(), String::from(value.#identifier));
As you know, the insert method takes a key and a value. You’ve told the compiler that both the key and value are of String
type. stringify
is a built-in macro in the standard library, that converts any Ident
type into its &str
equivalent. You use it here to convert your field identifiers into actual &str
. You then call to_string()
method on it to convert it to the String
type.
But what does the #identifier
represent?
quote
provides you with the ability to use any variables declared outside of the TokenStream
within it using the #
prefix. Think of it as {}
in format args. #identifier
in this case simply gets replaced with the field identifier we declared outside of the extend
call. So you basically call the stringify!()
macro on the field identifier directly.
Similarly, you can access the value of a field using the familiar struct_variable.field_name
syntax, but use the identifier variable instead of the field name instead. This is what you do when you pass the value to your insert statement: String::from(value.#identifier)
.
If you’ve looked at the code closely, you’ll realise where the value
came from, but if not, it’s just what the trait implementation method uses to declare its input argument further down.
Once you’ve built your implementation using the for loop for each field in the struct, you have a TokenStream
which, for representational purposes, contains the following code:
let mut hash_map = std::collections::HashMap::<String, String>::new();hash_map.insert("username".to_string(), String::from(value.username));hash_map.insert("first_name".to_string(), String::from(value.first_name));hash_map.insert("last_name".to_string(), String::from(value.last_name));
Moving on to finally generating the output of our macro, you have:
quote! { impl From<#struct_identifier> for std::collections::HashMap<String, String> { fn from(value: #struct_identifier) -> Self { #implementation hash_map } }}
Here you start by creating another TokenStream
using quote
. You write your From
trait implementation in this block.
The following line again uses the #
prefix syntax that we just looked at to declare that the trait implementation should be for your target struct, based on the identifier for the struct. In this case, this identifier will be replaced with User
if you apply the derive macro to User
struct.
impl From<#struct_identifier> for std::collections::HashMap<String, String> {}
Finally, you have the actual method body:
fn from(value: #struct_identifier) -> Self { #implementation hash_map}
As you can see, you can easily nest TokenStream
s into other TokenStreams
using the same #
syntax that lets you use external variables within the quote
macro.
Here, you declare that your hash map implementation should be inserted as the first few lines of the function. And then you simply return the same hash_map
. This completes your trait implementation.
As the very last step, you call .into()
on the return type of our match
block, which returns the output of quote
macro call. This converts the TokenStream
type used by quote
into the TokenStream
type that comes from the standard library and is expected by the compiler to be returned from a macro.
If it was harder to understand it when I broke it down line by line, you can look at the following complete but commented code in addition:
#[proc_macro_derive(IntoHashMap)]pub fn into_hash_map(item: TokenStream) -> TokenStream { let input = syn::parse_macro_input!(item as syn::DeriveInput); let struct_identifier = &input.ident; match &input.data { Data::Struct(syn::DataStruct { fields, .. }) => { let mut implementation = quote!{ let mut hash_map = std::collections::HashMap::<String, String>::new(); }; for field in fields { let identifier = field.ident.as_ref().unwrap(); implementation.extend(quote!{ hash_map.insert(stringify!(#identifier).to_string(), String::from(value.#identifier)); }); } quote! { impl From<#struct_identifier> for std::collections::HashMap<String, String> { fn from(value: #struct_identifier) -> Self { #implementation hash_map } } } } _ => unimplemented!() }.into()}
And that’s it. You’ve written your very first procedural macro in Rust!
It’s now time to enjoy the fruits of your labour.
How to Use Your Derive Macro
Coming back to your my-app/main.rs
, let’s debug-print the hashmap that you create using the macro you implemented. Your main.rs
should look like this:
use std::collections::HashMap;use my_app_macros::IntoHashMap;#[derive(IntoHashMap)]pub struct User { username: String, first_name: String, last_name: String,}fn main() { let user = User { username: "username".to_string(), first_name: "First".to_string(), last_name: "Last".to_string(), }; let hash_map = HashMap::<String, String>::from(user); dbg!(hash_map);}
If you run this using cargo run
, you should see the following output in your terminal:
[src/main.rs:20:5] hash_map = { "last_name": "Last", "first_name": "First", "username": "username",}
And there you go!
How to Improve Our Implementation
There is a better way to work with iterators and quote
that I skipped over in our original implementation – intentionally so, because it requires us to learn a bit more of the syntax specific to quote
.
Let’s see what it would have looked like with that, before we dive into how it works:
let input = syn::parse_macro_input!(item as syn::DeriveInput); let struct_identifier = &input.ident; match &input.data { Data::Struct(syn::DataStruct { fields, .. }) => { let field_identifiers = fields.iter().map(|item| item.ident.as_ref().unwrap()).collect::<Vec<_>>(); quote! { impl From<#struct_identifier> for std::collections::HashMap<String, String> { fn from(value: #struct_identifier) -> Self { let mut hash_map = std::collections::HashMap::<String, String>::new(); #( hash_map.insert(stringify!(#field_identifiers).to_string(), String::from(value.#field_identifiers)); )* hash_map } } } } _ => unimplemented!() }.into()
That looks so much more concise and easier to understand! Let’s look at the special bit of syntax that makes it possible – in particular, the following line:
#( hash_map.insert(stringify!(#field_identifiers).to_string(), String::from(value.#field_identifiers));)*
Let’s break it down. First, you wrap the entire block in a #()*
and your code goes inside the parentheses. This syntax is what allows you to make use of any iterator inside of the parenthesis, and it will repeat that block of code for all items in the iterator, while replacing the variable with correct item in each iteration.
In this case, you first create a field_identifiers
iterator, that is a collection of all the field identifiers in your target struct. You then write your hash_map
insert statement while using the iterator directly as if it is a single item. The #()*
wrapper converts this into the expected output of multiple lines, one for each item in the iterator.
A More Elaborate Derive Macro
Now that you’re comfortable writing a simple Derive macro, it’s time to move on and create something that will actually be useful in the real world – especially if you’re working with database models.
The DeriveCustomModel
Macro
You’re going to be building a Derive macro that helps you generate derived structs from your original struct. You’re going to be needing this all the time whenever you’re working with databases, and only want to load part of the data.
For example, if you have a User
struct, which has all of the user information, but you only want to load the name information for the User from the database, you’ll need a struct that only contains those fields – unless you want to make all the fields an Option, which isn’t the best idea.
We will also need to add an implementation of From
trait to automatically convert from User
struct to the derived struct. Another thing our macro needs is to be able to derive multiple models from the same target struct.
Let’s start by declaring it in lib.rs
:
#[proc_macro_derive(DeriveCustomModel, attributes(custom_model))]pub fn derive_custom_model(item: TokenStream) -> TokenStream { todo!()}
Most of this syntax should familiar to you by now from our previous example. The only addition we have here, is now we also define attributes(custom_model)
in the call to proc_macro_derive
, which basically tells the compiler to treat any attribute that begins with #[custom_model]
as an argument for this derive macro on that target.
For example, once you’ve defined this, you can apply #[custom_model(name = "SomeName")]
to the target struct, to define that the derived struct should have the name “SomeName”. You need to parse this yourself and handle it too, of course – the definition was only to tell the compiler to pass that through to your macro implementation and not treat it as an unknown attribute.
Let’s also create a new file that will contain the implementation detail of this macro. The macro rule states that it needs to be defined in lib.rs
, and we’ve done that. The implementation itself can live anywhere in the project.
Create a new file custom_model.rs
:
touch src/custom_model.rs
How to Separate the Implementation from the Declaration
Define a function that implements the DeriveCustomModel
macro. We’re also going to add all imports right away to avoid confusion later:
use syn::{ parse_macro_input, Data::Struct, DataStruct, DeriveInput, Field, Fields, Ident, Path,};use darling::util::PathList;use darling::{FromAttributes, FromDeriveInput, FromMeta};use proc_macro::TokenStream;use quote::{quote, ToTokens};pub(crate) fn derive_custom_model_impl(input: TokenStream) -> TokenStream { let original_struct = parse_macro_input!(input as DeriveInput); let DeriveInput { data, ident, .. } = original_struct.clone();}
This is just a Rust function, so there are no special rules here. You can call this from the declaration just like a regular Rust function.
#[proc_macro_derive(DeriveCustomModel, attributes(custom_model))]pub fn derive_custom_model(item: TokenStream) -> TokenStream { custom_model::custom_model_impl(item)}
How to Parse Derive Macro Arguments
To parse the arguments to our derive macro (which are usually provided using attributes applied to either the target or to its fields), we are going to rely on the darling
crate to make it as simple as defining the data type for them.
#[derive(FromDeriveInput, Clone)]#[darling(attributes(custom_model), supports(struct_named))]struct CustomModelArgs { #[darling(default, multiple, rename = "model")] pub models: Vec<CustomModel>,}
We’ve told darling
that for arguments to the struct, we should expect a list of model
arguments, and each one will define parameters for a single derived model. This allows us to use the macro to generate multiple derived structs from a single input struct.
Next, let’s define the arguments for each model:
#[derive(FromMeta, Clone)]struct CustomModel { name: String, fields: PathList, #[darling(default)] extra_derives: PathList,}
In this, we have two required arguments, name
and fields
, and one optional argument extra_derives
. It’s optional because of the #[darling(default)]
annotation on it.
How to Implement DeriveCustomModel
Now that we have all of our data types defined, let’s get to parsing – which is as simple as calling a method on our argument struct! The complete function implementation should like this:
pub(crate) fn derive_custom_model_impl(input: TokenStream) -> TokenStream { let original_struct = parse_macro_input!(input as DeriveInput); let DeriveInput { data, ident, .. } = original_struct.clone(); if let Struct(data_struct) = data { let DataStruct { fields, .. } = data_struct; let args = match CustomModelArgs::from_derive_input(&original_struct) { Ok(v) => v, Err(e) => { return TokenStream::from(e.write_errors()); } }; let CustomModelArgs { models } = args; let mut output = quote!(); if models.is_empty() { panic!( "Please specify at least 1 model using the `model` attribute" ) } for model in models { let generated_model = generate_custom_model(&fields, &model); output.extend(quote!(#generated_model)); } output.into() } else { panic!("DeriveCustomModel can only be used with named structs") }}
The code that generates tokens for each model has been extracted away to another function that we call generate_custom_model
. Let’s implement that as well:
How to Generate Each Custom Model
fn generate_custom_model(fields: &Fields, model: &CustomModel) -> proc_macro2::TokenStream { let CustomModel { name, fields: target_fields, extra_derives, } = model; let mut new_fields = quote!(); for Field { ident, attrs, vis, colon_token, ty, .. } in fields { let Some(ident) = ident else { panic!("Failed to get struct field identifier") }; let path = match Path::from_string(&ident.clone().to_string()) { Ok(path) => path, Err(error) => panic!("Failed to convert field identifier to path: {error:?}"), }; if !target_fields.contains(&path) { continue; } new_fields.extend(quote! { #(#attrs)* #vis #ident #colon_token #ty, }); } let struct_ident = match Ident::from_string(name) { Ok(ident) => ident, Err(error) => panic!("{error:?}"), }; let mut extra_derives_output = quote!(); if !extra_derives.is_empty() { extra_derives_output.extend(quote! { #(#extra_derives,)* }) } quote! { #[derive(#extra_derives_output)] pub struct #struct_ident { #new_fields } }}
How to Use Your DeriveCustomModel
Macro
Coming back to your my-app/main.rs
, let’s debug-print the generated hash-maps for your new structs that you create using the macro you implemented. Your main.rs
should look like this:
use macros::{DeriveCustomModel, IntoStringHashMap};use std::collections::HashMap;#[derive(DeriveCustomModel)]#[custom_model(model( name = "UserName", fields(first_name, last_name), extra_derives(IntoStringHashMap)))]#[custom_model(model(name = "UserInfo", fields(username, age), extra_derives(Debug)))]pub struct User2 { username: String, first_name: String, last_name: String, age: u32,}fn main() { let user_name = UserName { first_name: "first_name".to_string(), last_name: "last_name".to_string(), }; let hash_map = HashMap::<String, String>::from(user_name); dbg!(hash_map); let user_info = UserInfo { username: "username".to_string(), age: 27, }; dbg!(user_info);}
As you can see, extra_derives
is already useful to us since we need to derive Debug
and IntoStringHashMap
for the new models.
If you run this using cargo run
, you should see the following output in your terminal:
[src/main.rs:32:5] hash_map = { "last_name": "last_name", "first_name": "first_name",}[src/main.rs:39:5] user_info = UserInfo { username: "username", age: 27,}
We are going to wrap up the derive macros here.
A Simple Attribute Macro
In this section, you’re going to learn how to write an attribute macro.
The log_duration
Attribute
You are going to write a simple attribute macro that can be applied to any function (or method) that will log the total run time of that function each time the function is called.
How to Declare an Attribute Macro
You declare attribute macros by creating a function and annotating that function using the proc_macro_attribute
macro that tells the compiler to consider that function as a macro declaration. Let’s see what that looks like:
#[proc_macro_attribute]pub fn log_duration(args: TokenStream, item: TokenStream) -> TokenStream { log_duration_impl(args, item)}
For these macros, the function name is important, as that also becomes the name of the macro. As you can see, these take two different arguments. The first is the argument passed to the attribute macro, and the second is the target of the attribute macro.
Let’s also implement log_duration_impl
. Create a new file log_duration.rs
:
touch src/log_duration.rs
How to Implement the log_duration
Attribute Macro
I’m going to give you the complete implementation first, and then I’ll break down the parts that I haven’t used so far:
use proc_macro::TokenStream;use quote::quote;use syn::{parse_macro_input, ItemFn};pub(crate) fn log_duration_impl(_args: TokenStream, input: TokenStream) -> TokenStream { let input = parse_macro_input!(input as ItemFn); let ItemFn { sig, vis, block, attrs, } = input; let statements = block.stmts; let function_identifier = sig.ident.clone(); quote!( #(#attrs)* #vis #sig { let __start = std::time::Instant::now(); let __result = { #(#statements)* }; println!("{} took {}μs", stringify!(#function_identifier), __start.elapsed().as_micros()); return __result; } ) .into()}
The only things that you might not have seen previously are the sig
and the block
fields you get from parsing the input as ItemFn
. sig
contains the entire signature of a function while block
contains the entire body of the function. This is why, by using the following code, we can basically reconstruct the unmodified function:
#vis #sig #block
In this example, you want to modify the function body, which is why you create a new block that encapsulates the original function block.
How to Use Your log_duration
Macro
Coming back to main.rs
, using an attribute macro is simpler than you might think:
#[log_duration]#[must_use]fn function_to_benchmark() -> u16 { let mut counter = 0; for _ in 0..u16::MAX { counter += 1; } counter}fn main() { println!("{}", function_to_benchmark());}
When you run this, you should get the following output:
function_to_benchmark took 498μs65535
We are now ready to move on to a more complex use-case.
A More Elaborate Attribute Macro
The cached_fn
Attribute
You are going to write an attribute macro that will allow you to add caching capability to any function. For the purposes of this example, we’re going to assume that our function always has String
arguments and also returns a String
value.
Some of you might know this better as a “memoized” function.
In addition, you will need to allow the user of this macro to tell the macro how it can generate a dynamic key based on function args.
To help us facilitate the caching part so that we don’t get diverted, we’re going to use a dependency called cacache
. cacache
is a Rust library for managing local key and content caches. It works by writing the cache to the disk.
Let’s add it to the project by editing the Cargo.toml
file for my-app
directly:
workspace = { members = ["my-app-macros"] }[package]name = "my-app"version = "0.1.0"edition = "2021"resolver = "2"# See more keys and their definitions at https:[dependencies]# New dependencycacache = { version = "13.0.0", default-features = false, features = ["mmap"] }macros = { path = "./macros" }
How to Implement the cached_fn
Attribute Macro
Let’s start by declaring this macro in lib.rs
:
#[proc_macro_attribute]pub fn cached_fn(args: TokenStream, item: TokenStream) -> TokenStream { cached_fn_impl(args, item)}
Create a new file cached_fn.rs
to store the implementation:
touch my-app-macros/src/cached_fn.rs
Let’s define how our arguments should look before we go ahead and implement anything:
cached_fn
Attribute Arguments
#[derive(FromMeta)]struct CachedParams { keygen: Option<Expr>,}
The only argument is an optional keygen
, which is of type Expr
. Expr
represents any valid Rust expression, so it can be very dynamic. In this example, you’ll be passing an expression that generates the key based on function arguments of the target function.
As always, we’ll first see the entire implementation and then break down the parts that are new:
pub fn cached_fn_impl(args: TokenStream, item: TokenStream) -> TokenStream { let attr_args = match NestedMeta::parse_meta_list(args.into()) { Ok(v) => v, Err(e) => { return proc_macro::TokenStream::from(Error::from(e).write_errors()); } }; let CachedParams { keygen } = match CachedParams::from_list(&attr_args) { Ok(params) => params, Err(error) => { return proc_macro::TokenStream::from(Error::from(error).write_errors()); } }; let ItemFn { sig, vis, block, attrs, } = parse_macro_input!(item as ItemFn); let key_statement = if let Some(keygen) = keygen { quote! { let __cache_key = #keygen; } } else { let fn_name = sig.ident.clone().to_string(); quote! { let __cache_key = #fn_name; } }; quote!( #(#attrs)* #vis #sig { #key_statement match cacache::read_sync("./__cache", __cache_key.clone()) { Ok(value) => { println!("Data is fetched from cached"); from_utf8(&value).unwrap().to_string() }, Err(_) => { println!("Data is not fetched from cached"); let output = #block; cacache::write_sync("./__cache", __cache_key, output.as_bytes()).unwrap(); output } } } ) .into()}
Well, turns out that you’ve seen everything that we used in this one before.
The only new thing here is the use of the cacache
dependency, but that’s also pretty straightforward. You just give the location where you want to store the cached data as the first argument to the read_sync
and write_sync
functions provided by cacache
.
We’ve also added some logging to help us verify that the macro works as expected.
How to Use the cached_fn
Macro
To make any function memoized or cached, we simply annotate it using the cached_fn
attribute:
#[cached_fn(keygen = "format!(\"{first_name} {last_name}\")")]fn test_cache(first_name: String, last_name: String) -> String { format!("{first_name} {last_name}")}fn main() { test_cache("John".to_string(), "Appleseed".to_string()); test_cache("John".to_string(), "Appleseed".to_string()); test_cache("John".to_string(), "Doe".to_string());}
If you run this, you should see the following output:
Data is not fetched from cachedData is fetched from cachedData is not fetched from cached
Which clearly shows that if the function is called more than once for the same arguments, data is returned from cache. But if the arguments are different, it doesn’t return the value that was cached for a different set of arguments.
We did make a lot of assumptions for this one which don’t hold true for a real-world use case. As such, this is only for learning purposes, but depicts a real-world use case.
For example, I’ve written attribute macros to cache HTTP handler functions using redis
for production servers. Those have a very similar implementation to this, but contains a lot of bells and whistles to work with that particular use case.
A Simple Function-like Macro
It’s finally time to have some fun again. We are going to start simple, but the second example is going to include parsing custom syntax. Fun, right?
Disclaimer: If you’re familiar with declarative macros (using macro_rules!
syntax), you might realize that the following examples can easily be written using that syntax and don’t need to be procedural macros. Writing example procedural macros that cannot be written as declarative ones is extremely difficult if you also want to keep things simple, which is why the examples were chosen despite this.
The constant_string
Macro
We’re going to build a very simple macro that takes in a string literal (of type &str
) as input and creates a global public constant for it (the name of the variable being the same as the value). Basically, our macro will generate the following:
pub const STRING_LITERAL: &str = "STRING_LITERAL";
How to Declare a Function-like Macro
You declare function-like macros by creating a function and annotating that function using a proc_macro
macro. It tells the compiler to consider that function as a macro declaration. Let’s see what that looks like:
#[proc_macro]pub fn constant_string(item: TokenStream) -> TokenStream { constant_string_impl(item)}
For these macros, the function name is important, as that also becomes the name of the macro. As you can see, these only take a single argument, which is whatever you pass on to the macro. It can literally be anything, even custom syntax that’s not valid Rust code.
How to Implement the constant_string
Macro
For the implementation, let’s create a new file constant_string.rs
:
touch my-app-macros/src/constant_string.rs
The implementation is pretty simple:
use darling::FromMeta;use proc_macro::TokenStream;use quote::quote;use syn::{parse_macro_input, Ident, LitStr};pub fn constant_string_impl(item: TokenStream) -> TokenStream { let constant_value = parse_macro_input!(item as LitStr); let constant_value_name = Ident::from_string(&constant_value.value()).unwrap(); quote!(pub const #constant_value_name: &str = #constant_value;).into()}
All we’re doing is parsing input as a string literal. If you pass something to do this is not a string literal, it will throw an error. Then we take the string, create a identifier out of it, and generate the output code. Short and simple.
How to Use the constant_string
Macro
The usage of this macro is also pretty simple:
constant_string!("SOME_CONSTANT_STRING_VALUE");
The above code will expand to this:
pub const SOME_CONSTANT_STRING_VALUE: &str = "SOME_CONSTANT_STRING_VALUE";
A More Elaborate Function-like Macro
Function-like macros, as the name might suggest, can be used in a similar way to calling a function. You can also use them in any place where you can call a function, and beyond.
The hash_mapify
Macro
Moving on to the interesting parts: the macro you’re going to write now will allow you to generate a HashMap
by simply passing in a list of key-value pairs. For example:
let variable = "Some variable";hash_mapify!( &str, key = "value", key2 = "value2", key3 = "value3", key4 = variable);
As you can see, we want the first argument to be the type of the value, and the subsequent arguments to be the key-value pairs. And we’ll need to parse all of this ourselves.
To keep things simple, since this can easily get out of hand, we’re only going to support primitive values such as strings, integers, floats and booleans. So we’re not going to support creating a hash_map
with non-string keys or enum
and struct
as values.
How to Implement the hash_mapify
Macro
We’re going to start as usual by declaring our macro:
#[proc_macro]pub fn hash_mapify(item: TokenStream) -> TokenStream { hash_mapify_impl(item)}
Next, you’re going to define a data structure to hold your input data. In this case, you need to know the value type passed, as well as a list of key-value pairs.
We are going to extract the implementation to a separate file, which is where you’ll also implement the data types and parsing logic.
Create new file hash_mapify.rs
and declare the data type to hold input data:
touch my-app-macros/src/hash_mapify.rs
How to Parse hash_mapify
‘s Input
use proc_macro::TokenStream;use quote::{quote, ToTokens};use syn::parse::{Parse, ParseStream};use syn::{parse_macro_input, Lit, LitStr, Token, Type};pub struct ParsedMapEntry(String, proc_macro2::TokenStream);pub struct ParsedMap { value_type: Type, entries: Vec<ParsedMapEntry>,}
You store the value as TokenStream
directly because you need to support both literal values as well as variables, both of which only have 1 common type in this context, TokenStream
.
You also might have noticed that we save the value_type
as Type
which is a type provided by syn
crate which is an enum of the possible types that a Rust value could have. That was a mouthful!
You won’t need to handle each variant of this enum, since this type can also directly be converted to TokenStream
. You’ll better understand what that means shortly.
Next, you need to implement the syn::parse::Parse
trait for ParsedMap
declared previously, so that it can be computed from the TokenStream
passed as arguments to the macro.
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { let mut entries = Vec::<ParsedMapEntry>::new(); }}
input
, which is of type ParsedStream
in this case, works similar to an iterator. You need to parse tokens out of the input using the method parse
on it, which will also advance the stream to the beginning of the next token.
For example, if you have a stream of tokens representing [a, b, c]
, as soon as you parse [
out of this stream, the stream will be mutated to only contain a, b, c]
. This is very similar to iterators, where as soon as you take a value out, the iterator is advanced by one position and only holds the remaining items.
Before you parse anything, you need to check if input is empty, and panic if it is:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { if input.is_empty() { panic!("At least a type must be specified for an empty hashmap"); } }}
Since we expect the first argument passed to the macro to be the type of the value in our hashmap, let’s parse that out of the token stream:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { let ty = input.parse::<Type>()?; }}
Parse takes a single type argument which represents what to parse.
If the first argument cannot be parsed as a valid type, an error will be returned. Do note that this doesn’t verify if the type you passed actually exists or not, this will only validate whether the tokens in the first argument are valid for a type definition, and that’s all.
This means that if you pass SomeRandomType
where SomeRandomType
isn’t actually defined, the parsing will still succeed. It will only fail after expanding the macro during compile time.
Moving on, we also expect the user to use ,
to separate the arguments. Let’s parse that as the next token after type:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { input.parse::<Token![,]>()?; }}
You might notice the usage of the Token!
macro when providing the type argument for the parse
method. It’s a macro provided by syn
to easily convert built-ins such as keywords (type
, async
, fn
and so on) as well as punctuation marks (,
, .
, ;
and so on) and delimiters ({
, [
, (
and so on). This macro takes a single argument, which is the keyword/punctuation/delimiter literal for which the type is needed.
The official docs define it as:
A type-macro that expands to the name of the Rust type representation of a given token.
Now that you have the type of value as well as the first separator (comma), it’s time to start parsing key-value pairs. All of the key-value pairs follow the same structure key = value
and are separated by commas.
Do note that white-space isn’t important, as that is entirely handled during the tokenization process and isn’t something that you need to handle.
Since you won’t know how many key-value pairs are passed, you need something to tell you when all of it is parsed:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { while !input.is_empty() { } }}
As I explained previously, tokens are taken out of the stream and it’s advanced each time you parse something. This means that when all of the tokens are parsed, the stream will be empty. We utilise this fact here to figure out when to break out of the loop.
Each key-value pair can be parsed in a similar fashion as you parsed the type argument:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { while !input.is_empty() { let key = if let Ok(key) = input.parse::<syn::Ident>() { key.to_string() } else if let Ok(key) = input.parse::<LitStr>() { key.value() } else { panic!("Key must be either a string literal or an identifier!"); }; input.parse::<Token![=]>()?; let value = if let Ok(value) = input.parse::<syn::Ident>() { value.to_token_stream() } else if let Ok(value) = input.parse::<Lit>() { value.to_token_stream() } else { panic!("Value must be either a literal or an identifier!"); }; entries.push(ParsedMapEntry(key, value)); if input.peek(Token![,]) { input.parse::<Token![,]>()?; } } }}
The only thing here that is new, is the call to peek
method at the end. This is a special method that returns a boolean if the token that is passed to peek
is the next token in the stream, and false otherwise.
As the name might suggest, this only performs a check, so it doesn’t take that token out of the stream or advance the stream in any form.
Once all of the parsing is done, you just return the information as part of ParsedMap
struct we declared earlier. The complete implementation for this trait is as below if that’s easier for you to read through:
impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { let mut entries = Vec::<ParsedMapEntry>::new(); if input.is_empty() { panic!("At least a type must be specified for an empty hashmap"); } let ty = input.parse::<Type>()?; input.parse::<Token![,]>()?; while !input.is_empty() { let key = if let Ok(key) = input.parse::<syn::Ident>() { key.to_string() } else if let Ok(key) = input.parse::<LitStr>() { key.value() } else { panic!("Key must be either a string literal or an identifier!"); }; input.parse::<Token![=]>()?; let value = if let Ok(value) = input.parse::<syn::Ident>() { value.to_token_stream() } else if let Ok(value) = input.parse::<Lit>() { value.to_token_stream() } else { panic!("Value must be either a literal or an identifier!"); }; entries.push(ParsedMapEntry(key, value)); if input.peek(Token![,]) { input.parse::<Token![,]>()?; } } Ok(ParsedMap { value_type: ty, entries, }) }}
How to Generate the Output Code
You can now finally write the actual macro implementation, which is going to be pretty-straightforward:
pub fn hash_mapify_impl(item: TokenStream) -> TokenStream { let input = parse_macro_input!(item as ParsedMap); let key_value_pairs = input.entries; let ty = input.value_type; quote!({ let mut hash_map = std::collections::HashMap::<String, #ty>::new(); #( hash_map.insert(#key_value_pairs); )* hash_map }) .into()}
If you’re coding along with me, or if you have a keen eye, you might have noticed that there is an error here. The type of variable key_value_pairs
is Vec<ParsedMapEntry>
. We are trying to use it in the output as:
#(hash_map.insert(#key_value_pairs);)*
which is the correct syntax for working with lists, but the underlying type ParsedMapEntry
is a custom type. And neither syn
nor quote
would know how to convert it to a token stream. So we cannot use it with this syntax.
But if we try to manually write an implementation where we loop it ourselves, generate a separate tokens stream in each loop, and extend the existing one, it’s going to be quite tedious. Wouldn’t it be great if there was a better solution? Turns out there is: ToTokens
trait.
How to Convert Custom Data Types to Output Tokens
This trait can be implemented for any of our custom types and defines how the type looks like when converted into the token stream.
impl ToTokens for ParsedMapEntry { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { let key = self.0.clone(); let value = self.1.clone(); tokens.extend(quote!(String::from(#key), #value)); }}
As part of the implementation, you need to mutate the tokens
argument and extend it to contain the token stream that we want our type to generate. The syntax I used to do that should all be familiar by now.
Once you’ve done this, quote
can now easily convert the problematic code to token stream. So this: #(hash_map.insert(#key_value_pairs);)*
will now work directly.
As usual, here’s the complete implementation if that’s easier to understand:
use proc_macro::TokenStream;use quote::{quote, ToTokens};use syn::parse::{Parse, ParseStream};use syn::{parse_macro_input, Lit, LitStr, Token, Type};pub struct ParsedMapEntry(String, proc_macro2::TokenStream);pub struct ParsedMap { value_type: Type, entries: Vec<ParsedMapEntry>,}impl ToTokens for ParsedMapEntry { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { let key = self.0.clone(); let value = self.1.clone(); tokens.extend(quote!(String::from(#key), #value)); }}impl Parse for ParsedMap { fn parse(input: ParseStream) -> syn::Result<Self> { let mut entries = Vec::<ParsedMapEntry>::new(); if input.is_empty() { panic!("At least a type must be specified for an empty hashmap"); } let ty = input.parse::<Type>()?; input.parse::<Token![,]>()?; while !input.is_empty() { let key = if let Ok(key) = input.parse::<syn::Ident>() { key.to_string() } else if let Ok(key) = input.parse::<LitStr>() { key.value() } else { panic!("Key must be either a string literal or an identifier!"); }; input.parse::<Token![=]>()?; let value = if let Ok(value) = input.parse::<syn::Ident>() { value.to_token_stream() } else if let Ok(value) = input.parse::<Lit>() { value.to_token_stream() } else { panic!("Value must be either a literal or an identifier!"); }; entries.push(ParsedMapEntry(key, value)); if input.peek(Token![,]) { input.parse::<Token![,]>()?; } } Ok(ParsedMap { value_type: ty, entries, }) }}pub fn hash_mapify_impl(item: TokenStream) -> TokenStream { let input = parse_macro_input!(item as ParsedMap); let key_value_pairs = input.entries; let ty = input.value_type; quote!({ let mut hash_map = std::collections::HashMap::<String, #ty>::new(); #( hash_map.insert(#key_value_pairs); )* hash_map }) .into()}
How to Use the hash_mapify
Macro
We can verify that our macro works by writing a simple usage:
fn main() { test_hashmap();}fn test_hashmap() { let some_variable = "Some variable value"; let hash_map = hash_mapify!( &str, "first_key" = "first_value", "second_variable" = some_variable, some_key = "value for variable key", ); let number_hash_map = hash_mapify!(usize, "first_key" = 1, "second_variable" = 2, some_key = 3,); dbg!(hash_map); dbg!(number_hash_map);}
If you run this code, you should see the following output:
[src/main.rs:62:5] hash_map = { "first_key": "first_value", "some_key": "value for variable key", "second_variable": "Some variable value",}[src/main.rs:63:5] number_hash_map = { "second_variable": 2, "first_key": 1, "some_key": 3,}
which is what we would expect to happen.
And now that we’ve covered all three types of procedural macros, we’re going to wrap up the examples here.
Beyond Writing Macros
Now that you’ve learned how to write basic derive macros, I’d like to take some time to quickly introduce some additional tools and techniques that will be helpful when working with macros. I’ll also point out some drawbacks of why and when to avoid them.
cargo-expand
This is a CLI tool that can generate macro expanded code for any file in your project. Another great project by David Tolnay. You do need the nightly toolchain for Rust to use this, though. Don’t worry – it’s only required for the tool itself to work. You don’t need to make your project use the nightly toolchain as well. Your project can stay in the stable zone.
Install nightly toolchain:
rustup toolchain install nightly
Install cargo-expand
:
cargo install cargo-expand
Now that this is done, you can see what the actual expansion of your code in main looks like. Simply run the following in the my-app
project directory:
cargo expand
and it will output the expanded code in the terminal output. You will see some unfamiliar stuff as well, such as what the dbg!
macro expands to, but you can ignore those.
trybuild & macrotest
These are 2 crates that are extremely useful if you want to unit-test your procedural macros’ expanded forms, or assert any expected compilation errors.
Downsides of Macros
Debugging (or lack thereof)
You cannot put a breakpoint into any line of code that is generated by the macro. Nor can you get to it from the stacktrace of an error. This makes debugging generated code very difficult.
In my usual workflow, I either put logging into the generated code, or if that is not enough, I replace the usage of macro with the code given to me by cargo expand
temporarily to debug it, make changes, and then update the macro code based on that.
There might be better ways out there, and if you know any, I’d be grateful if you can share them with me.
Compile Time Costs
There’s a non-zero cost for macro expansion that the compiler needs to run and process, and then check that the code it generated is valid. This becomes even more expensive when recursive macros are involved.
As a very crude estimation, each macro expansion adds 10ms to the compile time of the project. If you’re interested, I encourage you to read through this introduction on how the compiler processes macros internally.
Lack of Auto-complete and Code Checks
Code written as part of a macro output isn’t presently supported fully by any IDE, nor is it supported by rust-analyzer. So in most cases, you’re writing code without relying on features such as auto-complete, auto-suggestions, and so on.
Where Do We Draw the Line?
Given the insane potential of macros, it’s very easy to get carried away with them. It’s important to remember all of the drawbacks and make decisions accordingly, ensuring that you are not indulging yourselves into premature abstraction.
As a general rule, I personally avoid implementing any “business logic” with macros, nor do I attempt to write macros for generating code that I will need to step through with a debugger time and again. Or the code that I will need to make micro changes in for performance testing and improvement.
Wrapping Up
This was a long journey! But I wanted anyone with basic knowledge and experience with Rust to be able to follow and come out of this able to write macros in their own projects.
Hopefully, I was able to do that for you. I will be writing a lot more about macros in general, so stay tuned for that.
You can find the complete code for everything we looked at in this article in https://github.com/anshulsanghi-blog/macros-handbook repository.
Also, feel free to contact me if you have any questions or opinions on this topic.
Enjoying my work?
Consider buying me a coffee to support my work!
☕Buy me a coffee
Till next time, happy coding and wishing you clear skies!