NFT Metadata
Metadata: Detailed Guide for NFT Creators
This section provides a comprehensive overview of how metadata is structured in this NFT system and how you can use it within your contracts and off-chain tools. We will reference the code from @metadata.ts
, @address.ts
, @text.ts
, @tree.ts
, @pin.ts
, and @metadata.test.ts
to illustrate how it all fits together. We will also show how these structures integrate with the NFT and Collection contracts (@nft.ts
and @collection.ts
).
Overview of the Metadata Flow
-
Creation of Metadata
- You instantiate a
Metadata
object with core fields likename
,image
,description
, etc. - Additional traits (key-value pairs) can be added to the
Metadata
object usingaddTrait()
. - Sensitive or hidden traits can be marked as private, ensuring they don't appear in the public JSON representation.
- You instantiate a
-
Merkle Tree Structures
- For text fields, a specialized
Text
Merkle tree allows large strings to be hashed securely. - For an entire set of key-value pairs, a
MetadataMap
(and optionally aMetadataTree
) ensures proof-of-inclusion can be performed without revealing the entirety of the data.
- For text fields, a specialized
-
Pinning and Storing
- Once your
Metadata
object is ready, you can store it off-chain (for example, on IPFS) using the helper functionpinMetadata()
in@pin.ts
. - The
pinMetadata()
function returns anipfsHash
and themetadataRoot
. This root is crucial for verifying the integrity of the metadata inside your NFT contract.
- Once your
Core Classes and Their Roles
-
Metadata
(frommetadata.ts
)- Represents the primary container for your NFT’s data.
- Stores universal fields like
name
,image
,description
, plus a flexibleplugins
array. - Uses
MetadataMap
under the hood, which is anIndexedMerkleMap
for storing traits in a Merkle tree. - You can serialize it into JSON (with or without private traits) and also rebuild the structure using
Metadata.fromJSON()
orMetadata.fromOpenApiJSON()
.
-
Text
(fromtext.ts
)- Encodes a string into a Merkle tree at creation, ensuring large strings can be hashed in a tamper-evident way.
- When you add a trait of type
"text"
, you can store the entire text as aText
tree.
-
MinaAddress
(fromaddress.ts
)- Similar idea for addresses: it turns a
PublicKey
into a hashedField
. - Useful if you want to store addresses or partial addresses in your metadata.
- Similar idea for addresses: it turns a
-
MetadataTree
(fromtree.ts
)- An optional specialized data structure for storing sets of
(key, value)
pairs in a Merkle tree. - If you have very large or complex data, you can create a subtree of metadata, reference only its root, and prove the existence of particular entries as needed.
- An optional specialized data structure for storing sets of
Key Steps to Implement Metadata in Your NFT Project
-
Create a
Metadata
Objectconst customMetadata = new Metadata({
name: "MyAwesomeNFT",
image: "https://example.com/awesome.png",
description: "A truly awesome NFT",
plugins: [], // optionally, you could include something like [new ColorPlugin()]
}); -
Add Traits
customMetadata.addTrait({
key: "artist",
type: "string",
value: "Alice",
});
customMetadata.addTrait({
key: "color",
type: "color", // loaded via plugin (e.g., ColorPlugin)
value: "red",
});
customMetadata.addTrait({
key: "secretNote",
type: "text",
value: "A hidden surprise for the NFT owner",
isPrivate: true,
}); -
Pin It (Optional)
const { ipfsHash, metadataRoot } = await pinMetadata(customMetadata);
console.log("Pinned at:", ipfsHash);
console.log("Root is:", metadataRoot); -
Reference the Metadata in Your Contract
- When minting the NFT, add
metadataRoot
andipfsHash
to metadata and storage fields ofMintParams
. - Ensure your contract logic checks the relevant flags (e.g.,
canChangeMetadata
) before letting themetadata
be changed.
- When minting the NFT, add
-
Verifying Off-Chain
- Any user or dApp can retrieve your NFT’s metadata from IPFS and compare the
metadataRoot
in the on-chain state with the root from the pinned file. - If they match, the user knows the data is authentic and unaltered. It is possible when all the data is public. If you have a private data, then you need to prepare a proof for specific trait.
- Any user or dApp can retrieve your NFT’s metadata from IPFS and compare the
5. Testing and Examples
- Take a look at
metadata.test.ts
for real usage examples. It shows how you can create random metadata, serialize to JSON, and then deserialize it back to aMetadata
object for verification. - The test also demonstrates how to handle private vs. public traits. The JSON produced by
toJSON(false)
omits private trait data, whiletoJSON(true)
includes everything.
6. Best Practices
-
Use Private Traits Sparingly
- Mark fields as private only if you genuinely need to hide them from the public JSON.
- Any data you place in the contract state might still be discoverable if you store it fully on-chain.
-
Pin or Otherwise Persist Metadata
- Relying on ephemeral storage can break your NFT’s data integrity over time.
- IPFS or decentralized networks are common choices, but always ensure the pinned data is widely hosted.
-
Validate Before Changing
- If your NFT’s contract logic allows metadata updates, ensure an owner or admin signature is required, or that a proof is provided, especially if
canChangeMetadata
is a condition.
- If your NFT’s contract logic allows metadata updates, ensure an owner or admin signature is required, or that a proof is provided, especially if
-
Plan for Upgrades
- If you foresee changing the proof system for your metadata, keep
canChangeMetadataVerificationKeyHash = true
. - This ensures you can adopt new ZK proof mechanisms for verifying your NFT data.
- If you foresee changing the proof system for your metadata, keep
With these steps, you can confidently integrate the metadata classes (Metadata
, Text
, MinaAddress
, MetadataTree
, etc.) into your own NFT or Collection contract, ensuring your NFT data is secure, verifiable, and customizable.
Metadata Examples
Example 1
{
"name": "Arlette",
"description": "Arlette is a vivid embodiment of cultural fusion, capturing the essence of global artistry in radiant colors and fluid shapes. Seamlessly blending modern design with traditional motifs, this NFT portrays resilience, creativity, and boundless imagination. Each detail of Arlette’s composition tells a story of unity and hope, inviting observers to journey through a world where art transcends borders.",
"image": "https://picsum.photos/seed/6222197/540/670",
"metadataRoot": "8089567988850885440162712273330394926387804623534417398971006566162209699819",
"traits": [
{
"key": "name",
"type": "string",
"value": "Arlette"
},
{
"key": "image",
"type": "image",
"value": "https://picsum.photos/seed/8255952/540/670"
},
{
"key": "description",
"type": "text",
"value": "Arlette is a vivid embodiment of cultural fusion, capturing the essence of global artistry in radiant colors and fluid shapes. Seamlessly blending modern design with traditional motifs, this NFT portrays resilience, creativity, and boundless imagination. Each detail of Arlette’s composition tells a story of unity and hope, inviting observers to journey through a world where art transcends borders."
},
{
"key": "data",
"type": "field",
"value": "6554118177673577367935032062634210489286477101599894267034075633452018550482"
},
{
"key": "Willi",
"type": "url",
"value": "https://example.com/Zaria"
}
]
}
Example 2
{
"name": "Prissie",
"image": "https://picsum.photos/seed/7871885/540/670",
"metadataRoot": "23522541297594634759038873120165711491386314828070311826395608643850011570839",
"traits": [
{
"key": "name",
"type": "string",
"value": "Prissie"
},
{
"key": "image",
"type": "image",
"value": "https://picsum.photos/seed/7871885/540/670"
},
{
"key": "Flossie",
"type": "text",
"isPrivate": true,
"value": "Flossie is a vivid portrayal of boundless creativity, intertwining contemporary vibrancy with timeless motifs. She radiates captivating cultural harmonies, uniting diverse influences into one mesmerizing tapestry of color and modern design. Flossie represents resilience, inspiration, and an unbridled sense of wonder."
},
{
"key": "Maggi",
"type": "text",
"isPrivate": true,
"value": "Maggi is an enchanting NFT that captures the essence of global diversity through vibrant colors and intricate patterns. This piece represents a harmonious blend of cultural elements from around the world, showcasing the beauty of unity in diversity. The delicate balance of warm and cool tones creates a mesmerizing visual experience that invites viewers to explore its depths. Maggi embodies resilience, creativity, and the transformative power of art across boundaries."
},
{
"key": "Annice",
"type": "image",
"isPrivate": true,
"value": "https://picsum.photos/seed/9634474/540/670"
}
]
}