以太幣交易所 以太幣交易所
Ctrl+D 以太幣交易所
ads
首頁 > 酷幣 > Info

還記得當年的百度貼吧嗎? 今天有人寫了一個去中心化的..._POST

Author:

Time:1900/1/1 0:00:00

今天科技圈最大的新聞莫過于百度李彥宏被“澆水”一事了,微博、微信、今日頭條可謂是炸開了鍋,但想想要是10年前,討論最火的地方可能不是這些app,無疑是百度貼吧了,但可能卻面臨刪帖的危險...

這時,區塊鏈的不可篡改行就幫上了大忙!

今天營長就使用DApp開發框架Embark,手把手教你構建一個去中心化的社交新聞網站,從主要分以下三個部分:

明確DApp需求,部署智能合約;

使用DApp開發框架Embark的JavaScript程序庫EmbarkJS測試智能合約;

使用JavaScript用戶界面框架React構建DApp的前端。

明確DApp需求,部署智能合約。

百度貼吧是一個功能非常復雜的平臺,因此我們無法做到把它全部推倒重建,我們只會構建出百度貼吧的一些核心功能,并在構建中詳細介紹如何使用Embark框架構建DApp。

我們的構想非常簡單:首先我們給DApp取名為DReddit(去中心化的百度貼吧),它允許用戶在其中發布帖子,而其他用戶可以憑興趣以及帖子的質量對帖子進行好評和差評的投票。為了簡化開發,DReddit直接使用以太坊錢包賬戶作為用戶帳戶,也就是說每個以太坊錢包賬戶都是該應用程序的有效帳戶,用戶可以使用基于瀏覽器的以太坊輕錢包Metamask等擴展程序進行身份認證。

我們將創建一個智能合約來實現發布帖子以及對帖子投票的功能。同時為了簡化用戶的交互過程,我們還會使用React框架構建一個用戶界面。

1、應用程序設置

首先,安裝Embark框架,命令如下:

npminstall-gembark

使用new命令來創建并設置應用程序:

embarknewdredditcddreddit

使用cd命令進入文件夾之后,我們可以看到應用程序的文件結構,在其中最重要的文件夾是用來存放智能合約的contracts,以及用來存放前端程序的app。

Bitfinex將于5月17日支持CryptoGPT(GPT)更名為LayerAI(LAI):5月11日消息,Bitfinex將于2023年5月17日上午11點(UTC時間)支持CryptoGPT(GPT)更名為LayerAI(LAI)。Bitfinex上的CryptoGPT(GPT)交易、存取款將于2023年5月16日上午11:00(UTC時間)停止,請客戶在此之前取消任何與CryptoGPT(GPT)的未結訂單,屆時系統將取消所有剩余的未結訂單。LayerAI(LAI)的交易、存取款預計將在更名完成后不久開放(預計在2023年5月17日),支持LAI/USD和LAI/USDT交易對。[2023/5/11 14:56:54]

2、創建智能合約

使用Solidity語言編寫智能合約,在其中加入創建帖子功能和投票功能。

在contracts文件夾下創建智能合約文件DReddit

上述結構體只能用來存儲單個帖子,在多個帖子場景中,我們需要添加一個數組來存儲多個帖子結構體,代碼如下:

Postpublicposts;

a)新建帖子

創建函數createPost,其中參數_description是用來表示帖子內容的字節型數據。

functioncreatePost(bytes_description)public{uintpostId=posts

在函數中,我們為存儲的帖子創建一個序號id,然后使用剛剛定義的帖子結構體Post創建一個新的實例。

b)發布帖子

創建一個新的事件類型NewPost,代碼如下:

eventNewPost(uintindexedpostId,addressowner,bytesdescription)

定義完成后,在新建帖子函數createPost中使用所需的數據執行NewPost:

Candy Digital未受Fanatics撤資影響推出新款MLB NFT:金色財經報道,體育NFT公司Candy Digital未受Fanatics撤資影響推出新款MLB NFT,據悉美國職棒大聯盟將與其共同推出2023賽季數字藏品系列“2023 MLB Shouwstopper ICON”,其中包含43為MLB球星。此外,Candy Digital首席執行官Scott Lawin還透露Fanatics目前仍然是其投資方之一,只是規模較小,而且暗示未來可能還有合作的機會。(decrypt)[2023/3/30 13:35:08]

functioncreatePost(bytes_description)public{..

c)好評/差評

