巧妙的合約設計,看看stETH如何按天自動發放收益?

筆者把自己為數不多的ETH 兌換為stETH 後,發現stETH 每天都在自然增長,不斷獲取收益。但是卻沒有看到帳戶有交易產生,這是為什麼呢?本文帶大家一起來看看背後的巧妙設計,揭開收益發放的秘密。

巧妙的合約設計,看看stETH如何按天自動發放收益?

1 個stETH 過去幾天後已經有了一些收益

在這之前先介紹stETH 賺取收益背後的邏輯,也就是以太坊的質押(Staking),已經了解這部分概念的讀者可以直接跳到後面。

巧妙的合約設計,看看stETH如何按天自動發放收益?

最初的以太坊和比特幣一樣是透過工作量證明(Proof of Work,PoW)來作為它的共識機制,但是PoW 因為耗電以及其它安全性和性能上發展的考慮,以太坊從2022 年9 月開始升級為權益證明(Proof of Stake,PoS)。

原本依靠算力挖礦來吸引礦工實現共識的以太坊,搖身一變,變成了依靠大家透過質押ETH 獲取投票權,透過投票來獲取收益,從而激勵大家透過PoS 的方式來實現共識。

透過質押32 個ETH 可以加入以太坊網絡,可以成為驗證者,負責儲存資料、處理交易以及向區塊鏈添加新區塊。只要運行將交易正確打包為新區塊並檢查其他驗證者工作,就會獲得ETH 獎勵,這樣就相當於你可以透過質押的方式讓ETH 可以擁有相對穩定的收益。

但是這樣的質押對一般使用者來說還是太麻煩,畢竟32 個以太坊和一台要能夠全年無休接入以太坊網路的專用電腦還是有一定的門檻的。而且質押ETH 會使得喪失了這部分ETH 的流動性。於是就有了流動性質押衍生品(Liquid Staking Derivatives,LSD),它旨在解決傳統質押中的門檻和流動性問題,允許用戶質押32 個以下的ETH,以及不需要自己擁有節點,而是把ETH 委託給第三方質押,並獲取相應的質押代幣(如Lido 的stETH 或Rocket Pool 的rETH),這些流動性代幣可以在其他平台上交易、借貸或用於其他金融活動,這樣,用戶既能更便捷參與質押中獲得獎勵,又能保持資金的彈性。

巧妙的合約設計,看看stETH如何按天自動發放收益?

所以stETH 本質的邏輯就是把ETH 給到Lido,Lido 會用這些ETH 去參與以太坊的PoS 以獲得收益,用戶會得到對應的stETH 作為憑證。接下來就是Lido 要把收益發給這些擁有stETH 的地址。

我們可以看到stETH 的收益每天都會自動更新,下圖是我們測試的收益狀況,對應每天都可以檢查加密錢包驗證相關內容。

巧妙的合約設計,看看stETH如何按天自動發放收益?

但到這裡我想熟悉智能合約開發的同學就會疑惑了:每天發放這麼少的收益,可能收益都不夠付GAS 的。

確實,如果Lido 按照最簡單的做法來發放收益的話,那確實難以覆蓋GAS 的成本。從我們的直覺來看,要往如此眾多的地址發送代幣,GAS 是難以想像的。

但確實Lido 就實現了錢包中的stETH 收益自動增長,而且我們並沒有發現該地址有任何交易,這是怎麼實現的呢?

我們找到了Lido 的合約追溯到合約的balanceOf 方法:

巧妙的合約設計,看看stETH如何按天自動發放收益?

balanceOf 是符合ERC20 規範的方法,錢包就是透過此方法取得使用者有多少token 的。

我們可以看到stETH 的合約中這裡呼叫了getPooledEthByShares方法。此方法入參是mapping (address => uint256) private shares;。這代表使用者有多少stETH?顯然不是,不然每天都需要更新每個地址的數據,雖然這樣也可以做到只要調用合約中的方法來更新shares來實現一次交易就能更新所有地址的token,但是顯然這樣做GAS 的消耗同樣也是巨大的。

想必到這裡大家已經要猜到合約是怎麼實現的了,我們繼續來看getPooledEthByShares方法。

巧妙的合約設計,看看stETH如何按天自動發放收益?

可以看到最終回傳的結果是用位址中sharesAmount 乘以_getTotalPooledEther()再除_getTotalShares。

_getTotalPooledEther代表總共有多少stETH(依照stETH 兌ETH 為1:1 的話也代表有多少ETH),_getTotalShares代表有多少份額。這樣一算每個位址有多少stETH 就是動態計算出來的了。

舉例說如果現在總共有1000 個份額(Shares,也就是_getTotalShares方法回傳的數量),其中A 位址有100 份(對應上面的sharesAmount)。這1000 個份額對應1000 個stETH(也就是_getTotalPooledEther)回傳的數量。那麼按照這個計算,A 位址就對應有100 個stETH。那Lido 拿這總得1000 個ETH 去質押獲取到1 個ETH 的收益後對應更新_getTotalPooledEther 為1001,也就是最初總共1000 個的stETH 變多變成1001 個了,那麼新的計算出來A 地址就擁有了100 * 1001 / 1000 = 100.1 個stETH。

**簡單點說就是每個地址擁有的股份不變,股份對應的stETH 變多了,那麼一計算自然stETH 就變多了。 **

我們繼續看程式碼,_getTotalPooledEther 中的邏輯是會受到handleOracleReport方法影響,而這個方法則會更新合約中的相關資料。具體的呼叫是會透過合約定期呼叫submitReportData 更新資料(submitReportData 中會呼叫Lido 合約的handleOracleReport):

巧妙的合約設計,看看stETH如何按天自動發放收益?

我們可以看到每天都會有調用更新相關內容,這就是為什麼雖然我們無法看到我們的地址中有發放收益的交易,但是金額仍然每天在變化的原因。

這背後其實體現了以太坊ERC20 智能合約的一個特點,就是這些ERC20 的合約擁有多少代幣並不是寫死在地址上的,而是合約方法返回的,所以可能會出現帳號雖然沒有任何交易,但是代幣的數量也可能會改變。這一方面讓ERC20 合約更靈活,但是另一方面也給很多對合約不熟悉的朋友帶來了很多困惑,希望本文可以幫助大家更多的理解智能合約,更安全的和智能合約互動。

另外,雖然透過將ETH 質押為stETH 能夠獲得看似穩定的質押收益,但是依然有可能的風險存在,本文只作為對質押合約的技術研究參考,不構成任何投資建議。

另外之前我們發布了一篇關於『再質押』的文章,有興趣的讀者可以繼續閱讀,深入了解更多內容。

查看原文
  • 讚賞
  • 留言
  • 分享
留言
0/400
暫無留言
交易,隨時隨地
掃碼下載 Gate.io APP
社群列表
繁體中文
  • 简体中文
  • English
  • Tiếng Việt
  • 繁體中文
  • Español
  • Русский
  • Français (Afrique)
  • Português (Portugal)
  • ไทย
  • Indonesia
  • 日本語
  • بالعربية
  • Українська
  • Português (Brasil)