Skip to content

Commit

Permalink
add more timelock tests and upgrades
Browse files Browse the repository at this point in the history
  • Loading branch information
Ramarti committed Oct 19, 2024
1 parent e78b963 commit 4d3a52d
Show file tree
Hide file tree
Showing 4 changed files with 216 additions and 82 deletions.
82 changes: 0 additions & 82 deletions contracts/script/TestPrecompileUpgrades.s.sol

This file was deleted.

105 changes: 105 additions & 0 deletions contracts/test/timelock/Timelock.t.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.23;

import { Test } from "../utils/Test.sol";
import { IPTokenStaking } from "../../src/protocol/IPTokenStaking.sol";
import { TimelockController } from "@openzeppelin/contracts/governance/TimelockController.sol";

contract TimelockTest is Test {
function setUp() public override {
super.setUp();
}

function testCancelBeforeExecute() public {
// Prepare a sample operation
address target = address(ipTokenStaking);
uint256 value = 0;
bytes memory data = abi.encodeWithSelector(IPTokenStaking.setFee.selector, 2 ether);
bytes32 predecessor = bytes32(0);
bytes32 salt = keccak256("TEST_SALT");
uint256 delay = timelock.getMinDelay();

// Schedule the operation
vm.prank(admin);
timelock.schedule(target, value, data, predecessor, salt, delay);

// Ensure the operation is pending
bytes32 operationId = timelock.hashOperation(target, value, data, predecessor, salt);
assertTrue(timelock.isOperationPending(operationId));

// Cancel the operation
vm.prank(admin);
timelock.cancel(operationId);

// Ensure the operation is no longer pending
assertFalse(timelock.isOperationPending(operationId));

// Wait for the delay to pass
vm.warp(block.timestamp + delay + 1);

// Try to execute the cancelled operation
vm.prank(executor);
vm.expectRevert(
abi.encodeWithSelector(
TimelockController.TimelockUnexpectedOperationState.selector,
operationId,
bytes32(1 << uint8(TimelockController.OperationState.Ready))
)
);

timelock.execute(target, value, data, predecessor, salt);

// Verify that the fee wasn't changed
assertEq(ipTokenStaking.fee(), 1 ether);
}

function testExecuteSequenceWithPredecessors() public {
// Prepare sample operations
address target = address(ipTokenStaking);
uint256 value = 0;
bytes memory data1 = abi.encodeWithSelector(IPTokenStaking.setFee.selector, 2 ether);
bytes memory data2 = abi.encodeWithSelector(IPTokenStaking.setFee.selector, 3 ether);
bytes memory data3 = abi.encodeWithSelector(IPTokenStaking.setFee.selector, 4 ether);
bytes32 salt1 = keccak256("SALT_1");
bytes32 salt2 = keccak256("SALT_2");
bytes32 salt3 = keccak256("SALT_3");
uint256 delay = timelock.getMinDelay();

// Schedule the first operation
vm.prank(admin);
timelock.schedule(target, value, data1, bytes32(0), salt1, delay);
bytes32 id1 = timelock.hashOperation(target, value, data1, bytes32(0), salt1);

// Schedule the second operation with the first as predecessor
vm.prank(admin);
timelock.schedule(target, value, data2, id1, salt2, delay);
bytes32 id2 = timelock.hashOperation(target, value, data2, id1, salt2);

// Schedule the third operation with the second as predecessor
vm.prank(admin);
timelock.schedule(target, value, data3, id2, salt3, delay);

// Wait for the delay to pass
vm.warp(block.timestamp + delay + 1);

// Execute the first operation
vm.prank(executor);
timelock.execute(target, value, data1, bytes32(0), salt1);
assertEq(ipTokenStaking.fee(), 2 ether);

// Try to execute the third operation (should fail due to unexecuted predecessor)
vm.prank(executor);
vm.expectRevert(abi.encodeWithSelector(TimelockController.TimelockUnexecutedPredecessor.selector, id2));
timelock.execute(target, value, data3, id2, salt3);

// Execute the second operation
vm.prank(executor);
timelock.execute(target, value, data2, id1, salt2);
assertEq(ipTokenStaking.fee(), 3 ether);

// Finally, execute the third operation
vm.prank(executor);
timelock.execute(target, value, data3, id2, salt3);
assertEq(ipTokenStaking.fee(), 4 ether);
}
}
111 changes: 111 additions & 0 deletions contracts/test/upgrades/PredeployUpgrades.t.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
// SPDX-License-Identifier: GPL-3.0-only
pragma solidity ^0.8.23;

