From e0df7f0f5bfc8320fdab1c6c7aecf397c4ad3675 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 14:38:13 +0200 Subject: [PATCH 01/16] Added L2VotingPowerPaused and L2GovernorPaused contract with test cases --- src/L2/paused/L2GovernorPaused.sol | 151 ++++++++++++++++++++++ src/L2/paused/L2VotingPowerPaused.sol | 38 ++++++ test/L2/paused/L2GovernorPaused.t.sol | 155 +++++++++++++++++++++++ test/L2/paused/L2VotingPowerPaused.t.sol | 110 ++++++++++++++++ 4 files changed, 454 insertions(+) create mode 100644 src/L2/paused/L2GovernorPaused.sol create mode 100644 src/L2/paused/L2VotingPowerPaused.sol create mode 100644 test/L2/paused/L2GovernorPaused.t.sol create mode 100644 test/L2/paused/L2VotingPowerPaused.t.sol diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol new file mode 100644 index 00000000..b857f698 --- /dev/null +++ b/src/L2/paused/L2GovernorPaused.sol @@ -0,0 +1,151 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { Ownable2StepUpgradeable } from "@openzeppelin-upgradeable/contracts/access/Ownable2StepUpgradeable.sol"; +import { Initializable } from "@openzeppelin-upgradeable/contracts/proxy/utils/Initializable.sol"; +import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol"; +import { TimelockControllerUpgradeable } from + "@openzeppelin-upgradeable/contracts/governance/extensions/GovernorTimelockControlUpgradeable.sol"; +import { L2Governor } from "../L2Governor.sol"; + +contract L2GovernorPaused is L2Governor { + /// @notice Setting global params. + function initializePaused() public reinitializer(2) { } + + function version() public pure virtual override returns (string memory) { + return string.concat(super.version(), "-paused"); + } + + /// @notice Override the cancel function to prevent staking from being processed. + function cancel( + address[] memory, + uint256[] memory, + bytes[] memory, + bytes32 + ) + public + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the castVote function to prevent staking from being processed. + function castVote(uint256, uint8) public virtual override returns (uint256) { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the castVoteBySig function to prevent staking from being processed. + function castVoteBySig(uint256, uint8, address, bytes memory) public virtual override returns (uint256) { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the castVoteWithReason function to prevent staking from being processed. + function castVoteWithReason(uint256, uint8, string calldata) public virtual override returns (uint256) { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the castVoteWithReasonAndParams function to prevent staking from being processed. + function castVoteWithReasonAndParams( + uint256, + uint8, + string calldata, + bytes memory + ) + public + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the castVoteWithReasonAndParamsBySig function to prevent staking from being processed. + function castVoteWithReasonAndParamsBySig( + uint256, + uint8, + address, + string calldata, + bytes memory, + bytes memory + ) + public + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the execute function to prevent staking from being processed. + function execute( + address[] memory, + uint256[] memory, + bytes[] memory, + bytes32 + ) + public + payable + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the propose function to prevent staking from being processed. + function propose( + address[] memory, + uint256[] memory, + bytes[] memory, + string memory + ) + public + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the queue function to prevent staking from being processed. + function queue( + address[] memory, + uint256[] memory, + bytes[] memory, + bytes32 + ) + public + virtual + override + returns (uint256) + { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the relay function to prevent staking from being processed. + function relay(address, uint256, bytes memory) public payable virtual override { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the setProposalThreshold function to prevent staking from being processed. + function setProposalThreshold(uint256) public virtual override { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the setVotingDelay function to prevent staking from being processed. + function setVotingDelay(uint48) public virtual override { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the setVotingPeriod function to prevent staking from being processed. + function setVotingPeriod(uint32) public virtual override { + revert("L2GovernorPaused: Governor is paused"); + } + + /// @notice Override the updateTimelock function to prevent staking from being processed. + function updateTimelock(TimelockControllerUpgradeable) public virtual override { + revert("L2GovernorPaused: Governor is paused"); + } +} diff --git a/src/L2/paused/L2VotingPowerPaused.sol b/src/L2/paused/L2VotingPowerPaused.sol new file mode 100644 index 00000000..bcaacc85 --- /dev/null +++ b/src/L2/paused/L2VotingPowerPaused.sol @@ -0,0 +1,38 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { Ownable2StepUpgradeable } from "@openzeppelin-upgradeable/contracts/access/Ownable2StepUpgradeable.sol"; +import { Initializable } from "@openzeppelin-upgradeable/contracts/proxy/utils/Initializable.sol"; +import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol"; +import { IL2LockingPosition } from "../../interfaces/L2/IL2LockingPosition.sol"; +import { L2VotingPower } from "../L2VotingPower.sol"; + +contract L2VotingPowerPaused is L2VotingPower { + /// @notice Setting global params. + function initializePaused() public reinitializer(2) { + version = "1.0.0-paused"; + } + + /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + function adjustVotingPower( + address, + IL2LockingPosition.LockingPosition memory, + IL2LockingPosition.LockingPosition memory + ) + public + virtual + override + { + revert("L2VotingPowerPaused: VotingPower is paused"); + } + + /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + function delegate(address) public virtual override { + revert("L2VotingPowerPaused: VotingPower is paused"); + } + + /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + function delegateBySig(address, uint256, uint256, uint8, bytes32, bytes32) public virtual override { + revert("L2VotingPowerPaused: VotingPower is paused"); + } +} diff --git a/test/L2/paused/L2GovernorPaused.t.sol b/test/L2/paused/L2GovernorPaused.t.sol new file mode 100644 index 00000000..932e6a4d --- /dev/null +++ b/test/L2/paused/L2GovernorPaused.t.sol @@ -0,0 +1,155 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { IVotes } from "@openzeppelin-upgradeable/contracts/governance/extensions/GovernorVotesUpgradeable.sol"; +import { OwnableUpgradeable } from "@openzeppelin-upgradeable/contracts/access/OwnableUpgradeable.sol"; +import { TimelockController } from "@openzeppelin/contracts/governance/TimelockController.sol"; +import { ERC1967Proxy } from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import { TimelockControllerUpgradeable } from + "@openzeppelin-upgradeable/contracts/governance/extensions/GovernorTimelockControlUpgradeable.sol"; +import { Test, console } from "forge-std/Test.sol"; +import { L2Governor } from "src/L2/L2Governor.sol"; +import { L2GovernorPaused } from "src/L2/paused/L2GovernorPaused.sol"; +import { Utils } from "script/contracts/Utils.sol"; + +contract MockL2GovernorV2 is L2Governor { + function version() public pure virtual override returns (string memory) { + return "2.0.0"; + } +} + +contract L2GovernorPausedTest is Test { + Utils public utils; + L2Governor public l2GovernorImplementation; + L2Governor public l2Governor; + + IVotes votingPower; + address[] executors; + TimelockController timelock; + address initialOwner; + + function setUp() public { + utils = new Utils(); + + // set initial values + votingPower = IVotes(address(0x1)); + executors.push(address(0)); // executor array contains address(0) such that anyone can execute proposals + timelock = new TimelockController(0, new address[](0), executors, address(this)); + initialOwner = address(this); + + console.log("L2GovernorTest address is: %s", address(this)); + + // deploy L2Governor Implementation contract + l2GovernorImplementation = new L2Governor(); + + // deploy L2Governor contract via proxy and initialize it at the same time + l2Governor = L2Governor( + payable( + address( + new ERC1967Proxy( + address(l2GovernorImplementation), + abi.encodeWithSelector(l2Governor.initialize.selector, votingPower, timelock, initialOwner) + ) + ) + ) + ); + + assertEq(l2Governor.name(), "Lisk Governor"); + assertEq(l2Governor.version(), "1.0.0"); + assertEq(l2Governor.votingDelay(), 0); + assertEq(l2Governor.votingPeriod(), 604800); + assertEq(l2Governor.proposalThreshold(), 300_000 * 10 ** 18); + assertEq(l2Governor.timelock(), address(timelock)); + assertEq(l2Governor.quorum(0), 24_000_000 * 10 ** 18); + assertEq(address(l2Governor.token()), address(votingPower)); + assertEq(l2Governor.owner(), initialOwner); + + // assure that address(0) is in executors role + assertEq(timelock.hasRole(timelock.EXECUTOR_ROLE(), address(0)), true); + + // Upgrade from L2Governor to L2GovernorPaused + L2GovernorPaused l2GovernorPausedImplementation = new L2GovernorPaused(); + l2Governor.upgradeToAndCall(address(l2GovernorPausedImplementation), ""); + assertEq(l2Governor.version(), "1.0.0-paused"); + } + + function test_Cancel_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.cancel(new address[](1), new uint256[](1), new bytes[](1), 0); + } + + function test_CastVote_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.castVote(0, 0); + } + + function test_CastVoteBySig_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.castVoteBySig(0, 0, address(0), ""); + } + + function test_CastVoteWithReason_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.castVoteWithReason(0, 0, ""); + } + + function test_CastVoteWithReasonAndParams_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.castVoteWithReasonAndParams(0, 0, "", ""); + } + + function test_CastVoteWithReasonAndParamsBySig_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.castVoteWithReasonAndParamsBySig(0, 0, address(0), "", "", ""); + } + + function test_Execute_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.execute(new address[](1), new uint256[](1), new bytes[](1), 0); + } + + function test_Propose_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.propose(new address[](1), new uint256[](1), new bytes[](1), ""); + } + + function test_Queue_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.queue(new address[](1), new uint256[](1), new bytes[](1), ""); + } + + function test_Relay_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.relay(address(0), 0, ""); + } + + function test_SetProposalThreshold_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.setProposalThreshold(0); + } + + function test_SetVotingDelay_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.setVotingDelay(0); + } + + function test_SetVotingPeriod_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.setVotingPeriod(0); + } + + function test_UpdateTimelock_Paused() public { + vm.expectRevert("L2GovernorPaused: Governor is paused"); + l2Governor.updateTimelock(TimelockControllerUpgradeable(payable(address(0)))); + } + + function test_UpgradeToAndCall_CanUpgradeFromPausedContractToNewContract() public { + MockL2GovernorV2 mockL2GovernorV2Implementation = new MockL2GovernorV2(); + + // upgrade contract + l2Governor.upgradeToAndCall(address(mockL2GovernorV2Implementation), ""); + + // new version updated + assertEq(l2Governor.version(), "2.0.0"); + } +} diff --git a/test/L2/paused/L2VotingPowerPaused.t.sol b/test/L2/paused/L2VotingPowerPaused.t.sol new file mode 100644 index 00000000..45fb31fb --- /dev/null +++ b/test/L2/paused/L2VotingPowerPaused.t.sol @@ -0,0 +1,110 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { IVotes } from "@openzeppelin/contracts/governance/utils/IVotes.sol"; +import { ERC1967Proxy } from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import { OwnableUpgradeable } from "@openzeppelin-upgradeable/contracts/access/OwnableUpgradeable.sol"; +import { IAccessControl } from "@openzeppelin/contracts/access/IAccessControl.sol"; +import { IERC20Errors } from "@openzeppelin/contracts/interfaces/draft-IERC6093.sol"; +import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; +import { Test, console } from "forge-std/Test.sol"; +import { IL2LockingPosition } from "src/interfaces/L2/IL2LockingPosition.sol"; +import { L2VotingPower } from "src/L2/L2VotingPower.sol"; +import { L2VotingPowerPaused } from "src/L2/paused/L2VotingPowerPaused.sol"; +import { Utils } from "script/contracts/Utils.sol"; + +contract MockL2VotingPowerV2 is L2VotingPower { + uint256 public testNumber; + + function initializeV2(uint256 _testNumber) public reinitializer(3) { + version = "2.0.0"; + testNumber = _testNumber; + } + + function onlyV2() public pure returns (string memory) { + return "Only L2VotingPowerV2 have this function"; + } +} + +contract L2VotingPowerPausedTest is Test { + Utils public utils; + L2VotingPower public l2VotingPowerImplementation; + L2VotingPower public l2VotingPower; + + address lockingPositionContractAddress; + + function setUp() public { + utils = new Utils(); + + // set initial values + lockingPositionContractAddress = address(0xdeadbeefdeadbeefdeadbeef); + + console.log("L2VotingPowerTest address is: %s", address(this)); + + // deploy L2VotingPower Implementation contract + l2VotingPowerImplementation = new L2VotingPower(); + + // deploy L2VotingPower contract via proxy and initialize it at the same time + l2VotingPower = L2VotingPower( + address( + new ERC1967Proxy( + address(l2VotingPowerImplementation), + abi.encodeWithSelector(l2VotingPower.initialize.selector, lockingPositionContractAddress) + ) + ) + ); + + assertEq(l2VotingPower.lockingPositionAddress(), lockingPositionContractAddress); + assertEq(l2VotingPower.version(), "1.0.0"); + assertEq(l2VotingPower.name(), "Lisk Voting Power"); + assertEq(l2VotingPower.symbol(), "vpLSK"); + + // Upgrade from L2VotingPower to L2VotingPowerPaused + L2VotingPowerPaused l2VotingPowerPausedImplementation = new L2VotingPowerPaused(); + l2VotingPower.upgradeToAndCall(address(l2VotingPowerPausedImplementation), ""); + + // Call initializePaused with new interface + L2VotingPowerPaused(address(l2VotingPower)).initializePaused(); + + // l2VotingPower pointing to paused contract + assertEq(l2VotingPower.version(), "1.0.0-paused"); + } + + function test_adjustVotingPower_Paused() public { + IL2LockingPosition.LockingPosition memory positionBefore = + IL2LockingPosition.LockingPosition(address(this), 0, 0, 0); + IL2LockingPosition.LockingPosition memory positionAfter = + IL2LockingPosition.LockingPosition(address(this), 50, 0, 0); + + vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); + l2VotingPower.adjustVotingPower(address(0), positionBefore, positionAfter); + } + + function test_delegate_Paused() public { + vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); + l2VotingPower.delegate(address(0)); + } + + function test_delegateBySig_Paused() public { + vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); + l2VotingPower.delegateBySig(address(0), 0, 0, 0, "", ""); + } + + function test_UpgradeToAndCall_CanUpgradeFromPausedContractToNewContract() public { + MockL2VotingPowerV2 mockL2VotingPowerV2Implementation = new MockL2VotingPowerV2(); + + uint256 testNumber = 123; + + // upgrade contract, and also change some variables by reinitialize + l2VotingPower.upgradeToAndCall( + address(mockL2VotingPowerV2Implementation), + abi.encodeWithSelector(mockL2VotingPowerV2Implementation.initializeV2.selector, testNumber) + ); + + // new testNumber variable introduced + assertEq(MockL2VotingPowerV2(address(l2VotingPower)).testNumber(), testNumber); + + // new version updated + assertEq(l2VotingPower.version(), "2.0.0"); + } +} From 5ac3c6fec4e4655f751324a6bc9b2e1b92cf42f2 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 16:14:07 +0200 Subject: [PATCH 02/16] Add deployment script --- .../L2/paused/L2GovernorPaused.s.sol | 53 ++++++++++++++++++ .../L2/paused/L2VotingPowerPaused.s.sol | 56 +++++++++++++++++++ script/contracts/Utils.sol | 4 ++ 3 files changed, 113 insertions(+) create mode 100644 script/contracts/L2/paused/L2GovernorPaused.s.sol create mode 100644 script/contracts/L2/paused/L2VotingPowerPaused.s.sol diff --git a/script/contracts/L2/paused/L2GovernorPaused.s.sol b/script/contracts/L2/paused/L2GovernorPaused.s.sol new file mode 100644 index 00000000..66b5719d --- /dev/null +++ b/script/contracts/L2/paused/L2GovernorPaused.s.sol @@ -0,0 +1,53 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { ERC1967Proxy } from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import { Upgrades } from "openzeppelin-foundry-upgrades/Upgrades.sol"; +import { Options } from "openzeppelin-foundry-upgrades/Options.sol"; +import { Script, console2 } from "forge-std/Script.sol"; +import { L2GovernorPaused } from "src/L2/paused/L2GovernorPaused.sol"; +import "script/contracts/Utils.sol"; + +/// @title L2GovernorPausedScript - L2GovernorPaused contract deployment script +/// @notice This contract is used to deploy L2GovernorPaused contract and write its address to JSON file. +contract L2GovernorPausedScript is Script { + /// @notice Utils contract which provides functions to read and write JSON files containing L1 and L2 addresses. + Utils utils; + + function setUp() public { + utils = new Utils(); + } + + /// @notice This function deploys L2GovernorPaused contract and writes its address to JSON file. + function run() public { + // Deployer's private key. This key is used to deploy the contract. + uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + + // Validate L2GovernorPaused contract if it is implemented correctly so that it may be used as new + // implementation for the proxy contract. + Options memory opts; + opts.referenceContract = "L2Governor.sol"; + opts.unsafeAllow = "constructor,external-library-linking"; + Upgrades.validateUpgrade("L2GovernorPaused.sol", opts); + + console2.log("Deploying L2GovernorPaused contract..."); + + // deploy L2GovernorPaused contract + vm.startBroadcast(deployerPrivateKey); + L2GovernorPaused l2GovernorPaused = new L2GovernorPaused(); + vm.stopBroadcast(); + + assert(address(l2GovernorPaused) != address(0)); + + // ERC1967Utils: keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1. + assert(l2GovernorPaused.proxiableUUID() == bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1)); + + console2.log("L2GovernorPaused contract successfully deployed!"); + console2.log("L2GovernorPaused (Implementation) address: %s", address(l2GovernorPaused)); + + // write L2GovernorPaused address to l2addresses.json + Utils.L2AddressesConfig memory l2AddressesConfig = utils.readL2AddressesFile(); + l2AddressesConfig.L2GovernorPaused = address(l2GovernorPaused); + utils.writeL2AddressesFile(l2AddressesConfig); + } +} diff --git a/script/contracts/L2/paused/L2VotingPowerPaused.s.sol b/script/contracts/L2/paused/L2VotingPowerPaused.s.sol new file mode 100644 index 00000000..7a1da787 --- /dev/null +++ b/script/contracts/L2/paused/L2VotingPowerPaused.s.sol @@ -0,0 +1,56 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { ERC1967Proxy } from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import { Upgrades } from "openzeppelin-foundry-upgrades/Upgrades.sol"; +import { Options } from "openzeppelin-foundry-upgrades/Options.sol"; +import { Script, console2 } from "forge-std/Script.sol"; +import { L2VotingPowerPaused } from "src/L2/paused/L2VotingPowerPaused.sol"; +import "script/contracts/Utils.sol"; + +/// @title L2VotingPowerPausedScript - L2VotingPowerPaused contract deployment script +/// @notice This contract is used to deploy L2VotingPowerPaused contract and write its address to JSON file. +contract L2VotingPowerPausedScript is Script { + /// @notice Utils contract which provides functions to read and write JSON files containing L1 and L2 addresses. + Utils utils; + + function setUp() public { + utils = new Utils(); + } + + /// @notice This function deploys L2VotingPowerPaused contract and writes its address to JSON file. + function run() public { + // Deployer's private key. This key is used to deploy the contract. + uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + + // Validate L2VotingPowerPaused contract if it is implemented correctly so that it may be used as new + // implementation for the proxy contract. + Options memory opts; + opts.referenceContract = "L2VotingPower.sol"; + opts.unsafeAllow = "constructor,external-library-linking"; + Upgrades.validateUpgrade("L2VotingPowerPaused.sol", opts); + + console2.log("Deploying L2VotingPowerPaused contract..."); + + // deploy L2VotingPowerPaused contract + vm.startBroadcast(deployerPrivateKey); + L2VotingPowerPaused l2VotingPowerPausedImplementation = new L2VotingPowerPaused(); + vm.stopBroadcast(); + + assert(address(l2VotingPowerPausedImplementation) != address(0)); + + // ERC1967Utils: keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1. + assert( + l2VotingPowerPausedImplementation.proxiableUUID() + == bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1) + ); + + console2.log("L2VotingPowerPaused contract successfully deployed!"); + console2.log("L2VotingPowerPaused (Implementation) address: %s", address(l2VotingPowerPausedImplementation)); + + // write L2VotingPowerPaused address to l2addresses.json + Utils.L2AddressesConfig memory l2AddressesConfig = utils.readL2AddressesFile(); + l2AddressesConfig.L2VotingPowerPaused = address(l2VotingPowerPausedImplementation); + utils.writeL2AddressesFile(l2AddressesConfig); + } +} diff --git a/script/contracts/Utils.sol b/script/contracts/Utils.sol index 3402ebf8..693a44f4 100644 --- a/script/contracts/Utils.sol +++ b/script/contracts/Utils.sol @@ -29,6 +29,8 @@ contract Utils is Script { address L2Governor; /// @notice The Current implementation of L2 Governor Contract. address L2GovernorImplementation; + /// @notice The Current implementation of L2GovernorPaused Contract. + address L2GovernorPaused; /// @notice L2 Lisk token address. address L2LiskToken; /// @notice L2 Locking Position contract (in Proxy), which users interact with. @@ -51,6 +53,8 @@ contract Utils is Script { address L2VotingPower; /// @notice The Current implementation of L2 Voting Power Contract. address L2VotingPowerImplementation; + /// @notice The Current implementation of L2VotingPowerPaused Contract. + address L2VotingPowerPaused; } /// @notice This struct is used to read MerkleRoot from JSON file. From 3f038e82d48eed786ad7c6992002b14c96c9b2ef Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 17:25:12 +0200 Subject: [PATCH 03/16] Update git submodules --- .gitmodules | 8 +++++++- lib/forge-std | 2 +- lib/openzeppelin-foundry-upgrades | 1 + lib/properties | 1 + 4 files changed, 10 insertions(+), 2 deletions(-) create mode 160000 lib/openzeppelin-foundry-upgrades create mode 160000 lib/properties diff --git a/.gitmodules b/.gitmodules index bd03c845..feb437a3 100644 --- a/.gitmodules +++ b/.gitmodules @@ -6,4 +6,10 @@ url = https://github.com/OpenZeppelin/openzeppelin-contracts [submodule "lib/openzeppelin-contracts-upgradeable"] path = lib/openzeppelin-contracts-upgradeable - url = https://github.com/OpenZeppelin/openzeppelin-contracts-upgradeable \ No newline at end of file + url = https://github.com/OpenZeppelin/openzeppelin-contracts-upgradeable +[submodule "lib/properties"] + path = lib/properties + url = https://github.com/crytic/properties +[submodule "lib/openzeppelin-foundry-upgrades"] + path = lib/openzeppelin-foundry-upgrades + url = https://github.com/OpenZeppelin/openzeppelin-foundry-upgrades diff --git a/lib/forge-std b/lib/forge-std index ae570fec..bb4ceea9 160000 --- a/lib/forge-std +++ b/lib/forge-std @@ -1 +1 @@ -Subproject commit ae570fec082bfe1c1f45b0acca4a2b4f84d345ce +Subproject commit bb4ceea94d6f10eeb5b41dc2391c6c8bf8e734ef diff --git a/lib/openzeppelin-foundry-upgrades b/lib/openzeppelin-foundry-upgrades new file mode 160000 index 00000000..4cd15fc5 --- /dev/null +++ b/lib/openzeppelin-foundry-upgrades @@ -0,0 +1 @@ +Subproject commit 4cd15fc50b141c77d8cc9ff8efb44d00e841a299 diff --git a/lib/properties b/lib/properties new file mode 160000 index 00000000..bb1b7854 --- /dev/null +++ b/lib/properties @@ -0,0 +1 @@ +Subproject commit bb1b78542b3f38e4ae56cf87389cd3ea94387f48 From ac513fd8916eaca6f2f6f1f4540d5dbbd331fc96 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 18:05:21 +0200 Subject: [PATCH 04/16] Add foundry.toml --- foundry.toml | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/foundry.toml b/foundry.toml index 8847712d..d42f8bf1 100644 --- a/foundry.toml +++ b/foundry.toml @@ -12,11 +12,20 @@ remappings = [ 'ds-test/=lib/forge-std/lib/ds-test/src/', 'erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/', 'forge-std/=lib/forge-std/src/', - '@openzeppelin/=lib/openzeppelin-contracts/', '@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts/', 'openzeppelin-contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/', 'openzeppelin-contracts/=lib/openzeppelin-contracts/', + '@openzeppelin/contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/contracts/', + 'ERC4626/=lib/properties/lib/ERC4626/contracts/', + 'properties/=lib/properties/contracts/', + 'solmate/=lib/properties/lib/solmate/src/', + 'openzeppelin-foundry-upgrades/=lib/openzeppelin-foundry-upgrades/src/', + 'solidity-stringutils/=lib/openzeppelin-foundry-upgrades/lib/solidity-stringutils/', ] +ffi = true +ast = true +build_info = true +extra_output = ["storageLayout"] [fmt] line_length = 120 From 69e0594b1ba29980da20c20be2604bec65aff446 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 18:07:55 +0200 Subject: [PATCH 05/16] Change version function to be independent --- src/L2/paused/L2GovernorPaused.sol | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol index b857f698..23dbbd07 100644 --- a/src/L2/paused/L2GovernorPaused.sol +++ b/src/L2/paused/L2GovernorPaused.sol @@ -13,7 +13,7 @@ contract L2GovernorPaused is L2Governor { function initializePaused() public reinitializer(2) { } function version() public pure virtual override returns (string memory) { - return string.concat(super.version(), "-paused"); + return "1.0.0-paused"; } /// @notice Override the cancel function to prevent staking from being processed. From 8a21d2182251938c104af966e97e335a656fda8e Mon Sep 17 00:00:00 2001 From: Franco NG Date: Tue, 11 Jun 2024 20:01:30 +0200 Subject: [PATCH 06/16] Update comments --- src/L2/paused/L2GovernorPaused.sol | 29 ++++++++++++++------------- src/L2/paused/L2VotingPowerPaused.sol | 6 +++--- 2 files changed, 18 insertions(+), 17 deletions(-) diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol index 23dbbd07..8512d5dc 100644 --- a/src/L2/paused/L2GovernorPaused.sol +++ b/src/L2/paused/L2GovernorPaused.sol @@ -12,11 +12,12 @@ contract L2GovernorPaused is L2Governor { /// @notice Setting global params. function initializePaused() public reinitializer(2) { } + /// @notice Marking version as paused. function version() public pure virtual override returns (string memory) { return "1.0.0-paused"; } - /// @notice Override the cancel function to prevent staking from being processed. + /// @notice Override the cancel function to pause Governor interactions. function cancel( address[] memory, uint256[] memory, @@ -31,22 +32,22 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the castVote function to prevent staking from being processed. + /// @notice Override the castVote function to pause Governor interactions. function castVote(uint256, uint8) public virtual override returns (uint256) { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the castVoteBySig function to prevent staking from being processed. + /// @notice Override the castVoteBySig function to pause Governor interactions. function castVoteBySig(uint256, uint8, address, bytes memory) public virtual override returns (uint256) { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the castVoteWithReason function to prevent staking from being processed. + /// @notice Override the castVoteWithReason function to pause Governor interactions. function castVoteWithReason(uint256, uint8, string calldata) public virtual override returns (uint256) { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the castVoteWithReasonAndParams function to prevent staking from being processed. + /// @notice Override the castVoteWithReasonAndParams function to pause Governor interactions. function castVoteWithReasonAndParams( uint256, uint8, @@ -61,7 +62,7 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the castVoteWithReasonAndParamsBySig function to prevent staking from being processed. + /// @notice Override the castVoteWithReasonAndParamsBySig function to pause Governor interactions. function castVoteWithReasonAndParamsBySig( uint256, uint8, @@ -78,7 +79,7 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the execute function to prevent staking from being processed. + /// @notice Override the execute function to pause Governor interactions. function execute( address[] memory, uint256[] memory, @@ -94,7 +95,7 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the propose function to prevent staking from being processed. + /// @notice Override the propose function to pause Governor interactions. function propose( address[] memory, uint256[] memory, @@ -109,7 +110,7 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the queue function to prevent staking from being processed. + /// @notice Override the queue function to pause Governor interactions. function queue( address[] memory, uint256[] memory, @@ -124,27 +125,27 @@ contract L2GovernorPaused is L2Governor { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the relay function to prevent staking from being processed. + /// @notice Override the relay function to pause Governor interactions. function relay(address, uint256, bytes memory) public payable virtual override { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the setProposalThreshold function to prevent staking from being processed. + /// @notice Override the setProposalThreshold function to pause Governor interactions. function setProposalThreshold(uint256) public virtual override { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the setVotingDelay function to prevent staking from being processed. + /// @notice Override the setVotingDelay function to pause Governor interactions. function setVotingDelay(uint48) public virtual override { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the setVotingPeriod function to prevent staking from being processed. + /// @notice Override the setVotingPeriod function to pause Governor interactions. function setVotingPeriod(uint32) public virtual override { revert("L2GovernorPaused: Governor is paused"); } - /// @notice Override the updateTimelock function to prevent staking from being processed. + /// @notice Override the updateTimelock function to pause Governor interactions. function updateTimelock(TimelockControllerUpgradeable) public virtual override { revert("L2GovernorPaused: Governor is paused"); } diff --git a/src/L2/paused/L2VotingPowerPaused.sol b/src/L2/paused/L2VotingPowerPaused.sol index bcaacc85..5e0e8cf7 100644 --- a/src/L2/paused/L2VotingPowerPaused.sol +++ b/src/L2/paused/L2VotingPowerPaused.sol @@ -13,7 +13,7 @@ contract L2VotingPowerPaused is L2VotingPower { version = "1.0.0-paused"; } - /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. function adjustVotingPower( address, IL2LockingPosition.LockingPosition memory, @@ -26,12 +26,12 @@ contract L2VotingPowerPaused is L2VotingPower { revert("L2VotingPowerPaused: VotingPower is paused"); } - /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. function delegate(address) public virtual override { revert("L2VotingPowerPaused: VotingPower is paused"); } - /// @notice Override the modifyLockingPosition function to prevent staking from being processed. + /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. function delegateBySig(address, uint256, uint256, uint8, bytes32, bytes32) public virtual override { revert("L2VotingPowerPaused: VotingPower is paused"); } From 817bd5d22e421dced2bc2d0422f34cfa7b2d4422 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 10:50:32 +0200 Subject: [PATCH 07/16] Update gitsubmodule, added Custom Errors --- .gitmodules | 3 - foundry.toml | 4 +- lib/properties | 1 - .../L2/paused/L2GovernorPaused.s.sol | 2 +- .../L2/paused/L2VotingPowerPaused.s.sol | 2 +- script/contracts/Utils.sol | 12 +++- src/L2/paused/L2GovernorPaused.sol | 64 +++++++++++++++---- src/L2/paused/L2VotingPowerPaused.sol | 20 ++---- test/L2/paused/L2GovernorPaused.t.sol | 35 +++++----- test/L2/paused/L2VotingPowerPaused.t.sol | 25 ++------ 10 files changed, 93 insertions(+), 75 deletions(-) delete mode 160000 lib/properties diff --git a/.gitmodules b/.gitmodules index feb437a3..f27e450e 100644 --- a/.gitmodules +++ b/.gitmodules @@ -7,9 +7,6 @@ [submodule "lib/openzeppelin-contracts-upgradeable"] path = lib/openzeppelin-contracts-upgradeable url = https://github.com/OpenZeppelin/openzeppelin-contracts-upgradeable -[submodule "lib/properties"] - path = lib/properties - url = https://github.com/crytic/properties [submodule "lib/openzeppelin-foundry-upgrades"] path = lib/openzeppelin-foundry-upgrades url = https://github.com/OpenZeppelin/openzeppelin-foundry-upgrades diff --git a/foundry.toml b/foundry.toml index d42f8bf1..55084799 100644 --- a/foundry.toml +++ b/foundry.toml @@ -16,12 +16,10 @@ remappings = [ 'openzeppelin-contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/', 'openzeppelin-contracts/=lib/openzeppelin-contracts/', '@openzeppelin/contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/contracts/', - 'ERC4626/=lib/properties/lib/ERC4626/contracts/', - 'properties/=lib/properties/contracts/', - 'solmate/=lib/properties/lib/solmate/src/', 'openzeppelin-foundry-upgrades/=lib/openzeppelin-foundry-upgrades/src/', 'solidity-stringutils/=lib/openzeppelin-foundry-upgrades/lib/solidity-stringutils/', ] + ffi = true ast = true build_info = true diff --git a/lib/properties b/lib/properties deleted file mode 160000 index bb1b7854..00000000 --- a/lib/properties +++ /dev/null @@ -1 +0,0 @@ -Subproject commit bb1b78542b3f38e4ae56cf87389cd3ea94387f48 diff --git a/script/contracts/L2/paused/L2GovernorPaused.s.sol b/script/contracts/L2/paused/L2GovernorPaused.s.sol index 66b5719d..831fd01e 100644 --- a/script/contracts/L2/paused/L2GovernorPaused.s.sol +++ b/script/contracts/L2/paused/L2GovernorPaused.s.sol @@ -27,7 +27,7 @@ contract L2GovernorPausedScript is Script { // implementation for the proxy contract. Options memory opts; opts.referenceContract = "L2Governor.sol"; - opts.unsafeAllow = "constructor,external-library-linking"; + opts.unsafeAllow = "constructor"; Upgrades.validateUpgrade("L2GovernorPaused.sol", opts); console2.log("Deploying L2GovernorPaused contract..."); diff --git a/script/contracts/L2/paused/L2VotingPowerPaused.s.sol b/script/contracts/L2/paused/L2VotingPowerPaused.s.sol index 7a1da787..bc721c5b 100644 --- a/script/contracts/L2/paused/L2VotingPowerPaused.s.sol +++ b/script/contracts/L2/paused/L2VotingPowerPaused.s.sol @@ -27,7 +27,7 @@ contract L2VotingPowerPausedScript is Script { // implementation for the proxy contract. Options memory opts; opts.referenceContract = "L2VotingPower.sol"; - opts.unsafeAllow = "constructor,external-library-linking"; + opts.unsafeAllow = "constructor"; Upgrades.validateUpgrade("L2VotingPowerPaused.sol", opts); console2.log("Deploying L2VotingPowerPaused contract..."); diff --git a/script/contracts/Utils.sol b/script/contracts/Utils.sol index 693a44f4..334670f6 100644 --- a/script/contracts/Utils.sol +++ b/script/contracts/Utils.sol @@ -178,6 +178,10 @@ contract Utils is Script { l2AddressesConfig.L2GovernorImplementation = l2GovernorImplementation; } catch { } + try vm.parseJsonAddress(addressJson, ".L2GovernorPaused") returns (address l2GovernorPaused) { + l2AddressesConfig.L2GovernorPaused = l2GovernorPaused; + } catch { } + try vm.parseJsonAddress(addressJson, ".L2LiskToken") returns (address l2LiskToken) { l2AddressesConfig.L2LiskToken = l2LiskToken; } catch { } @@ -227,6 +231,9 @@ contract Utils is Script { ) { l2AddressesConfig.L2VotingPowerImplementation = l2VotingPowerImplementation; } catch { } + try vm.parseJsonAddress(addressJson, ".L2VotingPowerPaused") returns (address l2VotingPowerPaused) { + l2AddressesConfig.L2VotingPowerPaused = l2VotingPowerPaused; + } catch { } return l2AddressesConfig; } @@ -241,6 +248,7 @@ contract Utils is Script { vm.serializeAddress(json, "L2ClaimImplementation", cfg.L2ClaimImplementation); vm.serializeAddress(json, "L2Governor", cfg.L2Governor); vm.serializeAddress(json, "L2GovernorImplementation", cfg.L2GovernorImplementation); + vm.serializeAddress(json, "L2GovernorPaused", cfg.L2GovernorPaused); vm.serializeAddress(json, "L2LiskToken", cfg.L2LiskToken); vm.serializeAddress(json, "L2LockingPosition", cfg.L2LockingPosition); vm.serializeAddress(json, "L2LockingPositionImplementation", cfg.L2LockingPositionImplementation); @@ -251,8 +259,8 @@ contract Utils is Script { vm.serializeAddress(json, "L2TimelockController", cfg.L2TimelockController); vm.serializeAddress(json, "L2VestingWalletImplementation", cfg.L2VestingWalletImplementation); vm.serializeAddress(json, "L2VotingPower", cfg.L2VotingPower); - string memory finalJson = - vm.serializeAddress(json, "L2VotingPowerImplementation", cfg.L2VotingPowerImplementation); + vm.serializeAddress(json, "L2VotingPowerImplementation", cfg.L2VotingPowerImplementation); + string memory finalJson = vm.serializeAddress(json, "L2VotingPowerPaused", cfg.L2VotingPowerPaused); finalJson.write(string.concat("deployment/", network, "/l2addresses.json")); } diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol index 8512d5dc..c7789379 100644 --- a/src/L2/paused/L2GovernorPaused.sol +++ b/src/L2/paused/L2GovernorPaused.sol @@ -9,6 +9,8 @@ import { TimelockControllerUpgradeable } from import { L2Governor } from "../L2Governor.sol"; contract L2GovernorPaused is L2Governor { + error GovernorIsPaused(); + /// @notice Setting global params. function initializePaused() public reinitializer(2) { } @@ -29,22 +31,22 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the castVote function to pause Governor interactions. function castVote(uint256, uint8) public virtual override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the castVoteBySig function to pause Governor interactions. function castVoteBySig(uint256, uint8, address, bytes memory) public virtual override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the castVoteWithReason function to pause Governor interactions. function castVoteWithReason(uint256, uint8, string calldata) public virtual override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the castVoteWithReasonAndParams function to pause Governor interactions. @@ -59,7 +61,7 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the castVoteWithReasonAndParamsBySig function to pause Governor interactions. @@ -76,7 +78,7 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the execute function to pause Governor interactions. @@ -92,7 +94,7 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the propose function to pause Governor interactions. @@ -107,7 +109,7 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the queue function to pause Governor interactions. @@ -122,31 +124,65 @@ contract L2GovernorPaused is L2Governor { override returns (uint256) { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the relay function to pause Governor interactions. function relay(address, uint256, bytes memory) public payable virtual override { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the setProposalThreshold function to pause Governor interactions. function setProposalThreshold(uint256) public virtual override { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the setVotingDelay function to pause Governor interactions. function setVotingDelay(uint48) public virtual override { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the setVotingPeriod function to pause Governor interactions. function setVotingPeriod(uint32) public virtual override { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); } /// @notice Override the updateTimelock function to pause Governor interactions. function updateTimelock(TimelockControllerUpgradeable) public virtual override { - revert("L2GovernorPaused: Governor is paused"); + revert GovernorIsPaused(); + } + + function onERC1155BatchReceived( + address, + address, + uint256[] memory, + uint256[] memory, + bytes memory + ) + public + virtual + override + returns (bytes4) + { + revert GovernorIsPaused(); + } + + function onERC1155Received( + address, + address, + uint256, + uint256, + bytes memory + ) + public + virtual + override + returns (bytes4) + { + revert GovernorIsPaused(); + } + + function onERC721Received(address, address, uint256, bytes memory) public virtual override returns (bytes4) { + revert GovernorIsPaused(); } } diff --git a/src/L2/paused/L2VotingPowerPaused.sol b/src/L2/paused/L2VotingPowerPaused.sol index 5e0e8cf7..249e1a89 100644 --- a/src/L2/paused/L2VotingPowerPaused.sol +++ b/src/L2/paused/L2VotingPowerPaused.sol @@ -4,35 +4,23 @@ pragma solidity 0.8.23; import { Ownable2StepUpgradeable } from "@openzeppelin-upgradeable/contracts/access/Ownable2StepUpgradeable.sol"; import { Initializable } from "@openzeppelin-upgradeable/contracts/proxy/utils/Initializable.sol"; import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol"; -import { IL2LockingPosition } from "../../interfaces/L2/IL2LockingPosition.sol"; import { L2VotingPower } from "../L2VotingPower.sol"; contract L2VotingPowerPaused is L2VotingPower { + error VotingPowerIsPaused(); + /// @notice Setting global params. function initializePaused() public reinitializer(2) { version = "1.0.0-paused"; } - /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. - function adjustVotingPower( - address, - IL2LockingPosition.LockingPosition memory, - IL2LockingPosition.LockingPosition memory - ) - public - virtual - override - { - revert("L2VotingPowerPaused: VotingPower is paused"); - } - /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. function delegate(address) public virtual override { - revert("L2VotingPowerPaused: VotingPower is paused"); + revert VotingPowerIsPaused(); } /// @notice Override the modifyLockingPosition function to pause VotingPower interactions. function delegateBySig(address, uint256, uint256, uint8, bytes32, bytes32) public virtual override { - revert("L2VotingPowerPaused: VotingPower is paused"); + revert VotingPowerIsPaused(); } } diff --git a/test/L2/paused/L2GovernorPaused.t.sol b/test/L2/paused/L2GovernorPaused.t.sol index 932e6a4d..a7cdcb8e 100644 --- a/test/L2/paused/L2GovernorPaused.t.sol +++ b/test/L2/paused/L2GovernorPaused.t.sol @@ -67,79 +67,82 @@ contract L2GovernorPausedTest is Test { // assure that address(0) is in executors role assertEq(timelock.hasRole(timelock.EXECUTOR_ROLE(), address(0)), true); - // Upgrade from L2Governor to L2GovernorPaused + // Upgrade from L2Governor to L2GovernorPaused, and call initializePaused L2GovernorPaused l2GovernorPausedImplementation = new L2GovernorPaused(); - l2Governor.upgradeToAndCall(address(l2GovernorPausedImplementation), ""); + l2Governor.upgradeToAndCall( + address(l2GovernorPausedImplementation), + abi.encodeWithSelector(l2GovernorPausedImplementation.initializePaused.selector) + ); assertEq(l2Governor.version(), "1.0.0-paused"); } function test_Cancel_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.cancel(new address[](1), new uint256[](1), new bytes[](1), 0); } function test_CastVote_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.castVote(0, 0); } function test_CastVoteBySig_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.castVoteBySig(0, 0, address(0), ""); } function test_CastVoteWithReason_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.castVoteWithReason(0, 0, ""); } function test_CastVoteWithReasonAndParams_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.castVoteWithReasonAndParams(0, 0, "", ""); } function test_CastVoteWithReasonAndParamsBySig_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.castVoteWithReasonAndParamsBySig(0, 0, address(0), "", "", ""); } function test_Execute_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.execute(new address[](1), new uint256[](1), new bytes[](1), 0); } function test_Propose_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.propose(new address[](1), new uint256[](1), new bytes[](1), ""); } function test_Queue_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.queue(new address[](1), new uint256[](1), new bytes[](1), ""); } function test_Relay_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.relay(address(0), 0, ""); } function test_SetProposalThreshold_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.setProposalThreshold(0); } function test_SetVotingDelay_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.setVotingDelay(0); } function test_SetVotingPeriod_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.setVotingPeriod(0); } function test_UpdateTimelock_Paused() public { - vm.expectRevert("L2GovernorPaused: Governor is paused"); + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.updateTimelock(TimelockControllerUpgradeable(payable(address(0)))); } diff --git a/test/L2/paused/L2VotingPowerPaused.t.sol b/test/L2/paused/L2VotingPowerPaused.t.sol index 45fb31fb..b28e4d10 100644 --- a/test/L2/paused/L2VotingPowerPaused.t.sol +++ b/test/L2/paused/L2VotingPowerPaused.t.sol @@ -8,7 +8,6 @@ import { IAccessControl } from "@openzeppelin/contracts/access/IAccessControl.so import { IERC20Errors } from "@openzeppelin/contracts/interfaces/draft-IERC6093.sol"; import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { Test, console } from "forge-std/Test.sol"; -import { IL2LockingPosition } from "src/interfaces/L2/IL2LockingPosition.sol"; import { L2VotingPower } from "src/L2/L2VotingPower.sol"; import { L2VotingPowerPaused } from "src/L2/paused/L2VotingPowerPaused.sol"; import { Utils } from "script/contracts/Utils.sol"; @@ -59,34 +58,24 @@ contract L2VotingPowerPausedTest is Test { assertEq(l2VotingPower.name(), "Lisk Voting Power"); assertEq(l2VotingPower.symbol(), "vpLSK"); - // Upgrade from L2VotingPower to L2VotingPowerPaused + // Upgrade from L2VotingPower to L2VotingPowerPaused, and call initializePaused L2VotingPowerPaused l2VotingPowerPausedImplementation = new L2VotingPowerPaused(); - l2VotingPower.upgradeToAndCall(address(l2VotingPowerPausedImplementation), ""); - - // Call initializePaused with new interface - L2VotingPowerPaused(address(l2VotingPower)).initializePaused(); + l2VotingPower.upgradeToAndCall( + address(l2VotingPowerPausedImplementation), + abi.encodeWithSelector(l2VotingPowerPausedImplementation.initializePaused.selector) + ); // l2VotingPower pointing to paused contract assertEq(l2VotingPower.version(), "1.0.0-paused"); } - function test_adjustVotingPower_Paused() public { - IL2LockingPosition.LockingPosition memory positionBefore = - IL2LockingPosition.LockingPosition(address(this), 0, 0, 0); - IL2LockingPosition.LockingPosition memory positionAfter = - IL2LockingPosition.LockingPosition(address(this), 50, 0, 0); - - vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); - l2VotingPower.adjustVotingPower(address(0), positionBefore, positionAfter); - } - function test_delegate_Paused() public { - vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); + vm.expectRevert(L2VotingPowerPaused.VotingPowerIsPaused.selector); l2VotingPower.delegate(address(0)); } function test_delegateBySig_Paused() public { - vm.expectRevert("L2VotingPowerPaused: VotingPower is paused"); + vm.expectRevert(L2VotingPowerPaused.VotingPowerIsPaused.selector); l2VotingPower.delegateBySig(address(0), 0, 0, 0, "", ""); } From 84b1c26aedbc3b1ddba935a5d36d6ddd1cb4df15 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 11:56:29 +0200 Subject: [PATCH 08/16] Rearrange functions according to requirements --- src/L2/paused/L2GovernorPaused.sol | 71 ++++++++++++------------ test/L2/paused/L2GovernorPaused.t.sol | 26 ++++++--- test/L2/paused/L2VotingPowerPaused.t.sol | 13 ++++- 3 files changed, 65 insertions(+), 45 deletions(-) diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol index c7789379..0a2dc16f 100644 --- a/src/L2/paused/L2GovernorPaused.sol +++ b/src/L2/paused/L2GovernorPaused.sol @@ -97,6 +97,43 @@ contract L2GovernorPaused is L2Governor { revert GovernorIsPaused(); } + /// @notice Override the onERC1155BatchReceived function to pause Governor interactions. + function onERC1155BatchReceived( + address, + address, + uint256[] memory, + uint256[] memory, + bytes memory + ) + public + virtual + override + returns (bytes4) + { + revert GovernorIsPaused(); + } + + /// @notice Override the onERC1155Received function to pause Governor interactions. + function onERC1155Received( + address, + address, + uint256, + uint256, + bytes memory + ) + public + virtual + override + returns (bytes4) + { + revert GovernorIsPaused(); + } + + /// @notice Override the onERC721Received function to pause Governor interactions. + function onERC721Received(address, address, uint256, bytes memory) public virtual override returns (bytes4) { + revert GovernorIsPaused(); + } + /// @notice Override the propose function to pause Governor interactions. function propose( address[] memory, @@ -151,38 +188,4 @@ contract L2GovernorPaused is L2Governor { function updateTimelock(TimelockControllerUpgradeable) public virtual override { revert GovernorIsPaused(); } - - function onERC1155BatchReceived( - address, - address, - uint256[] memory, - uint256[] memory, - bytes memory - ) - public - virtual - override - returns (bytes4) - { - revert GovernorIsPaused(); - } - - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes memory - ) - public - virtual - override - returns (bytes4) - { - revert GovernorIsPaused(); - } - - function onERC721Received(address, address, uint256, bytes memory) public virtual override returns (bytes4) { - revert GovernorIsPaused(); - } } diff --git a/test/L2/paused/L2GovernorPaused.t.sol b/test/L2/paused/L2GovernorPaused.t.sol index a7cdcb8e..cf8bbd4d 100644 --- a/test/L2/paused/L2GovernorPaused.t.sol +++ b/test/L2/paused/L2GovernorPaused.t.sol @@ -28,6 +28,17 @@ contract L2GovernorPausedTest is Test { TimelockController timelock; address initialOwner; + function assetInitParamsEq() internal { + assertEq(l2Governor.name(), "Lisk Governor"); + assertEq(l2Governor.votingDelay(), 0); + assertEq(l2Governor.votingPeriod(), 604800); + assertEq(l2Governor.proposalThreshold(), 300_000 * 10 ** 18); + assertEq(l2Governor.timelock(), address(timelock)); + assertEq(l2Governor.quorum(0), 24_000_000 * 10 ** 18); + assertEq(address(l2Governor.token()), address(votingPower)); + assertEq(l2Governor.owner(), initialOwner); + } + function setUp() public { utils = new Utils(); @@ -54,15 +65,8 @@ contract L2GovernorPausedTest is Test { ) ); - assertEq(l2Governor.name(), "Lisk Governor"); + assetInitParamsEq(); assertEq(l2Governor.version(), "1.0.0"); - assertEq(l2Governor.votingDelay(), 0); - assertEq(l2Governor.votingPeriod(), 604800); - assertEq(l2Governor.proposalThreshold(), 300_000 * 10 ** 18); - assertEq(l2Governor.timelock(), address(timelock)); - assertEq(l2Governor.quorum(0), 24_000_000 * 10 ** 18); - assertEq(address(l2Governor.token()), address(votingPower)); - assertEq(l2Governor.owner(), initialOwner); // assure that address(0) is in executors role assertEq(timelock.hasRole(timelock.EXECUTOR_ROLE(), address(0)), true); @@ -74,6 +78,9 @@ contract L2GovernorPausedTest is Test { abi.encodeWithSelector(l2GovernorPausedImplementation.initializePaused.selector) ); assertEq(l2Governor.version(), "1.0.0-paused"); + + // Ensure all other params are unchanged after paused contract update + assetInitParamsEq(); } function test_Cancel_Paused() public { @@ -154,5 +161,8 @@ contract L2GovernorPausedTest is Test { // new version updated assertEq(l2Governor.version(), "2.0.0"); + + // Ensure all other params are unchanged after non-paused contract update + assetInitParamsEq(); } } diff --git a/test/L2/paused/L2VotingPowerPaused.t.sol b/test/L2/paused/L2VotingPowerPaused.t.sol index b28e4d10..3bf8eff4 100644 --- a/test/L2/paused/L2VotingPowerPaused.t.sol +++ b/test/L2/paused/L2VotingPowerPaused.t.sol @@ -32,6 +32,12 @@ contract L2VotingPowerPausedTest is Test { address lockingPositionContractAddress; + function assetInitParamsEq() internal { + assertEq(l2VotingPower.lockingPositionAddress(), lockingPositionContractAddress); + assertEq(l2VotingPower.name(), "Lisk Voting Power"); + assertEq(l2VotingPower.symbol(), "vpLSK"); + } + function setUp() public { utils = new Utils(); @@ -53,10 +59,8 @@ contract L2VotingPowerPausedTest is Test { ) ); - assertEq(l2VotingPower.lockingPositionAddress(), lockingPositionContractAddress); + assetInitParamsEq(); assertEq(l2VotingPower.version(), "1.0.0"); - assertEq(l2VotingPower.name(), "Lisk Voting Power"); - assertEq(l2VotingPower.symbol(), "vpLSK"); // Upgrade from L2VotingPower to L2VotingPowerPaused, and call initializePaused L2VotingPowerPaused l2VotingPowerPausedImplementation = new L2VotingPowerPaused(); @@ -67,6 +71,9 @@ contract L2VotingPowerPausedTest is Test { // l2VotingPower pointing to paused contract assertEq(l2VotingPower.version(), "1.0.0-paused"); + + // Ensure all other params are unchanged after paused contract update + assetInitParamsEq(); } function test_delegate_Paused() public { From afefffd06f06aa835a67213325d9bf73c622dc2d Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 12:03:36 +0200 Subject: [PATCH 09/16] pump forge-std to 1.8.2 --- lib/forge-std | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/forge-std b/lib/forge-std index bb4ceea9..978ac6fa 160000 --- a/lib/forge-std +++ b/lib/forge-std @@ -1 +1 @@ -Subproject commit bb4ceea94d6f10eeb5b41dc2391c6c8bf8e734ef +Subproject commit 978ac6fadb62f5f0b723c996f64be52eddba6801 From 2076e38f4ba360f00ced19bb3bf3aa96c690ff13 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 12:28:05 +0200 Subject: [PATCH 10/16] Add onReceive ERC721 and ERC1155 check --- src/L2/paused/L2GovernorPaused.sol | 16 +++++------ test/L2/paused/L2GovernorPaused.t.sol | 39 ++++++++++++++++++++++++++- test/mock/MockERC1155.sol | 15 +++++++++++ test/mock/MockERC721.sol | 15 +++++++++++ 4 files changed, 76 insertions(+), 9 deletions(-) create mode 100644 test/mock/MockERC1155.sol create mode 100644 test/mock/MockERC721.sol diff --git a/src/L2/paused/L2GovernorPaused.sol b/src/L2/paused/L2GovernorPaused.sol index 0a2dc16f..b617a922 100644 --- a/src/L2/paused/L2GovernorPaused.sol +++ b/src/L2/paused/L2GovernorPaused.sol @@ -105,10 +105,10 @@ contract L2GovernorPaused is L2Governor { uint256[] memory, bytes memory ) - public - virtual - override - returns (bytes4) + public + virtual + override + returns (bytes4) { revert GovernorIsPaused(); } @@ -121,10 +121,10 @@ contract L2GovernorPaused is L2Governor { uint256, bytes memory ) - public - virtual - override - returns (bytes4) + public + virtual + override + returns (bytes4) { revert GovernorIsPaused(); } diff --git a/test/L2/paused/L2GovernorPaused.t.sol b/test/L2/paused/L2GovernorPaused.t.sol index cf8bbd4d..7ea36897 100644 --- a/test/L2/paused/L2GovernorPaused.t.sol +++ b/test/L2/paused/L2GovernorPaused.t.sol @@ -7,10 +7,14 @@ import { TimelockController } from "@openzeppelin/contracts/governance/TimelockC import { ERC1967Proxy } from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; import { TimelockControllerUpgradeable } from "@openzeppelin-upgradeable/contracts/governance/extensions/GovernorTimelockControlUpgradeable.sol"; +import { ERC1155Holder } from "@openzeppelin/contracts/token/ERC1155/utils/ERC1155Holder.sol"; +import { ERC721Holder } from "@openzeppelin/contracts/token/ERC721/utils/ERC721Holder.sol"; import { Test, console } from "forge-std/Test.sol"; import { L2Governor } from "src/L2/L2Governor.sol"; import { L2GovernorPaused } from "src/L2/paused/L2GovernorPaused.sol"; import { Utils } from "script/contracts/Utils.sol"; +import { MockERC721 } from "test/mock/MockERC721.sol"; +import { MockERC1155 } from "test/mock/MockERC1155.sol"; contract MockL2GovernorV2 is L2Governor { function version() public pure virtual override returns (string memory) { @@ -18,7 +22,7 @@ contract MockL2GovernorV2 is L2Governor { } } -contract L2GovernorPausedTest is Test { +contract L2GovernorPausedTest is Test, ERC1155Holder, ERC721Holder { Utils public utils; L2Governor public l2GovernorImplementation; L2Governor public l2Governor; @@ -118,6 +122,39 @@ contract L2GovernorPausedTest is Test { l2Governor.execute(new address[](1), new uint256[](1), new bytes[](1), 0); } + function test_OnERC1155BatchReceived_Paused() public { + MockERC1155 mockERC1155 = new MockERC1155(); + mockERC1155.mint(address(this), 0, 10, ""); + mockERC1155.mint(address(this), 1, 20, ""); + + uint256[] memory ids = new uint256[](2); + ids[0] = 0; + ids[1] = 1; + + uint256[] memory values = new uint256[](2); + values[0] = 10; + values[1] = 20; + + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); + mockERC1155.safeBatchTransferFrom(address(this), address(l2Governor), ids, values, ""); + } + + function test_OnERC1155Received_Paused() public { + MockERC1155 mockERC1155 = new MockERC1155(); + mockERC1155.mint(address(this), 0, 10, ""); + + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); + mockERC1155.safeTransferFrom(address(this), address(l2Governor), 0, 10, ""); + } + + function test_OnERC721Received_Paused() public { + MockERC721 mockERC721 = new MockERC721(); + mockERC721.mint(address(this), 0); + + vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); + mockERC721.safeTransferFrom(address(this), address(l2Governor), 0); + } + function test_Propose_Paused() public { vm.expectRevert(L2GovernorPaused.GovernorIsPaused.selector); l2Governor.propose(new address[](1), new uint256[](1), new bytes[](1), ""); diff --git a/test/mock/MockERC1155.sol b/test/mock/MockERC1155.sol new file mode 100644 index 00000000..897371bb --- /dev/null +++ b/test/mock/MockERC1155.sol @@ -0,0 +1,15 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { ERC1155 } from "@openzeppelin/contracts/token/ERC1155/ERC1155.sol"; + +/// @title MockERC1155 +/// @notice MockERC1155 is a mock implementation of ERC1155 token. +/// IT SHOULD NEVER BE USED IN PRODUCTION. +contract MockERC1155 is ERC1155 { + constructor() ERC1155("") { } + + function mint(address _to, uint256 _id, uint256 _value, bytes memory _data) public { + _mint(_to, _id, _value, _data); + } +} diff --git a/test/mock/MockERC721.sol b/test/mock/MockERC721.sol new file mode 100644 index 00000000..18ea7af8 --- /dev/null +++ b/test/mock/MockERC721.sol @@ -0,0 +1,15 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity 0.8.23; + +import { ERC721 } from "@openzeppelin/contracts/token/ERC721/ERC721.sol"; + +/// @title MockERC721 +/// @notice MockERC721 is a mock implementation of ERC721 token. +/// IT SHOULD NEVER BE USED IN PRODUCTION. +contract MockERC721 is ERC721 { + constructor() ERC721("MockERC721", "ERC721") { } + + function mint(address _to, uint256 _id) public { + _mint(_to, _id); + } +} From 935067d2af1cddcbb5d97d9d54c68dde18ac1300 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 12:34:22 +0200 Subject: [PATCH 11/16] Test adjustVotingPower is unpaused --- test/L2/paused/L2VotingPowerPaused.t.sol | 19 +++++++++++++++++-- 1 file changed, 17 insertions(+), 2 deletions(-) diff --git a/test/L2/paused/L2VotingPowerPaused.t.sol b/test/L2/paused/L2VotingPowerPaused.t.sol index 3bf8eff4..2dc83645 100644 --- a/test/L2/paused/L2VotingPowerPaused.t.sol +++ b/test/L2/paused/L2VotingPowerPaused.t.sol @@ -10,6 +10,7 @@ import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import { Test, console } from "forge-std/Test.sol"; import { L2VotingPower } from "src/L2/L2VotingPower.sol"; import { L2VotingPowerPaused } from "src/L2/paused/L2VotingPowerPaused.sol"; +import { IL2LockingPosition } from "src/interfaces/L2/IL2LockingPosition.sol"; import { Utils } from "script/contracts/Utils.sol"; contract MockL2VotingPowerV2 is L2VotingPower { @@ -76,12 +77,26 @@ contract L2VotingPowerPausedTest is Test { assetInitParamsEq(); } - function test_delegate_Paused() public { + function test_AdjustVotingPower_NotPaused() public { + IL2LockingPosition.LockingPosition memory positionBefore = + IL2LockingPosition.LockingPosition(address(this), 50, 0, 0); + IL2LockingPosition.LockingPosition memory positionAfter = + IL2LockingPosition.LockingPosition(address(this), 100, 0, 0); + + // call it as LockingPosition contract + vm.prank(lockingPositionContractAddress); + + // Throws the require check on the first line, proving this function is not paused + vm.expectRevert("L2VotingPower: owner address cannot be 0"); + l2VotingPower.adjustVotingPower(address(0), positionBefore, positionAfter); + } + + function test_Delegate_Paused() public { vm.expectRevert(L2VotingPowerPaused.VotingPowerIsPaused.selector); l2VotingPower.delegate(address(0)); } - function test_delegateBySig_Paused() public { + function test_DelegateBySig_Paused() public { vm.expectRevert(L2VotingPowerPaused.VotingPowerIsPaused.selector); l2VotingPower.delegateBySig(address(0), 0, 0, 0, "", ""); } From f31590d3b60b26da28c1b7e876d1819248bdfdcf Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 13:39:56 +0200 Subject: [PATCH 12/16] Add deployPausedDAO.sh --- script/paused/deployPausedDAO.sh | 60 ++++++++++++++++++++++++++++++++ 1 file changed, 60 insertions(+) create mode 100755 script/paused/deployPausedDAO.sh diff --git a/script/paused/deployPausedDAO.sh b/script/paused/deployPausedDAO.sh new file mode 100755 index 00000000..06ef853c --- /dev/null +++ b/script/paused/deployPausedDAO.sh @@ -0,0 +1,60 @@ +#!/usr/bin/env bash + +echo "Instructing the shell to exit immediately if any command returns a non-zero exit status..." +set -e +echo "Done." + +echo "Navigating to the root directory of the project..." +cd ../ +echo "Done." + +echo "Setting environment variables..." +source .env +echo "Done." + +echo "Creating $NETWORK directory inside deployment directory..." +if [ -z "$NETWORK" ] +then + echo "NETWORK variable inside .env file is not set. Please set NETWORK environment variable." + exit 1 +else + if [ -d "deployment/$NETWORK" ] + then + echo "Directory deployment/$NETWORK already exists." + else + mkdir deployment/$NETWORK + fi +fi +echo "Done." + +echo "Deploying and if enabled verifying L2GovernorPaused smart contract..." +if [ -z "$CONTRACT_VERIFIER" ] +then + forge script --rpc-url="$L2_RPC_URL" --broadcast -vvvv script/contracts/L2/paused/L2GovernorPaused.s.sol:L2GovernorPausedScript +else + if [ $CONTRACT_VERIFIER = "blockscout" ] + then + forge script --rpc-url="$L2_RPC_URL" --broadcast --verify --verifier blockscout --verifier-url $L2_VERIFIER_URL -vvvv script/contracts/L2/paused/L2GovernorPaused.s.sol:L2GovernorPausedScript + fi + if [ $CONTRACT_VERIFIER = "etherscan" ] + then + forge script --rpc-url="$L2_RPC_URL" --broadcast --verify --verifier etherscan --etherscan-api-key="$L2_ETHERSCAN_API_KEY" -vvvv script/contracts/L2/paused/L2GovernorPaused.s.sol:L2GovernorPausedScript + fi +fi +echo "Done." + +echo "Deploying and if enabled verifying L2VotingPowerPaused smart contract..." +if [ -z "$CONTRACT_VERIFIER" ] +then + forge script --rpc-url="$L2_RPC_URL" --broadcast -vvvv script/contracts/L2/paused/L2VotingPowerPaused.s.sol:L2VotingPowerPausedScript +else + if [ $CONTRACT_VERIFIER = "blockscout" ] + then + forge script --rpc-url="$L2_RPC_URL" --broadcast --verify --verifier blockscout --verifier-url $L2_VERIFIER_URL -vvvv script/contracts/L2/paused/L2VotingPowerPaused.s.sol:L2VotingPowerPausedScript + fi + if [ $CONTRACT_VERIFIER = "etherscan" ] + then + forge script --rpc-url="$L2_RPC_URL" --broadcast --verify --verifier etherscan --etherscan-api-key="$L2_ETHERSCAN_API_KEY" -vvvv script/contracts/L2/paused/L2VotingPowerPaused.s.sol:L2VotingPowerPausedScript + fi +fi +echo "Done." \ No newline at end of file From f35342f6b1b519710f4d393162b7ab81b3bf8174 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 13:46:14 +0200 Subject: [PATCH 13/16] Add forge clean to deploy script --- script/paused/deployPausedDAO.sh | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/script/paused/deployPausedDAO.sh b/script/paused/deployPausedDAO.sh index 06ef853c..b4165080 100755 --- a/script/paused/deployPausedDAO.sh +++ b/script/paused/deployPausedDAO.sh @@ -12,6 +12,10 @@ echo "Setting environment variables..." source .env echo "Done." +echo "Cleaning up the build artifacts to be able to deploy the contract..." +forge clean +echo "Done." + echo "Creating $NETWORK directory inside deployment directory..." if [ -z "$NETWORK" ] then From 2a5d9b8bd51cbc9865a09f55f49b6294e9f31535 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 13:49:18 +0200 Subject: [PATCH 14/16] Clean before every forge script runs --- script/paused/deployPausedDAO.sh | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/script/paused/deployPausedDAO.sh b/script/paused/deployPausedDAO.sh index b4165080..5b63d67e 100755 --- a/script/paused/deployPausedDAO.sh +++ b/script/paused/deployPausedDAO.sh @@ -12,10 +12,6 @@ echo "Setting environment variables..." source .env echo "Done." -echo "Cleaning up the build artifacts to be able to deploy the contract..." -forge clean -echo "Done." - echo "Creating $NETWORK directory inside deployment directory..." if [ -z "$NETWORK" ] then @@ -31,6 +27,10 @@ else fi echo "Done." +echo "Cleaning up the build artifacts to be able to deploy the contract..." +forge clean +echo "Done." + echo "Deploying and if enabled verifying L2GovernorPaused smart contract..." if [ -z "$CONTRACT_VERIFIER" ] then @@ -47,6 +47,10 @@ else fi echo "Done." +echo "Cleaning up the build artifacts to be able to deploy the contract..." +forge clean +echo "Done." + echo "Deploying and if enabled verifying L2VotingPowerPaused smart contract..." if [ -z "$CONTRACT_VERIFIER" ] then From e67d16a195cded411d3d4bdf7fa9973238d45410 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 13:53:57 +0200 Subject: [PATCH 15/16] Used `../../` to navigate paused script --- script/paused/deployPausedDAO.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/script/paused/deployPausedDAO.sh b/script/paused/deployPausedDAO.sh index 5b63d67e..513c31a8 100755 --- a/script/paused/deployPausedDAO.sh +++ b/script/paused/deployPausedDAO.sh @@ -5,7 +5,7 @@ set -e echo "Done." echo "Navigating to the root directory of the project..." -cd ../ +cd ../../ echo "Done." echo "Setting environment variables..." From f2a6a9957943e7cd40fd1c212dbed00b0d4a51e3 Mon Sep 17 00:00:00 2001 From: Franco NG Date: Wed, 12 Jun 2024 14:36:22 +0200 Subject: [PATCH 16/16] rename assertInitParamsEq --- test/L2/paused/L2GovernorPaused.t.sol | 8 ++++---- test/L2/paused/L2VotingPowerPaused.t.sol | 6 +++--- 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/test/L2/paused/L2GovernorPaused.t.sol b/test/L2/paused/L2GovernorPaused.t.sol index 7ea36897..41cd0720 100644 --- a/test/L2/paused/L2GovernorPaused.t.sol +++ b/test/L2/paused/L2GovernorPaused.t.sol @@ -32,7 +32,7 @@ contract L2GovernorPausedTest is Test, ERC1155Holder, ERC721Holder { TimelockController timelock; address initialOwner; - function assetInitParamsEq() internal { + function assertInitParamsEq() internal { assertEq(l2Governor.name(), "Lisk Governor"); assertEq(l2Governor.votingDelay(), 0); assertEq(l2Governor.votingPeriod(), 604800); @@ -69,7 +69,7 @@ contract L2GovernorPausedTest is Test, ERC1155Holder, ERC721Holder { ) ); - assetInitParamsEq(); + assertInitParamsEq(); assertEq(l2Governor.version(), "1.0.0"); // assure that address(0) is in executors role @@ -84,7 +84,7 @@ contract L2GovernorPausedTest is Test, ERC1155Holder, ERC721Holder { assertEq(l2Governor.version(), "1.0.0-paused"); // Ensure all other params are unchanged after paused contract update - assetInitParamsEq(); + assertInitParamsEq(); } function test_Cancel_Paused() public { @@ -200,6 +200,6 @@ contract L2GovernorPausedTest is Test, ERC1155Holder, ERC721Holder { assertEq(l2Governor.version(), "2.0.0"); // Ensure all other params are unchanged after non-paused contract update - assetInitParamsEq(); + assertInitParamsEq(); } } diff --git a/test/L2/paused/L2VotingPowerPaused.t.sol b/test/L2/paused/L2VotingPowerPaused.t.sol index 2dc83645..0fdccb88 100644 --- a/test/L2/paused/L2VotingPowerPaused.t.sol +++ b/test/L2/paused/L2VotingPowerPaused.t.sol @@ -33,7 +33,7 @@ contract L2VotingPowerPausedTest is Test { address lockingPositionContractAddress; - function assetInitParamsEq() internal { + function assertInitParamsEq() internal { assertEq(l2VotingPower.lockingPositionAddress(), lockingPositionContractAddress); assertEq(l2VotingPower.name(), "Lisk Voting Power"); assertEq(l2VotingPower.symbol(), "vpLSK"); @@ -60,7 +60,7 @@ contract L2VotingPowerPausedTest is Test { ) ); - assetInitParamsEq(); + assertInitParamsEq(); assertEq(l2VotingPower.version(), "1.0.0"); // Upgrade from L2VotingPower to L2VotingPowerPaused, and call initializePaused @@ -74,7 +74,7 @@ contract L2VotingPowerPausedTest is Test { assertEq(l2VotingPower.version(), "1.0.0-paused"); // Ensure all other params are unchanged after paused contract update - assetInitParamsEq(); + assertInitParamsEq(); } function test_AdjustVotingPower_NotPaused() public {