Skip to content

Commit

Permalink
Pulled alpha governor tests out into a separate suite (#14)
Browse files Browse the repository at this point in the history
Pulled the alpha governor tests out for readability

Resolves #6
  • Loading branch information
jferas authored Dec 7, 2023
1 parent 3613326 commit 3123686
Show file tree
Hide file tree
Showing 2 changed files with 212 additions and 195 deletions.
195 changes: 0 additions & 195 deletions test/RadworksGovernor.t.sol
Original file line number Diff line number Diff line change
Expand Up @@ -28,201 +28,6 @@ abstract contract Constructor is RadworksGovernorTest {
}

abstract contract Propose is ProposalTest {
function test_GovernorUpgradeProposalIsSubmittedCorrectly() public {
// Proposal has been recorded
assertEq(governorAlpha.proposalCount(), initialProposalCount + 1);

// Proposal is in the expected state
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, PENDING);

// Proposal actions correspond to Governor upgrade
(
address[] memory _targets,
uint256[] memory _values,
string[] memory _signatures,
bytes[] memory _calldatas
) = governorAlpha.getActions(upgradeProposalId);
assertEq(_targets.length, 2);
assertEq(_targets[0], TIMELOCK);
assertEq(_targets[1], address(governorBravo));
assertEq(_values.length, 2);
assertEq(_values[0], 0);
assertEq(_values[1], 0);
assertEq(_signatures.length, 2);
assertEq(_signatures[0], "setPendingAdmin(address)");
assertEq(_signatures[1], "__acceptAdmin()");
assertEq(_calldatas.length, 2);
assertEq(_calldatas[0], abi.encode(address(governorBravo)));
assertEq(_calldatas[1], "");
}

function test_UpgradeProposalActiveAfterDelay() public {
_jumpToActiveUpgradeProposal();

// Ensure proposal has become active the block after the voting delay
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, ACTIVE);
}

function testFuzz_UpgradeProposerCanCastVote(bool _willSupport) public {
_jumpToActiveUpgradeProposal();
uint256 _proposerVotes =
ERC20VotesComp(RAD_TOKEN).getPriorVotes(PROPOSER, _upgradeProposalStartBlock());

vm.prank(PROPOSER);
governorAlpha.castVote(upgradeProposalId, _willSupport);

IGovernorAlpha.Receipt memory _receipt = governorAlpha.getReceipt(upgradeProposalId, PROPOSER);
assertEq(_receipt.hasVoted, true);
assertEq(_receipt.support, _willSupport);
assertEq(_receipt.votes, _proposerVotes);
}

function test_UpgradeProposalSucceedsWhenAllDelegatesVoteFor() public {
_passUpgradeProposal();

// Ensure proposal state is now succeeded
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, SUCCEEDED);
}

function test_UpgradeProposalDefeatedWhenAllDelegatesVoteAgainst() public {
_defeatUpgradeProposal();

// Ensure proposal state is now defeated
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, DEFEATED);
}

function test_UpgradeProposalCanBeQueuedAfterSucceeding() public {
_passUpgradeProposal();
governorAlpha.queue(upgradeProposalId);

// Ensure proposal can be queued after success
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, QUEUED);

(
address[] memory _targets,
uint256[] memory _values,
string[] memory _signatures,
bytes[] memory _calldatas
) = governorAlpha.getActions(upgradeProposalId);

uint256 _eta = block.timestamp + timelock.delay();

for (uint256 _index = 0; _index < _targets.length; _index++) {
// Calculate hash of transaction in Timelock
bytes32 _txHash = keccak256(
abi.encode(_targets[_index], _values[_index], _signatures[_index], _calldatas[_index], _eta)
);

// Ensure transaction is queued in Timelock
bool _isQueued = timelock.queuedTransactions(_txHash);
assertEq(_isQueued, true);
}
}

function test_UpgradeProposalCanBeExecutedAfterDelay() public {
_passAndQueueUpgradeProposal();
_jumpPastProposalEta();

// Execute the proposal
governorAlpha.execute(upgradeProposalId);

// Ensure the proposal is now executed
uint8 _state = governorAlpha.state(upgradeProposalId);
assertEq(_state, EXECUTED);

// Ensure the governorBravo is now the admin of the timelock
assertEq(timelock.admin(), address(governorBravo));
}

//
// Post proposal tests
//
function testFuzz_OldGovernorSendsETHAfterProposalIsDefeated(uint128 _amount, address _receiver)
public
{
_assumeReceiver(_receiver);

// Counter-intuitively, the Governor (not the Timelock) must hold the ETH.
// See the deployed GovernorAlpha, line 281:
// https://etherscan.io/address/0x31c8EAcBFFdD875c74b94b077895Bd78CF1E64A3#code
// The governor transfers ETH to the Timelock in the process of executing
// the proposal. The Timelock then just passes that ETH along.
vm.deal(address(governorAlpha), _amount);

uint256 _receiverETHBalance = _receiver.balance;
uint256 _governorETHBalance = address(governorAlpha).balance;

// Defeat the proposal to upgrade the Governor
_defeatUpgradeProposal();

// Create a new proposal to send the ETH.
address[] memory _targets = new address[](1);
uint256[] memory _values = new uint256[](1);
_targets[0] = _receiver;
_values[0] = _amount;

_queueAndVoteAndExecuteProposalWithAlphaGovernor(
_targets,
_values,
new string[](1), // No signature needed for an ETH send.
new bytes[](1), // No calldata needed for an ETH send.
true // GovernorAlpha is still the Timelock admin.
);

// Ensure the ETH has been transferred to the receiver
assertEq(
address(governorAlpha).balance,
_governorETHBalance - _amount,
"Governor alpha ETH balance is incorrect"
);
assertEq(_receiver.balance, _receiverETHBalance + _amount, "Receiver ETH balance is incorrect");
}

function testFuzz_OldGovernorCanNotSendTokensAfterUpgradeCompletes(
uint256 _amount,
address _receiver,
uint256 _seed
) public {
_assumeReceiver(_receiver);
IERC20 _token = _randomERC20Token(_seed);

uint256 _receiverTokenBalance = _token.balanceOf(_receiver);
uint256 _timelockTokenBalance = _token.balanceOf(TIMELOCK);
// bound by the number of tokens the timelock currently controls
_amount = bound(_amount, 0, _timelockTokenBalance);

// Pass and execute the proposal to upgrade the Governor
_upgradeToBravoGovernor();

// Craft a new proposal to send the token.
address[] memory _targets = new address[](1);
uint256[] memory _values = new uint256[](1);
string[] memory _signatures = new string[](1);
bytes[] memory _calldatas = new bytes[](1);

_targets[0] = address(_token);
_values[0] = 0;
_signatures[0] = "transfer(address,uint256)";
_calldatas[0] = abi.encode(_receiver, _amount);

_queueAndVoteAndExecuteProposalWithAlphaGovernor(
_targets,
_values,
_signatures,
_calldatas,
false // GovernorAlpha is not the Timelock admin anymore.
);

// Ensure no tokens have been transferred from the timelock to the receiver.
assertEq(_token.balanceOf(TIMELOCK), _timelockTokenBalance, "Timelock balance is incorrect");
assertEq(_token.balanceOf(_receiver), _receiverTokenBalance, "Receiver balance is incorrect");
}

//
// Bravo proposal tests
//
Expand Down
Loading

0 comments on commit 3123686

Please sign in to comment.