Events
In this tutorial, you'll learn about the events standard and how to implement it in your smart contract.
Introduction
To get started, either switch to the 6.royalty
branch from our GitHub repository, or continue your work from the previous tutorials.
git checkout 6.royalty
7.events
branch. ::: :::Understanding the use case
Have you ever wondered how the wallet knows which NFTs you own and how it can display them in the collectibles tab? Originally, an indexer was used and it listened for any functions starting with nft_
on your account. These contracts were then flagged on your account as likely NFT contracts. Originally, an indexer used to listen for any functions calls starting with nft_
on your account. These contracts were then flagged on your account as likely NFT contracts.
When you navigated to your collectibles tab, the wallet would then query all those contracts for the list of NFTs you owned using the nft_tokens_for_owner
function you saw in the enumeration tutorial.
The problem
This method of flagging contracts was not reliable as each NFT-driven application might have its own way of minting or transferring NFTs. In addition, it's common for apps to transfer or mint many tokens at a time using batch functions. In addition, it's common for apps to transfer or mint many tokens at a time using batch functions.
The solution
A standard was introduced so that smart contracts could emit an event anytime NFTs were transferred, minted, or burnt. This event was in the form of a log. No matter how a contract implemented the functionality, an indexer could now listen for those standardized logs. This event was in the form of a log. No matter how a contract implemented the functionality, an indexer could now listen for those standardized logs.
As per the standard, you need to implement a logging functionality that gets fired when NFTs are transferred or minted. In this case, the contract doesn't support burning so you don't need to worry about that for now. In this case, the contract doesn't support burning so you don't need to worry about that for now.
It's important to note the standard dictates that the log should begin with "EVENT_JSON:"
. The structure of your log should, however, always contain the 3 following things: The structure of your log should, however, always contain the 3 following things:
- standard: the current name of the standard (e.g. nep171)
- version: the version of the standard you're using (e.g. 1.0.0)
- event: a list of events you're emitting.
The event interface differs based on whether you're recording transfers or mints. The interface for both events is outlined below. The interface for both events is outlined below.
Transfer events:
- Optional - authorized_id: the account approved to transfer on behalf of the owner.
- old_owner_id: the old owner of the NFT.
- new_owner_id: the new owner that the NFT is being transferred to.
- token_ids: a list of NFTs being transferred.
- Optional - memo: an optional message to include with the event.
Minting events:
- owner_id: the owner that the NFT is being minted to.
- token_ids: a list of NFTs being transferred.
- Optional - memo: an optional message to include with the event.
Examples
In order to solidify your understanding of the standard, let's walk through three scenarios and see what the logs should look like.
Scenario A - simple mint
In this scenario, Benji wants to mint an NFT to Mike with a token ID "team-token"
and he doesn't include a message. The log should look as follows. The log should look as follows.
EVENT_JSON:{
"standard": "nep171",
"version": "1.0.0",
"event": "nft_mint",
"data": [
{"owner_id": "mike.testnet", "token_ids": ["team-token"]}
]
}
Scenario B - batch mint
In this scenario, Benji wants to perform a batch mint. He will mint an NFT to Mike, Damian, Josh, and Dorian. Dorian, however, will get two NFTs. Each token ID will be "team-token"
followed by an incrementing number. The log is as follows.
EVENT_JSON:{
"standard": "nep171",
"version": "1.0.0",
"event": "nft_mint",
"data": [
{"owner_id": "mike.testnet", "token_ids": ["team-token0"]},
{"owner_id": "damian.testnet", "token_ids": ["team-token1"]},
{"owner_id": "josh.testnet", "token_ids": ["team-token2"]}
{"owner_id": "dorian.testnet", "token_ids": ["team-token3", "team-token4"]},
]
}
Scenario C - transfer NFTs
In this scenario, Mike is transferring both his team tokens to Josh. The log should look as follows. The log should look as follows.
EVENT_JSON:{
"standard": "nep171",
"version": "1.0.0",
"event": "nft_transfer",
"data": [
{"old_owner_id": "mike.testnet", "new_owner_id": "josh.testnet", "token_ids": ["team-token", "team-token0"], "memo": "Go Team!"}
]
}
]
}
Modifications to the contract
At this point, you should have a good understanding of what the end goal should be so let's get to work!
Logging minted tokens
Since the contract will only be minting tokens in one place, it's trivial where you should place the log. Open the nft-contract/src/mint.ts
file and navigate to the bottom of the file. This is where you'll construct the log for minting. Anytime someone successfully mints an NFT, it will now correctly emit a log. Open the nft-contract/src/mint.ts
file and navigate to the bottom of the file. This is where you'll construct the log for minting. Anytime someone successfully mints an NFT, it will now correctly emit a log.
// Construct the mint log as per the events standard.
let nftMintLog = {
// Standard name ("nep171").
standard: NFT_STANDARD_NAME,
// Version of the standard ("nft-1.0.0").
version: NFT_METADATA_SPEC,
// The data related with the event stored in a vector.
event: "nft_mint",
data: [
{
// Owner of the token.
owner_id: token.owner_id,
// Vector of token IDs that were minted.
token_ids: [tokenId],
}
]
}
// Log the json.
near.log(`EVENT_JSON:${JSON.stringify(nftMintLog)}`);
Loading...
Logging transfers
Let's open the nft-contract/src/internal.ts
file and navigate to the internalTransfer
function. This is the location where you'll build your transfer logs. Whenever an NFT is transferred, this function is called and so you'll correctly be logging the transfers.
// Construct the transfer log as per the events standard.
let nftTransferLog = {
// Standard name ("nep171").
standard: NFT_STANDARD_NAME,
// Version of the standard ("nft-1.0.0").
version: NFT_METADATA_SPEC,
// The data related with the event stored in a vector.
event: "nft_transfer",
data: [
{
// The optional authorized account ID to transfer the token on behalf of the old owner.
authorized_id: authorizedId,
// The old owner's account ID.
old_owner_id: token.owner_id,
// The account ID of the new owner of the token.
new_owner_id: receiverId,
// A vector containing the token IDs as strings.
token_ids: [tokenId],
// An optional memo to include.
memo,
}
]
}
// Log the serialized json.
near.log(`EVENT_JSON:${JSON.stringify(nftTransferLog)}`);
Loading...
This solution, unfortunately, has an edge case which will break things. This solution, unfortunately, has an edge case which will break things. If an NFT is transferred via the nft_transfer_call
function, there's a chance that the transfer will be reverted if the nft_on_transfer
function returns true
. Taking a look at the logic for nft_transfer_call
, you can see why this is a problem. Taking a look at the logic for nft_transfer_call
, you can see why this is a problem.
When nft_transfer_call
is invoked, it will:
- Call
internalTransfer
to perform the actual transfer logic. - Initiate a cross-contract call and invoke the
nft_on_transfer
function. - Resolve the promise and perform logic in
internalResolveTransfer
.- This will either return true meaning the transfer went fine or it will revert the transfer and return false.
Let's open the nft-contract/src/internal.ts
file and navigate to the internalTransfer
function. This is the location where you'll build your transfer logs. Whenever an NFT is transferred, this function is called and so you'll correctly be logging the transfers. If the transfer is reverted during internalResolveTransfer
, however, that event should also be emitted. If you only place the log in the internalTransfer
function, the log will be emitted and the indexer will think that the NFT was transferred. If the transfer is reverted during internalResolveTransfer
, however, that event should also be emitted. Anywhere that an NFT could be transferred, we should add logs. Replace the internalResolveTransfer
with the following code. Replace the internalResolveTransfer
with the following code.
Loading...
With that finished, you've successfully implemented the events standard and it's time to start testing.
Deploying the contract
For the purpose of readability and ease of development, instead of redeploying the contract to the same account, let's create a sub-account and deploy to that instead. For the purpose of readability and ease of development, instead of redeploying the contract to the same account, let's create a sub-account and deploy to that instead. You could have deployed to the same account as none of the changes you implemented in this tutorial would have caused errors.
Creating a sub-account
Run the following command to create a sub-account events
of your main account with an initial balance of 25 NEAR which will be transferred from the original to your new account.
near create-account events.$NFT_CONTRACT_ID --masterAccount $NFT_CONTRACT_ID --initialBalance 25
Next, you'll want to export an environment variable for ease of development:
export EVENTS_NFT_CONTRACT_ID=events.$NFT_CONTRACT_ID
Using the build script, build the deploy the contract as you did in the previous tutorials:
yarn build && near deploy --wasmFile build/nft.wasm --accountId $EVENTS_NFT_CONTRACT_ID
Initialization and minting
Since this is a new contract, you'll need to initialize and mint a token. Use the following command to initialize the contract: Use the following command to initialize the contract:
near call $EVENTS_NFT_CONTRACT_ID init '{"owner_id": "'$EVENTS_NFT_CONTRACT_ID'"}' --accountId $EVENTS_NFT_CONTRACT_ID
Next, you'll need to mint a token. Next, you'll need to mint a token. By running this command, you'll mint a token with a token ID "events-token"
and the receiver will be your new account. In addition, you're passing in a map with two accounts that will get perpetual royalties whenever your token is sold. In addition, you're passing in a map with two accounts that will get perpetual royalties whenever your token is sold.
near call $EVENTS_NFT_CONTRACT_ID nft_mint '{"token_id": "events-token", "metadata": {"title": "Events Token", "description": "testing out the new events extension of the standard", "media": "https://bafybeiftczwrtyr3k7a2k4vutd3amkwsmaqyhrdzlhvpt33dyjivufqusq.ipfs.dweb.link/goteam-gif.gif"}, "receiver_id": "'$EVENTS_NFT_CONTRACT_ID'"}' --accountId $EVENTS_NFT_CONTRACT_ID --amount 0.1
You can check to see if everything went through properly by looking at the output in your CLI:
Doing account.functionCall()
Receipts: F4oxNfv54cqwUwLUJ7h74H1iE66Y3H7QDfZMmGENwSxd, BJxKNFRuLDdbhbGeLA3UBSbL8UicU7oqHsWGink5WX7S
Log [events.goteam.examples.testnet]: EVENT_JSON:{"standard":"nep171","version":"1.0.0","event":"nft_mint","data":[{"owner_id":"events.goteam.examples.testnet","token_ids":["events-token"]}]}
Transaction Id 4Wy2KQVTuAWQHw5jXcRAbrz7bNyZBoiPEvLcGougciyk
To see the transaction in the transaction explorer, please open this url in your browser
https://testnet.nearblocks.io/txns/4Wy2KQVTuAWQHw5jXcRAbrz7bNyZBoiPEvLcGougciyk
''
You can see that the event was properly logged!
Transferring
You can now test if your transfer log works as expected by sending benjiman.testnet
your NFT.
near call $EVENTS_NFT_CONTRACT_ID nft_transfer '{"receiver_id": "benjiman.testnet", "token_id": "events-token", "memo": "Go Team :)", "approval_id": 0}' --accountId $EVENTS_NFT_CONTRACT_ID --depositYocto 1
This should return an output similar to the following:
Doing account.functionCall()
Receipts: EoqBxrpv9Dgb8KqK4FdeREawVVLWepEUR15KPNuZ4fGD, HZ4xQpbgc8EfU3PiV72LvfXb2f3dVC1n9aVTbQds9zfR
Log [events.goteam.examples.testnet]: Memo: Go Team :)
Log [events.goteam.examples.testnet]: EVENT_JSON:{"standard":"nep171","version":"1.0.0","event":"nft_transfer","data":[{"authorized_id":"events.goteam.examples.testnet","old_owner_id":"events.goteam.examples.testnet","new_owner_id":"benjiman.testnet","token_ids":["events-token"],"memo":"Go Team :)"}]}
Transaction Id 4S1VrepKzA6HxvPj3cK12vaT7Dt4vxJRWESA1ym1xdvH
To see the transaction in the transaction explorer, please open this url in your browser
https://testnet.nearblocks.io/txns/4S1VrepKzA6HxvPj3cK12vaT7Dt4vxJRWESA1ym1xdvH
''
Hurray! Hurray! At this point, your NFT contract is fully complete and the events standard has been implemented.
Conclusion
Today you went through the events standard and implemented the necessary logic in your smart contract. Today you went through the events standard and implemented the necessary logic in your smart contract. You created events for minting and transferring NFTs. You then deployed and tested your changes by minting and transferring NFTs. You then deployed and tested your changes by minting and transferring NFTs.
In the next tutorial, you'll look at the basics of a marketplace contract and how it was built.
At the time of this writing, this example works with the following versions:
- near-cli:
3.0.0
- NFT standard: NEP171, version
1.0.0
- Events standard: NEP297 extension, version
1.0.0