比特幣行情 比特幣行情
Ctrl+D 比特幣行情
ads
首頁 > Filecoin > Info

ALL:二十三種 DeFi 安全事故匯總:智能合約風險與防范

Author:

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

撰文:AustinZhang,JonLi,AsymmetriesTechnologies

智能合約的安全性問題一直是業界的一個重點話題,由于程序員的某些疏忽造成了思維和邏輯上的漏洞,從而導致黑客有了可乘之機。我們搜集了目前在DeFi領域已經發生了安全事故的智能合約,并根據我們編寫的示例代碼來實證分析其中的原因,希望能給到同事和同行們一些啟示。

重入攻擊

主要攻擊方式之一:合約調用惡意外部合約結束之前,惡意外部合約函數反向調用原合約函數利用相關漏洞。

示例代碼

案例1:2021年12月22日UniswapV3流動性管理協議Visor被盜120ETH

事故原因:deposit函數沒有防重入鎖也沒有驗證from地址是否是合法的Visor合約地址。攻擊者傳入攻擊合約地址,重復調用deposit函數繞過取款金額檢查多次取款。

案例2:2021年6月5日BurgerSwap被盜700萬美金

事故原因:類似Uniswap的原創dex,分為Platform和Pool兩個合約。Platform類似Uniswap的Router,Pair類似Uniswap的Pool,開發者錯誤的將K值校驗放在Platform計算,攻擊者在Platform中進行重入攻擊,多次以舊的K值換取代幣,造成流動性提供者損失。

解決方案:調用外部合約前確保所有中間狀態變量已更新并使用再入鎖。

未檢查函數返回值

調用外部合約函數時,有些函數調用失敗不會拋出錯誤回滾交易而是返回false,如果忘記檢查函數返回值會導致誤以為調用成功。

示例代碼

案例:2021年4月4日ForceDao到被攻擊損失183ETH

Asproex(阿波羅)生態通證MOON完成第二十七期回購銷毀:官方消息,2月23日,Asproex(阿波羅)生態通證MOON完成第二十七期回購銷毀,銷毀數量為?1,783,001枚MOON,區塊高度為11911762?,銷毀交易哈希值為

0x2662a1cc9f76dd797f202fc73c9b8cb8fd03be8a13e88a92f7e1ecd776011b5f。據悉,此次銷毀價值總金額為523311美金,MOON最新價格為0.2930U(實時數據),截止當前時間,Asproex(阿波羅)二十七期回購累計銷毀35,608,025枚MOON。

Asproex(阿波羅)作為一家離岸銀行控股持牌交易平臺,涵蓋CTO(Corporate Token Offering)企業通證上市、合約跟單、ETT指數通證、數字礦業、Digital Bank板塊并持有5國合法牌照,致力于為全球中小微企業提供數字化上市一站式服務。[2021/2/23 17:43:46]

事故原因:Force代幣的transferFrom余額不足時返回false而不是直接回滾交易,合約中未做判斷導致轉賬失敗時也被認為成功,可以換取到對應代幣。

解決方案:使用call函數調用外部合約時必須檢查調用是否成功。注:call調用外部合約未匹配到函數時,會調用外部合約fallback或者receive函數,如果外部合約有定義receive函數且call函數未攜帶calldata則會調用外部合約receive函數,其他情況調用fallback函數。

未正確設置函數可見性

Solidity中函數默認為public,可以被外部調用,一旦未將關鍵函數設置為Private,就會導致安全風險。

示例代碼

案例1:2022年1月22日DexCrosswise被攻擊損失80萬美金

Asproex(阿波羅)生態通證MOON完成第二十六期回購銷毀:官方消息,2月18日,Asproex(阿波羅)生態通證MOON完成第二十六期回購銷毀,銷毀數量為?1,729,593枚MOON,區塊高度為11878324,銷毀交易哈希值為

0x25e4a008f05eace85f2ac73e48b796484f71513f21a912086314e0750143c8c0。據悉,此次銷毀價值總金額高達501582美金,MOON最新價格為0.2919U(實時數據),截止當前時間,Asproex(阿波羅)二十六期回購累計銷毀33,825,024枚MOON。

Asproex(阿波羅)作為一家離岸銀行控股持牌交易平臺,涵蓋CTO(Corporate Token Offering)企業通證上市、合約跟單、ETT指數通證、數字礦業、Digital Bank板塊并持有5國合法牌照。[2021/2/18 17:26:36]

事故原因:Crosswise雖然實現了權限驗證函數onlyOwner,但忘記設置setTrustedForwarder為private,導致被攻擊者利用,將自己設置為池子的Owner將代幣全部轉走。

