13
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

[ERC5489] NFTにハイパーリンクをつける機能を理解しよう!

Last updated at Posted at 2023-07-15

はじめに

初めまして。
CryptoGamesというブロックチェーンゲーム企業でエンジニアをしている cardene(かるでね) です!
スマートコントラクトを書いたり、フロントエンド・バックエンド・インフラと幅広く触れています。

代表的なゲームはクリプトスペルズというブロックチェーンゲームです。

今回は、NFTにハイパーリンクを埋め込み、ユーザーが任意のNFTをクリックして、所有者が設定した任意のURLに移動できるようにするERC5489についてまとめていきます!

以下にまとめられているものを翻訳・要約・補足しながらまとめていきます。

概要

ERC721規格のNFTの新たな拡張を提案します。
nft-hyperlink-extention(hNFT) とは、NFTにハイパーリンクを組み込むもので、hNFTsと呼ばれます。
hNFTsの所有者は、URLスロットの変更権限を特定のアドレス(外部所有アカウント[EOA]、またはコントラクトアドレス)に承認することができ、いつでもその承認を取り消すことができます。
権限を持つアドレスは、そのスロットのURLを管理することができます。

スロットとは、メタデータ、特にハイパーリンクを保存する場所を示します。
各スロットはアドレスをキー、URIを値として保持している。
例){"0xabc123...": "metadata"}
スロットをアドレスを承認」とは、特定のアドレスが特定のスロットを管理できるように許可することを意味する。

動機

NFTが注目を集めるにつれて、Web3の主要な媒体になる可能性があります。
現在、エンドユーザーはNFTにリッチテキスト、ビデオ、画像を添付することができません。
そのため、多くの業界がこのようなリッチコンテンツ添付する機能を熱望しています。
高度にカスタマイズされた情報の添付、編集、表示は、標準化されていると有用であるため、「NFT上の高度にカスタマイズされた添付ファイル」の形としてハイパーリンクを使用し、NFTにどのように添付、編集、表示するかも指定します。

仕様

インターフェース

// SPDX-License-Identifier: CC0-1.0
pragma solidity ^0.8.0;

interface IERC5489 {
    event SlotAuthorizationCreated(uint256 indexed tokenId, address indexed slotManagerAddr);

    event SlotAuthorizationRevoked(uint256 indexed tokenId, address indexed slotManagerAddr);

    event SlotUriUpdated(uint256 indexed tokenId, address indexed slotManagerAddr, string uri);

    function authorizeSlotTo(uint256 tokenId, address slotManagerAddr) external;

    function revokeAuthorization(uint256 tokenId, address slotManagerAddr) external;

    function revokeAllAuthorizations(uint256 tokenId) external;

    function setSlotUri(
        uint256 tokenId,
        string calldata newUri
    ) external;

    function getSlotUri(uint256 tokenId, address slotManagerAddr)
        external
        view
        returns (string memory);
}

Event

SlotAuthorizationCreated

特定のアドレスが、特定のNFTのURIの変更権限を得られた時に発行される。

SlotAuthorizationRevoked

特定のアドレスが、特定のNFTのURIの変更権限を取り消された時に発行される。
進行中のインセンティブや権利を停止することができます。

SlotUriUpdated

特定のアドレスが、特定のNFTのURI変更した時に発行される。

関数

authorizeSlotTo

slotManagerAddrに指定されたアドレスに、tokenIdに一致するNFTのハイバーリンク変更権限を付与する。
publicもしくはexternalとして実装して良い。

revokeAuthorization

slotManagerAddrに指定されたアドレスの、tokenIdに一致するNFTのハイバーリンク変更権限を取り消す。
publicもしくはexternalとして実装して良い。

revokeAllAuthorizations

tokenIdに一致するNFTのハイバーリンク変更権限をすべてのアドレスから取り消す。
publicもしくはexternalとして実装して良い。

setSlotUri

slotManagerAddrに指定されたアドレスがtokenIdに一致するNFTのURIを設定する。
authorizeSlotToによって権限を付与されたアドレスのみ実行できる。
publicもしくはexternalとして実装して良い。

getSlotUri

tokenIdに一致するNFTのURIを返す。
URIは「EIP5489 Metadata JSON schema」に準拠したJSONファイルである必要があります。
pureもしくはviewとして実装して良い。

認証

authorizeSlotTo関数、revokeAuthorization関数、revokeAllAuthorizations関数は、メッセージ送信者がトークンの所有者である場合にのみ実行されます。

Metadata JSON schema

{
    "title": "AD Metadata",
    "type": "object",
    "properties": {
        "icon": {
            "type": "string",
            "description": "A URI pointing to a resource with mime type image/* representing the slot's occupier. Consider making any images at a width between 48 and 1080 pixels and aspect ration between 1.91:1 and 4:5 inclusive. Suggest to show this as an thumbnail of the target resource"
        },
        "description": {
            "type": "string",
            "description": "A paragraph which briefly introduce what is the target resource"
        },
        "target": {
            "type": "string",
            "description": "A URI pointing to target resource, sugguest to follow 30X status code to support more redirections, the mime type and content rely on user's setting"
        } 
    }
}

補足

NFTにハイパーリンクを拡張

URIはさまざまなユースケースを処理するための十分な柔軟性を確保します。

スロットをアドレスに承認

我々は、スロットのキーを表現するためにアドレスを使用し、すべてのユースケースを処理するための十分な柔軟性を確保します。

後方互換性

完全にEIP721と互換性を持っている。

参照実装

//SPDX-License-Identifier: CC0-1.0
pragma solidity ^0.8.0;

