💪

ノンプログラマーでもできる!! フルオンチェーン NFTの作り方(初心者向け)

2023/01/15に公開

フルオンチェーンNFTのスマートコントラクトを簡単に作成する方法を紹介します。

初めてブロックチェーンのプログラムを触るという方向けに、できるだけわかりやすく書きたいと思いますので、ここがよくわからん、などあればフィードバックいただけると嬉しいです。

0.はじめに

Ethereumに大きなデータを書き込むとガス代が大変なことになるので、ある程度綺麗な画像をフルオンチェーンにしようとすると、jpegやpngでなく、SVGという形式を使います。
そして、ブラウザで表示するときには、更にSVGをBase64という形式に変換しないといけません。
(SVG, Base64の詳細は各自でお調べください)

今回はSVGの作成、Base64の変換は別で実施し、作成するコントラクトでは変換したデータを書き込むようにします。

1.画像データの準備

SVG画像は、イラストレータで作成した画像の出力形式をSVGに指定して出力したり、巷の画像変換サービスを利用したりします。

  • googleで検索したら https://convertio.co/ja/ などが出てきました。このようなサービスを使ってSVG形式の画像ファイルを作成します。
  • こういったサービスで作成したSVGは余分なコードが多く含まれているため、これを削減するテクニックがあるそうです(ここでは割愛します)

SVGファイルができたら、今度はこのファイルをbase64形式に変換します。
これも巷に無料の変換サービスがあるので、使ってください。

2.作成するスマートコントラクトの概要

今回作成するスマートコントラクトは、3つのファイルで構成されています。

一つは、 ERC721SVG.sol というもので、これがフルオンチェーンNFTの中核をなします。トークンIDごとの画像データを保管します。

二つ目は、 SampleSVG.sol というERC721SVGを継承したものです。継承という専門用語を使いましたが、これは継承元に書かれているプログラムを簡単に使えるようにする仕組みです。
これを読んでいる皆さんが修正するのは、このファイルのみです。

三つ目は、Base64.solで、ERC721SVGから使用します。コントラクトからトークンIDの画像をリクエストされた際、ブラウザで解釈できるようにBase64形式に変換します。

3.実践

いよいよ実践です。
今回は、Ethereumのテスト用ネットワークである、Goerliチェーンにデプロイ(アップロード)します。
Goerliの設定は https://qiita.com/ItodaiCrypto/items/563214a28239c6f9cf0e を参考にどうぞ。
Goerliチェーンにデプロイするときには、Goerli専用のETHが必要となります。 ETHは https://goerlifaucet.com/ で無料取得することができますが、Alchemyというサービスにユーザ登録が必要だったりで、ちょっとめんどいです。難しい場合はお送りするのでTwitterDMでご連絡ください。

3-1.REMIXプロジェクトの作成

デプロイには、REMIXというオンラインの統合開発環境を使用します。
https://remix.ethereum.org/ へアクセスしてください。Chromeからの使用が安定しているようです。

アクセスしたら、WORKSPACE横の”+”ボタン

テンプレートに "Blank"を指定してOK

3-2.プログラムコードのコピペ

新規ファイル作成のボタンをクリックして、"ERC721SVG.sol"の空ファイルを作成します。
同様に、"SampleSVG.sol", "Base64.sol"も作成してください。

"ERC721SVG.sol"に以下のコードをコピーして貼り付けてください。

ERC721SVG.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.14;

import "@openzeppelin/contracts/token/ERC721/extensions/ERC721URIStorage.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import "./Base64.sol";

abstract contract ERC721SVG is ERC721URIStorage, Ownable {
    bytes32 public constant CONTRACT_ADMIN = keccak256("CONTRACT_ADMIN");
    uint256 public currentTokenId;

    function setTokenURI(
        uint256 tokenId,
        string memory _encodedData,
        string memory _name,
        string memory _description
    ) public virtual onlyOwner {
        string memory _tokenURI;

        _tokenURI = string(
            abi.encodePacked(
                "data:application/json;base64,",
                Base64.encode(
                    bytes(
                        abi.encodePacked(
                            '{"name": "',
                            _name,
                            '", "description": "',
                            _description,
                            '", "image": "data:image/svg+xml;base64,',
                            _encodedData,
                            '"}'
                        )
                    )
                )
            )
        );

        _setTokenURI(tokenId, _tokenURI);
    }

    function createToken(
        address _to,
        string memory _encodedData,
        string memory _name,
        string memory _description
    ) public virtual onlyOwner returns (uint256 _tokenId) {
        _tokenId = ++currentTokenId;
        _safeMint(_to, _tokenId);
        setTokenURI(_tokenId, _encodedData, _name, _description);
    }

    function supportsInterface(bytes4 interfaceId)
        public
        view
        virtual
        override(ERC721)
        returns (bool)
    {
        return ERC721.supportsInterface(interfaceId);
    }
}