案例2:2020年6月18日跨鏈橋BancorNetwork被攻擊損失14萬美金

事故原因:合約用于轉賬的函數默認為public,攻擊者可以直接調用轉走合約中的代幣。

解決方案:提款函數事關合約資產的轉移,需謹慎設置權限控制,確保初始化函數只能運行一次。

未驗證Map中Key不存在的情況

Solidity中的Mapping在獲取對應Key的Value時,如果Key不存在,會返回對應類型的默認值,而不是報錯。例如Mapping(int→int),如果對應int的Key不存在,會返回默認值0。

動態 | 中鐵二十三局成功上線人民普惠平臺 首筆供應鏈業務正式落地:12月6日,中鐵二十三局集團有限公司(以下簡稱“中鐵二十三局”)在人民普惠平臺上完成首筆供應鏈融資業務落地。人民普惠平臺是人民金服推出的一款集成了人工智能、生物識別、區塊鏈存證等多項前沿技術的供應鏈金融科技服務平臺。平臺以區塊鏈作為數據存儲與智能合約的底層技術,旨在將鏈上所有關鍵節點實現智能合約應用,可實時監督溯源,可為完整交易做最詳盡的安全管控。所有核心審批操作,貿易憑證等原始材料均有區塊鏈存證備份。(人民網)[2019/12/9]

示例代碼

案例:2021年7月11日跨鏈橋ChainSwap被攻擊損失400萬美金

事故原因:ChainSwap依賴其網絡中的validator進行轉賬。為了限制validator一次轉走超過其質押的代幣,設置了配額。結果合約中存在漏洞可以繞過配額限制,當地址變量signatory不存在時,authQuotes和lasttimeUpdateQuoteOf會返回0,導致配額計算錯誤返回預期外的大量配額。

解決方案:使用map時必須檢查key是否存在。

在狀態變更前進行轉賬

轉賬時有可能被重入,利用未變更的狀態進行攻擊。

案例:2021年8月17日XSURGE被攻擊損失500萬美金

事故原因:在轉賬后才修改totalSupply,轉賬時被重入另外一個未加重入鎖的函數損失500萬美金。

解決方案:使用了再入鎖也要在所有狀態變更之后在轉賬。

初始化函數未做調用和權限限制

很多合約需要初始化子合約,例如Uniswap需要通過Factory合約初始化Pool合約,這時候如果忘記對子合約的初始化函數做權限和重復初始化限制,可能被攻擊者進行惡意初始化。

全球市值前二十加密貨幣 BCN獨漲:據coinmarketcap平臺數據,截至目前,市值排名第19的BCN(Bytecoin)全球均價為0.042元,漲幅5.43%。與此同時,全球市值前二十的其他加密貨幣均有不同程度下跌。BCN是第一個基于cryptonote技術,致力于匿名反機槍池的超前的一種貨幣,于2012年發布。[2018/5/31]

案例:2021年8月11日PunkProtocol被攻擊損失400萬美金

事故原因:池子的initialize函數未做權限和重復調用限制,攻擊者調用該函數將自己設置為Forge管理員權限,并調用withdrawToForge將池子所有資金都發送到攻擊者地址。

解決方案:初始化函數必須設置成只能初始化一次。

未正確檢查對應合約函數實現

通常智能合約被調用的函數不存在時會報錯,但如果合約實現了fallback函數,則會自動調用fallback函數。有時fallback函數并不會報錯,導致調用方誤以為調用成功。

案例:2022年1月18日跨鏈橋Multichain被攻擊損失450ETH

事故原因:通常ERC20的合約會實現permit函數,用于簽名檢查與授權操作。但WETH、PERI、OMT、WBNB、MATIC、AVAX六種代幣的合約沒有實現permit卻實現了fallback,Multichain在檢查這些代幣的權限時誤以為用戶已經授權轉賬給攻擊者,導致代幣被盜。

解決方案:不同代幣的實現方式不同,引入新代幣之前應仔細檢查其具體實現。

未正確處理帶轉賬費的代幣

有些代幣在轉賬時會銷毀一部分轉賬費用,導致實際收到的代幣余額偏少,如果開發者沒考慮到這一點,以轉賬值計算,會導致出現偏差。

二十國集團財長和央行行長會議開幕:就加密貨幣等展開討論:新華社布宜諾斯艾利斯3月19日消息,二十國集團(G20)財政部長和中央銀行行長會議19日在阿根廷首都布宜諾斯艾利斯開幕,與會代表將就基礎設施投資、未來就業形勢、加密貨幣等議題展開討論與磋商。[2018/3/20]

案例:2021年8月19日Pinecone被盜20萬美金