import "./IERC5489.sol";

contract ERC5489 is IERC5489, ERC721Enumerable, Ownable {
    using EnumerableSet for EnumerableSet.AddressSet;

    mapping(uint256 => EnumerableSet.AddressSet) tokenId2AuthroizedAddresses;
    mapping(uint256 => mapping(address=> string)) tokenId2Address2Value;
    mapping(uint256 => string) tokenId2ImageUri;

    string private _imageURI;
    string private _name;

    constructor() ERC721("Hyperlink NFT Collection", "HNFT") {}

    modifier onlyTokenOwner(uint256 tokenId) {
        require(_msgSender() == ownerOf(tokenId), "should be the token owner");
        _;
    }

    modifier onlySlotManager(uint256 tokenId) {
        require(_msgSender() == ownerOf(tokenId) || tokenId2AuthroizedAddresses[tokenId].contains(_msgSender()), "address should be authorized");
        _;
    }

    function setSlotUri(uint256 tokenId, string calldata value) override external onlySlotManager(tokenId) {
        tokenId2Address2Value[tokenId][_msgSender()] = value;

        emit SlotUriUpdated(tokenId, _msgSender(), value);
    }

    function getSlotUri(uint256 tokenId, address slotManagerAddr) override external view returns (string memory) {
        return tokenId2Address2Value[tokenId][slotManagerAddr];
    }

    function authorizeSlotTo(uint256 tokenId, address slotManagerAddr) override external onlyTokenOwner(tokenId) {
        require(!tokenId2AuthroizedAddresses[tokenId].contains(slotManagerAddr), "address already authorized");
        
        _authorizeSlotTo(tokenId, slotManagerAddr);
    }

    function _authorizeSlotTo(uint256 tokenId, address slotManagerAddr) private {
        tokenId2AuthroizedAddresses[tokenId].add(slotManagerAddr);
        emit SlotAuthorizationCreated(tokenId, slotManagerAddr);
    }

    function revokeAuthorization(uint256 tokenId, address slotManagerAddr) override external onlyTokenOwner(tokenId) {
        tokenId2AuthroizedAddresses[tokenId].remove(slotManagerAddr);
        delete tokenId2Address2Value[tokenId][slotManagerAddr];

        emit SlotAuthorizationRevoked(tokenId, slotManagerAddr);
    }

    function revokeAllAuthorizations(uint256 tokenId) override external onlyTokenOwner(tokenId) {
        for (uint256 i = tokenId2AuthroizedAddresses[tokenId].length() - 1;i > 0; i--) {
            address addr = tokenId2AuthroizedAddresses[tokenId].at(i);
            tokenId2AuthroizedAddresses[tokenId].remove(addr);
            delete tokenId2Address2Value[tokenId][addr];

            emit SlotAuthorizationRevoked(tokenId, addr);
        }

        if (tokenId2AuthroizedAddresses[tokenId].length() > 0) {
            address addr = tokenId2AuthroizedAddresses[tokenId].at(0);
            tokenId2AuthroizedAddresses[tokenId].remove(addr);
            delete tokenId2Address2Value[tokenId][addr];

            emit SlotAuthorizationRevoked(tokenId, addr);
        }
    }

    function isSlotManager(uint256 tokenId, address addr) public view returns (bool) {
        return tokenId2AuthroizedAddresses[tokenId].contains(addr);
    }

    // !!expensive, should call only when no gas is needed;
    function getSlotManagers(uint256 tokenId) external view returns (address[] memory) {
        return tokenId2AuthroizedAddresses[tokenId].values();
    }

    function _mintToken(uint256 tokenId, string calldata imageUri) private {
        _safeMint(msg.sender, tokenId);
        tokenId2ImageUri[tokenId] = imageUri;
    }

    function mint(string calldata imageUri) external {
        uint256 tokenId = totalSupply() + 1;
        _mintToken(tokenId, imageUri);
    }

    function mintAndAuthorizeTo(string calldata imageUri, address slotManagerAddr) external {
        uint256 tokenId = totalSupply() + 1;
        _mintToken(tokenId, imageUri);
        _authorizeSlotTo(tokenId, slotManagerAddr);
    }

    function tokenURI(uint256 _tokenId) public view override returns (string memory) {
        require(
            _exists(_tokenId),
            "URI query for nonexistent token"
        );

        return
            string(
                abi.encodePacked(
                    "data:application/json;base64,",
                    Base64.encode(
                        bytes(
                            abi.encodePacked(
                                '{"name":"',
                                abi.encodePacked(
                                    _name,
                                    " # ",
                                    Strings.toString(_tokenId)
                                ),
                                '",',
                                '"description":"Hyperlink NFT collection created with Parami Foundation"',
                                '}'
                            )
                        )
                    )
                )
            );
    }
}

最後に

今回は「NFTにハイパーリンクを埋め込み、ユーザーが任意のNFTをクリックして、所有者が設定した任意のURLに移動できるようにするERC5489」についてまとめてきました。
いかがだったでしょうか?
実装については今後追記していきます。

質問などがある方は以下のTwitterのDMなどからお気軽に質問してください!

Twitter @cardene777

採用強化中!

CryptoGamesでは一緒に働く仲間を大募集中です。

この記事で書いた自分の経験からもわかるように、裁量権を持って働くことができて一気に成長できる環境です。
「ブロックチェーンやWeb3、NFTに興味がある」、「スマートコントラクトの開発に携わりたい」など、少しでも興味を持っている方はまずはお話ししましょう!

13
6
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
13
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?