当前位置: 首页 > news >正文

在REMIX中使用OpenZeppelin集成透明升级合约和在HARDHAT中集成透明升级合约演示

目录

1.在REMIX中集成透明升级合约

2.在HARDHAT中集成透明升级合约


合约一旦部署,是不可以更改了,项目初期必须提前设计,决定是否需要升级,否则是无法升级的,只要提前用代理模式,就能升级。演示一下升级的过程。BoxV1升级到BoxV2过程。

1.在REMIX中集成透明升级合约

BoxV1.sol代码:

// SPDX-License-Identifier: MIT pragma solidity ^0.8.26; import "@openzeppelin/contracts/proxy/utils/Initializable.sol"; contract BoxV1 is Initializable { uint public x; function initialize(uint _val) external initializer { x = _val; // set initial value in initializer } function call() external { x += 1; } function showInvoke() external pure returns (bytes memory) { return abi.encodeWithSelector(this.initialize.selector, 1); } }

BoxV2.sol

// SPDX-License-Identifier: MIT pragma solidity ^0.8.26; import "@openzeppelin/contracts/proxy/utils/Initializable.sol"; contract BoxV2 is Initializable { uint public x; function initialize(uint _val) external initializer { x = _val; // set initial value in initializer } function call() external { x *= 2; } function showInvoke() external pure returns (bytes memory) { return abi.encodeWithSelector(this.initialize.selector, 1); } }

TPUProxy.sol

// SPDX-License-Identifier: MIT pragma solidity ^0.8.26; import "@openzeppelin/contracts/proxy/transparent/TransparentUpgradeableProxy.sol"; contract TPUProxy is TransparentUpgradeableProxy { constructor( address _logic, address _initialOwner, bytes memory _data ) payable TransparentUpgradeableProxy(_logic, _initialOwner, _data) {} function proxAdmin() external view returns (address) { return _proxyAdmin(); } function getImplementation() external view returns (address) { return _implementation(); } }

在remix,分别编译和部署BoxV1,BoxV2合约

然后部署TPUProxy合约,部署前注意填写数据

参数解释:

参数类型作用
_logicaddress业务逻辑合约地址(如BoxV1)
_initialOwneraddress管理员地址(可以升级合约的人)
_databytes初始化数据(调用逻辑合约initialize函数)

_logic的值是复制BoxV1的地址

然后点击部署,部署TPUProxy时就已经调用了BoxV1的initialize(1)方法,在 TPUProxy 的存储上下文中,执行 BoxV1.initialize(1),initializer 来自 Initializable,该函数只能调用一次。

复制TPUProxy合约地址

然后挂载到BoxV1上

Remix 的界面会把 “At Address 绑定的操作入口” 和 “直接部署的合约” 放在同一区域,显示名称都是 “BoxV1”,操作本质:给 TPUProxy 套了个 BoxV1 的 “操作界面”,用 BoxV1 的按钮控制代理

现在要升级到BoxV2,只能由管理合约来升级

找管理员合约

可以对应上的。如果发现ProxyAdmin合约地址跟proxAdmin按钮显示的地址不一样,明显就是错了,那么不管填写什么值,都会报错。

参数意思怎么填
proxy要升级的代理地址你的TPUProxy地址
implementation新逻辑合约地址你的BoxV2地址
data升级后要执行的函数 calldata0x(不执行)或abi.encode(...)

然后点击“transact",意思是发起一笔区块链交易,请求将代理合约(TPUProxy)的逻辑升级到新版本(如 BoxV2),并在升级后立即执行指定的初始化或迁移函数。

接下来,把代理合约挂载到BoxV2中,因为TPUProxy已经变了,我们再次复制这个TPUProxy合约地址

点击”Ar Address

点击“x:显示的值是之前Boxv1的值,状态没丢,然后点“call",发现值已经是乘法,升级成功了

完成合约。

用户/前端始终与 同一个地址 交互:TPUProxy,即使升级了 10 次逻辑合约,这个地址永远不会变,看这个TPUProxy地址,始终都没变。

2.在HARDHAT中集成透明升级合约

首先安装依赖

npm install --save-dev @openzeppelin/hardhat-upgrades 或 yarn add -D @openzeppelin/hardhat-upgrades