事故原因:Pinecone使用其代幣PCT作為資金池的質押代幣,PCT轉賬會有手續費的損耗。合約并沒有考慮相關損耗導致用戶份額和質押的PCT總額出現偏差,被攻擊者利用領取多余的獎勵。

解決方案:謹記不是所有的代幣轉賬費都為nativetoken。

簽名驗證漏洞

簽名被重復使用,或者利用橢圓曲線簽名算法的對稱性,根據已有簽名構造合法簽名。

案例:2021年7月12日AnySwap被盜800萬美金

事故原因:對交易簽名除了私鑰外需要一個隨機數R,但是Anyswap部署新合約失誤,導致在BSC上的V3路由器MPC帳戶下有兩個交易具有相同的R值簽名,攻擊者反推到這個MPC賬戶的私鑰轉走了被盜資金。

解決方案:使用EIP-712標準驗證簽名,參考OpenZeppelin的實現:https://docs.openzeppelin.com/contracts/3.x/api/drafts。

未考慮合約余額可能產生的變化

礦工挖出塊時或者智能合約調用selfdestruct函數銷毀自己時可以向任意地址強行打幣改變其原生代幣的余額。當使用余額函數返回值作為判斷條件時,余額有可能被強行改變導致風險,極端情況下甚至導致合約拒絕服務。

示例代碼

即使捐贈合約不能接受代幣轉賬,合約余額也可能在部署后被改變,嚴格檢查已空投總量與合約余額之和等于總供應量可能導致捐贈合約拒絕服務。

解決方案:在合約中避免對合約余額做嚴格相等的檢查。

使用delegatecall調用外部合約

delegatecall可以將對應合約的函數代碼內嵌到當前上下文中執行,就像調用內置函數一般。如果不小心調用了惡意合約極易導致攻擊。

示例代碼

當攻擊者調用forward函數并傳入Attack合約地址以及函數setOwner()作為參數時,Proxy合約owner將被修改為攻擊者地址。

解決方案:不推薦使用delegatecall調用外部合約。

授權tx.origin

tx.origin是交易的發起者地址,合約如果使用tx.origin做權限檢查,當合約的授權用戶與惡意合約交互時,惡意合約調用合約即可通過合約權限檢查。

示例代碼

當MyWallet合約owner使用transferTo函數向Attack合約轉賬時,Attack合約會重入MyWallet合約,并調用transferTo函數,此時tx.origin仍然為MyWalletowner,require條件滿足,MyWallet余額將被全部轉移至Attack合約。

解決方案:不使用tx.origin做權限檢查。

交易排序競爭

全節點運行者可以在交易被確認之前獲取交易信息,進而根據獲取的交易信息,構造高手續費交易,讓礦工優先打包自己的交易以執行對自己有利的策略。例如,謎語合約獎勵最快找出謎底的用戶,惡意用戶可以在獲悉誠實用戶提交的謎底后,構造高手續費交易優先誠實用戶提交謎底,從而獲取獎勵;又如當用戶更新授權額度時,被授權用戶可以在更新授權額度交易被確認之前轉移舊的授權額度,如此,被授權人實際獲得的授權額度為兩次授權額度之和。

解決方案:針對謎語合約,獲得謎底的用戶先提交「隨機數+自身地址+謎底」的哈希值,謎語合約存儲該哈希值后,用戶再提交隨機信息與答案,合約檢查哈希值匹配后再發放獎勵;更新授權額度時先置零授權額度。

使用block.timestamp或者block.number作為合約時間參考

block.timestamp與block.number都不能獲得精確都時間,用作智能合約的時間參考會引入潛在的風險。

解決方案:使用oracle獲取時間信息。

Denial-of-Service(DoS)拒絕服務

調用外部合約可能永久失敗導致本合約不能接受新的指令,例如當合約主動對另外一個合約轉賬,而被轉賬合約沒有接受轉賬的函數時,轉賬失敗,此時合約可能進入拒絕服務狀態。

示例代碼

當合約向其中一個賬號轉賬失敗會導致所有轉賬全部失敗。

解決方案:合約調用外部合約時可能出現的失敗,合約需包含處理調用失敗情況的代碼,防止合約進入拒絕服務狀態。

使用鏈屬性作為隨機源

鏈屬性如block.timestamp,blockhash,bock.difficulty以及其他屬性可被礦工操控,存在風險。

解決方案:考慮使用RANDAO,oracle或比特幣區塊hash作為隨機源。

繼承順序錯誤

多個被繼承合約都定義了同一個函數時,繼承合約調用該函數的優先級由繼承順序決定,錯誤的繼承順序將導致函數調用錯誤。

