KeyFrame

Web3 重入漏洞讲解 | Solidity 智能合约安全 5分钟看懂重入攻击

bugTheAnswer·4月12日週日·5 min中文

三句話摘要

以 Foundry 模擬區塊鏈重入漏洞(Reentrancy Attack)的攻擊原理與兩種防禦方法。 提款函數務必遵循「先更新狀態、再轉移資產」的 CEI 順序,或加掛 `nonReentrant` 鎖,否則任何先轉帳後清帳的設計都是潛在的重入攻擊缺口。 正確與錯誤提款流程的根本差異:正確流程先扣除餘額再發送資產;錯誤流程先發送資產,餘額留到最後才更新,導致攻擊視窗開放。

重點整理

重點
  • 1

    正確與錯誤提款流程的根本差異:正確流程先扣除餘額再發送資產;錯誤流程先發送資產,餘額留到最後才更新,導致攻擊視窗開放。

  • 2

    `receive()` 函數是重入攻擊的觸發器:當合約向外部地址發送 ETH 時,若接收方是合約,其 `receive()` 函數會自動執行,攻擊者在此處再次呼叫 `withdraw()`,形成遞迴循環。

  • 3

    CEI 模式(Checks-Effects-Interactions)是最輕量的防禦:只需將 `balances[msg.sender] = 0` 移到 `.call()` 之前,攻擊者第二次進入時餘額已為零,提款直接 revert。

  • 4

    OpenZeppelin `nonReentrant` 提供合約層級鎖:透過 mutex 狀態旗標,確保整個提款函數執行期間無法被重新進入,適合邏輯複雜、難以手動套用 CEI 的場景。

實用技巧與重點

乾貨
  • 攻擊數字:受害者存入 10 ETH,攻擊者存入 1 ETH,攻擊成功後攻擊合約持有 11 ETH
  • 漏洞合約名稱:`Etherstone`,含 `deposit()` 與 `withdraw()` 兩個函數
  • 測試框架:Foundry(使用 `TestAttack` 函數撰寫攻擊測試,`forge test` 通過驗證漏洞存在)
  • 防禦工具一:OpenZeppelin 函式庫,`nonReentrant` modifier
  • 防禦工具二:CEI 模式,將 `balances[msg.sender] = 0` 置於 `.call{value: amount}("")` 之前
  • 修復後測試結果:`withdraw()` 因 `balance == 0` 條件失敗而 revert,攻擊無效

結論

結論

提款函數務必遵循「先更新狀態、再轉移資產」的 CEI 順序,或加掛 `nonReentrant` 鎖,否則任何先轉帳後清帳的設計都是潛在的重入攻擊缺口。

完整解析

詳細

區塊鏈智能合約中的重入漏洞(Reentrancy Vulnerability)是 DeFi 生態中最經典也最危險的安全問題之一。其根本成因在於提款函數的執行順序錯誤:正確的設計應先從帳本扣除使用者餘額,再將資產發送出去;而存在漏洞的版本卻反其道而行,先呼叫 `.call()` 將 ETH 傳給呼叫者,等到轉帳完成後才清空餘額。這個時序上的破口,給了攻擊者可乘之機。

影片以一個名為 `Etherstone` 的合約為例說明攻擊流程。該合約的 `deposit()` 函數負責累積使用者存入的 ETH,`withdraw()` 則先確認餘額大於零,再以 `.call{value: balance}("")` 形式轉出全部餘額,最後才將餘額欄位歸零。攻擊者部署一個惡意合約,並在其 `receive()` 函數中植入再次呼叫 `withdraw()` 的邏輯。當受害者存入 10 ETH、攻擊者存入 1 ETH 後,攻擊者呼叫 `withdraw()`:合約先把 1 ETH 轉給攻擊合約,觸發 `receive()`,此時餘額尚未歸零,攻擊合約立刻再次呼叫 `withdraw()`,如此循環,直到抽乾合約資金池。最終 Foundry 測試顯示攻擊合約成功持有 11 ETH,而僅投入了 1 ETH,完整驗證漏洞的嚴重性。

針對此漏洞,影片提供了兩種開發層面的防禦策略。第一種是引入 OpenZeppelin 函式庫,在 `withdraw()` 加上 `nonReentrant` modifier。該 modifier 使用互斥鎖(mutex)機制,在函數尚未執行完畢時拒絕任何重入呼叫,適合邏輯複雜、呼叫鏈較深的合約。第二種方法更為輕量,即遵循「Checks-Effects-Interactions(CEI)」模式:在發送 ETH 之前就先將 `balances[msg.sender]` 清零。如此一來,攻擊者第二次進入 `withdraw()` 時,餘額已為零,條件檢查失敗,函數直接 revert。修改後重新執行 Foundry 測試,結果顯示提款函數因 `balance == 0` 而 revert,攻擊流程被完全阻斷。

關鍵時刻

Pipeline v2

帶時間戳的重點,會在逐字稿層級分析上線後產生。目前請先透過原始影片觀看。

事實查核

Pipeline v2

說法查證是下一次管線升級的一部分。KeyFrame 只會顯示它真正能驗證的內容。

更多「Web3 安全」的內容

How Safe is Your Bitcoin?!
編輯精選
88 min
Web3 安全英文6月19日

How Safe is Your Bitcoin?!

Maple Bitcoin

  • 假 App 詐騙手法已低門檻化:現今 AI 工具讓複製官方應用介面的技術門檻大幅降低,任何知名品牌(Ledger、Sparrow)都是仿冒目標。搜尋結果排名靠前的 App 不等於安全,用戶須從官方網站取得正確下載連結。
  • 硬體錢包的核心價值是隔離私鑰,但仍需配合正確行為:硬體錢包本身無法防護用戶主動輸入助記詞至惡意網站的行為,且無螢幕的錢包(Tangem、Trezor 基本款等)無法讓用戶在設備上核驗交易細節,形成地址掉包漏洞。Coldcard 搭配 Sparrow 的組合最大程度縮小攻擊面。
  • 威脅模式思考是自管的必要功課:火災、地震、竊盜、意外失憶、家人繼承等情境都需事先規劃。主持人建議用「不提示任何資訊、讓家人嘗試恢復錢包」作為壓力測試,確認繼承流程確實可行。
深度观察:Q2黑客攻击创历史新高,DeFi安全体系全面溃败 | 七十起攻击与7.46亿损失 | 从亡羊补牢到未雨绸缪
10 min
Web3 安全中文6月16日

深度观察:Q2黑客攻击创历史新高,DeFi安全体系全面溃败 | 七十起攻击与7.46亿损失 | 从亡羊补牢到未雨绸缪

Web3观察哨

  • 廢棄合約即定時炸彈: 智能合約一旦部署於公鏈便永久運行,即使項目方已停止維護、合約不可升級,黑客仍可針對殘留餘額發動攻擊,S-Tech Connect 案例說明「代碼不朽」是公鏈安全的根本隱患。
  • 行業激勵錯位導致防禦失敗: 項目方融資後需優先砸錢做市場和拉高 TVL 以支撐估值,將預算投入審計與安全機制反而拖慢增長,市場機制本身就不獎勵「防禦故事」,系統性漏洞因此得不到修補。
  • 去中心化信仰與安全機制互相矛盾: 引入緊急熔斷或多方暫停機制在技術上可行,但社群將其視為中心化復辟;然而服務億級用戶的金融市場必須具備基本風控,這是 DeFi 無法迴避的現實悖論。