DReddit允許用戶對帖子進行好評差評投票。為實現這一功能,我們需要使用投票計數器來擴展之前定義的帖子結構體Post,并引入一個代表投票類型的枚舉結構。為了方便前端應用程序調用,我們需要添加一個新建投票事件NewVote。完成后,我們還需要添加一個用來執行投票的方法。

首先,定義一個表示投票種類的枚舉類型Ballot,其中可選的投票類型包括好評UPVOTE、差評DOWNVOTE、不投票NONE:

enumBallot{NONE,UPVOTE,DOWNVOTE}

為存儲每個帖子中的投票紀錄,我們需要在帖子結構體Post中相應地加入“好評”投票計數器和“差評”投票計數器。為確保用戶不會重復投票,我們還需要添加一個用來存儲所有已投票用戶以及投票的映射:

structPost{..

現在的新建投票事件NewVote應該如下所示:

eventNewVote(uintindexedpostId,addressowner,uint8vote);

由于帖子結構體Post中加入了投票計數器,需要用新的結構體更新createPost():

Ankr回應被盜:正在與交易平臺合作以立即停止交易:12月2日,Ankr在其官方社交平臺表示,其aBNB被盜,目前與交易平臺合作以立即停止交易。

此前報道,Ankr遭黑客攻擊,10萬億枚aBNBc被鑄造,其在Pancake上的交易池流動性被掏空,aBNBc價格已幾乎歸零。[2022/12/2 21:17:24]

functioncreatePost(bytes_description)public{..

現在萬事俱備,只欠投票函數vote()了!!!

函數的參數_vote就是我們剛剛定義的投票枚舉類型Ballot,它的取值為0、1、2這三個無符號整數,分別對應三種類型的投票。

使用Solidity的require()語句確保用戶只能對實際存在的帖子進行投票及用戶不能對同一個帖子多次投票。

在函數中,我們用當前的投票類型更新“好評”投票計數器或“差評”投票計數器,存儲已投票用戶的信息并發出新建投票事件NewVote:

functionvote(uint_postId,uint8_vote)public{Poststoragepost=posts;require(post

else{post

post

d)判斷用戶是否可以投票

在前端中,我們希望向用戶展示自己是否已經對帖子進行了投票。為此,定義一個可以判斷用戶能否對帖子投票的API將大大簡化這個過程。判斷用戶是否可以投票的過程非常簡單,只需要判斷該帖子中是否存在該用戶的投票,判斷代碼如下:

functioncanVote(uint_postId)publicviewreturns(bool){if(_postId>posts

e)獲取投票信息

如果你想瀏覽自己過去的投票信息怎么辦?很簡單,一個簡單的函數getVote()就可以實現,代碼如下:

Cardano基金會與歐洲交易所WhiteBIT達成合作伙伴關系:金色財經報道,據歐洲加密貨幣交易所 WhiteBIT 官方網站,Cardano基金會已與該交易所達成合作伙伴關系并上線 ADA 永續合約并支持 ADA 的 P2P轉賬,以及探討構建可擴展、安全、高效的去中心化網絡。[2022/11/17 13:15:33]

functiongetVote(uint_postId)publicviewreturns(uint8){Poststoragepost=posts;returnuint8(post

到這里,部署智能合約大功告成!

使用EmbarkJS測試智能合約

前面已經部署了DReddit智能合約,并在智能合約中實現了發布帖子和給帖子投票的功能,接下來就需要使用Embark框架為智能合約編寫一些測試。

1、編寫第一個測試

先從最簡單的功能開始測試。

首先,我們需要在測試文件夾test中創建一個測試文件DReddit_spec

);});

運行測試命令embarktest,輸出如下:

所有測試都成功通過,接下來測試一些實際的功能!

2、測試帖子的創建過程

測試創建帖子:首先以某種方式在JavaScript中導入DReddit智能合約的實例,然后調用智能合約中的各個方法測試它們能否正常工作,同時我們還需要配置測試環境來正確創建智能合約的實例。

a)導入智能合約實例

在運行測試時,Embark框架會在全局范圍加入一些必要的自定義函數和對象。其中一個就是自定義獲取函數require(),它可以幫助我們從特定的Embark路徑中導入智能合約實例。

就比如說,為了在測試中導入DReddit智能合約的實例,我們需要在spec文件中添加如下的命令:

NFT鑄造平臺Fair.xyz與OpenSea合作推出靈魂綁定代幣Minter Token:9月8日消息,針對創作者的NFT鑄造平臺Fair.xyz與OpenSea合作推出靈魂綁定代幣Minter Token,以作為創作者與OG社區互動的新方式,它可直接集成到智能合約中,允許為NFT項目的鑄幣者構建封閉式體驗,并兼容所有歷史、現在和未來的NFT。其中用戶可進行免費Claim,前500名免Gas費,同時可通過分享Minter Token以參與RTFKT贈品的抽獎活動。[2022/9/8 13:15:39]

constDReddit=require('Embark/contracts/DReddit');

DReddit現在被指定為一個EmbarkJS的智能合約實例,我們需要使用設置函數config()讓Embark框架知道,我們需要的智能合約都有哪些。設置函數config(),以便Embark框架知道我們需要哪些智能合約:

config({contracts:{DReddit:{}}});

這個操作與配置智能合約的操作非常相似,實際上它就相當于在測試環境中配置智能合約。我們將所需的智能合約作為參數,通過配置對象將它傳遞給設置函數config()。在我們這個應用程序中,需要設置的參數只有DReddit,這是因為我們的智能合約并不需要構造函數。

b)測試創建帖子函數createPost()

導入好智能合約實例之后,我們就可以測試智能合約的創建帖子函數createPost()了。不過在定義createPost函數時,我們指定了帖子的描述為字節形式,如何測試呢?

首先我們需要說明的是為什么要用字節形式的數據。我們都知道,帖子的長短不好控制,有些帖子很長,有些帖子很短,所以最好的方案就是將帖子的描述(內容)存儲在一個并不在意數據大小的地方,而在智能合約之中存儲的只是帖子描述的哈希值。通過使用哈希值我們可以保證數據的索引與數據一一對應,同時智能合約中存儲的數據索引始終具有相同的長度,所以我們將帖子真正的描述存儲在IPFS中,而創建帖子函數createPost中的帖子描述實際上是帖子描述的IPFS哈希值。

在得到帖子描述的哈希值后(代碼中選用之前準備好的哈希值),我們可以使用Web3程序庫的fromAscii()工具函數將該哈希值轉換為字節,然后使用智能合約的創建帖子函數createPost將它發送出去。在測試時,我們可以檢索剛才發出的事件,并檢查它的返回值,這些操作的代碼如下所示:

..

);});