解決方案:繼承順序說明請參考官方實例:https://solidity-by-example.org/inheritance/。

Gas不足攻擊

多簽情況下或者需要其他人幫自己代付Gas時,用戶準備好簽名交易并交給代執行人,代執行人再將用戶交易提交給執行合約,代執行人可以提前審查用戶代交易,惡意的代執行人或當交易內容不利于代執行人時,可以通過限制Gas的供給,使交易的執行失敗,從而阻止交易的執行。

示例代碼

當Relayer調用者通過限制Gas使用導致某個交易失敗,那么失敗的交易將永遠不能再被提交。

解決方案:選擇信任的代執行人,或者在執行合約中檢查代理人提供的Gas費是否足夠。

函數類型變量跳轉

solidity支持函數類型變量,當函數類型變量使用匯編指令賦值時,函數類型變量有可能被指向惡意構造當函數。

解決方案:如無必要,盡量避免在智能合約中使用匯編指令。

GasLimit服務拒絕攻擊

區塊設置有Gas使用上限,如果合約當執行超過了區塊Gas使用上限,則合約永遠不能被執行成功。

示例代碼

當操作的循環次數過大時,執行合約所需Gas將超過區塊上限,導致合約執行失敗。

解決方案:在智能合約中謹慎操作大數組,或循環。

abi.encodePacked()哈希碰撞

abi.encodePacked()采用非填充序列化,當序列化參數包含多個變長數組時,攻擊者可以在保持所有元素順序不變的前提下,改變兩個變長數組的元素,如此序列化的結果相同。

示例代碼

通過構造addUser的輸入,攻擊者可以將regularUsers的成員加入admins成員,但是構造的輸入和原輸入的簽名相同。

解決方案:使用定長數組,或者不讓調用者傳入abi.encodePacked()的參數,或者使用abi.encode()。

transfer()和send()函數Gas不足

transfer()和send()函數使用2300gas以防止重入攻擊,公鏈升級后可能導致gas不足。

解決方案:推薦使用call()函數,但需做好重入攻擊防護。

鏈上未加密隱私數據

鏈上數據完全透明,合約的private關鍵字不能阻止合約的隱私數據泄漏。

示例代碼

雖然players為private,但攻擊者仍然可以通過解析鏈上數據讀取players。

解決方案:隱私數據需要加密放在鏈上。

以上是我們分析和總結的二十三種安全事故類型匯總,希望能夠給到您些許參考和啟示。

Tags:ALLACKGASMOOtrustwallet什么錢包blackcoinDexigasMOO幣

Filecoin
NFT:晚報 | Voyager Digital 稱三箭資本的未償還貸款價值約 6.6 億美元;1confirmation 宣布推出 1 億美元 NFT 基金

整理:胡韜,鏈捕手 “過去24小時都發生了哪些重要事件”1、VoyagerDigital:三箭資本的未償還貸款價值約6.6億美元.

1900/1/1 0:00:00
比特幣:MicroStrategy “All in Bitcoin”的背后是否另有隱情?

撰文:ForesightNews去年加密貨幣的大牛市吸引了一眾美股上市公司,ElonMusk也從之前喊單美股改為喊單比特幣,特斯拉等上市公司在主營業務之余也買入了不少比特幣.

1900/1/1 0:00:00
NFT:OpenSea 內容治理困境:從 Okay Bears 最熱仿盤下架事件談起

作者:念青,鏈捕手 編輯:Demian、谷昱Solana上生態上的NFT勢頭正起,而作為Solana鏈上最具爆發力的藍籌NFT項目“淡定熊”OkayBears也在近期持續大熱.

1900/1/1 0:00:00
SAM:加密行業頂級白帽黑客 samczsun 是如何誕生的?

作者:谷昱,鏈捕手 “Uup?” 這句來自samczsun的詢問,是任何DeFi項目方最害怕收到的消息之一,因為這很可能意味著samczsun發現了該項目智能合約存在嚴重漏洞.

1900/1/1 0:00:00
ELE:借貸協議 Maple:資金池可能出現流動性問題,放貸方或無法提款

鏈捕手消息,據機構借貸協議MapleFinance官方公告表示,本周資金池可能出現流動性問題,放貸用戶可能無法提款,必須等待借款用戶還款.

1900/1/1 0:00:00
NFT:如何使用NFT、社交通證與分數構建鏈上聲譽系統?

作者:深潮TechFlow原文標題:《NFTs、社交通證與分數:鏈上聲譽系統應如何構建?》聲譽分數的未來與社區息息相關現實世界中,我們可以用駕照來證明自己的身份.

1900/1/1 0:00:00
ads