一文入門 NFT 製作與 OpenSea 部署

買賣虛擬貨幣

來源 |freecodecamp.org

作者 | Patrick Collins

在這篇文章裡,筆者將展示如何在沒有軟體公共技術背景下製作 NFT。然後,我們將學習如何用 Brownie、 Python、和 Chainlink 製作無限制可自定義的 NFT。以及我們也會看到如何在 OpenSea 的 NFT 市場上部署和銷售我們的創作。

如果你在找使用 Truffle、JavaScript 和有趣的中世紀人物的教程,可以看看如何《構造、部署、和出售你的NFT》 。

什麼是 NFT?

NFTs (Non-Fungible Tokens,不可置換代幣) 可以用一個詞來總結:唯一性。這些是部署在區塊鏈上的智慧合約,代表著某種東西的唯一性。

ERC20 vs ERC721

NFT 是一種區塊鏈代幣標準,與 ERC20 相似,像 AAVE、SNX、和LINK (準確來說是 ERC677)。ERC20 是“可置換”代幣,這意味著那些代幣是可被替換的 (replaceable) 和可互相兌換 (interchangeable)。

例如,無論你使用的是哪種美元鈔票,你的美元鈔票都將價值 1 美元。美元鈔票上的序列號可能不一樣,但鈔票間是可以互相兌換的,而且它們仍然價值1美元。

另一方面,NFT 是“不可置換的”,它們遵守的是它們自己的代幣標準 ERC721。例如,蒙羅麗莎是“不可置換的”。即使有人可以製作一個副本,永遠也只會有一幅蒙羅麗莎。如果蒙羅麗莎是在區塊鏈上創造的,那麼它就是個 NFT。

原始影象來自 Wikipedia

NFT 是做什麼的?

NFT 給創作者、藝術家、遊戲設計師和其他職業者提供價值,儲存在鏈上的部署還能永久儲存。

你總能知道是誰創作了 NFT、誰擁有了 NFT、NFT 的來源,等等,它們給這些創作者帶來的價值比傳統藝術更大。在傳統藝術裡,分辨贗品會有點難,但鏈上歷史是很容追蹤到的。

由於智慧合約和 NFT 是100%可程式設計的,NFT 還可以新增內建的版稅和其他功能。給藝術家補助一直是一個問題,因為情況經常是藝術家的作品廣泛傳播而人們不知道作品是誰的。

越來越多的藝術家和工程師正在躍躍欲試,參與到這個有巨大價值的領域,因為這最終是讓藝術家獲得作品報酬的好途徑。不止於此,NFT 是展示你的創造力以及在數字世界成為收藏者的有趣方式。

NFT 的價值

NFT 已經走了很長一段路了,並且我們不斷看到有拍賣價格破紀錄的 NFT,比如“Everyday:The First 5,000 Days” 以 6930 萬美元售出。

影象來自 Twitter

因此,這裡蘊含著很多價值,它也是在數字世界創作藝術和學習建立智慧合約的有趣、動態、和很有魅力的方式。那麼,現在我將教你如何製作 NFT 。

如何製作 NFT ?

文章不會涉及的內容

現在,製作 NFT 最簡單的方法是到像 Opensea、 Rarible 或 Mintible 這些平臺,跟著它們的逐步指引在它們平臺部署。

你可以完全採用這個方法,但你可能會被繫結在平臺上,受限於平臺提供的功能。你不能實現無限制的自定義製作,或真正使用 NFT 的優勢。但如果你是一個初級軟體工程師,或沒有非常豐富的技術知識,這是適合你的方法。

如果你想成為一名很強的軟體工程師,學習一些 solidity 的知識,且有能力可以用無限的創造力創造一些東西,請繼續往下看吧!

如果你剛接觸 solidity,不用擔心,我們會從基礎講起。

如何用無限制的定製製作一個 NFT

我將用這個 NFT Brownie Mix 讓你快速入門。這是一個工作程式碼庫,有非常多樣板程式碼。

準備要求

我們需要安裝下列內容才能開始:

Python

Nodejs 和 npm and npm

Metamask

如果你不熟悉 Metamask,你可以按著這個教程設定。