運行測試命令embarktest,兩條測試都測試通過!

3、測試數據的正確性

需要測試的另外一個功能是,存儲的數據(帖子的描述,所有者)是否能解析回正確的數據。這就要用到先前定義的全局可見的帖子序號postId。我們還需要執行與先前測試類似的檢查,如果要測試帖子的所有者數據是否正確,我們首先需要訪問創建帖子的帳戶。

Embark框架的設置函數config可以讓我們輕松地訪問錢包帳戶,我們所要做的就是將一個解析處理程序加入到設置函數config中并存儲傳遞的值:

..

);

完成了操作后,測試代碼如下:

it('postshouldhavecorrectdata',async()=>{constpost=awaitDReddit

);

注意到,代碼中引用了帳戶accounts,但僅僅通過查看代碼,我們無法確定賬戶account是否是我們指定的那個賬戶。而Embark框架可以幫助我們解決這個問題,在設置完帳戶后,Embark框架會自動將錢包的第一個帳戶(accounts)設置為用于發起交易的默認帳戶。這種特性讓我們可以確定,賬戶accounts會是帖子的所有者。

另一種方法是將所有帳戶發送給智能合約的send()函數,在這種情況下,我們可以決定使用哪個賬戶發起交易。

4、測試能否投票函數canVote()

接下來我們來測試能否投票函數canVote()是否按預期的方式工作。很簡單,用戶不能給不存在的帖子投票,因此測試只需要用能否投票函數判斷不存在的帖子序號postId。測試代碼如下:

it('shouldnotbeabletovoteinanunexistingpost',async()=>{constuserCanVote=awaitDReddit

);

不過,當用戶確實可以給某個帖子投票時,我們要確保能否投票函數canVote()的返回值是能true,我們需要用該函數來判斷之前存儲的帖子序號postId:

it('shouldbeabletovoteinapostifaccounthasnotvotedbefore',async()=>{constuserCanVote=awaitDReddit

);

很棒,我們現在完成了5個測試!

5、測試投票函數vote()

投票功能可謂是我們應用程序的核心功能,因而對它的測試是重中之重。我們有許多種不同的方法驗證投票函數vote()的功能是否符合預期,但在本教程中,我們只檢查新建投票事件NewVote發出投票的所有者帳戶是否與真正執行投票的帳戶相同,在代碼實現中我們可以借鑒先前的測試:

it("shouldbeabletovoteinapost",async()=>{constreceipt=awaitDReddit

);

5、測試每個用戶每個帖子只能投一票

在智能合約定義中,我們設置了每個用戶對每個帖子只能投一票,因而最后一個也是最必要的一個測試就是檢查智能合約是否允許用戶在同一帖子上多次投票。這個測試中我們又用到了async/await異步操作的方法,同時還用到了try/catch來更好地進行測試。當用戶對一個已經投過票的帖子再次進行投票時,投票函數vote()將執行失敗,這個操作我們可以使用斷言(assert)方法來實現:

it('shouldnotbeabletovotetwice',async()=>{try{constreceipt=awaitDReddit

catch(error){assert(error

});

代碼看起來可能會讓你有些困惑,但實際上它的邏輯非常直接。如果投票函數vote()執行失敗,我們不應該調用函數assert.fail(),而應該立即進入catch()部分。如果結果不是這樣,那么就說明測試發現了問題,這種測試方法其實就是大名鼎鼎的負向(Negative)測試。

到這里,也就是我們最后一次運行embarktest進行測試了,如果一切正常的話,測試的輸出應該如下所示,也就是說,我們已經完全覆蓋了所有的測試!快為自己點個贊!

?embarktestCompilingcontractsDReddit?shouldwork(0ms)-?shouldbeabletocreateapostandreceiveitviacontractevent(60ms)-?postshouldhavecorrectdata(18ms)-?shouldnotbeabletovoteinanunexistingpost(14ms)-?shouldbeabletovoteinapostifaccounthasn'tvotedbefore(12ms)-?shouldbeabletovoteinapost(42ms)-?shouldn'tbeabletovotetwice(37ms)-7passing(5s)->Alltestspassed

由于下一部分篇幅過長,我們將在下一篇文章中介紹如何使用React框架作為客戶端前端JavaScript庫來構建DReddit前端界面。主要包括以下5部分:

渲染組件

構建創建帖子組件CreatePost

構建帖子組件Post

構建帖子列表組件List

添加投票功能

老鐵們,敬請期待

Tags:POSTOSTPOSVOTEMNPoSTreeTPOS幣VOTES幣

酷幣
Newton 項目評級:BB 展望穩定 | TokenInsight_NEW

牛頓項目致力于打造社群經濟基礎設施,通過提供治理、協作和激勵等服務,將公司經濟升級為“人人貢獻、人人受益”的全新社群經濟.

1900/1/1 0:00:00
Jameson Lopp:我與比特幣的二三事_比特幣

作者|dyorpodcast 編譯|Jhonny 前言: 本文系比特幣核心開發者、前比特幣安全平臺BitGo首席開發工程師JamesonLopp近期受邀參加TomBuonincontri主持的博.

1900/1/1 0:00:00
Libra“涼了” 我們可以坐下來聊聊它的真相了_LIBRA

大殿的角石, 并不高于那最低的基石。 Libra是Facebook發起的數字貨幣,最近最重磅的市場炸彈。我說的Libra“涼了”,是指它給幣圈帶來的這波初夏的熱度,開始有明顯的退溫跡象.

1900/1/1 0:00:00
DragonEx 項目2019年進度簡報(2019.6.29-2019.7.5)_DRAG

DragonEx項目2019年進度簡報 2019-07-05 一.重大事件 1、7月1日開放平臺應用——瘋狂動物園權益認購.

1900/1/1 0:00:00
Facebook Libra項目遭四大監管機構聯合施壓 扎克伯格被點名_FACE

北京時間7月3日消息,美國立法機構的四位主席已正式向FacebookCEO馬克·扎克伯格等多名高管發出公開信,要求其暫停所有有關Libra加密貨幣的開發.

1900/1/1 0:00:00
DragonEx上線NEWS/USDT交易對,充值送DT體驗金 百萬交易大賽!_EWS

DragonEx上線NEWS/USDT交易對,充值送DT體驗金百萬交易大賽! 2019-07-03 親愛的用戶: DragonEx將于2019年7月3日17:00(UTC8)在創新區上線NEWS.

1900/1/1 0:00:00
ads