We are going to update the cis2-multi contract in this link. First, remind us what we want to achieve. We would like to have a minting dApp that uses Concordiumβs ID layer and according to our scenario, you should prove that you are older than 18 in order to mint a token. The most obvious solution to this is could be to ask for ID proofs from the smart contract but the proofs are not available to use like that from directly the contract. Instead, to create the same logic using theverifier backend server we will implement it in the following:
We assume the owner of the verifier and the smart contract instance owner (dApp owner) is the same person. When we run the verifier backend server, we will use an accounts sign and verify keys.
While creating a new instance of the contract, the owner has to send its verify key(public key) and the contract will keep it in the state. (To verify the signature)
When a user wants to mint a token, the dApp will ask for a challenge and a statement from the verifier.
Using that challenge, the dApp expects that the user accepts the information from his/her wallet that is going to be shared with it.
The proof will be shared with the verifier backend, and if they are verified then the verifier will sign the public address of the user with its sign key which is given as an input parameter while starting the server.
This signature will be the input of the minting function and using the public key stored in the state, the smart contract will verify that it's coming from the verifier and is valid really.
The mint() function will be executed.
Create a project for your smart contract called cis2-multi.
cargo new cis2-multi
Add the following dependencies to Cargo.toml file.
Create a lib.rs file and copy&paste the code from the example contract. Start modification with the State struct and add a variable for the verify_key and add it to the stateβs empty() function. Please note that, unlike a regular empty() function, this one takes the verify_key as a parameter.
/// The contract state,
///
/// Note: The specification does not specify how to structure the contract state
/// and this could be structured in a more space efficient way.
#[derive(Serial, DeserialWithState, StateClone)]
#[concordium(state_parameter = "S")]
struct State<S> {
/// The state of addresses.
state: StateMap<Address, AddressState<S>, S>,
/// All of the token IDs
tokens: StateMap<ContractTokenId, MetadataUrl, S>,
/// Map with contract addresses providing implementations of additional
/// standards.
implementors: StateMap<StandardIdentifierOwned, Vec<ContractAddress>, S>,
verify_key: PublicKeyEd25519,
}
impl<S: HasStateApi> State<S> {
/// Construct a state with no tokens
fn empty(state_builder: &mut StateBuilder<S>, verify_key: PublicKeyEd25519) -> Self {
State {
state: state_builder.new_map(),
tokens: state_builder.new_map(),
implementors: state_builder.new_map(),
verify_key,
}
}
/// rest of the state functions below
}
While creating a new instance, we will need to store the verify_key or the public key of the owner. So create a struct for getting that input parameter called InitParam.
And for the last part of the state, while minting we will require the signature in order to verify so our mint parameters should send it, add it like below.
/// The parameter for the contract function `mint` which mints a number of
/// token types and/or amounts of tokens to a given address.
#[derive(Serial, Deserial, SchemaType)]
struct MintParams {
/// Owner of the newly minted tokens.
owner: Address,
/// A collection of tokens to mint.
tokens: collections::BTreeMap<ContractTokenId, (TokenMetadata, ContractTokenAmount)>,
/// Signature from the owner of the contract
signature: SignatureEd25519,
}
Finally, we will update the mint() function. Basically add
#[receive(
contract = "CIS2-Multi",
name = "mint",
crypto_primitives,
parameter = "MintParams",
error = "ContractError",
enable_logger,
mutable
)]
fn contract_mint<S: HasStateApi>(
ctx: &impl HasReceiveContext,
host: &mut impl HasHost<State<S>, StateApiType = S>,
logger: &mut impl HasLogger,
crypto_primitives: &impl HasCryptoPrimitives,
) -> ContractResult<()> {
// Get the sender of the transaction
let sender: AccountAddress = match ctx.sender() {
Address::Account(a) => a,
Address::Contract(_) => bail!(ContractError::Custom(CustomContractError::AccountOnly)),
};
// Parse the parameter.
let params: MintParams = ctx.parameter_cursor().get()?;
let (state, builder) = host.state_and_builder();
// Verifying that the signature belongs to the public key which was added at the time of init.
ensure!(
crypto_primitives.verify_ed25519_signature(state.verify_key, params.signature, &sender.0),
ContractError::Unauthorized
);
for (token_id, token_info) in params.tokens {
ensure!(
state.contains_token(&token_id).eq(&false),
ContractError::Custom(CustomContractError::TokenAlreadyMinted)
);
// Mint the token in the state.
state.mint(
&token_id,
&token_info.0,
token_info.1,
¶ms.owner,
builder,
);
// Event for minted token.
logger.log(&Cis2Event::Mint(MintEvent {
token_id,
amount: token_info.1,
owner: params.owner,
}))?;
// Metadata URL for the token.
logger.log(&Cis2Event::TokenMetadata::<_, ContractTokenAmount>(
TokenMetadataEvent {
token_id,
metadata_url: token_info.0.to_metadata_url(),
},
))?;
}
Ok(())
}
Nice, we are done with the smart contract. Without slowing down, now the front-end development starts.