From 7b6414017020f71807f1438e891a7080e1f7eb6d Mon Sep 17 00:00:00 2001 From: Hadrien Croubois Date: Tue, 16 Nov 2021 15:03:22 +0100 Subject: [PATCH] GovernorExtendedVoting testing --- .../mocks/GovernorExtendedVotingMock.sol | 54 ++++ contracts/mocks/GovernorMock.sol | 2 +- .../mocks/GovernorTimelockCompoundMock.sol | 46 +--- .../mocks/GovernorTimelockControlMock.sol | 46 +--- test/governance/GovernorWorkflow.behavior.js | 2 +- .../extensions/GovernorExtendedVoting.test.js | 230 ++++++++++++++++++ 6 files changed, 294 insertions(+), 86 deletions(-) create mode 100644 contracts/mocks/GovernorExtendedVotingMock.sol create mode 100644 test/governance/extensions/GovernorExtendedVoting.test.js diff --git a/contracts/mocks/GovernorExtendedVotingMock.sol b/contracts/mocks/GovernorExtendedVotingMock.sol new file mode 100644 index 00000000000..2485acb9e8d --- /dev/null +++ b/contracts/mocks/GovernorExtendedVotingMock.sol @@ -0,0 +1,54 @@ +// SPDX-License-Identifier: MIT + +pragma solidity ^0.8.0; + +import "./GovernorMock.sol"; +import "../governance/extensions/GovernorExtendedVoting.sol"; + +contract GovernorExtendedVotingMock is GovernorMock, GovernorExtendedVoting { + uint256 private _quorum; + + constructor( + string memory name_, + ERC20Votes token_, + uint256 votingDelay_, + uint256 votingPeriod_, + uint256 quorum_, + uint64 votingDelayExtention_ + ) + GovernorMock(name_, token_, votingDelay_, votingPeriod_, 0) + GovernorExtendedVoting(votingDelayExtention_) + { + _quorum = quorum_; + } + + function quorum(uint256) public view virtual override returns (uint256) { + return _quorum; + } + + function proposalDeadline(uint256 proposalId) public view virtual override(Governor, GovernorExtendedVoting) returns (uint256) { + return super.proposalDeadline(proposalId); + } + + function proposalThreshold() public view virtual override(Governor, GovernorMock) returns (uint256) { + return super.proposalThreshold(); + } + + function propose( + address[] memory targets, + uint256[] memory values, + bytes[] memory calldatas, + string memory description + ) public virtual override(Governor, GovernorMock) returns (uint256) { + return super.propose(targets, values, calldatas, description); + } + + function _castVote( + uint256 proposalId, + address account, + uint8 support, + string memory reason + ) internal virtual override(Governor, GovernorExtendedVoting) returns (uint256) { + return super._castVote(proposalId, account, support, reason); + } +} diff --git a/contracts/mocks/GovernorMock.sol b/contracts/mocks/GovernorMock.sol index cc96dcd2775..b1701e3acb9 100644 --- a/contracts/mocks/GovernorMock.sol +++ b/contracts/mocks/GovernorMock.sol @@ -45,7 +45,7 @@ contract GovernorMock is return super.getVotes(account, blockNumber); } - function proposalThreshold() public view override(Governor, GovernorSettings) returns (uint256) { + function proposalThreshold() public view virtual override(Governor, GovernorSettings) returns (uint256) { return super.proposalThreshold(); } diff --git a/contracts/mocks/GovernorTimelockCompoundMock.sol b/contracts/mocks/GovernorTimelockCompoundMock.sol index 848f4b409b3..bf3d0a6efb9 100644 --- a/contracts/mocks/GovernorTimelockCompoundMock.sol +++ b/contracts/mocks/GovernorTimelockCompoundMock.sol @@ -2,17 +2,10 @@ pragma solidity ^0.8.0; +import "./GovernorMock.sol"; import "../governance/extensions/GovernorTimelockCompound.sol"; -import "../governance/extensions/GovernorSettings.sol"; -import "../governance/extensions/GovernorCountingSimple.sol"; -import "../governance/extensions/GovernorVotesQuorumFraction.sol"; -contract GovernorTimelockCompoundMock is - GovernorSettings, - GovernorTimelockCompound, - GovernorVotesQuorumFraction, - GovernorCountingSimple -{ +contract GovernorTimelockCompoundMock is GovernorMock, GovernorTimelockCompound { constructor( string memory name_, ERC20Votes token_, @@ -21,11 +14,8 @@ contract GovernorTimelockCompoundMock is ICompoundTimelock timelock_, uint256 quorumNumerator_ ) - Governor(name_) + GovernorMock(name_, token_, votingDelay_, votingPeriod_, quorumNumerator_) GovernorTimelockCompound(timelock_) - GovernorSettings(votingDelay_, votingPeriod_, 0) - GovernorVotes(token_) - GovernorVotesQuorumFraction(quorumNumerator_) {} function supportsInterface(bytes4 interfaceId) @@ -38,24 +28,6 @@ contract GovernorTimelockCompoundMock is return super.supportsInterface(interfaceId); } - function quorum(uint256 blockNumber) - public - view - override(IGovernor, GovernorVotesQuorumFraction) - returns (uint256) - { - return super.quorum(blockNumber); - } - - function cancel( - address[] memory targets, - uint256[] memory values, - bytes[] memory calldatas, - bytes32 salt - ) public returns (uint256 proposalId) { - return _cancel(targets, values, calldatas, salt); - } - /** * Overriding nightmare */ @@ -69,7 +41,7 @@ contract GovernorTimelockCompoundMock is return super.state(proposalId); } - function proposalThreshold() public view override(Governor, GovernorSettings) returns (uint256) { + function proposalThreshold() public view override(Governor, GovernorMock) returns (uint256) { return super.proposalThreshold(); } @@ -92,16 +64,6 @@ contract GovernorTimelockCompoundMock is return super._cancel(targets, values, calldatas, salt); } - function getVotes(address account, uint256 blockNumber) - public - view - virtual - override(IGovernor, GovernorVotes) - returns (uint256) - { - return super.getVotes(account, blockNumber); - } - function _executor() internal view virtual override(Governor, GovernorTimelockCompound) returns (address) { return super._executor(); } diff --git a/contracts/mocks/GovernorTimelockControlMock.sol b/contracts/mocks/GovernorTimelockControlMock.sol index 4d9e97fd522..bff09a2d12b 100644 --- a/contracts/mocks/GovernorTimelockControlMock.sol +++ b/contracts/mocks/GovernorTimelockControlMock.sol @@ -2,17 +2,10 @@ pragma solidity ^0.8.0; +import "./GovernorMock.sol"; import "../governance/extensions/GovernorTimelockControl.sol"; -import "../governance/extensions/GovernorSettings.sol"; -import "../governance/extensions/GovernorCountingSimple.sol"; -import "../governance/extensions/GovernorVotesQuorumFraction.sol"; -contract GovernorTimelockControlMock is - GovernorSettings, - GovernorTimelockControl, - GovernorVotesQuorumFraction, - GovernorCountingSimple -{ +contract GovernorTimelockControlMock is GovernorMock, GovernorTimelockControl { constructor( string memory name_, ERC20Votes token_, @@ -21,11 +14,8 @@ contract GovernorTimelockControlMock is TimelockController timelock_, uint256 quorumNumerator_ ) - Governor(name_) + GovernorMock(name_, token_, votingDelay_, votingPeriod_, quorumNumerator_) GovernorTimelockControl(timelock_) - GovernorSettings(votingDelay_, votingPeriod_, 0) - GovernorVotes(token_) - GovernorVotesQuorumFraction(quorumNumerator_) {} function supportsInterface(bytes4 interfaceId) @@ -38,24 +28,6 @@ contract GovernorTimelockControlMock is return super.supportsInterface(interfaceId); } - function quorum(uint256 blockNumber) - public - view - override(IGovernor, GovernorVotesQuorumFraction) - returns (uint256) - { - return super.quorum(blockNumber); - } - - function cancel( - address[] memory targets, - uint256[] memory values, - bytes[] memory calldatas, - bytes32 descriptionHash - ) public returns (uint256 proposalId) { - return _cancel(targets, values, calldatas, descriptionHash); - } - /** * Overriding nightmare */ @@ -69,7 +41,7 @@ contract GovernorTimelockControlMock is return super.state(proposalId); } - function proposalThreshold() public view override(Governor, GovernorSettings) returns (uint256) { + function proposalThreshold() public view override(Governor, GovernorMock) returns (uint256) { return super.proposalThreshold(); } @@ -92,16 +64,6 @@ contract GovernorTimelockControlMock is return super._cancel(targets, values, calldatas, descriptionHash); } - function getVotes(address account, uint256 blockNumber) - public - view - virtual - override(IGovernor, GovernorVotes) - returns (uint256) - { - return super.getVotes(account, blockNumber); - } - function _executor() internal view virtual override(Governor, GovernorTimelockControl) returns (address) { return super._executor(); } diff --git a/test/governance/GovernorWorkflow.behavior.js b/test/governance/GovernorWorkflow.behavior.js index 70319cd44d3..3256792cca9 100644 --- a/test/governance/GovernorWorkflow.behavior.js +++ b/test/governance/GovernorWorkflow.behavior.js @@ -65,7 +65,7 @@ function runGovernorWorkflow () { // vote if (tryGet(this.settings, 'voters')) { this.receipts.castVote = []; - for (const voter of this.settings.voters) { + for (const voter of this.settings.voters.filter(({ support }) => !!support)) { if (!voter.signature) { this.receipts.castVote.push( await getReceiptOrRevert( diff --git a/test/governance/extensions/GovernorExtendedVoting.test.js b/test/governance/extensions/GovernorExtendedVoting.test.js new file mode 100644 index 00000000000..2298105a94a --- /dev/null +++ b/test/governance/extensions/GovernorExtendedVoting.test.js @@ -0,0 +1,230 @@ +const { BN, expectEvent, expectRevert, time } = require('@openzeppelin/test-helpers'); +const Enums = require('../../helpers/enums'); + +const { + runGovernorWorkflow, +} = require('../GovernorWorkflow.behavior'); + +const Token = artifacts.require('ERC20VotesCompMock'); +const Governor = artifacts.require('GovernorExtendedVotingMock'); +const CallReceiver = artifacts.require('CallReceiverMock'); + +contract('GovernorExtendedVoting', function (accounts) { + const [ owner, proposer, voter1, voter2, voter3, voter4 ] = accounts; + + const name = 'OZ-Governor'; + // const version = '1'; + const tokenName = 'MockToken'; + const tokenSymbol = 'MTKN'; + const tokenSupply = web3.utils.toWei('100'); + const votingDelay = new BN(4); + const votingPeriod = new BN(16); + const votingDelayExtention = new BN(8); + const quorum = web3.utils.toWei('1'); + + beforeEach(async function () { + this.owner = owner; + this.token = await Token.new(tokenName, tokenSymbol); + this.mock = await Governor.new(name, this.token.address, votingDelay, votingPeriod, quorum, votingDelayExtention); + this.receiver = await CallReceiver.new(); + await this.token.mint(owner, tokenSupply); + await this.token.delegate(voter1, { from: voter1 }); + await this.token.delegate(voter2, { from: voter2 }); + await this.token.delegate(voter3, { from: voter3 }); + await this.token.delegate(voter4, { from: voter4 }); + }); + + it('deployment check', async function () { + expect(await this.mock.name()).to.be.equal(name); + expect(await this.mock.token()).to.be.equal(this.token.address); + expect(await this.mock.votingDelay()).to.be.bignumber.equal(votingDelay); + expect(await this.mock.votingPeriod()).to.be.bignumber.equal(votingPeriod); + expect(await this.mock.quorum(0)).to.be.bignumber.equal(quorum); + expect(await this.mock.votingDelayExtention()).to.be.bignumber.equal(votingDelayExtention); + }); + + describe('nominal is unaffected', function () { + beforeEach(async function () { + this.settings = { + proposal: [ + [ this.receiver.address ], + [ 0 ], + [ this.receiver.contract.methods.mockFunction().encodeABI() ], + '', + ], + proposer, + tokenHolder: owner, + voters: [ + { voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For, reason: 'This is nice' }, + { voter: voter2, weight: web3.utils.toWei('7'), support: Enums.VoteType.For }, + { voter: voter3, weight: web3.utils.toWei('5'), support: Enums.VoteType.Against }, + { voter: voter4, weight: web3.utils.toWei('2'), support: Enums.VoteType.Abstain }, + ], + }; + }); + + afterEach(async function () { + expect(await this.mock.hasVoted(this.id, owner)).to.be.equal(false); + expect(await this.mock.hasVoted(this.id, voter1)).to.be.equal(true); + expect(await this.mock.hasVoted(this.id, voter2)).to.be.equal(true); + + await this.mock.proposalVotes(this.id).then(result => { + for (const [key, value] of Object.entries(Enums.VoteType)) { + expect(result[`${key.toLowerCase()}Votes`]).to.be.bignumber.equal( + Object.values(this.settings.voters).filter(({ support }) => support === value).reduce( + (acc, { weight }) => acc.add(new BN(weight)), + new BN('0'), + ), + ); + } + }); + + const startBlock = new BN(this.receipts.propose.blockNumber).add(votingDelay); + const endBlock = new BN(this.receipts.propose.blockNumber).add(votingDelay).add(votingPeriod); + expect(await this.mock.proposalSnapshot(this.id)).to.be.bignumber.equal(startBlock); + expect(await this.mock.proposalDeadline(this.id)).to.be.bignumber.equal(endBlock); + + expectEvent( + this.receipts.propose, + 'ProposalCreated', + { + proposalId: this.id, + proposer, + targets: this.settings.proposal[0], + // values: this.settings.proposal[1].map(value => new BN(value)), + signatures: this.settings.proposal[2].map(() => ''), + calldatas: this.settings.proposal[2], + startBlock, + endBlock, + description: this.settings.proposal[3], + }, + ); + + this.receipts.castVote.filter(Boolean).forEach(vote => { + const { voter } = vote.logs.find(Boolean).args; + expectEvent( + vote, + 'VoteCast', + this.settings.voters.find(({ address }) => address === voter), + ); + }); + expectEvent( + this.receipts.execute, + 'ProposalExecuted', + { proposalId: this.id }, + ); + await expectEvent.inTransaction( + this.receipts.execute.transactionHash, + this.receiver, + 'MockFunctionCalled', + ); + }); + runGovernorWorkflow(); + }); + + describe('Delay is extended to prevent last minute take-over', function () { + beforeEach(async function () { + this.settings = { + proposal: [ + [ this.receiver.address ], + [ 0 ], + [ this.receiver.contract.methods.mockFunction().encodeABI() ], + '', + ], + proposer, + tokenHolder: owner, + voters: [ + { voter: voter1, weight: web3.utils.toWei('0.2'), support: Enums.VoteType.Against, reason: 'This is not going to reach quorum' }, + { voter: voter2, weight: web3.utils.toWei('1.0') }, // do not actually vote, only getting tokens + { voter: voter3, weight: web3.utils.toWei('0.9') }, // do not actually vote, only getting tokens + ], + steps: { + wait: { enable: false }, + execute: { enable: false }, + }, + }; + }); + + afterEach(async function () { + expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Active); + + const startBlock = new BN(this.receipts.propose.blockNumber).add(votingDelay); + const endBlock = new BN(this.receipts.propose.blockNumber).add(votingDelay).add(votingPeriod); + expect(await this.mock.proposalSnapshot(this.id)).to.be.bignumber.equal(startBlock); + expect(await this.mock.proposalDeadline(this.id)).to.be.bignumber.equal(endBlock); + + // wait until the vote is almost over + await time.advanceBlockTo(endBlock.subn(1)); + expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Active); + + // try to overtake the vote at the last minute + const tx = await this.mock.castVote(this.id, Enums.VoteType.For, { from: voter2 }); + + // vote duration is extended + const extendedBlock = new BN(tx.receipt.blockNumber).add(votingDelayExtention); + expect(await this.mock.proposalDeadline(this.id)).to.be.bignumber.equal(extendedBlock); + + // vote is still active after expected end + await time.advanceBlockTo(endBlock.addn(1)); + expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Active); + + // Still possible to vote + await this.mock.castVote(this.id, Enums.VoteType.Against, { from: voter3 }); + + // proposal fails + await time.advanceBlockTo(extendedBlock.addn(1)); + expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Defeated); + }); + runGovernorWorkflow(); + }); + + describe('setVotingDelayExtention', function () { + beforeEach(async function () { + this.newVotingDelayExtention = new BN(0); // disable voting delay extention + }); + + it('protected', async function () { + await expectRevert( + this.mock.setVotingDelayExtention(this.newVotingDelayExtention), + 'Governor: onlyGovernance', + ); + }); + + describe('using workflow', function () { + beforeEach(async function () { + this.settings = { + proposal: [ + [ this.mock.address ], + [ web3.utils.toWei('0') ], + [ this.mock.contract.methods.setVotingDelayExtention(this.newVotingDelayExtention).encodeABI() ], + '', + ], + proposer, + tokenHolder: owner, + voters: [ + { voter: voter1, weight: web3.utils.toWei('1.0'), support: Enums.VoteType.For }, + ], + }; + }); + afterEach(async function () { + expectEvent( + this.receipts.propose, + 'ProposalCreated', + { proposalId: this.id }, + ); + expectEvent( + this.receipts.execute, + 'ProposalExecuted', + { proposalId: this.id }, + ); + expectEvent( + this.receipts.execute, + 'VotingDelayExtentionSet', + { oldVotingDelayExtention: votingDelayExtention, newVotingDelayExtention: this.newVotingDelayExtention }, + ); + expect(await this.mock.votingDelayExtention()).to.be.bignumber.equal(this.newVotingDelayExtention); + }); + runGovernorWorkflow(); + }); + }); +});