本文学习《深度剖析智能合约升级——inherited storage》实现,过程中遇到的坑在这里记录
过程中多次请教文章的作者王铁塔,这里也表示感谢
1、拉取代码
git clone https://github.com/hammeWang/Proxy.git
2、打开remix,导入代码
导入后编译,有几个地方需要改一下
a、UpgradeabilityStorage.sol中
import "openzeppelin-solidity/contracts/ownership/Ownable.sol";
改import "./Ownable.sol";
这个文件来自openzeppelin-solidity
当然先git clone https://github.com/OpenZeppelin/openzeppelin-solidity.git
然后只把Ownable.sol导入
b、Registry.sol中
import "openzeppelin-solidity/contracts/ownership/Ownable.sol";
改import "./Ownable.sol";
c、Token.sol中
import '../Upgradeable.sol';
改import './Upgradeable.sol';
3、 如何初始化
a、部署Registry合约:
先要修改一下这个合约,增加变量
UpgradeabilityProxy public proxy ;
修改函数createProxy中
UpgradeabilityProxy proxy = new UpgradeabilityProxy(version);
为
proxy = new UpgradeabilityProxy(version);
主要这个proxy后面多次用到,添加变量方便直接获取
然后在run面板,选择Registry,点击deploy
b、部署逻辑合约的初始版本(V1),并确保它继承了Upgradeable合约
在remix选择文件Token.sol,选择TokenV1_0,点击deploy
c、向Registry合约中注册这个最初版本(V1)的地址
直接在发布后的TokenV1_0上复制其地址0x11f0c46dc617619d51aa60e02daca7608dc714e7,然后打开发布后的Registry,调用函数addVersion,参数:"1",0x11f0c46dc617619d51aa60e02daca7608dc714e7
d、要求Registry合约创建一个UpgradeabilityProxy实例
在发布后的Registry中调用函数createProxy,参数"1"
这个时候查看变量proxy的值:0x7Fc933Fa49555816C4874616f6751F565Af4028F
特别要做的一步是文章中的第三步转移proxy合约所有权,必现先做,否则后续的升级无法调用
更改proxy的owner:
在发布后的Registry中调用函数transferProxyOwnership,参数0x7Fc933Fa49555816C4874616f6751F565Af4028F,0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c
其中0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c也是Registry的owner
这样修改后,后续调用就不会因为owner不对,执行升级后代码出错的情况了
e、调用你的UpgrageabilityProxy实例来升级到你最初版本(V1)
如图所示,
选择文件Token.sol,选择TokenV1_0,在At address填写proxy的地址:0x7Fc933Fa49555816C4874616f6751F565Af4028F,然后点击At address,强制类型转换为TokenV1_0,在最下面就可以看到新发布的TokenV1_0,我们调用TokenV1_0中的函数balanceOf,参数0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c,其值为10000
然后调用函数mint,参数0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c,1000,得到结果11000
4. 如何升级
a、部署一个继承了你最初版本合约的新版本(V2),V2必须继承V1
在remix选择文件Token.sol,选择TokenV1_1,点击deploy
b、向Registry中注册合约的新版本V2
直接在发布后的TokenV1_1上复制其地址0x27bf09ca43cb0142bd148e9bdf0e3fa1b483f5c6,打开发布后的Registry,调用函数addVersion,参数:"2",0x27bf09ca43cb0142bd148e9bdf0e3fa1b483f5c6
c、调用你的UpgradeabilityProxy实例来升级到最新注册的版本
如下图所示
打开文件UpgradeabilityProxy.sol,选择UpgradeabilityProxy,在At address填写proxy的地址:0x7Fc933Fa49555816C4874616f6751F565Af4028F,然后点击At address,强制类型转换为UpgradeabilityProxy,在最下面就可以看到新发布的UpgradeabilityProxy,然后调用其函数upgradeTo,参数"2"
要观察这个是否执行成功,如果失败说明第一步的d中没有做所有权转移
如图所示,
选择文件Token.sol,选择TokenV1_1,在At address填写proxy的地址:0x7Fc933Fa49555816C4874616f6751F565Af4028F,然后点击At address,强制类型转换为TokenV1_1,在最下面就可以看到新发布的TokenV1_1,我们调用TokenV1_1中的函数balanceOf,参数0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c,其值为11000
说明更新的合约满足了存储可继承
调用approve,参数0xCA35b7d915458EF540aDe6068dFe2F44E8fa733c,100,调用成功,说明合约升级成功
5. 如何转移proxy合约所有权
调用Registry中的transferProxyOwnership方法进行所有权转移;
这一步我们已经在如何初始化的d里面做了