Rinkeby測試網上的ETH和LINK

我們還需要在 Rinkeby 以太坊測試網上工作,因此我們能免費把我們的合約部署到一個真正的區塊鏈網路上。

測試網是用來測試我們的智慧合約在現實世界執行得如何的好方法。我們需要有 Rinkeby 網路的 ETH和 LINK 測試幣,我們可以在 Chainlink 文件中最新的水龍頭連結裡免費獲得。

我們還需要新增 Rinkeby 的LINK測試幣到我們的 metamask 裡,我們可以跟著這個獲取 LINK 文件的步驟。

如果你還是很疑惑,你可以跟著這個影片去做,只是要注意是用的 Rinkeby 而不是 Ropsten。

當在像 Ethereum 這樣的智慧合約平臺上工作時,我們需要支付少量的 ETH,在需要從鏈下獲取資料時,我們需要支付少量的 LINK。這就是為什麼我們需要測試網的 LINK 和 ETH 測試幣。

正式開始吧!這是我們準備部署到 OpenSea 的 NFT。

快速開始

git clone https://github.com/PatrickAlphaC/nft-mixcd nft-mix

現在我們需要安裝ganache-cli和eth-brownie

pip install eth-brownienpm install -g ganache-cli

現在我們可以設定環境變數。如果你對環境變數不熟悉,你可以只把它們新增到你的.env文件,然後執行:

source .env

你剛剛克隆的程式碼庫裡應該有一個.env (環境)樣本,它是加了環境變數註釋的。取消註釋就可以使用它們了!

你將需要一個WEB3_INFURA_PROJECT_ID(Web3 Infura 專案 ID) 和一個 PRIVATE_KEY(私鑰)。註冊一個免費 Infura 賬號就會有這個 WEB3_INFURA_PROJECT_ID。它會使我們可以傳送交易到區塊鏈上。

你還需要一個金鑰,你可以在你的 Metamask 裡獲得。點一下那三個小點,然後點選Account Details(賬戶詳情)和Export Private Key(匯出私鑰)。如果你準備把真錢放進去的話,不要把私鑰告訴別人。

export PRIVATE_KEY=YOUR_KEY_HEREexport WEB3_INFURA_PROJECT_ID=YOUR_PROJECT_ID_HERE

.env

現在我們可以用以下兩個命令部署我們的 NFT 合約和建立我們第一件收藏品了。

brownie run scripts/simple_collectible/deploy_simple.py --network rinkebybrownie run scripts/simple_collectible/create_collectible.py --network rinkeby

第一行命令用來把我們的 NFT 合約部署到 Rinkeby 區塊鏈,第二行命令用來建立我們第一件收藏品。

你剛已經部署了我們第一份智慧合約了!

它不做什麼事,但不用擔心。我將在這份教程的高階部分展示你可以如何把它部署到 OpenSea 上。但首先,讓我們看看 ERC721 代幣標準吧。

ERC721 代幣標準

讓我們在SimpleCollectible.sol文件裡看看剛部署的合約吧。

// SPDX-License-Identifier: MITpragma solidity 0.6.6;import "@openzeppelin/contracts/token/ERC721/ERC721.sol";contract SimpleCollectible is ERC721 {    uint256 public tokenCounter;    constructor () public ERC721 ("Dogie", "DOG"){        tokenCounter = 0;    }    function createCollectible(string memory tokenURI) public returns (uint256) {        uint256 newItemId = tokenCounter;        _safeMint(msg.sender, newItemId);        _setTokenURI(newItemId, tokenURI);        tokenCounter = tokenCounter + 1;        return newItemId;    }}

我們採用的是 OpenZepplin 的 ERC721 代幣標準。我們匯入的這個包 (package) 允許我們使用一個典型 ERC721 代幣的全部功能。這定義了我們的代幣將具有的所有功能,例如transfer意味著把我們的代幣轉移給新使用者,safeMint表示建立新的代幣,等等。

在 OpenZepplin 的ERC721 代幣合約裡,你可以找到賦予我們合約的所有功能。我們的合約是透過這個命令列承繼這些功能的:

contract SimpleCollectible is ERC721 {

solidity 的承繼就是這樣工作的。當我們部署合約時,constructor會被自動呼叫,它還需要一些引數。

constructor () public ERC721 ("Dogie", "DOG"){        tokenCounter = 0;    }

我們的建構函式也使用了ERC721的建構函式,我們只需要給它取一個名字和一個符號。在我們的例子中,是"Dogie"和"DOG"。這意味著每個我們創造的 NFT 都是 Dogie/DOG 類。

這就像每張 Pokemon 卡片都是一個 pokermon,或每個在集換式卡片上的棒球手都是一名棒球手。每個棒球手是獨一無二的,但他們全部是棒球手。我們只是在用DOG類。

在頂部我們有tokenCounter,用來計數我們建立了多少這個型別的 NFT。每個新代幣都會獲得一個基於當前 tokenCounter 的 tokenId。

實際上,我們可以用 createCollectible 函式建立一個 NFT。這就是在我們的create_collectible.py指令碼里調取的函式。

function createCollectible(string memory tokenURI) public returns (uint256) {        uint256 newItemId = tokenCounter;        _safeMint(msg.sender, newItemId);        _setTokenURI(newItemId, tokenURI);        tokenCounter = tokenCounter + 1;        return newItemId;    }

_safeMint 函式建立新的 NFT,並把它分給任何呼叫了createdCollectible 的賬戶,即msg.sender,且會有一個從tokenCounter派生的 newItemId。這就是我們追蹤誰擁有什麼的方法——檢視所有者的 tokenId。

你會發現我們還呼叫 _setTokenURI。

什麼是 NFT 後設資料和 TokenURI

當智慧合約被建立了,NFT 就被建立了,人們很快會發現把大量資料部署到區塊鏈真的超級超級貴。試想一下,儲存1 KB 的資料可能就需要100萬美元。

對於 NFT 來說,這顯然是一個問題,因為擁有創意藝術意味著你必須把這個資訊儲存在某處。他們還想要一種輕量方法儲存 NFT 的屬性資訊——這就是需要 tokenURI 和後設資料的地方。

TokenURI

一個 NFT 上的 tokenURI是能展示這個代幣的唯一識別符號。一個 URI 可以是HTTPS上的 API 呼叫、一個 IPFS 的雜湊、或任何其他代表唯一的東西。

它們遵循一套顯示後設資料的標準,如下:

{    "name": "name",    "description": "description",    "image": "https://ipfs.io/ipfs/QmTgqnhFBMkfT9s8PHKcdXBn1f5bG3Q5hmBaR4U6hoTvb1?filename=Chainlink_Elf.png",    "attributes": [        {            "trait_type": "trait",            "value": 100        }    ]}

這些資料展現了一個 NFT 是什麼樣的以及它的屬性。image部分指向另一個展示該 NFT 的 URI。這使得 NFT 可以更容易在像 Opensea、Rarible和 Mintable 這些 NFT 平臺上部署,因為它們都需要這種後設資料。

鏈下後設資料 vs 鏈上後設資料

現在你可能會想“等等,如果後設資料不在鏈上,這是否意味著我的 NFT 可能會在某個時候消失了”?你的想法是對的。

如果你認為鏈下後設資料意味著你無法用那後設資料讓你的智慧合約互相互動,你的想法也是對的。

這也是為什麼我們專注在鏈上後設資料上,這樣我們才可以對我們的 NFT 進行程式設計,使它們互相互動。

儘管如此,我們仍然需要image部分的鏈下後設資料,因為我們還沒有在鏈上儲存大影象的好方法。但不要擔心,透過使用 IPFS,我們還是可以免費在一個去中心化網路實現這一點。

這裡是 IPFS 的一個 imageURI 示例,這個影象是按照《龍與地下城》教程建立的 Chainlink Elf。

The Chainlink Elf

我們沒有給簡單的 NFT 設一個 tokenURI 是因為我們只想展示一個基本示例。

現在進入高階 NFT 部分,我們可以看到用鏈上後設資料實現的一些很棒的功能,把 NFT 部署到 Opensea 上,把我們的 Dogie 放上去!

如果你想看我們剛講過部分的重溫影片,可以點選這裡。

動態和高階的 NFT

動態 NFT 是那種隨時間變化的,或具有我們可以用來互相互動的鏈上功能的 NFT。這些 NFT 給了我們無限制的自定義空間去製作整個遊戲、構建世界、以及某種互動藝術。現在讓我們進入高階部分吧。

高階部分快速開始

確認你的 metamask 裡有足夠的 ETH 和 LINK 測試幣,然後執行下面的命令列:

brownie run scripts/advanced_collectible/deploy_advanced.py --network rinkebybrownie run scripts/advanced_collectible/create_collectible.py --network rinkeby

我們這裡的收藏品是從 Chainlink VRF (Virtual Routing and Forwarding,虛擬路由和轉發) 返回的一個隨機犬種。Chainlink VRF 能提供可驗證隨機數,也就我們 NFT 真正稀缺性來源。然後,我們就要建立它的後設資料了。

brownie run scripts/advanced_collectible/create_metadata.py --network rinkeby

然後,我們可以選擇性地上載這個資料到 IPFS,這樣我們就會獲得一個 tokenURI。我在後面會展示給你看要怎麼做。現在,我只需要用下面這個樣本 tokenURI。

https://ipfs.io/ipfs/Qmd9MCGtdVz2miNumBHDbvj8bigSgTwnr4SbyH6DNnpWdt?filename=1-PUG.json

如果你下載 IPFS Companion 擴充套件程式到你的瀏覽器,你可以使用該 URL 來檢視這個 URI 返回的是什麼。它應該是下面展示的這樣:

{    "name": "PUG",    "description": "An adorable PUG pup!",    "image": "https://ipfs.io/ipfs/QmSsYRx3LpDAb1GZQm7zZ1AuHZjfbPkD6J7s9r41xu1mf8?filename=pug.png",    "attributes": [        {            "trait_type": "cuteness",            "value": 100        }    ]}

然後執行我們的 set_tokenuri.py 指令碼:

brownie run scripts/advanced_collectible/set_tokenuri.py --network rinkeby

我們得到的輸出如下:

Running 'scripts/advanced_collectible/set_tokenuri.py::main'...Working on rinkebyTransaction sent: 0x8a83a446c306d6255952880c0ca35fa420248a84ba7484c3798d8bbad421f88e  Gas price: 1.0 gwei   Gas limit: 44601   Nonce: 354  AdvancedCollectible.setTokenURI confirmed - Block: 8331653   Gas used: 40547 (90.91%)Awesome! You can view your NFT at https://testnets.opensea.io/assets/0x679c5f9adC630663a6e63Fa27153B215fe021b34/0Please give up to 20 minutes, and hit the "refresh metadata" button

我們可以點選裡面的連結看看它在 Opensea 上怎麼樣的。你可能需要點 refresh metadata ,等上幾分鐘。

隨機犬種

現在說一下我們剛做了什麼。這是我們的 AdvancedCollectible.sol:

pragma solidity 0.6.6;import "@openzeppelin/contracts/token/ERC721/ERC721.sol";import "@chainlink/contracts/src/v0.6/VRFConsumerBase.sol";contract AdvancedCollectible is ERC721, VRFConsumerBase {    uint256 public tokenCounter;    enum Breed{PUG, SHIBA_INU, BRENARD}    // add other things    mapping(bytes32 => address) public requestIdToSender;    mapping(bytes32 => string) public requestIdToTokenURI;    mapping(uint256 => Breed) public tokenIdToBreed;    mapping(bytes32 => uint256) public requestIdToTokenId;    event requestedCollectible(bytes32 indexed requestId);     bytes32 internal keyHash;    uint256 internal fee;    uint256 public randomResult;    constructor(address _VRFCoordinator, address _LinkToken, bytes32 _keyhash)    public     VRFConsumerBase(_VRFCoordinator, _LinkToken)    ERC721("Dogie", "DOG")    {        tokenCounter = 0;        keyHash = _keyhash;        fee = 0.1 * 10 ** 18;    }    function createCollectible(string memory tokenURI, uint256 userProvidedSeed)         public returns (bytes32){            bytes32 requestId = requestRandomness(keyHash, fee, userProvidedSeed);            requestIdToSender[requestId] = msg.sender;            requestIdToTokenURI[requestId] = tokenURI;            emit requestedCollectible(requestId);    }    function fulfillRandomness(bytes32 requestId, uint256 randomNumber) internal override {        address dogOwner = requestIdToSender[requestId];        string memory tokenURI = requestIdToTokenURI[requestId];        uint256 newItemId = tokenCounter;        _safeMint(dogOwner, newItemId);        _setTokenURI(newItemId, tokenURI);        Breed breed = Breed(randomNumber % 3);         tokenIdToBreed[newItemId] = breed;        requestIdToTokenId[requestId] = newItemId;        tokenCounter = tokenCounter + 1;    }    function setTokenURI(uint256 tokenId, string memory _tokenURI) public {        require(            _isApprovedOrOwner(_msgSender(), tokenId),            "ERC721: transfer caller is not owner nor approved"        );        _setTokenURI(tokenId, _tokenURI);    }}

我們使用 Chainlink VRF 從 PUG, SHIBA_INU, BRENARD列表裡建立一個隨機犬種。此時當我們呼叫 createCollectible 時,我們實際上在鏈下向 Chainlink VRF 節點發出了一個請求,然後返回一個隨機數,以三個犬種裡的一種來建立這個 NFT。

在你的 NFT 裡使用一個真正的隨機數是創造真正稀缺性的好方法,使用 Chainlink 預言機的隨機數意味著你的數字具有可驗證的隨機性,無法被礦工修改。

你可以在 Chainlink VRF 文件裡瞭解更多。

Chainlink 節點透過呼叫 fulfillRandomness函式來響應,並基於隨機數建立收藏品。然後,我們還是必須呼叫 _setTokenURI 來給 NFT 提供所需的外觀。

我們此時不設定我們 NFT 的屬性,但屬性對提升我們 NFT 的競爭力和互動性非常有用。你可以在龍與地下城的例子中看到設定了屬性的 NFT 是怎樣的。

來自 IPFS 的後設資料

我們正在使用 IPFS 來儲存兩個文件:

1.NFT 的影象 (八哥的形象)

2. tokenURI文件 (還包含了影象連結的 JSON 文件)

我們使用 IPFS 因為它是免費的去中心化平臺。透過下載 IPFS 桌面版,點選 import按鈕,我們可以把 tokenURI 和影象新增到 IPFS。

新增文件到 IPFS

然後,點選文件傍邊的三個點,點 share link,然後複製提供的連結, 我們就可以分享 URI 了。然後,我們可以把這個連結新增到我們的 set_tokenuri.py 文件,把代 tokenURI 改為我們想要使用的。

堅持去中心化

但是,如果 tokenURI 只在我們自己的節點上,這意味著如果我們的節點離線了,其他人就看不到它了。因此,我們希望別人可以 pin (掛)我們的 NFT。我們可以使用像 Pinata 這樣的服務商,即使我們的 IPFS 節點處於離線狀態,他們也可以幫助我們保持資料線上上。

我想,以後會有越來越多的後設資料儲存在 IPFS 和去中心化儲存平臺。中心化伺服器會式微,也就是說在上面的藝術品會永遠消失。因此,請務必檢查你的NFT使用的 tokenURI 的位置!

我預計以後會有越來越多人使用像 Filecoin 這樣的去中心化儲存平臺,因為使用 pinning 服務並不非常去中心化。

繼續前進

如果你想看一個高階 NFT 講解影片,你可以看這個。

現在你已經有了製作好看、有趣、可自定義、具有互動性的 NFT,並把它部署到交易平臺的技能了。

NFT 是能給藝術家的辛勤創作提供精確回報的一種有趣、強大的方式。祝你好運,享受這個過程吧!

原文連結:https://www.freecodecamp.org/news/how-to-make-an-nft-and-render-on-opensea-marketplace/

免責聲明:

  1. 本文版權歸原作者所有,僅代表作者本人觀點,不代表鏈報觀點或立場。
  2. 如發現文章、圖片等侵權行爲,侵權責任將由作者本人承擔。
  3. 鏈報僅提供相關項目信息,不構成任何投資建議

推荐阅读