Let’s continue with the client development, first, we will create a new instance of our deployed contract with the module reference we have. For structuring things a bit, we will create another folder for our client with cargo.
cargo new cis2-rust-sdk-minting
As the beginning step, we need to add our dependencies to Cargo.toml file. Add dependencies shown below including SDK, web client components, command line & error handlers, serialization, and time libraries.
[dependencies]
concordium-rust-sdk="2"
tokio = { version = "1", features = ["full"] }
warp = "0.3"
log = "0.4.11"
env_logger = "0.9"
clap = { version = "4", features = ["derive"] }
anyhow = "1.0"
chrono = "0.4.19"
thiserror = "1"
structopt = "0.3.26"
serde = { version = "1.0", features = ["derive"] }
serde_json = "1.0"
strum = "0.24"
strum_macros = "0.24"
Then in main.rs start adding the necessary libraries to our program, we need to command line argument parser (clap) for getting parameters and structopt for parsing the input parameters as a struct. The path library to manipulate file paths for example path of our wallet file, and many internal concordium-rust-sdk functions for making the transactions, serialization, accounts and more.
use crate::clap::AppSettings;
use anyhow::Context;
use concordium_rust_sdk::{
common::{self, types::TransactionTime, SerdeDeserialize, SerdeSerialize},
smart_contracts::{
common as concordium_std,
common::Amount,
types::{OwnedContractName, OwnedReceiveName},
},
types::{
smart_contracts::{ModuleReference, OwnedParameter, WasmModule},
transactions::{send, BlockItem, InitContractPayload, UpdateContractPayload},
AccountInfo, ContractAddress, WalletAccount,
},
v2,
};
use std::path::PathBuf;
use structopt::*;
use strum_macros::EnumString;
For simplicity, we will create an enum that represents the actions that we would like to do. In that Action enum, we specify the transaction types as following: Deploy, Init, WithSchema. In Deploy Action, we just need to specify our module's path and in Init Action, we need only the deployed module reference. In the WithSchema Action, we need to specify the transaction parameter for minting, transferring, or viewing the contract state. Please note that all will be invokable with the schema file.
#[derive(StructOpt)]
enum Action {
#[structopt(about = "Deploy the module")]
Deploy {
#[structopt(long = "module", help = "Path to the contract module.")]
module_path: PathBuf,
},
#[structopt(about = "Initialize the CIS-2 NFT contract")]
Init {
#[structopt(
long,
help = "The module reference used for initializing the contract instance."
)]
module_ref: ModuleReference,
},
#[structopt(
about = "Update the contract and set the provided using JSON parameters and a \
schema."
)]
WithSchema {
#[structopt(long, help = "Path of the JSON parameter.")]
parameter: PathBuf,
#[structopt(long, help = "Path to the schema.")]
schema: PathBuf,
#[structopt(long, help = "The contract to update.")]
address: ContractAddress,
#[structopt(long, help = "Transaction Type")]
transaction_type_: TransactionType,
},
}
We add an enum to distinguish all transactions that require a schema that comes with the WithSchema parameter. We need the schema file for both state-changing and view functions (to print in a human-readable form).
We have a TransactionResult enum that will be used for escaping an error for incompatible type error for returning different results from each match. Every state change after each invocation, including init_contract, deploy_contract and update_contract, needs to be treated differently than the tokenMetadata() and the view() functions. In order to call these view functions -meaning won't cause any state changes-the invoke_instance function should be called which has a return type. So having a parent enum helps us to return the same types, but gives us the ability to manipulate each one individually.
Now, we need to set the base configurations including node setup. Since we are going to deploy this contract to testnet, we use the testnet node gRPC endpoint as the default provided by Concordium which is testnet.node.concordium.com. We also need our key file path -the file exported from the wallet- and the Action. All these should be configurable from the terminal as an input parameter.
/// Node connection, key path and the action input struct
#[derive(StructOpt)]
struct App {
#[structopt(
long = "node",
help = "GRPC interface of the node.",
default_value = "http://node.testnet.concordium.com:20000"
)]
endpoint: v2::Endpoint,
#[structopt(long = "account", help = "Path to the account key file.")]
keys_path: PathBuf,
#[structopt(subcommand, help = "The action you want to perform.")]
action: Action,
}
Now we can create our main() function. As you can see from the code below, it is a multi-threaded runtime that can handle multiple requests simultaneously. It reads the inputs from the terminal and creates a connection with Concordium by creating a client. We upload our key file by providing its path, and get the nonce of the last finalized block to have the full list of the accounts onboarded. Then we check the action type, to decide whether this is going to be a Deploy, Init or WithSchema transaction in a match or switch case statement. (in rust there is no switch case statement). Let’s start coding with Deploy and Init first, then continue with WithSchema.
#[tokio::main(flavor = "multi_thread")]
async fn main() -> anyhow::Result<()> {
use base64::{engine::general_purpose, Engine as _};
let app = {
let app = App::clap().global_setting(AppSettings::ColoredHelp);
let matches = app.get_matches();
App::from_clap(&matches)
};
let mut client = v2::Client::new(app.endpoint)
.await
.context("Cannot connect.")?;
// load account keys and sender address from a file
let keys: WalletAccount =
WalletAccount::from_json_file(app.keys_path).context("Could not read the keys file.")?;
// Get the initial nonce at the last finalized block.
let acc_info: AccountInfo = client
.get_account_info(&keys.address.into(), &v2::BlockIdentifier::Best)
.await?
.response;
let nonce = acc_info.account_nonce;
// set expiry to now + 5min
let expiry: TransactionTime =
TransactionTime::from_seconds((chrono::Utc::now().timestamp() + 300) as u64);
Deploy Contract
In order to deploy the contract and all other transactions, we use the send() wrapper from the concordium-rust-sdk under the transactions library. We read the wasm compiled smart contract module, and after deserializing it invoke the deploy_module() function from the same library. For structuring the our directory a bit better, we created a folder called nft-params and copy and paste the exported wallet file and the module from “concordium-out” into it.
let tx = match app.action {
Action::Deploy { module_path } => {
let contents = std::fs::read(module_path).context("Could not read contract module.")?;
let payload: WasmModule =
common::Deserial::deserial(&mut std::io::Cursor::new(contents))?;
TransactionResult::StateChanging(send::deploy_module(
&keys,
keys.address,
nonce,
expiry,
payload,
))
}
Let’s build our file first, then run the executable in the target/debug folder with the below command.
cargo build
cd target/debug
./cis2-rust-sdk-minting --account ../../nft-params/wallet.export deploy --module ../../nft-params/module.wasm.v1
Congrats! We have successfully deployed our smart contract!
Init Contract
Now we will create a new instance of the deployed contract. In the first match, we check whether the action is Init and then we add an empty OwnedParam this is because our smart contract init function doesn't require an input parameter and similarly there is no Amount for this function as a payment. But the init function itself requires the modulereference that we had in the previous step. Use that and call the init_contract() function from send wrapper of the transactions library.
In the following sections, we will use the schema file either while changing the state with transfer(),mint() functions or print return values in the form of JSON from the contract.
Using Schema in View and State Changing Functions
We will need the schema file when calling mint() and transfer() functions and any view function’s printing including tokenMetadata() and view(). First, we need to read and load schema from the .bs64 output file, for convenience copy and paste it from “concordium-out” to “nft-params” folder. Please note that base64 encoding is without padding, so we decode it accordingly. Then we have the TransactionType enum which helps us to distinguish the transactions cause each one needs different parameters, invokes different functions, and uses different parts of the schema.
For the sake of the match statement’s return type mismatch error, after every transaction the return type is TransactionResult. Depending on the transaction it returns TransactionResult::StateChanging (If it’s a mint or transfer)or TransactionResult::None (If it’s a view function)
Action::WithSchema {
parameter,
schema,
address,
transaction_type_,
} => {
let parameter: serde_json::Value = serde_json::from_slice(
&std::fs::read(parameter.unwrap()).context("Unable to read parameter file.")?,
)
.context("Unable to parse parameter JSON.")?;
let schemab64 = std::fs::read(schema).context("Unable to read the schema file.")?;
let schema_source = general_purpose::STANDARD_NO_PAD.decode(schemab64);
let schema = concordium_std::from_bytes::<concordium_std::schema::VersionedModuleSchema>(
&schema_source?,
)?;
// schema_global = schema;
match transaction_type_ {
TransactionType::Mint => {
let param_schema =
schema.get_receive_param_schema("rust_sdk_minting_tutorial", "mint")?;
let serialized_parameter = param_schema.serial_value(¶meter)?;
let message = OwnedParameter::try_from(serialized_parameter).unwrap();
let payload = UpdateContractPayload {
amount: Amount::zero(),
address,
receive_name: OwnedReceiveName::new_unchecked(
"rust_sdk_minting_tutorial.mint".to_string(),
),
message,
};
TransactionResult::StateChanging(send::update_contract(
&keys,
keys.address,
nonce,
expiry,
payload,
10000u64.into(),
))
}
//// Transfer Transaction which changes the state
TransactionType::Transfer => {
let param_schema =
schema.get_receive_param_schema("rust_sdk_minting_tutorial", "transfer")?;
let serialized_parameter = param_schema.serial_value(¶meter)?;
let message = OwnedParameter::try_from(serialized_parameter).unwrap();
let payload = UpdateContractPayload {
amount: Amount::zero(),
address,
receive_name: OwnedReceiveName::new_unchecked(
"rust_sdk_minting_tutorial.transfer".to_string(),
),
message,
};
//// call update contract with the payload
TransactionResult::StateChanging(send::update_contract(
&keys,
keys.address,
nonce,
expiry,
payload,
10000u64.into(),
))
}
/// Token Metadata function with no state change
TransactionType::TokenMetadata => {
let param_schema = schema
.get_receive_param_schema("rust_sdk_minting_tutorial", "tokenMetadata")?;
let rv_schema = schema.get_receive_return_value_schema(
"rust_sdk_minting_tutorial",
"tokenMetadata",
)?;
let serialized_parameter = param_schema.serial_value(¶meter)?;
let context = ContractContext {
invoker: None, //Account(AccountAddress),
contract: address,
amount: Amount::zero(),
method: OwnedReceiveName::new_unchecked(
"rust_sdk_minting_tutorial.tokenMetadata".to_string(),
),
parameter: OwnedParameter::try_from(serialized_parameter).unwrap(), //Default::default(),
energy: 1000000.into(),
};
// invoke instance
let info = client
.invoke_instance(&BlockIdentifier::Best, &context)
.await?;
match info.response {
concordium_rust_sdk::types::smart_contracts::InvokeContractResult::Success { return_value, .. } => {
let bytes: concordium_rust_sdk::types::smart_contracts::ReturnValue = return_value.unwrap();
// deserialize and print return value
println!( "{}",rv_schema.to_json_string_pretty(&bytes.value)?);//jsonxf::pretty_print(¶m_schema.to_json_string_pretty(&bytes.value)?).unwrap());
}
_ => {
println!("Could'nt succesfully invoke the instance. Check the parameters.")
}
}
TransactionResult::None
// info
}
TransactionType::View => {
let rv_schema = schema
.get_receive_return_value_schema("rust_sdk_minting_tutorial", "view")?;
let context = ContractContext {
invoker: None, //Account(AccountAddress),
contract: address,
amount: Amount::zero(),
method: OwnedReceiveName::new_unchecked(
"rust_sdk_minting_tutorial.view".to_string(),
),
parameter: Default::default(),
energy: 1000000.into(),
};
// invoke instance
let info = client
.invoke_instance(&BlockIdentifier::Best, &context)
.await?;
match info.response {
concordium_rust_sdk::types::smart_contracts::InvokeContractResult::Success { return_value, .. } => {
let bytes: concordium_rust_sdk::types::smart_contracts::ReturnValue = return_value.unwrap();
// deserialize and print return value
println!( "{}",rv_schema.to_json_string_pretty(&bytes.value)?);//jsonxf::pretty_print(¶m_schema.to_json_string_pretty(&bytes.value)?).unwrap());
}
_ => {
println!("Could'nt succesfully invoke the instance. Check the parameters.")
}
}
TransactionResult::None
// info
}
}
}
Finally, for the transaction output, we have one final match statement with TransactionResult, which will print the transaction details including module reference when deployed, contract address when initialized, and rejection reason if it's rejected by looking at the BlockSummaryDetails. The program will print the view functions’ returns in the previous section so in this final match they are just gracefully exiting.
match tx {
TransactionResult::StateChanging(result) => {
let item = BlockItem::AccountTransaction(result);
// submit the transaction to the chain
let transaction_hash = client.send_block_item(&item).await?;
println!(
"Transaction {} submitted (nonce = {}).",
transaction_hash, nonce,
);
let (bh, bs) = client.wait_until_finalized(&transaction_hash).await?;
println!("Transaction finalized in block {}.", bh);
match bs.details {
BlockItemSummaryDetails::AccountTransaction(ad) => {
match ad.effects {
AccountTransactionEffects::ModuleDeployed { module_ref } => {
println!("module ref is {}", module_ref);
}
AccountTransactionEffects::ContractInitialized { data } => {
println!("Contract address is {}", data.address);
}
AccountTransactionEffects::None {
transaction_type,
reject_reason,
} => {
println!("The Rejection Outcome is {:#?}", reject_reason);
}
_ => (),
};
}
BlockItemSummaryDetails::AccountCreation(_) => (),
BlockItemSummaryDetails::Update(_) => {
println!("Transaction finalized in block {:?}.", bs.details);
()
}
};
}
TransactionResult::None => {
println!("No state changes, already printed, gracefully exiting.");
}
}
Mint Function
Now we can call the mint() function from our new instance. For the complete minting tutorial you can follow this from our developer portal. Let’s create a file to mint our token in the nft-params folder called nft-params.json similar to the tutorial and add your address and token ID. And copy the schema file from concordium-out folder to the nft-params.
If you want to check the parameters you can always use — help keyword.
We will call with-schema which requires the contract address, parameters, schema, and transaction type. Since there could be multiple transaction types like mint, transfer, view, burn, etc. we have added another enum TransactionType specification of the transaction type is necessary while starting the program using the command line. We are also expected to provide the JSON parameters and the schema file both will be read from the provided path. If you need more details use — help again.
Use the command below to invoke the mint() function.
Congrats! You have successfully minted your first token using Concordium Rust-SDK!
TokenMetadata Function
Let’s check our token’s metadata URL. We should invoke the tokenMetada() function of cis2-nft, it requires token_id. Create a JSON file like below and add any token_ids to send as a parameter.
Finally, we will invoke the view() function, which simply returns the current state of the contract instance. It doesn't necessarily require a parameter to be invoked, but our program waits for a parameter so you can use the same token_id.json file to display the state.