In this tutorial, we are going to show you how to access the MoveVM hosted inside the pallet-move
, integrated inside the template node.
To make Move interoperable with Substrate, we provide the package manager smove, which can compile your Move projects, create bundles, access and act with the network node and more.
This tutorial shows a summary of a workflow with Move on a Substrate template node. We will publish a module and execute a script which uses that module's functionality.
Therefore, the package manager smove
will be used to compile those resources, estimate the required amount of gas, and create the script-transaction
.
Finally, we publish the module and execute a script via using polkadot.js).
-
Bob (with address:
5FHneW46xGXgs5mUiveU4sbTyGBzmstUspZC92UhjJM694ty
)- who will publish the module and enable the module so other users can use it.
-
Alice (with address
5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY
)- who will use Bob's module to register her account in his smart contract and then use the module's functionality (buying and spending a token).
Prerequisites:
- Install smove.
- Setup our template node with pallet-move integrated and run it in the background - the instructions can be found in our tech guide.
- Either use polkadot.js or run a local frontend for the node running in the background.
- Switch the current working directory to the code example directory in
pallet-move/pallet/src/assets/move-projects/car-wash-example
.
In our example, a simple car wash is modelized, where washing coins are used to control the usage of the car wash. Users can register, buy washing coins and use them to start the car wash. For the sake of simplicity, we show here only the function headers of the Move module, the full module can be seen here.
module DeveloperBob::CarWash {
/// Simple solution, fixed price for one washing coin.
const COIN_PRICE: u128 = 1000000000000; // equals 1 UNIT
/// Struct stores number of coins for each user.
struct Balance has key, store {
coins: u8
}
/// Method executes the ICO without money. The module owner (also the car wash owner) gets deposited the minted washing coins.
public fun initial_coin_minting(module_owner: &signer) {}
/// Registers a new user. The account address will be added to the storage Balance with zero initial washing coins.
public fun register_new_user(account: &signer) {}
/// Buys `count` washing coin(s) for the car wash. Therfore, `COIN_PRICE` * `count` will be withdrawn from the user's account.
public fun buy_coin(user: &signer, count: u8) acquires Balance {}
/// Initiates the washing process by paying one washing coin.
public fun wash_car(user: &signer) acquires Balance {}
}
All public module methods can be used in an executable script, for example:
script {
use DeveloperBob::CarWash;
fun initial_coin_minting(account: signer) {
CarWash::initial_coin_minting(&account);
}
}
Now, let's compile this project to be ready for the estimation of needed gas and publication of the module.
smove build
Gas can be seen as weights in the Substrate. It will adjust the fees for the execution or publication of Move resources. Hereby, we prevent malicious scripts from running forever.
To estimate the optimal amount of gas for your module, use smove
tool:
smove node rpc estimate-gas-publish-module --account-id 5FHneW46xGXgs5mUiveU4sbTyGBzmstUspZC92UhjJM694ty --module-path build/car-wash-example/bytecode_modules/CarWash.mv
The successful result will look like:
Gas estimation:
used gas: 732
total extrinsic weight cost with the above gas:
Weight { ref_time: 20205232845, proof_size: 3576 }
vm_status_code: EXECUTED
The compiled bytecode file can be found in the subfolder
build/car-wash-example/bytecode_modules
Do the following steps in polkadot.js GUI:
- Switch to menu Developer->Extrinsics.
- Select the Move pallet. Our template node is called
moveModule
. - For this pallet, choose extrinsic
publishModule(bytecode, gasLimit)
and selectBob
as the user who shall submit the transaction.
Parameter explanation:
- bytecode represents the compiled module bytecode. Fill it up by uploading the compiled file
CarWash.mv
(from the previous compilation). - gasLimit - use the estimated optimal amount of gas for publishing this module from the previous subsection. Using less than that will make the extrinsic fail, while using more is unnecessary (and more costly).
Publish a module using polkadot.js |
Note that the module can only be published if the specified module address DeveloperBob
in the Move.toml
file of the Move project matches the user's address (in this case, Bob's wallet address).
Additionally, we can verify that the module has successfully been published by requesting the module's ABI specification using smove
:
smove node rpc get-module-abi --address 5FHneW46xGXgs5mUiveU4sbTyGBzmstUspZC92UhjJM694ty --name CarWash
You should get a longer response that starts with:
Module ABI: Some(ModuleAbi { # ...
That is the proof that the module has been published on the chain successfully.
Compiled move scripts must be passed to pallet-move's extrinsic calls in serialized transactions, which are created with the create-transaction
command. If scripts require additional function parameters, those input parameters also have to be provided to our command:
smove create-transaction --compiled-script-path build/car-wash-example/bytecode_scripts/initial_coin_minting.mv --args signer:5FHneW46xGXgs5mUiveU4sbTyGBzmstUspZC92UhjJM694ty
An important note here - if the script function requires a signer, that signer's address needs to be the same as the Substrate account which will use this serialized transaction to execute this script.
In the above case, we used Bob's address since he owns the module and wants to enable the module by executing the initial_coin_minting
script.
If you see the following message:
Script transaction is created at: # ...
It means the script and provided parameters have been serialized into the specified output file (serialized transaction), which can now be used in polkadot.js:
Now - like when publishing a module - the optimal amount of needed gas for the script execution can also be estimated by using smove
:
smove node rpc estimate-gas-execute-script -s build/car-wash-example/script_transactions/initial_coin_minting.mvt
with response:
Gas estimation:
used gas: 343
total extrinsic weight cost with the above gas:
Weight { ref_time: 2829763107, proof_size: 17037 }
vm_status_code: EXECUTED
To execute a Move script in pallet's MoveVM, use the execute(transactionBc, gasLimit, chequeLimit)
extrinsic with the following parameters:
Execute a script with parameters in polkadot.js |
Parameters are:
- transactionBc represents the serialized script transaction generated by the
smove create-transaction
command. Fill it up by uploading the generated fileinitial_coin_minting.mvt
. This encoded file contains the actual Move script bytecode and the script parameter list (since the same Move script can use different input parameters). - gasLimit is a limitation for the maximum gas the script is allowed to use in the MoveVM. If the script requires more gas than provided, MoveVM will fail to execute the script, and the user will need to retry again with more gas provided.
- chequeLimit is your balance limit for the optional funds transfer between accounts. If set to zero, no funds can be withdrawn during the script execution. In this example, the script
initial_coin_minting
will not try to charge any funds, butbuy_coin
from the next chapter will (and it shall fail if the user hasn't written the cheque when signing the extrinsic).
You have successfully published a Move module on your Polkadot blockchain and executed a Move script successfully.
Observe the balance before we execute some new scripts which will charge the user |
Now, let's execute the following actions:
- Using Alice's account, let's register her as a customer by executing the
register_new_user.mv
script
-
Hint 1: This script requires one signer, who has to be Alice's account ID.
-
Hint 2: Use the estimation RPC method to calculate gas.
-
Hint 3:
cheque_amount
is unused in this script, so set it to zero (it's always safer not to allow scripts to charge you anything if not necessary).Click here to unlock the hidden command for the above action.
# First, let's create a transcation script: smove create-transaction --compiled-script-path build/car-wash-example/bytecode_scripts/register_new_user.mv --args signer:5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY # Now let's estimate gas for this transaction-script: smove node rpc estimate-gas-execute-script -s build/car-wash-example/script_transactions/register_new_user.mvt
- Now, again using Alice's account, buy a single coin with a
buy_coin.mv
script:
-
Hint: Because the price of a washing coin is
1 UNIT
, we need to write a cheque with a value of at least1000000000000
.Click here to unlock the hidden command for the above action.
# The script will try to transfer the amount of funds required to buy a specified (via the script arguments in the create-transaction command) number # of wash coins - so the `cheque_limit` needs to have an appropriate value so that the script won't fail while trying to charge the user. # # Notice the `u8:1` script argument - it indicates the number of wash coins, and we are buying here just one. smove create-transaction --compiled-script-path build/car-wash-example/bytecode_scripts/buy_coin.mv --args signer:5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY u8:1 # Now let's estimate gas for this transaction-script. # `cheque_limit` isn't used in the RPC command for gas estimation, the RPC command assume the user has enough funds: smove node rpc estimate-gas-execute-script -s build/car-wash-example/script_transactions/buy_coin.mvt
Script execution with the adjusted chequeLimit
-
(Optional) Wash the car with Alice's account.
Click here to unlock the hidden command for the above action.
# The check_limit isn't required here (so it should be set to zero for safety reasons) since Alice will burn the car wash coin in order to wash her car. smove create-transaction --compiled-script-path build/car-wash-example/bytecode_scripts/wash_car.mv --args signer:5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY # Now let's estimate gas for this transaction-script: smove node rpc estimate-gas-execute-script -s build/car-wash-example/script_transactions/wash_car.mvt
After buying a washing coin, check how the above operations have affected Alice's balance |
We hope you finished our tutorial without any issues! And we hope you liked it. To see the full API for this pallet, check our design document here.
To learn about the differences in executing a script with multiple signers, have a look at this additional tutorial.
If you want to report feedback, please feel free to do so. We are always available at [email protected].