import { ProxyAdmin } from "@openzeppelin/contracts/proxy/transparent/ProxyAdmin.sol";
// solhint-disable-next-line max-line-length
import { ITransparentUpgradeableProxy } from "@openzeppelin/contracts/proxy/transparent/TransparentUpgradeableProxy.sol";

import { IPTokenStaking } from "../../src/protocol/IPTokenStaking.sol";
import { UpgradeEntrypoint } from "../../src/protocol/UpgradeEntrypoint.sol";
import { UBIPool } from "../../src/protocol/UBIPool.sol";

import { EIP1967Helper } from "../../script/utils/EIP1967Helper.sol";
import { Predeploys } from "../../src/libraries/Predeploys.sol";
import { Test } from "../utils/Test.sol";

abstract contract MockNewFeatures {
function foo() external pure returns (string memory) {
return "bar";
}
}

contract IPTokenStakingV2 is IPTokenStaking, MockNewFeatures {
constructor(uint256 stakingRounding, uint256 defaultMinFee) IPTokenStaking(stakingRounding, defaultMinFee) {}
}

contract UpgradeEntrypointV2 is UpgradeEntrypoint, MockNewFeatures {}

contract UBIPoolV2 is UBIPool, MockNewFeatures {
constructor(uint32 maxUBIPercentage) UBIPool(maxUBIPercentage) {}
}

/**
* @title PrecompileUpgrades
* @dev A script to test upgrading the precompile contracts
*/
contract PrecompileUpgrades is Test {
function testUpgradeStaking() public {
// ---- Staking
address newImpl = address(
new IPTokenStakingV2(
1 gwei, // stakingRounding
1 ether
)
);
ProxyAdmin proxyAdmin = ProxyAdmin(EIP1967Helper.getAdmin(Predeploys.Staking));
assertEq(proxyAdmin.owner(), address(timelock));

performTimelocked(
address(proxyAdmin),
abi.encodeWithSelector(
ProxyAdmin.upgradeAndCall.selector,
ITransparentUpgradeableProxy(Predeploys.Staking),
newImpl,
""
)
);

assertEq(EIP1967Helper.getImplementation(Predeploys.Staking), newImpl, "Staking not upgraded");
assertEq(
keccak256(abi.encode(IPTokenStakingV2(Predeploys.Staking).foo())),
keccak256(abi.encode("bar")),
"Upgraded to wrong iface"
);
}

function testUpgradeUpgradeEntrypoint() public {
// ---- Upgrades
address newImpl = address(new UpgradeEntrypointV2());
ProxyAdmin proxyAdmin = ProxyAdmin(EIP1967Helper.getAdmin(Predeploys.Upgrades));
assertEq(proxyAdmin.owner(), address(timelock));

performTimelocked(
address(proxyAdmin),
abi.encodeWithSelector(
ProxyAdmin.upgradeAndCall.selector,
ITransparentUpgradeableProxy(Predeploys.Upgrades),
newImpl,
""
)
);
assertEq(EIP1967Helper.getImplementation(Predeploys.Upgrades), newImpl, "Upgrades not upgraded");
assertEq(
keccak256(abi.encode(IPTokenStakingV2(Predeploys.Upgrades).foo())),
keccak256(abi.encode("bar")),
"Upgraded to wrong iface"
);
}

function testUpgradeUBIPool() public {
// ---- UBIPool
address newImpl = address(new UBIPoolV2(10_00));
ProxyAdmin proxyAdmin = ProxyAdmin(EIP1967Helper.getAdmin(Predeploys.UBIPool));
assertEq(proxyAdmin.owner(), address(timelock));

performTimelocked(
address(proxyAdmin),
abi.encodeWithSelector(
ProxyAdmin.upgradeAndCall.selector,
ITransparentUpgradeableProxy(Predeploys.UBIPool),
newImpl,
""
)
);
assertEq(EIP1967Helper.getImplementation(Predeploys.UBIPool), newImpl, "Upgrades not upgraded");
assertEq(
keccak256(abi.encode(IPTokenStakingV2(Predeploys.UBIPool).foo())),
keccak256(abi.encode("bar")),
"Upgraded to wrong iface"
);
}
}

0 comments on commit 4d3a52d

Please sign in to comment.