Adding Call Macro to Balances
Let's start by adding the #[macros::call]
macro to our Balances Pallet.
The Call Macro
The purpose of the #[macros::call]
macro is to automatically generate the enum Call
from the functions of the pallet and the pallet level Dispatch
logic found in each Pallet.
We can place the #[macros::call]
attribute over our impl<T: Config> Pallet<T>
where the callable functions are implemented. From there, the macro can parse the whole object, and extract the data it needs. Not all of your functions are intended to be callable, so you can isolate the functions which should be in their own impl<T: Config> Pallet<T>
as the template does.
Parse
In order to generate the code that we want, we need to keep track of:
- Each callable function that the developer wants to expose through the Runtime.
- The name of that function.
- The argument names and types of that function.
- The name of the
struct
where those functions are implemented. Normally this isPallet
, but we can allow the developer flexibility in their naming.
These things are tracked with CallDef
and CallVariantDef
.
Also, during the parsing process, we might want to check for certain consistencies in the code being parsed. In this case, we require that every callable function must have caller
as their first parameter with type T::AccountId
. This should make sense to you since you have designed a number of different callable functions, and they all follow this pattern.
This checking logic is handled by fn check_caller_arg
.
Expand
Once we have parsed all the data we need, generating the code is pretty straight forward.
If you jump down to let dispatch_impl = quote!
you will see a bunch of code that looks like the templates we used earlier in the tutorial. We just left markers where the macro generation logic should place all the information to write the code we need.
Macro Quirks
Macros are often very "quirky" when you use them. Since all of the input going into the macro is other code, sometimes the format of that code might not match what you expect.
For example, the original Call
enum we have constructed looks like:
#![allow(unused)] fn main() { pub enum Call<T: Config> { Transfer { to: T::AccountId, amount: T::Balance }, } }
The variant is called Transfer
because the function it represents is named fn transfer
.
However, if we want to generate the Call
enum, and we only have fn transfer
, where will we get the specific string Transfer
with a capital T
?
It is possible to do string manipulation and adjust everything to make it consistent to what Rust expects, but in this case it is better for our macros to make minimal modifications to user written code.
What does this mean?
When the #[macros::call]
macro generates our enum Call
, it will actually look like this:
#![allow(unused)] fn main() { #[allow(non_camel_case_types)] pub enum Call<T: Config> { transfer { to: T::AccountId, amount: T::Balance }, } }
Here you see that transfer
is exactly the string which comes from the name of the function. Normally all enum variants should be CamelCase
, but since rust functions are snake_case
, our enum will have variants which are also snake_case
. We won't see any warnings about this because we enabled #[allow(non_camel_case_types)]
.
Ultimately, this has no significant impact on your underlying code. It is just ergonomics and expectations.
Indeed, macros can be quirky, but the amount of time they save you makes them worth it.
Time to Add Your Call Macro
- If you haven't, move your
transfer
function into its ownimpl<T: Config> Pallet<T>
. We only want to apply the macro to this one function, so we need to isolate it from the other functions which are not meant to be callable. - Add the
#[macros::call]
attribute over this newimpl<T: Config> Pallet<T>
. - Delete your existing
enum Call
. - Delete your existing implementation of
Dispatch for Pallet
. - Then, in your
main.rs
file, change instances ofbalances::Call::Transfer
tobalances::Call::transfer
with a lowercaset
.
At this point, everything should compile just like before! We are witnessing the power of macros to generate code for us auto-magically!
#![allow(unused)] fn main() { use num::traits::{CheckedAdd, CheckedSub, Zero}; use std::collections::BTreeMap; /// The configuration trait for the Balances Module. /// Contains the basic types needed for handling balances. pub trait Config: crate::system::Config { /// A type which can represent the balance of an account. /// Usually this is a large unsigned integer. type Balance: Zero + CheckedSub + CheckedAdd + Copy; } /// This is the Balances Module. /// It is a simple module which keeps track of how much balance each account has in this state /// machine. #[derive(Debug)] pub struct Pallet<T: Config> { // A simple storage mapping from accounts to their balances. balances: BTreeMap<T::AccountId, T::Balance>, } impl<T: Config> Pallet<T> { // Create a new instance of the balances module. pub fn new() -> Self { Self { balances: BTreeMap::new() } } /// Set the balance of an account `who` to some `amount`. pub fn set_balance(&mut self, who: &T::AccountId, amount: T::Balance) { self.balances.insert(who.clone(), amount); } /// Get the balance of an account `who`. /// If the account has no stored balance, we return zero. pub fn balance(&self, who: &T::AccountId) -> T::Balance { *self.balances.get(who).unwrap_or(&T::Balance::zero()) } } /* TODO: Add the `#[macros::call]` attribute right here. */ impl<T: Config> Pallet<T> { /// Transfer `amount` from one account to another. /// This function verifies that `from` has at least `amount` balance to transfer, /// and that no mathematical overflows occur. pub fn transfer( &mut self, caller: T::AccountId, to: T::AccountId, amount: T::Balance, ) -> crate::support::DispatchResult { let caller_balance = self.balance(&caller); let to_balance = self.balance(&to); let new_caller_balance = caller_balance.checked_sub(&amount).ok_or("Not enough funds.")?; let new_to_balance = to_balance.checked_add(&amount).ok_or("Overflow")?; self.balances.insert(caller, new_caller_balance); self.balances.insert(to, new_to_balance); Ok(()) } } // A public enum which describes the calls we want to expose to the dispatcher. // We should expect that the caller of each call will be provided by the dispatcher, // and not included as a parameter of the call. /* TODO: Remove `enum Call`, this is being generated automatically by `#[macros::call]`. */ pub enum Call<T: Config> { Transfer { to: T::AccountId, amount: T::Balance }, } /// Implementation of the dispatch logic, mapping from `BalancesCall` to the appropriate underlying /// function we want to execute. /* TODO: Remove this `Dispatch` impl, this is also being generated by `#[macros::call]`. */ impl<T: Config> crate::support::Dispatch for Pallet<T> { type Caller = T::AccountId; type Call = Call<T>; fn dispatch( &mut self, caller: Self::Caller, call: Self::Call, ) -> crate::support::DispatchResult { match call { Call::Transfer { to, amount } => { self.transfer(caller, to, amount)?; }, } Ok(()) } } #[cfg(test)] mod tests { struct TestConfig; impl crate::system::Config for TestConfig { type AccountId = String; type BlockNumber = u32; type Nonce = u32; } impl super::Config for TestConfig { type Balance = u128; } #[test] fn init_balances() { let mut balances = super::Pallet::<TestConfig>::new(); assert_eq!(balances.balance(&"alice".to_string()), 0); balances.set_balance(&"alice".to_string(), 100); assert_eq!(balances.balance(&"alice".to_string()), 100); assert_eq!(balances.balance(&"bob".to_string()), 0); } #[test] fn transfer_balance() { let mut balances = super::Pallet::<TestConfig>::new(); assert_eq!( balances.transfer("alice".to_string(), "bob".to_string(), 51), Err("Not enough funds.") ); balances.set_balance(&"alice".to_string(), 100); assert_eq!(balances.transfer("alice".to_string(), "bob".to_string(), 51), Ok(())); assert_eq!(balances.balance(&"alice".to_string()), 49); assert_eq!(balances.balance(&"bob".to_string()), 51); assert_eq!( balances.transfer("alice".to_string(), "bob".to_string(), 51), Err("Not enough funds.") ); } } }
mod balances; mod proof_of_existence; mod support; mod system; use crate::support::Dispatch; // These are the concrete types we will use in our simple state machine. // Modules are configured for these types directly, and they satisfy all of our // trait requirements. mod types { pub type AccountId = String; pub type Balance = u128; pub type BlockNumber = u32; pub type Nonce = u32; pub type Extrinsic = crate::support::Extrinsic<AccountId, crate::RuntimeCall>; pub type Header = crate::support::Header<BlockNumber>; pub type Block = crate::support::Block<Header, Extrinsic>; pub type Content = &'static str; } // These are all the calls which are exposed to the world. // Note that it is just an accumulation of the calls exposed by each module. pub enum RuntimeCall { Balances(balances::Call<Runtime>), ProofOfExistence(proof_of_existence::Call<Runtime>), } // This is our main Runtime. // It accumulates all of the different pallets we want to use. #[derive(Debug)] pub struct Runtime { system: system::Pallet<Self>, balances: balances::Pallet<Self>, proof_of_existence: proof_of_existence::Pallet<Self>, } impl system::Config for Runtime { type AccountId = types::AccountId; type BlockNumber = types::BlockNumber; type Nonce = types::Nonce; } impl balances::Config for Runtime { type Balance = types::Balance; } impl proof_of_existence::Config for Runtime { type Content = types::Content; } impl Runtime { // Create a new instance of the main Runtime, by creating a new instance of each pallet. fn new() -> Self { Self { system: system::Pallet::new(), balances: balances::Pallet::new(), proof_of_existence: proof_of_existence::Pallet::new(), } } // Execute a block of extrinsics. Increments the block number. fn execute_block(&mut self, block: types::Block) -> support::DispatchResult { self.system.inc_block_number(); if block.header.block_number != self.system.block_number() { return Err("block number does not match what is expected"); } // An extrinsic error is not enough to trigger the block to be invalid. We capture the // result, and emit an error message if one is emitted. for (i, support::Extrinsic { caller, call }) in block.extrinsics.into_iter().enumerate() { self.system.inc_nonce(&caller); let _res = self.dispatch(caller, call).map_err(|e| { eprintln!( "Extrinsic Error\n\tBlock Number: {}\n\tExtrinsic Number: {}\n\tError: {}", block.header.block_number, i, e ) }); } Ok(()) } } impl crate::support::Dispatch for Runtime { type Caller = <Runtime as system::Config>::AccountId; type Call = RuntimeCall; // Dispatch a call on behalf of a caller. Increments the caller's nonce. // // Dispatch allows us to identify which underlying module call we want to execute. // Note that we extract the `caller` from the extrinsic, and use that information // to determine who we are executing the call on behalf of. fn dispatch( &mut self, caller: Self::Caller, runtime_call: Self::Call, ) -> support::DispatchResult { // This match statement will allow us to correctly route `RuntimeCall`s // to the appropriate pallet level function. match runtime_call { RuntimeCall::Balances(call) => { self.balances.dispatch(caller, call)?; }, RuntimeCall::ProofOfExistence(call) => { self.proof_of_existence.dispatch(caller, call)?; }, } Ok(()) } } fn main() { // Create a new instance of the Runtime. // It will instantiate with it all the modules it uses. let mut runtime = Runtime::new(); let alice = "alice".to_string(); let bob = "bob".to_string(); let charlie = "charlie".to_string(); // Initialize the system with some initial balance. runtime.balances.set_balance(&alice, 100); // Here are the extrinsics in our block. // You can add or remove these based on the modules and calls you have set up. let block_1 = types::Block { header: support::Header { block_number: 1 }, extrinsics: vec![ support::Extrinsic { caller: alice.clone(), /* TODO: Update the enum name to match what is generated with the macro. */ call: RuntimeCall::Balances(balances::Call::Transfer { to: bob.clone(), amount: 30, }), }, support::Extrinsic { caller: alice.clone(), call: RuntimeCall::Balances(balances::Call::Transfer { to: charlie, amount: 20 }), }, ], }; let block_2 = types::Block { header: support::Header { block_number: 2 }, extrinsics: vec![ support::Extrinsic { caller: alice.clone(), call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, support::Extrinsic { caller: bob.clone(), call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, ], }; let block_3 = types::Block { header: support::Header { block_number: 3 }, extrinsics: vec![ support::Extrinsic { caller: alice, call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::RevokeClaim { claim: "Hello, world!", }), }, support::Extrinsic { caller: bob, call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, ], }; // Execute the extrinsics which make up our blocks. // If there are any errors, our system panics, since we should not execute invalid blocks. runtime.execute_block(block_1).expect("invalid block"); runtime.execute_block(block_2).expect("invalid block"); runtime.execute_block(block_3).expect("invalid block"); // Simply print the debug format of our runtime state. println!("{:#?}", runtime); }
#![allow(unused)] fn main() { use num::traits::{CheckedAdd, CheckedSub, Zero}; use std::collections::BTreeMap; /// The configuration trait for the Balances Module. /// Contains the basic types needed for handling balances. pub trait Config: crate::system::Config { /// A type which can represent the balance of an account. /// Usually this is a large unsigned integer. type Balance: Zero + CheckedSub + CheckedAdd + Copy; } /// This is the Balances Module. /// It is a simple module which keeps track of how much balance each account has in this state /// machine. #[derive(Debug)] pub struct Pallet<T: Config> { // A simple storage mapping from accounts to their balances. balances: BTreeMap<T::AccountId, T::Balance>, } impl<T: Config> Pallet<T> { // Create a new instance of the balances module. pub fn new() -> Self { Self { balances: BTreeMap::new() } } /// Set the balance of an account `who` to some `amount`. pub fn set_balance(&mut self, who: &T::AccountId, amount: T::Balance) { self.balances.insert(who.clone(), amount); } /// Get the balance of an account `who`. /// If the account has no stored balance, we return zero. pub fn balance(&self, who: &T::AccountId) -> T::Balance { *self.balances.get(who).unwrap_or(&T::Balance::zero()) } } #[macros::call] impl<T: Config> Pallet<T> { /// Transfer `amount` from one account to another. /// This function verifies that `from` has at least `amount` balance to transfer, /// and that no mathematical overflows occur. pub fn transfer( &mut self, caller: T::AccountId, to: T::AccountId, amount: T::Balance, ) -> crate::support::DispatchResult { let caller_balance = self.balance(&caller); let to_balance = self.balance(&to); let new_caller_balance = caller_balance.checked_sub(&amount).ok_or("Not enough funds.")?; let new_to_balance = to_balance.checked_add(&amount).ok_or("Overflow")?; self.balances.insert(caller, new_caller_balance); self.balances.insert(to, new_to_balance); Ok(()) } } #[cfg(test)] mod tests { struct TestConfig; impl crate::system::Config for TestConfig { type AccountId = String; type BlockNumber = u32; type Nonce = u32; } impl super::Config for TestConfig { type Balance = u128; } #[test] fn init_balances() { let mut balances = super::Pallet::<TestConfig>::new(); assert_eq!(balances.balance(&"alice".to_string()), 0); balances.set_balance(&"alice".to_string(), 100); assert_eq!(balances.balance(&"alice".to_string()), 100); assert_eq!(balances.balance(&"bob".to_string()), 0); } #[test] fn transfer_balance() { let mut balances = super::Pallet::<TestConfig>::new(); assert_eq!( balances.transfer("alice".to_string(), "bob".to_string(), 51), Err("Not enough funds.") ); balances.set_balance(&"alice".to_string(), 100); assert_eq!(balances.transfer("alice".to_string(), "bob".to_string(), 51), Ok(())); assert_eq!(balances.balance(&"alice".to_string()), 49); assert_eq!(balances.balance(&"bob".to_string()), 51); assert_eq!( balances.transfer("alice".to_string(), "bob".to_string(), 51), Err("Not enough funds.") ); } } }
mod balances; mod proof_of_existence; mod support; mod system; use crate::support::Dispatch; // These are the concrete types we will use in our simple state machine. // Modules are configured for these types directly, and they satisfy all of our // trait requirements. mod types { pub type AccountId = String; pub type Balance = u128; pub type BlockNumber = u32; pub type Nonce = u32; pub type Extrinsic = crate::support::Extrinsic<AccountId, crate::RuntimeCall>; pub type Header = crate::support::Header<BlockNumber>; pub type Block = crate::support::Block<Header, Extrinsic>; pub type Content = &'static str; } // These are all the calls which are exposed to the world. // Note that it is just an accumulation of the calls exposed by each module. pub enum RuntimeCall { Balances(balances::Call<Runtime>), ProofOfExistence(proof_of_existence::Call<Runtime>), } // This is our main Runtime. // It accumulates all of the different pallets we want to use. #[derive(Debug)] pub struct Runtime { system: system::Pallet<Self>, balances: balances::Pallet<Self>, proof_of_existence: proof_of_existence::Pallet<Self>, } impl system::Config for Runtime { type AccountId = types::AccountId; type BlockNumber = types::BlockNumber; type Nonce = types::Nonce; } impl balances::Config for Runtime { type Balance = types::Balance; } impl proof_of_existence::Config for Runtime { type Content = types::Content; } impl Runtime { // Create a new instance of the main Runtime, by creating a new instance of each pallet. fn new() -> Self { Self { system: system::Pallet::new(), balances: balances::Pallet::new(), proof_of_existence: proof_of_existence::Pallet::new(), } } // Execute a block of extrinsics. Increments the block number. fn execute_block(&mut self, block: types::Block) -> support::DispatchResult { self.system.inc_block_number(); if block.header.block_number != self.system.block_number() { return Err("block number does not match what is expected"); } // An extrinsic error is not enough to trigger the block to be invalid. We capture the // result, and emit an error message if one is emitted. for (i, support::Extrinsic { caller, call }) in block.extrinsics.into_iter().enumerate() { self.system.inc_nonce(&caller); let _res = self.dispatch(caller, call).map_err(|e| { eprintln!( "Extrinsic Error\n\tBlock Number: {}\n\tExtrinsic Number: {}\n\tError: {}", block.header.block_number, i, e ) }); } Ok(()) } } impl crate::support::Dispatch for Runtime { type Caller = <Runtime as system::Config>::AccountId; type Call = RuntimeCall; // Dispatch a call on behalf of a caller. Increments the caller's nonce. // // Dispatch allows us to identify which underlying module call we want to execute. // Note that we extract the `caller` from the extrinsic, and use that information // to determine who we are executing the call on behalf of. fn dispatch( &mut self, caller: Self::Caller, runtime_call: Self::Call, ) -> support::DispatchResult { // This match statement will allow us to correctly route `RuntimeCall`s // to the appropriate pallet level function. match runtime_call { RuntimeCall::Balances(call) => { self.balances.dispatch(caller, call)?; }, RuntimeCall::ProofOfExistence(call) => { self.proof_of_existence.dispatch(caller, call)?; }, } Ok(()) } } fn main() { // Create a new instance of the Runtime. // It will instantiate with it all the modules it uses. let mut runtime = Runtime::new(); let alice = "alice".to_string(); let bob = "bob".to_string(); let charlie = "charlie".to_string(); // Initialize the system with some initial balance. runtime.balances.set_balance(&alice, 100); // Here are the extrinsics in our block. // You can add or remove these based on the modules and calls you have set up. let block_1 = types::Block { header: support::Header { block_number: 1 }, extrinsics: vec![ support::Extrinsic { caller: alice.clone(), call: RuntimeCall::Balances(balances::Call::transfer { to: bob.clone(), amount: 30, }), }, support::Extrinsic { caller: alice.clone(), call: RuntimeCall::Balances(balances::Call::transfer { to: charlie, amount: 20 }), }, ], }; let block_2 = types::Block { header: support::Header { block_number: 2 }, extrinsics: vec![ support::Extrinsic { caller: alice.clone(), call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, support::Extrinsic { caller: bob.clone(), call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, ], }; let block_3 = types::Block { header: support::Header { block_number: 3 }, extrinsics: vec![ support::Extrinsic { caller: alice, call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::RevokeClaim { claim: "Hello, world!", }), }, support::Extrinsic { caller: bob, call: RuntimeCall::ProofOfExistence(proof_of_existence::Call::CreateClaim { claim: "Hello, world!", }), }, ], }; // Execute the extrinsics which make up our blocks. // If there are any errors, our system panics, since we should not execute invalid blocks. runtime.execute_block(block_1).expect("invalid block"); runtime.execute_block(block_2).expect("invalid block"); runtime.execute_block(block_3).expect("invalid block"); // Simply print the debug format of our runtime state. println!("{:#?}", runtime); }
diff --git a/src/balances.rs b/src/balances.rs
index 947ad847..42d66d0f 100644
--- a/src/balances.rs
+++ b/src/balances.rs
@@ -34,7 +34,10 @@ impl<T: Config> Pallet<T> {
pub fn balance(&self, who: &T::AccountId) -> T::Balance {
*self.balances.get(who).unwrap_or(&T::Balance::zero())
}
+}
+/* TODO: Add the `#[macros::call]` attribute right here. */
+impl<T: Config> Pallet<T> {
/// Transfer `amount` from one account to another.
/// This function verifies that `from` has at least `amount` balance to transfer,
/// and that no mathematical overflows occur.
@@ -60,12 +63,14 @@ impl<T: Config> Pallet<T> {
// A public enum which describes the calls we want to expose to the dispatcher.
// We should expect that the caller of each call will be provided by the dispatcher,
// and not included as a parameter of the call.
+/* TODO: Remove `enum Call`, this is being generated automatically by `#[macros::call]`. */
pub enum Call<T: Config> {
Transfer { to: T::AccountId, amount: T::Balance },
}
/// Implementation of the dispatch logic, mapping from `BalancesCall` to the appropriate underlying
/// function we want to execute.
+/* TODO: Remove this `Dispatch` impl, this is also being generated by `#[macros::call]`. */
impl<T: Config> crate::support::Dispatch for Pallet<T> {
type Caller = T::AccountId;
type Call = Call<T>;
diff --git a/src/main.rs b/src/main.rs
index d2e384e3..69ba65b5 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -125,6 +125,7 @@ fn main() {
extrinsics: vec![
support::Extrinsic {
caller: alice.clone(),
+ /* TODO: Update the enum name to match what is generated with the macro. */
call: RuntimeCall::Balances(balances::Call::Transfer {
to: bob.clone(),
amount: 30,
diff --git a/src/balances.rs b/src/balances.rs
index 42d66d0f..68b2bb3a 100644
--- a/src/balances.rs
+++ b/src/balances.rs
@@ -36,7 +36,7 @@ impl<T: Config> Pallet<T> {
}
}
-/* TODO: Add the `#[macros::call]` attribute right here. */
+#[macros::call]
impl<T: Config> Pallet<T> {
/// Transfer `amount` from one account to another.
/// This function verifies that `from` has at least `amount` balance to transfer,
@@ -60,35 +60,6 @@ impl<T: Config> Pallet<T> {
}
}
-// A public enum which describes the calls we want to expose to the dispatcher.
-// We should expect that the caller of each call will be provided by the dispatcher,
-// and not included as a parameter of the call.
-/* TODO: Remove `enum Call`, this is being generated automatically by `#[macros::call]`. */
-pub enum Call<T: Config> {
- Transfer { to: T::AccountId, amount: T::Balance },
-}
-
-/// Implementation of the dispatch logic, mapping from `BalancesCall` to the appropriate underlying
-/// function we want to execute.
-/* TODO: Remove this `Dispatch` impl, this is also being generated by `#[macros::call]`. */
-impl<T: Config> crate::support::Dispatch for Pallet<T> {
- type Caller = T::AccountId;
- type Call = Call<T>;
-
- fn dispatch(
- &mut self,
- caller: Self::Caller,
- call: Self::Call,
- ) -> crate::support::DispatchResult {
- match call {
- Call::Transfer { to, amount } => {
- self.transfer(caller, to, amount)?;
- },
- }
- Ok(())
- }
-}
-
#[cfg(test)]
mod tests {
struct TestConfig;
diff --git a/src/main.rs b/src/main.rs
index 69ba65b5..eb1e404d 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -125,15 +125,14 @@ fn main() {
extrinsics: vec![
support::Extrinsic {
caller: alice.clone(),
- /* TODO: Update the enum name to match what is generated with the macro. */
- call: RuntimeCall::Balances(balances::Call::Transfer {
+ call: RuntimeCall::Balances(balances::Call::transfer {
to: bob.clone(),
amount: 30,
}),
},
support::Extrinsic {
caller: alice.clone(),
- call: RuntimeCall::Balances(balances::Call::Transfer { to: charlie, amount: 20 }),
+ call: RuntimeCall::Balances(balances::Call::transfer { to: charlie, amount: 20 }),
},
],
};