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多平台公布