哪些信息持有 NFT?

Which information hold an NFT?

我正在使用 solidity 开发 NFT 市场,特别是我正在 OpenZeppelin 的 ERC-721 智能合约之上创建我自己的智能合约。我现在的NFT图片有5个属性(id、image、description、collection和image),我保存了上传时ipfs生成的hash。

我的问题是在哪里保存所有这些属性,因为我有具有上述属性的图像结构,我将它添加到一个数组中,并使用数组中图像对象的 ID 和创建者的地址。我的意思是,我保存了 ERC-721 合约之外的所有信息,所以我不太了解 NFT 是什么,因为属性不是来自 NFT,而是 NFT 是我的结构的属性。

我是否正确执行它,ERC-721 标准只是 NFT 的必要功能,还是我将信息保存在它不接触的地方?

我的代码目前如下:

pragma solidity ^0.5.0;

import "./ERC721Full.sol";

contract NftShop is ERC721Full {
  string public name;
  Image[] public nft;
  uint public imageId = 0;
  mapping(uint => bool) public _nftExists;
  mapping(uint => Image) public images;

  struct Image {
    uint id;                  //id of the nft
    string hash;              //hash of the ipfs            
    string description;       //nft description
    string collection;        //what collection the nft bellongs
    address payable author;   //creator of the nft
  }

  //Event used when new Token is created
  event TokenCreated(
    uint id,
    string hash,
    string description,
    string collection,
    address payable author
  );

  constructor() public payable ERC721Full("NftShop", "NFTSHOP") {
    name = "NftShop";
  }

  //uploadImage to the blockchain and mint the nft.
  function uploadImage(string memory _imgHash, string memory _description, string memory _collection) public {
    // Make sure the image hash exists
    require(bytes(_imgHash).length > 0);
    // Make sure image description exists
    require(bytes(_description).length > 0);
    // Make sure collectionage exists
    require(bytes(_collection).length > 0);
    // Make sure uploader address exists
    require(msg.sender!=address(0));

    // Increment image id
    imageId ++;

    // Add Image to the contract
    images[imageId] = Image(imageId, _imgHash, _description, _collection, msg.sender);

    //Mint the token
    require(!_nftExists[imageId]);
    uint _id = nft.push(images[imageId]);
    _mint(msg.sender, _id);
    _nftExists[imageId] = true;

    // Trigger an event
    emit TokenCreated(imageId, _imgHash, _description, _collection, msg.sender);
  }
} 

如果有任何奇怪的地方,欢迎提出任何关于如何改进代码的建议。

我希望这不是一个荒谬的问题,我是从以太坊的世界开始的。

非常感谢。

是的,图像地址等信息存储在继承自ERC721的合约中。 ERC721 主要跟踪代币的所有权、铸造和转移。

您可能要考虑的一件事是令牌的唯一性。在您的示例中,可以存在许多具有相同图像和描述的标记。 您可能希望通过存储 _imgHash、_description、_collection 的哈希值并要求它是唯一的来防止这种情况发生,这样任何用户都无法创建现有令牌的“副本”。

类似于:

keccak256(abi.encodePacked(_imgHash, _description, _collection))

另一个经常使用的标准是 Opensea 元数据标准,其中 tokenURI 存储在链上,元数据位于区块链之外。 https://docs.opensea.io/docs/metadata-standards