From ecb9f0fb45e7978875bf0c598a258ea4304a5b4d Mon Sep 17 00:00:00 2001 From: Kyryl Riabov Date: Wed, 15 May 2024 16:38:41 +0300 Subject: [PATCH 1/2] Added `upgradeToWithSigAndCall` function to the UUPSSignableUpgradeable --- contracts/utils/UUPSSignableUpgradeable.sol | 17 +++++++++++++++++ test/bridge/Upgradeable.test.ts | 8 ++++++++ 2 files changed, 25 insertions(+) diff --git a/contracts/utils/UUPSSignableUpgradeable.sol b/contracts/utils/UUPSSignableUpgradeable.sol index 71bf894..c315b2b 100644 --- a/contracts/utils/UUPSSignableUpgradeable.sol +++ b/contracts/utils/UUPSSignableUpgradeable.sol @@ -30,4 +30,21 @@ abstract contract UUPSSignableUpgradeable is UUPSUpgradeable { _authorizeUpgrade(newImplementation_, signatures_); _upgradeToAndCallUUPS(newImplementation_, new bytes(0), false); } + + /** + * @notice Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call + * encoded in `data`. + * + * Calls {_authorizeUpgrade} with the provided `newImplementation` and `signatures`. + * + * Emits an {Upgraded} event. + */ + function upgradeToWithSigAndCall( + address newImplementation_, + bytes[] calldata signatures_, + bytes calldata data_ + ) external virtual onlyProxy { + _authorizeUpgrade(newImplementation_, signatures_); + _upgradeToAndCallUUPS(newImplementation_, data_, true); + } } diff --git a/test/bridge/Upgradeable.test.ts b/test/bridge/Upgradeable.test.ts index d20228a..e066ab5 100644 --- a/test/bridge/Upgradeable.test.ts +++ b/test/bridge/Upgradeable.test.ts @@ -50,6 +50,10 @@ describe("Upgradeable", () => { await expect(newBridge.upgradeToWithSig(await newBridge.getAddress(), [])).to.be.rejectedWith( "Function must be called through delegatecall", ); + + await expect(newBridge.upgradeToWithSigAndCall(await newBridge.getAddress(), [], "0x")).to.be.rejectedWith( + "Function must be called through delegatecall", + ); }); it("should upgrade implementation", async () => { @@ -85,6 +89,10 @@ describe("Upgradeable", () => { await expect(proxyBridge.connect(SECOND).upgradeToWithSig(await newBridge.getAddress(), [])).to.be.rejectedWith( "Ownable: caller is not the owner", ); + + await expect( + proxyBridge.connect(SECOND).upgradeToWithSigAndCall(await newBridge.getAddress(), [], "0x"), + ).to.be.rejectedWith("Ownable: caller is not the owner"); }); it("should receive ether through proxy", async () => { From f7d05003c579980449252386f8f3074ba0834b72 Mon Sep 17 00:00:00 2001 From: Kyryl Riabov Date: Wed, 15 May 2024 16:51:19 +0300 Subject: [PATCH 2/2] Changed `forceCall` to false --- contracts/utils/UUPSSignableUpgradeable.sol | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/utils/UUPSSignableUpgradeable.sol b/contracts/utils/UUPSSignableUpgradeable.sol index c315b2b..dd2e4ae 100644 --- a/contracts/utils/UUPSSignableUpgradeable.sol +++ b/contracts/utils/UUPSSignableUpgradeable.sol @@ -45,6 +45,6 @@ abstract contract UUPSSignableUpgradeable is UUPSUpgradeable { bytes calldata data_ ) external virtual onlyProxy { _authorizeUpgrade(newImplementation_, signatures_); - _upgradeToAndCallUUPS(newImplementation_, data_, true); + _upgradeToAndCallUUPS(newImplementation_, data_, false); } }