在hardhat.config.ts文件添加依赖

require("@nomicfoundation/hardhat-toolbox"); require("@openzeppelin/hardhat-upgrades")

把BoxV1.sol和BoxV2.sol复制到hardhat中

然后在test目录下新建BoxV1.js文件

const hre = require("hardhat"); async function deploy() { const BOXV1 = await hre.ethers.getContractFactory("BoxV1"); // 通过v1版本部署代理 const v1 = await hre.upgrades.deployProxy(BOXV1, [1], { initializer: "initialize", }); await v1.waitForDeployment(); console.log(await v1.getAddress()); console.log(await v1.x()); await v1.call(); console.log(await v1.x()); } deploy();

然后在package.json文件所在目录,打开命令提示符,输入

#清理可能的缓存 npx hardhat clean #重新编译 npx hardhat compile #启动节点 npx hardhat node

启动另一个命令窗口,输入

npx hardhat run test\BoxV1.js --network localhost

复制BoxV1.js变成BoxV2.js

const hre = require("hardhat"); async function deploy() { //V2版本工厂 const BOXV2 = await hre.ethers.getContractFactory("BoxV2"); // 通过v2版本部署代理 const v2 = await hre.upgrades.upgradeProxy( "0xe7f1725E7734CE288F8367e1Bb143E90bb3F0512", BOXV2 ); await v2.waitForDeployment(); console.log(await v2.getAddress()); console.log(await v2.x()); await v2.call(); console.log(await v2.x()); } deploy();

再次要命令行执行

npx hardhat run test\BoxV2.js --network localhost

成功了

http://www.jsqmd.com/news/83693/

相关文章:

  • 光刻胶增感剂用正丁胺
  • 汽车变速器电控系统Simulink模型:从原理到实现
  • MPK(Mirage Persistent Kernel)源码笔记(3)--- 系统接口
  • vs2010卸载安装后报错未能正确加载 “Microsoft.Entity.Design.BootstrapPackage.BootstrapPackage,Microsoft.Data.Entity
  • SmartCrop.js智能图像裁剪库升级完全攻略
  • 光刻胶用增感剂:乙氧基/丙氧基改性吡唑啉有机物
  • 在 Yocto 中配置 OP-TEE 的工程优势
  • 深度学习python项目--垃圾图像分类识别 关键模型:VGG19DenseNet121Res...
  • “STM32语音智能窗帘(轻松上手)”
  • 5分钟掌握IOPaint集成:从零部署到深度定制全攻略
  • WOA-XGBoost回归+SHAP分析+新数据预测!Matlab代码实现
  • 图像增强与滤波
  • PAT 1151 LCA in a Binary Tree
  • docker网络模式详解
  • 快速上手shadcn-svelte:简单高效的Svelte组件库配置指南
  • 智投助手发布更新v0.0.5
  • 2025年终大盘点:实验室离心机知名企业/优秀企业/优质厂家及行业趋势总结 - 品牌推荐大师1
  • 1213总结
  • CMake-变量+条件判断+添加宏定义+编译QML
  • TCN-GRU回归+特征贡献SHAP分析+新数据预测+多输出,MATLAB代码
  • CAD坐标标注插件终极指南:快速提升绘图效率的5个技巧
  • 基于冠豪猪CPO优化核极限学习机KELM的分类及性能评估报告:包含分类效果图、迭代优化图、混淆...
  • [特殊字符]️ 深度解析我的 Overleaf 私有化部署:一份稳定、高兼容性的 `docker-compose.yaml`
  • 5分钟掌握SplitJoin.vim:终极代码格式化神器
  • 当算力博弈升级为网络战争:拆解DDoS攻击背后的技术攻防战——从DeepSeek遇袭看全球网络安全新趋势
  • Milkdown编辑器终极指南:如何选择最适合你的Markdown解决方案
  • 实践测评:Windows Sandbox 入门教程:快速打造安全的测试环境, windows沙盒环境
  • 创客匠人峰会洞察:AI 时代教育知识变现的重构 —— 从 “刷题记忆” 到 “成长赋能” 的革命
  • 混沌工程基本原理
  • 拉盖尔高斯光束透射石英基底石墨烯涂层的光强分布特性研究:深入探索与实验分析