当前位置: 首页 > 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.rkmt.cn/news/93885.html

相关文章:

  • vs2010卸载安装后报错未能正确加载 “Microsoft.Entity.Design.BootstrapPackage.BootstrapPackage,Microsoft.Data.Entity
  • SmartCrop.js智能图像裁剪库升级完全攻略
  • 光刻胶用增感剂:乙氧基/丙氧基改性吡唑啉有机物
  • 图像增强与滤波
  • “STM32语音智能窗帘(轻松上手)”
  • WOA-XGBoost回归+SHAP分析+新数据预测!Matlab代码实现
  • 1213总结
  • TCN-GRU回归+特征贡献SHAP分析+新数据预测+多输出,MATLAB代码
  • 零基础学黑客:90% 的人会踩的 4 个坑!
  • 3000亿参数AI大模型部署终极指南:4步实现低成本企业级应用
  • 5分钟掌握SplitJoin.vim:终极代码格式化神器
  • CAD坐标标注插件终极指南:快速提升绘图效率的5个技巧
  • 基于冠豪猪CPO优化核极限学习机KELM的分类及性能评估报告:包含分类效果图、迭代优化图、混淆...
  • 当算力博弈升级为网络战争:拆解DDoS攻击背后的技术攻防战——从DeepSeek遇袭看全球网络安全新趋势
  • 混沌工程基本原理
  • 终极Sionna入门指南:5分钟快速上手下一代物理层研究
  • 快速上手Codebox:开源云端IDE的终极配置指南
  • 毕方Talon:鸿蒙开发的编译时安全守护神
  • ANSYS Fluent用户自定义函数开发指南(2020R2版)技术文档解析
  • 如何用强化学习提升数学推理能力:SimpleRL-reason完整指南
  • 软件测试工程师如何利用LinkedIn吸引优质面试机会
  • 如何快速定制Android系统:终极GApps安装指南
  • 入门】使用Node.js开发一个MCP服务器(STDIO方式)介绍
  • Java AI工具箱终极指南:免费离线AI算法一站式解决方案
  • 企业AI接入的核心痛点解法:JBoltAI智能模型路由网关的技术实践
  • shadPS4模拟器实用排障指南:让PS4游戏在PC上畅玩
  • Java,集合框架体系
  • 深度解析vscode-neovim状态栏:从零开始打造个性化编辑界面
  • 企业AI落地破局:从分散消耗到战略运营,JBoltAI路由网关的核心价值
  • 见过哪些醍醐灌顶的Java代码:从“卧槽“到“原来如此“的顿悟