ペーストすると本当にコピーして良いかの確認ダイアログが表示されます。
変なコードは仕込んでいませんが(笑)、自己責任でご確認ください。

同様に、"Base64.sol","SampleSVG.sol" にも以下のコードをコピーして貼り付けてください。

Base64.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

// @title Base64
// @author Brecht Devos <brecht@loopring.org>
// @notice Provides a function for encoding some bytes in base64
library Base64 {
    string internal constant TABLE =
        "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

    function encode(bytes memory data) internal pure returns (string memory) {
        if (data.length == 0) return "";

        // load the table into memory
        string memory table = TABLE;

        // multiply by 4/3 rounded up
        uint256 encodedLen = 4 * ((data.length + 2) / 3);

        // add some extra buffer at the end required for the writing
        string memory result = new string(encodedLen + 32);

        assembly {
            // set the actual output length
            mstore(result, encodedLen)

            // prepare the lookup table
            let tablePtr := add(table, 1)

            // input ptr
            let dataPtr := data
            let endPtr := add(dataPtr, mload(data))

            // result ptr, jump over length
            let resultPtr := add(result, 32)

            // run over the input, 3 bytes at a time
            for {

            } lt(dataPtr, endPtr) {

            } {
                dataPtr := add(dataPtr, 3)

                // read 3 bytes
                let input := mload(dataPtr)

                // write 4 characters
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(18, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(12, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(6, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(input, 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
            }

            // padding with '='
            switch mod(mload(data), 3)
            case 1 {
                mstore(sub(resultPtr, 2), shl(240, 0x3d3d))
            }
            case 2 {
                mstore(sub(resultPtr, 1), shl(248, 0x3d))
            }
        }

        return result;
    }
}
SampleSVG.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.17;

import "./ERC721SVG.sol";

contract SampleSVG is ERC721SVG {
    constructor() ERC721("SampleSVG", "SSVG") {
    }
}

3-3.プログラムコードの修正

修正する箇所は、SampleSVG.solの2箇所のみです。

  • 6行目: contract SampleSVG is ERC721SVG {
    SampleSVGをご自分の発行するコントラクト名に変更してください。
    ※contact名とファイル名は原則一致させるため、ファイル名も6行目に合わせて修正してください。(修正しなくても続行可能)
  • 7行目: constructor() ERC721("SampleSVG", "SSVG") {
    SampleSVGとSSVGも同様にご自分の発行するものに書き換えてください。SSVGは任意の短縮名です。

3-4.コンパイル

通常、 SampleSVG.solを修正すると、プログラムは自動でコンパイルされます。
コンパイル結果は、左メニューのコンパイルマークをクリックして確認できます。 SampleSVG.solを選択した状態でクリックしてください。

3-5.デプロイ

Goerliチェーンにデプロイします。
左メニューからデプロイマークをクリック後、ENVIRONMENTに InjectProvider - MetaMaskを指定してください。

このとき、MetaMaskのネットワークに「Goerliテストネットワーク」が選択されていることを確認してください。

問題なければ、以下のようにENVIRONMENTの下に「Goerli(5) network」、Accountにご自分のMetamaskアドレス、Contractに「SampleSVG」が表示されているはずです。

これを確認後、Deployボタンをクリックすると、Metamaskの確認画面が立ち上がり、確認ボタンでデプロイのトランザクションが走ります。

トランザクションが通ると、左画面下側のDeployed Contractにコントラクトアドレスが表示されます。

右側のコピーマークをクリックして、コントラクトアドレスを控えておいてください。(後ほどOpenseaで確認します)

3-6.SVGデータの登録

「1.画像データの準備」で準備したデータを使用します。
ここにサンプルデータを貼り付けていますので、とりあえずお試しされる場合はこちらをご利用ください。

base64変換後のSVG
<svg width="320" height="320" viewBox="0 0 320 320" xmlns="http://www.w3.org/2000/svg" shape-rendering="crispEdges"><rect width="100%" height="100%" fill="#e1d7d5" /><rect width="140" height="10" x="90" y="210" fill="#343235" /><rect width="140" height="10" x="90" y="220" fill="#343235" /><rect width="140" height="10" x="90" y="230" fill="#343235" /><rect width="140" height="10" x="90" y="240" fill="#343235" /><rect width="20" height="10" x="90" y="250" fill="#343235" /><rect width="110" height="10" x="120" y="250" fill="#343235" /><rect width="20" height="10" x="90" y="260" fill="#343235" /><rect width="110" height="10" x="120" y="260" fill="#343235" /><rect width="20" height="10" x="90" y="270" fill="#343235" /><rect width="110" height="10" x="120" y="270" fill="#343235" /><rect width="20" height="10" x="90" y="280" fill="#343235" /><rect width="110" height="10" x="120" y="280" fill="#343235" /><rect width="20" height="10" x="90" y="290" fill="#343235" /><rect width="110" height="10" x="120" y="290" fill="#343235" /><rect width="20" height="10" x="90" y="300" fill="#343235" /><rect width="110" height="10" x="120" y="300" fill="#343235" /><rect width="20" height="10" x="90" y="310" fill="#343235" /><rect width="110" height="10" x="120" y="310" fill="#343235" /><rect width="10" height="10" x="120" y="210" fill="#ffc110" /><rect width="10" height="10" x="190" y="210" fill="#ffc110" /><rect width="10" height="10" x="120" y="220" fill="#ffc110" /><rect width="10" height="10" x="190" y="220" fill="#ffc110" /><rect width="10" height="10" x="130" y="230" fill="#ffc110" /><rect width="10" height="10" x="180" y="230" fill="#ffc110" /><rect width="40" height="10" x="140" y="240" fill="#ffc110" /><rect width="20" height="10" x="150" y="250" fill="#ffc110" /><rect width="40" height="10" x="140" y="260" fill="#ffc110" /><rect width="20" height="10" x="140" y="270" fill="#ffc110" /><rect width="20" height="10" x="160" y="280" fill="#ffc110" /><rect width="40" height="10" x="140" y="290" fill="#ffc110" /><rect width="20" height="10" x="150" y="300" fill="#ffc110" /><rect width="20" height="10" x="150" y="10" fill="#ffc110" /><rect width="20" height="10" x="150" y="20" fill="#ffc110" /><rect width="80" height="10" x="120" y="30" fill="#ffc110" /><rect width="20" height="10" x="110" y="40" fill="#ffc110" /><rect width="10" height="10" x="130" y="40" fill="#ffffff" /><rect width="70" height="10" x="140" y="40" fill="#ffc110" /><rect width="20" height="10" x="100" y="50" fill="#ffc110" /><rect width="10" height="10" x="120" y="50" fill="#ffffff" /><rect width="90" height="10" x="130" y="50" fill="#ffc110" /><rect width="20" height="10" x="90" y="60" fill="#ffc110" /><rect width="10" height="10" x="110" y="60" fill="#ffffff" /><rect width="110" height="10" x="120" y="60" fill="#ffc110" /><rect width="140" height="10" x="90" y="70" fill="#ffc110" /><rect width="140" height="10" x="90" y="80" fill="#ffc110" /><rect width="160" height="10" x="80" y="90" fill="#ffc110" /><rect width="160" height="10" x="80" y="100" fill="#ffc110" /><rect width="160" height="10" x="80" y="110" fill="#ffc110" /><rect width="160" height="10" x="80" y="120" fill="#ffc110" /><rect width="160" height="10" x="80" y="130" fill="#ffc110" /><rect width="160" height="10" x="80" y="140" fill="#d08b11" /><rect width="160" height="10" x="80" y="150" fill="#d08b11" /><rect width="200" height="10" x="60" y="160" fill="#ffc110" /><rect width="10" height="10" x="60" y="170" fill="#ffc110" /><rect width="10" height="10" x="70" y="170" fill="#ffffff" /><rect width="180" height="10" x="80" y="170" fill="#ffc110" /><rect width="120" height="10" x="40" y="180" fill="#ffc110" /><rect width="30" height="10" x="160" y="180" fill="#e8705b" /><rect width="90" height="10" x="190" y="180" fill="#ffc110" /><rect width="240" height="10" x="40" y="190" fill="#ffc110" /><rect width="220" height="10" x="50" y="200" fill="#ffc110" /><rect width="60" height="10" x="100" y="110" fill="#2b83f6" /><rect width="60" height="10" x="170" y="110" fill="#2b83f6" /><rect width="10" height="10" x="100" y="120" fill="#2b83f6" /><rect width="20" height="10" x="110" y="120" fill="#ffffff" /><rect width="20" height="10" x="130" y="120" fill="#000000" /><rect width="10" height="10" x="150" y="120" fill="#2b83f6" /><rect width="10" height="10" x="170" y="120" fill="#2b83f6" /><rect width="20" height="10" x="180" y="120" fill="#ffffff" /><rect width="20" height="10" x="200" y="120" fill="#000000" /><rect width="10" height="10" x="220" y="120" fill="#2b83f6" /><rect width="40" height="10" x="70" y="130" fill="#2b83f6" /><rect width="20" height="10" x="110" y="130" fill="#ffffff" /><rect width="20" height="10" x="130" y="130" fill="#000000" /><rect width="30" height="10" x="150" y="130" fill="#2b83f6" /><rect width="20" height="10" x="180" y="130" fill="#ffffff" /><rect width="20" height="10" x="200" y="130" fill="#000000" /><rect width="10" height="10" x="220" y="130" fill="#2b83f6" /><rect width="10" height="10" x="70" y="140" fill="#2b83f6" /><rect width="10" height="10" x="100" y="140" fill="#2b83f6" /><rect width="20" height="10" x="110" y="140" fill="#ffffff" /><rect width="20" height="10" x="130" y="140" fill="#000000" /><rect width="10" height="10" x="150" y="140" fill="#2b83f6" /><rect width="10" height="10" x="170" y="140" fill="#2b83f6" /><rect width="20" height="10" x="180" y="140" fill="#ffffff" /><rect width="20" height="10" x="200" y="140" fill="#000000" /><rect width="10" height="10" x="220" y="140" fill="#2b83f6" /><rect width="10" height="10" x="70" y="150" fill="#2b83f6" /><rect width="10" height="10" x="100" y="150" fill="#2b83f6" /><rect width="20" height="10" x="110" y="150" fill="#ffffff" /><rect width="20" height="10" x="130" y="150" fill="#000000" /><rect width="10" height="10" x="150" y="150" fill="#2b83f6" /><rect width="10" height="10" x="170" y="150" fill="#2b83f6" /><rect width="20" height="10" x="180" y="150" fill="#ffffff" /><rect width="20" height="10" x="200" y="150" fill="#000000" /><rect width="10" height="10" x="220" y="150" fill="#2b83f6" /><rect width="60" height="10" x="100" y="160" fill="#2b83f6" /><rect width="60" height="10" x="170" y="160" fill="#2b83f6" /></svg>

Deployed ContractのSampleSVGを展開すると、以下のように関数が一覧されます。

この中のcreateTokenがSVG画像のトークンを作成する関数です。右側の展開マークをクリックしてください。

以下のパラメタを設定し、「transact」をクリック、MetaMaskで確認するとトランザクションが実行されます。

  • _to: このSVGトークンをミントするアドレス
  • _encodedData: base64変換後のSVG
  • _name: このトークンの名前
  • _description: このトークンの説明
    トランザクションが通ると右下に以下のようなメッセージが表示されます。

同じコレクションに複数のSVGを登録したい場合は、このcreateTokenで別画像を登録します。

4.Openseaで確認

以上でSVGのトークンが発行されたため、Openseaで閲覧できる状態になりました。
テストネット版Opensea (https://testnets.opensea.io/ja) にアクセスし、「3-5.デプロイ」で控えたコントラクトアドレスを検索してください。
無事、登録したSVG画像が表示されれば成功です!!

Openseaへの反映まで時間がかかる場合があります。検索できない場合は時間が経ってから確認してみてください。

5.まとめ

以上、フルオンチェーンNFTを作成するコントラクトのデプロイ手順でした。
今回はGoerliテストネットへのデプロイしていますが、この手順に慣れたらEthereumのメインネットへデプロイすることも、そう難しくないと思います。どうぞ、お試しください。

Discussion