Add Governor contracts (#2672)
Co-authored-by: Francisco Giordano <frangio.1@gmail.com>
This commit is contained in:
819
test/governance/Governor.test.js
Normal file
819
test/governance/Governor.test.js
Normal file
@ -0,0 +1,819 @@
|
||||
const { BN, constants, expectEvent, expectRevert, time } = require('@openzeppelin/test-helpers');
|
||||
const ethSigUtil = require('eth-sig-util');
|
||||
const Wallet = require('ethereumjs-wallet').default;
|
||||
const Enums = require('../helpers/enums');
|
||||
const { EIP712Domain } = require('../helpers/eip712');
|
||||
const { fromRpcSig } = require('ethereumjs-util');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('./GovernorWorkflow.behavior');
|
||||
|
||||
const {
|
||||
shouldSupportInterfaces,
|
||||
} = require('../utils/introspection/SupportsInterface.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesMock');
|
||||
const Governor = artifacts.require('GovernorMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
contract('Governor', 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');
|
||||
|
||||
beforeEach(async function () {
|
||||
this.owner = owner;
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16, 0);
|
||||
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 });
|
||||
});
|
||||
|
||||
shouldSupportInterfaces([
|
||||
'ERC165',
|
||||
'Governor',
|
||||
]);
|
||||
|
||||
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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
expect(await this.mock.COUNTING_MODE()).to.be.equal('support=bravo&quorum=for,abstain');
|
||||
});
|
||||
|
||||
describe('scenario', function () {
|
||||
describe('nominal', function () {
|
||||
beforeEach(async function () {
|
||||
this.value = web3.utils.toWei('1');
|
||||
|
||||
await web3.eth.sendTransaction({ from: owner, to: this.mock.address, value: this.value });
|
||||
expect(await web3.eth.getBalance(this.mock.address)).to.be.bignumber.equal(this.value);
|
||||
expect(await web3.eth.getBalance(this.receiver.address)).to.be.bignumber.equal('0');
|
||||
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ this.value ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
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('10'), 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 },
|
||||
],
|
||||
};
|
||||
this.votingDelay = await this.mock.votingDelay();
|
||||
this.votingPeriod = await this.mock.votingPeriod();
|
||||
});
|
||||
|
||||
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'),
|
||||
),
|
||||
);
|
||||
}
|
||||
});
|
||||
|
||||
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: new BN(this.receipts.propose.blockNumber).add(this.votingDelay),
|
||||
endBlock: new BN(this.receipts.propose.blockNumber).add(this.votingDelay).add(this.votingPeriod),
|
||||
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',
|
||||
);
|
||||
|
||||
expect(await web3.eth.getBalance(this.mock.address)).to.be.bignumber.equal('0');
|
||||
expect(await web3.eth.getBalance(this.receiver.address)).to.be.bignumber.equal(this.value);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('vote with signature', function () {
|
||||
beforeEach(async function () {
|
||||
const chainId = await web3.eth.getChainId();
|
||||
// generate voter by signature wallet
|
||||
const voterBySig = Wallet.generate();
|
||||
this.voter = web3.utils.toChecksumAddress(voterBySig.getAddressString());
|
||||
// use delegateBySig to enable vote delegation for this wallet
|
||||
const { v, r, s } = fromRpcSig(ethSigUtil.signTypedMessage(
|
||||
voterBySig.getPrivateKey(),
|
||||
{
|
||||
data: {
|
||||
types: {
|
||||
EIP712Domain,
|
||||
Delegation: [
|
||||
{ name: 'delegatee', type: 'address' },
|
||||
{ name: 'nonce', type: 'uint256' },
|
||||
{ name: 'expiry', type: 'uint256' },
|
||||
],
|
||||
},
|
||||
domain: { name: tokenName, version: '1', chainId, verifyingContract: this.token.address },
|
||||
primaryType: 'Delegation',
|
||||
message: { delegatee: this.voter, nonce: 0, expiry: constants.MAX_UINT256 },
|
||||
},
|
||||
},
|
||||
));
|
||||
await this.token.delegateBySig(this.voter, 0, constants.MAX_UINT256, v, r, s);
|
||||
// prepare signature for vote by signature
|
||||
const signature = async (message) => {
|
||||
return fromRpcSig(ethSigUtil.signTypedMessage(
|
||||
voterBySig.getPrivateKey(),
|
||||
{
|
||||
data: {
|
||||
types: {
|
||||
EIP712Domain,
|
||||
Ballot: [
|
||||
{ name: 'proposalId', type: 'uint256' },
|
||||
{ name: 'support', type: 'uint8' },
|
||||
],
|
||||
},
|
||||
domain: { name, version, chainId, verifyingContract: this.mock.address },
|
||||
primaryType: 'Ballot',
|
||||
message,
|
||||
},
|
||||
},
|
||||
));
|
||||
};
|
||||
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: this.voter, signature, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
};
|
||||
});
|
||||
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(false);
|
||||
expect(await this.mock.hasVoted(this.id, voter2)).to.be.equal(false);
|
||||
expect(await this.mock.hasVoted(this.id, this.voter)).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'),
|
||||
),
|
||||
);
|
||||
}
|
||||
});
|
||||
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.receiver,
|
||||
'MockFunctionCalled',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('send ethers', function () {
|
||||
beforeEach(async function () {
|
||||
this.receiver = { address: web3.utils.toChecksumAddress(web3.utils.randomHex(20)) };
|
||||
this.value = web3.utils.toWei('1');
|
||||
|
||||
await web3.eth.sendTransaction({ from: owner, to: this.mock.address, value: this.value });
|
||||
expect(await web3.eth.getBalance(this.mock.address)).to.be.bignumber.equal(this.value);
|
||||
expect(await web3.eth.getBalance(this.receiver.address)).to.be.bignumber.equal('0');
|
||||
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ this.value ],
|
||||
[ '0x' ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
{ voter: voter2, weight: web3.utils.toWei('1'), support: Enums.VoteType.Abstain },
|
||||
],
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expect(await web3.eth.getBalance(this.mock.address)).to.be.bignumber.equal('0');
|
||||
expect(await web3.eth.getBalance(this.receiver.address)).to.be.bignumber.equal(this.value);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('receiver revert without reason', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ 0 ],
|
||||
[ this.receiver.contract.methods.mockFunctionRevertsNoReason().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'Governor: call reverted without message' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('receiver revert with reason', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ 0 ],
|
||||
[ this.receiver.contract.methods.mockFunctionRevertsReason().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'CallReceiverMock: reverting' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('missing proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.For,
|
||||
error: 'Governor: unknown proposal id',
|
||||
},
|
||||
{
|
||||
voter: voter2,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.Abstain,
|
||||
error: 'Governor: unknown proposal id',
|
||||
},
|
||||
],
|
||||
steps: {
|
||||
propose: { enable: false },
|
||||
wait: { enable: false },
|
||||
execute: { error: 'Governor: unknown proposal id' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('duplicate pending proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectRevert(this.mock.propose(...this.settings.proposal), 'Governor: proposal already exists');
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('duplicate executed proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
{ voter: voter2, weight: web3.utils.toWei('1'), support: Enums.VoteType.Abstain },
|
||||
],
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectRevert(this.mock.propose(...this.settings.proposal), 'Governor: proposal already exists');
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('Invalid vote type', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: new BN('255'),
|
||||
error: 'GovernorVotingSimple: invalid value for enum VoteType',
|
||||
},
|
||||
],
|
||||
steps: {
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('double cast', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.For,
|
||||
},
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.For,
|
||||
error: 'GovernorVotingSimple: vote already casted',
|
||||
},
|
||||
],
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('quorum not reached', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('0'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'Governor: proposal not successful' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('score not reached', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.Against },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'Governor: proposal not successful' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('vote not over', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
wait: { enable: false },
|
||||
execute: { error: 'Governor: proposal not successful' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
|
||||
describe('state', function () {
|
||||
describe('Unset', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
propose: { enable: false },
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectRevert(this.mock.state(this.id), 'Governor: unknown proposal id');
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('Pending & Active', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
propose: { noadvance: true },
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Pending);
|
||||
|
||||
await time.advanceBlockTo(this.snapshot);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Active);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('Defeated', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Defeated);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('Succeeded', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Succeeded);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('Executed', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Executed);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
|
||||
describe('Cancel', function () {
|
||||
describe('Before proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
propose: { enable: false },
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectRevert(
|
||||
this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: unknown proposal id',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('After proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
steps: {
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.castVote(this.id, new BN('100'), { from: voter1 }),
|
||||
'Governor: vote not currently active',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('After vote', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
wait: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('After deadline', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('After execution', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectRevert(
|
||||
this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not active',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
|
||||
describe('Proposal length', function () {
|
||||
it('empty', async function () {
|
||||
await expectRevert(
|
||||
this.mock.propose(
|
||||
[],
|
||||
[],
|
||||
[],
|
||||
'<proposal description>',
|
||||
),
|
||||
'Governor: empty proposal',
|
||||
);
|
||||
});
|
||||
|
||||
it('missmatch #1', async function () {
|
||||
await expectRevert(
|
||||
this.mock.propose(
|
||||
[ ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
),
|
||||
'Governor: invalid proposal length',
|
||||
);
|
||||
});
|
||||
|
||||
it('missmatch #2', async function () {
|
||||
await expectRevert(
|
||||
this.mock.propose(
|
||||
[ this.receiver.address ],
|
||||
[ ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
),
|
||||
'Governor: invalid proposal length',
|
||||
);
|
||||
});
|
||||
|
||||
it('missmatch #3', async function () {
|
||||
await expectRevert(
|
||||
this.mock.propose(
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ ],
|
||||
'<proposal description>',
|
||||
),
|
||||
'Governor: invalid proposal length',
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
133
test/governance/GovernorWorkflow.behavior.js
Normal file
133
test/governance/GovernorWorkflow.behavior.js
Normal file
@ -0,0 +1,133 @@
|
||||
const { expectRevert, time } = require('@openzeppelin/test-helpers');
|
||||
|
||||
async function getReceiptOrRevert (promise, error = undefined) {
|
||||
if (error) {
|
||||
await expectRevert(promise, error);
|
||||
return undefined;
|
||||
} else {
|
||||
const { receipt } = await promise;
|
||||
return receipt;
|
||||
}
|
||||
}
|
||||
|
||||
function tryGet (obj, path = '') {
|
||||
try {
|
||||
return path.split('.').reduce((o, k) => o[k], obj);
|
||||
} catch (_) {
|
||||
return undefined;
|
||||
}
|
||||
}
|
||||
|
||||
function runGovernorWorkflow () {
|
||||
beforeEach(async function () {
|
||||
this.receipts = {};
|
||||
this.descriptionHash = web3.utils.keccak256(this.settings.proposal.slice(-1).find(Boolean));
|
||||
this.id = await this.mock.hashProposal(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
});
|
||||
|
||||
it('run', async function () {
|
||||
// transfer tokens
|
||||
if (tryGet(this.settings, 'voters')) {
|
||||
for (const voter of this.settings.voters) {
|
||||
if (voter.weight) {
|
||||
await this.token.transfer(voter.voter, voter.weight, { from: this.settings.tokenHolder });
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// propose
|
||||
if (this.mock.propose && tryGet(this.settings, 'steps.propose.enable') !== false) {
|
||||
this.receipts.propose = await getReceiptOrRevert(
|
||||
this.mock.methods['propose(address[],uint256[],bytes[],string)'](
|
||||
...this.settings.proposal,
|
||||
{ from: this.settings.proposer },
|
||||
),
|
||||
tryGet(this.settings, 'steps.propose.error'),
|
||||
);
|
||||
|
||||
if (tryGet(this.settings, 'steps.propose.error') === undefined) {
|
||||
this.deadline = await this.mock.proposalDeadline(this.id);
|
||||
this.snapshot = await this.mock.proposalSnapshot(this.id);
|
||||
}
|
||||
|
||||
if (tryGet(this.settings, 'steps.propose.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.propose.delay'));
|
||||
}
|
||||
|
||||
if (
|
||||
tryGet(this.settings, 'steps.propose.error') === undefined &&
|
||||
tryGet(this.settings, 'steps.propose.noadvance') !== true
|
||||
) {
|
||||
await time.advanceBlockTo(this.snapshot);
|
||||
}
|
||||
}
|
||||
|
||||
// vote
|
||||
if (tryGet(this.settings, 'voters')) {
|
||||
this.receipts.castVote = [];
|
||||
for (const voter of this.settings.voters) {
|
||||
if (!voter.signature) {
|
||||
this.receipts.castVote.push(
|
||||
await getReceiptOrRevert(
|
||||
voter.reason
|
||||
? this.mock.castVoteWithReason(this.id, voter.support, voter.reason, { from: voter.voter })
|
||||
: this.mock.castVote(this.id, voter.support, { from: voter.voter }),
|
||||
voter.error,
|
||||
),
|
||||
);
|
||||
} else {
|
||||
const { v, r, s } = await voter.signature({ proposalId: this.id, support: voter.support });
|
||||
this.receipts.castVote.push(
|
||||
await getReceiptOrRevert(
|
||||
this.mock.castVoteBySig(this.id, voter.support, v, r, s),
|
||||
voter.error,
|
||||
),
|
||||
);
|
||||
}
|
||||
if (tryGet(voter, 'delay')) {
|
||||
await time.increase(tryGet(voter, 'delay'));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// fast forward
|
||||
if (tryGet(this.settings, 'steps.wait.enable') !== false) {
|
||||
await time.advanceBlockTo(this.deadline);
|
||||
}
|
||||
|
||||
// queue
|
||||
if (this.mock.queue && tryGet(this.settings, 'steps.queue.enable') !== false) {
|
||||
this.receipts.queue = await getReceiptOrRevert(
|
||||
this.mock.methods['queue(address[],uint256[],bytes[],bytes32)'](
|
||||
...this.settings.proposal.slice(0, -1),
|
||||
this.descriptionHash,
|
||||
{ from: this.settings.queuer },
|
||||
),
|
||||
tryGet(this.settings, 'steps.queue.error'),
|
||||
);
|
||||
this.eta = await this.mock.proposalEta(this.id);
|
||||
if (tryGet(this.settings, 'steps.queue.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.queue.delay'));
|
||||
}
|
||||
}
|
||||
|
||||
// execute
|
||||
if (this.mock.execute && tryGet(this.settings, 'steps.execute.enable') !== false) {
|
||||
this.receipts.execute = await getReceiptOrRevert(
|
||||
this.mock.methods['execute(address[],uint256[],bytes[],bytes32)'](
|
||||
...this.settings.proposal.slice(0, -1),
|
||||
this.descriptionHash,
|
||||
{ from: this.settings.executer },
|
||||
),
|
||||
tryGet(this.settings, 'steps.execute.error'),
|
||||
);
|
||||
if (tryGet(this.settings, 'steps.execute.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.execute.delay'));
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = {
|
||||
runGovernorWorkflow,
|
||||
};
|
||||
430
test/governance/compatibility/GovernorCompatibilityBravo.test.js
Normal file
430
test/governance/compatibility/GovernorCompatibilityBravo.test.js
Normal file
@ -0,0 +1,430 @@
|
||||
const { BN, expectEvent, expectRevert, time } = require('@openzeppelin/test-helpers');
|
||||
const Enums = require('../../helpers/enums');
|
||||
const RLP = require('rlp');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('../GovernorWorkflow.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesCompMock');
|
||||
const Timelock = artifacts.require('CompTimelock');
|
||||
const Governor = artifacts.require('GovernorCompatibilityBravoMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
async function getReceiptOrRevert (promise, error = undefined) {
|
||||
if (error) {
|
||||
await expectRevert(promise, error);
|
||||
return undefined;
|
||||
} else {
|
||||
const { receipt } = await promise;
|
||||
return receipt;
|
||||
}
|
||||
}
|
||||
|
||||
function tryGet (obj, path = '') {
|
||||
try {
|
||||
return path.split('.').reduce((o, k) => o[k], obj);
|
||||
} catch (_) {
|
||||
return undefined;
|
||||
}
|
||||
}
|
||||
|
||||
function makeContractAddress (creator, nonce) {
|
||||
return web3.utils.toChecksumAddress(web3.utils.sha3(RLP.encode([creator, nonce])).slice(12).substring(14));
|
||||
}
|
||||
|
||||
contract('GovernorCompatibilityBravo', function (accounts) {
|
||||
const [ owner, proposer, voter1, voter2, voter3, voter4, other ] = accounts;
|
||||
|
||||
const name = 'OZ-Governor';
|
||||
// const version = '1';
|
||||
const tokenName = 'MockToken';
|
||||
const tokenSymbol = 'MTKN';
|
||||
const tokenSupply = web3.utils.toWei('100');
|
||||
const proposalThreshold = web3.utils.toWei('10');
|
||||
|
||||
beforeEach(async function () {
|
||||
const [ deployer ] = await web3.eth.getAccounts();
|
||||
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
|
||||
// Need to predict governance address to set it as timelock admin with a delayed transfer
|
||||
const nonce = await web3.eth.getTransactionCount(deployer);
|
||||
const predictGovernor = makeContractAddress(deployer, nonce + 1);
|
||||
|
||||
this.timelock = await Timelock.new(predictGovernor, 2 * 86400);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16, proposalThreshold, this.timelock.address);
|
||||
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 });
|
||||
|
||||
await this.token.transfer(proposer, proposalThreshold, { from: owner });
|
||||
await this.token.delegate(proposer, { from: proposer });
|
||||
});
|
||||
|
||||
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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
expect(await this.mock.quorumVotes()).to.be.bignumber.equal('0');
|
||||
expect(await this.mock.COUNTING_MODE()).to.be.equal('support=bravo&quorum=bravo');
|
||||
});
|
||||
|
||||
describe('nominal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ], // targets
|
||||
[ web3.utils.toWei('0') ], // values
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ], // calldatas
|
||||
'<proposal description>', // description
|
||||
],
|
||||
proposer,
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.Abstain,
|
||||
},
|
||||
{
|
||||
voter: voter2,
|
||||
weight: web3.utils.toWei('10'),
|
||||
support: Enums.VoteType.For,
|
||||
},
|
||||
{
|
||||
voter: voter3,
|
||||
weight: web3.utils.toWei('5'),
|
||||
support: Enums.VoteType.Against,
|
||||
},
|
||||
{
|
||||
voter: voter4,
|
||||
support: '100',
|
||||
error: 'GovernorCompatibilityBravo: invalid vote type',
|
||||
},
|
||||
{
|
||||
voter: voter1,
|
||||
support: Enums.VoteType.For,
|
||||
error: 'GovernorCompatibilityBravo: vote already casted',
|
||||
skip: true,
|
||||
},
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
this.votingDelay = await this.mock.votingDelay();
|
||||
this.votingPeriod = await this.mock.votingPeriod();
|
||||
this.receipts = {};
|
||||
});
|
||||
afterEach(async function () {
|
||||
const proposal = await this.mock.proposals(this.id);
|
||||
expect(proposal.id).to.be.bignumber.equal(this.id);
|
||||
expect(proposal.proposer).to.be.equal(proposer);
|
||||
expect(proposal.eta).to.be.bignumber.equal(this.eta);
|
||||
expect(proposal.startBlock).to.be.bignumber.equal(this.snapshot);
|
||||
expect(proposal.endBlock).to.be.bignumber.equal(this.deadline);
|
||||
expect(proposal.canceled).to.be.equal(false);
|
||||
expect(proposal.executed).to.be.equal(true);
|
||||
|
||||
for (const [key, value] of Object.entries(Enums.VoteType)) {
|
||||
expect(proposal[`${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 action = await this.mock.getActions(this.id);
|
||||
expect(action.targets).to.be.deep.equal(this.settings.proposal[0]);
|
||||
// expect(action.values).to.be.deep.equal(this.settings.proposal[1]);
|
||||
expect(action.signatures).to.be.deep.equal(Array(this.settings.proposal[2].length).fill(''));
|
||||
expect(action.calldatas).to.be.deep.equal(this.settings.proposal[2]);
|
||||
|
||||
for (const voter of this.settings.voters.filter(({ skip }) => !skip)) {
|
||||
expect(await this.mock.hasVoted(this.id, voter.voter)).to.be.equal(voter.error === undefined);
|
||||
|
||||
const receipt = await this.mock.getReceipt(this.id, voter.voter);
|
||||
expect(receipt.hasVoted).to.be.equal(voter.error === undefined);
|
||||
expect(receipt.support).to.be.bignumber.equal(voter.error === undefined ? voter.support : '0');
|
||||
expect(receipt.votes).to.be.bignumber.equal(voter.error === undefined ? voter.weight : '0');
|
||||
}
|
||||
|
||||
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: new BN(this.receipts.propose.blockNumber).add(this.votingDelay),
|
||||
endBlock: new BN(this.receipts.propose.blockNumber).add(this.votingDelay).add(this.votingPeriod),
|
||||
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('proposalThreshold not reached', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ], // targets
|
||||
[ web3.utils.toWei('0') ], // values
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ], // calldatas
|
||||
'<proposal description>', // description
|
||||
],
|
||||
proposer: other,
|
||||
steps: {
|
||||
propose: { error: 'GovernorCompatibilityBravo: proposer votes below proposal threshold' },
|
||||
wait: { enable: false },
|
||||
queue: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('with compatibility interface', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ], // targets
|
||||
[ web3.utils.toWei('0') ], // values
|
||||
[ '' ], // signatures
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ], // calldatas
|
||||
'<proposal description>', // description
|
||||
],
|
||||
proposer,
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{
|
||||
voter: voter1,
|
||||
weight: web3.utils.toWei('1'),
|
||||
support: Enums.VoteType.Abstain,
|
||||
},
|
||||
{
|
||||
voter: voter2,
|
||||
weight: web3.utils.toWei('10'),
|
||||
support: Enums.VoteType.For,
|
||||
},
|
||||
{
|
||||
voter: voter3,
|
||||
weight: web3.utils.toWei('5'),
|
||||
support: Enums.VoteType.Against,
|
||||
},
|
||||
{
|
||||
voter: voter4,
|
||||
support: '100',
|
||||
error: 'GovernorCompatibilityBravo: invalid vote type',
|
||||
},
|
||||
{
|
||||
voter: voter1,
|
||||
support: Enums.VoteType.For,
|
||||
error: 'GovernorCompatibilityBravo: vote already casted',
|
||||
skip: true,
|
||||
},
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
this.votingDelay = await this.mock.votingDelay();
|
||||
this.votingPeriod = await this.mock.votingPeriod();
|
||||
this.receipts = {};
|
||||
});
|
||||
|
||||
afterEach(async function () {
|
||||
const proposal = await this.mock.proposals(this.id);
|
||||
expect(proposal.id).to.be.bignumber.equal(this.id);
|
||||
expect(proposal.proposer).to.be.equal(proposer);
|
||||
expect(proposal.eta).to.be.bignumber.equal(this.eta);
|
||||
expect(proposal.startBlock).to.be.bignumber.equal(this.snapshot);
|
||||
expect(proposal.endBlock).to.be.bignumber.equal(this.deadline);
|
||||
expect(proposal.canceled).to.be.equal(false);
|
||||
expect(proposal.executed).to.be.equal(true);
|
||||
|
||||
for (const [key, value] of Object.entries(Enums.VoteType)) {
|
||||
expect(proposal[`${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 action = await this.mock.getActions(this.id);
|
||||
expect(action.targets).to.be.deep.equal(this.settings.proposal[0]);
|
||||
// expect(action.values).to.be.deep.equal(this.settings.proposal[1]);
|
||||
expect(action.signatures).to.be.deep.equal(this.settings.proposal[2]);
|
||||
expect(action.calldatas).to.be.deep.equal(this.settings.proposal[3]);
|
||||
|
||||
for (const voter of this.settings.voters.filter(({ skip }) => !skip)) {
|
||||
expect(await this.mock.hasVoted(this.id, voter.voter)).to.be.equal(voter.error === undefined);
|
||||
|
||||
const receipt = await this.mock.getReceipt(this.id, voter.voter);
|
||||
expect(receipt.hasVoted).to.be.equal(voter.error === undefined);
|
||||
expect(receipt.support).to.be.bignumber.equal(voter.error === undefined ? voter.support : '0');
|
||||
expect(receipt.votes).to.be.bignumber.equal(voter.error === undefined ? voter.weight : '0');
|
||||
}
|
||||
|
||||
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],
|
||||
calldatas: this.settings.proposal[3],
|
||||
startBlock: new BN(this.receipts.propose.blockNumber).add(this.votingDelay),
|
||||
endBlock: new BN(this.receipts.propose.blockNumber).add(this.votingDelay).add(this.votingPeriod),
|
||||
description: this.settings.proposal[4],
|
||||
},
|
||||
);
|
||||
|
||||
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',
|
||||
);
|
||||
});
|
||||
|
||||
it('run', async function () {
|
||||
// transfer tokens
|
||||
if (tryGet(this.settings, 'voters')) {
|
||||
for (const voter of this.settings.voters) {
|
||||
if (voter.weight) {
|
||||
await this.token.transfer(voter.voter, voter.weight, { from: this.settings.tokenHolder });
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// propose
|
||||
if (this.mock.propose && tryGet(this.settings, 'steps.propose.enable') !== false) {
|
||||
this.receipts.propose = await getReceiptOrRevert(
|
||||
this.mock.methods['propose(address[],uint256[],string[],bytes[],string)'](
|
||||
...this.settings.proposal,
|
||||
{ from: this.settings.proposer },
|
||||
),
|
||||
tryGet(this.settings, 'steps.propose.error'),
|
||||
);
|
||||
|
||||
if (tryGet(this.settings, 'steps.propose.error') === undefined) {
|
||||
this.id = this.receipts.propose.logs.find(({ event }) => event === 'ProposalCreated').args.proposalId;
|
||||
this.snapshot = await this.mock.proposalSnapshot(this.id);
|
||||
this.deadline = await this.mock.proposalDeadline(this.id);
|
||||
}
|
||||
|
||||
if (tryGet(this.settings, 'steps.propose.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.propose.delay'));
|
||||
}
|
||||
|
||||
if (
|
||||
tryGet(this.settings, 'steps.propose.error') === undefined &&
|
||||
tryGet(this.settings, 'steps.propose.noadvance') !== true
|
||||
) {
|
||||
await time.advanceBlockTo(this.snapshot);
|
||||
}
|
||||
}
|
||||
|
||||
// vote
|
||||
if (tryGet(this.settings, 'voters')) {
|
||||
this.receipts.castVote = [];
|
||||
for (const voter of this.settings.voters) {
|
||||
if (!voter.signature) {
|
||||
this.receipts.castVote.push(
|
||||
await getReceiptOrRevert(
|
||||
this.mock.castVote(this.id, voter.support, { from: voter.voter }),
|
||||
voter.error,
|
||||
),
|
||||
);
|
||||
} else {
|
||||
const { v, r, s } = await voter.signature({ proposalId: this.id, support: voter.support });
|
||||
this.receipts.castVote.push(
|
||||
await getReceiptOrRevert(
|
||||
this.mock.castVoteBySig(this.id, voter.support, v, r, s),
|
||||
voter.error,
|
||||
),
|
||||
);
|
||||
}
|
||||
if (tryGet(voter, 'delay')) {
|
||||
await time.increase(tryGet(voter, 'delay'));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// fast forward
|
||||
if (tryGet(this.settings, 'steps.wait.enable') !== false) {
|
||||
await time.advanceBlockTo(this.deadline);
|
||||
}
|
||||
|
||||
// queue
|
||||
if (this.mock.queue && tryGet(this.settings, 'steps.queue.enable') !== false) {
|
||||
this.receipts.queue = await getReceiptOrRevert(
|
||||
this.mock.methods['queue(uint256)'](this.id, { from: this.settings.queuer }),
|
||||
tryGet(this.settings, 'steps.queue.error'),
|
||||
);
|
||||
this.eta = await this.mock.proposalEta(this.id);
|
||||
if (tryGet(this.settings, 'steps.queue.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.queue.delay'));
|
||||
}
|
||||
}
|
||||
|
||||
// execute
|
||||
if (this.mock.execute && tryGet(this.settings, 'steps.execute.enable') !== false) {
|
||||
this.receipts.execute = await getReceiptOrRevert(
|
||||
this.mock.methods['execute(uint256)'](this.id, { from: this.settings.executer }),
|
||||
tryGet(this.settings, 'steps.execute.error'),
|
||||
);
|
||||
if (tryGet(this.settings, 'steps.execute.delay')) {
|
||||
await time.increase(tryGet(this.settings, 'steps.execute.delay'));
|
||||
}
|
||||
}
|
||||
});
|
||||
});
|
||||
});
|
||||
87
test/governance/extensions/GovernorComp.test.js
Normal file
87
test/governance/extensions/GovernorComp.test.js
Normal file
@ -0,0 +1,87 @@
|
||||
const { BN, expectEvent } = require('@openzeppelin/test-helpers');
|
||||
const Enums = require('../../helpers/enums');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('./../GovernorWorkflow.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesCompMock');
|
||||
const Governor = artifacts.require('GovernorCompMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
contract('GovernorComp', function (accounts) {
|
||||
const [ owner, voter1, voter2, voter3, voter4 ] = accounts;
|
||||
|
||||
const name = 'OZ-Governor';
|
||||
// const version = '1';
|
||||
const tokenName = 'MockToken';
|
||||
const tokenSymbol = 'MTKN';
|
||||
const tokenSupply = web3.utils.toWei('100');
|
||||
|
||||
beforeEach(async function () {
|
||||
this.owner = owner;
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16);
|
||||
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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
});
|
||||
|
||||
describe('voting with comp token', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
{ voter: voter2, weight: web3.utils.toWei('10'), 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);
|
||||
expect(await this.mock.hasVoted(this.id, voter3)).to.be.equal(true);
|
||||
expect(await this.mock.hasVoted(this.id, voter4)).to.be.equal(true);
|
||||
|
||||
this.receipts.castVote.filter(Boolean).forEach(vote => {
|
||||
const { voter } = vote.logs.find(Boolean).args;
|
||||
expectEvent(
|
||||
vote,
|
||||
'VoteCast',
|
||||
this.settings.voters.find(({ address }) => address === voter),
|
||||
);
|
||||
});
|
||||
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'),
|
||||
),
|
||||
);
|
||||
}
|
||||
});
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
432
test/governance/extensions/GovernorTimelockCompound.test.js
Normal file
432
test/governance/extensions/GovernorTimelockCompound.test.js
Normal file
@ -0,0 +1,432 @@
|
||||
const { constants, expectEvent, expectRevert } = require('@openzeppelin/test-helpers');
|
||||
const { expect } = require('chai');
|
||||
const Enums = require('../../helpers/enums');
|
||||
const RLP = require('rlp');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('../GovernorWorkflow.behavior');
|
||||
|
||||
const {
|
||||
shouldSupportInterfaces,
|
||||
} = require('../../utils/introspection/SupportsInterface.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesMock');
|
||||
const Timelock = artifacts.require('CompTimelock');
|
||||
const Governor = artifacts.require('GovernorTimelockCompoundMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
function makeContractAddress (creator, nonce) {
|
||||
return web3.utils.toChecksumAddress(web3.utils.sha3(RLP.encode([creator, nonce])).slice(12).substring(14));
|
||||
}
|
||||
|
||||
contract('GovernorTimelockCompound', function (accounts) {
|
||||
const [ admin, voter ] = accounts;
|
||||
|
||||
const name = 'OZ-Governor';
|
||||
// const version = '1';
|
||||
const tokenName = 'MockToken';
|
||||
const tokenSymbol = 'MTKN';
|
||||
const tokenSupply = web3.utils.toWei('100');
|
||||
|
||||
beforeEach(async function () {
|
||||
const [ deployer ] = await web3.eth.getAccounts();
|
||||
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
|
||||
// Need to predict governance address to set it as timelock admin with a delayed transfer
|
||||
const nonce = await web3.eth.getTransactionCount(deployer);
|
||||
const predictGovernor = makeContractAddress(deployer, nonce + 1);
|
||||
|
||||
this.timelock = await Timelock.new(predictGovernor, 2 * 86400);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16, this.timelock.address, 0);
|
||||
this.receiver = await CallReceiver.new();
|
||||
await this.token.mint(voter, tokenSupply);
|
||||
await this.token.delegate(voter, { from: voter });
|
||||
});
|
||||
|
||||
shouldSupportInterfaces([
|
||||
'ERC165',
|
||||
'Governor',
|
||||
'GovernorTimelock',
|
||||
]);
|
||||
|
||||
it('post 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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
|
||||
expect(await this.mock.timelock()).to.be.equal(this.timelock.address);
|
||||
expect(await this.timelock.admin()).to.be.equal(this.mock.address);
|
||||
});
|
||||
|
||||
describe('nominal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.queue,
|
||||
'ProposalQueued',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.queue.transactionHash,
|
||||
this.timelock,
|
||||
'QueueTransaction',
|
||||
{ eta: this.eta },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.timelock,
|
||||
'ExecuteTransaction',
|
||||
{ eta: this.eta },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.receiver,
|
||||
'MockFunctionCalled',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('not queued', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { enable: false },
|
||||
execute: { error: 'GovernorTimelockCompound: proposal not yet queued' },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Succeeded);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('to early', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'Timelock::executeTransaction: Transaction hasn\'t surpassed time lock' },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Queued);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('to late', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 30 * 86400 },
|
||||
execute: { error: 'Governor: proposal not successful' },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Expired);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('deplicated underlying call', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
Array(2).fill(this.token.address),
|
||||
Array(2).fill(web3.utils.toWei('0')),
|
||||
Array(2).fill(this.token.contract.methods.approve(this.receiver.address, constants.MAX_UINT256).encodeABI()),
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: {
|
||||
error: 'GovernorTimelockCompound: identical proposal action already queued',
|
||||
},
|
||||
execute: {
|
||||
error: 'GovernorTimelockCompound: proposal not yet queued',
|
||||
},
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('re-queue / re-execute', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Executed);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.queue(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('cancel before queue prevents scheduling', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Succeeded);
|
||||
|
||||
expectEvent(
|
||||
await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'ProposalCanceled',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.queue(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('cancel after queue prevents executing', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Queued);
|
||||
|
||||
const receipt = await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
expectEvent(
|
||||
receipt,
|
||||
'ProposalCanceled',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
receipt.receipt.transactionHash,
|
||||
this.timelock,
|
||||
'CancelTransaction',
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('updateTimelock', function () {
|
||||
beforeEach(async function () {
|
||||
this.newTimelock = await Timelock.new(this.mock.address, 7 * 86400);
|
||||
});
|
||||
|
||||
it('protected', async function () {
|
||||
await expectRevert(
|
||||
this.mock.updateTimelock(this.newTimelock.address),
|
||||
'Governor: onlyGovernance',
|
||||
);
|
||||
});
|
||||
|
||||
describe('using workflow', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[
|
||||
this.timelock.address,
|
||||
this.mock.address,
|
||||
],
|
||||
[
|
||||
web3.utils.toWei('0'),
|
||||
web3.utils.toWei('0'),
|
||||
],
|
||||
[
|
||||
this.timelock.contract.methods.setPendingAdmin(admin).encodeABI(),
|
||||
this.mock.contract.methods.updateTimelock(this.newTimelock.address).encodeABI(),
|
||||
],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'TimelockChange',
|
||||
{ oldTimelock: this.timelock.address, newTimelock: this.newTimelock.address },
|
||||
);
|
||||
expect(await this.mock.timelock()).to.be.bignumber.equal(this.newTimelock.address);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
|
||||
describe('transfer timelock to new governor', function () {
|
||||
beforeEach(async function () {
|
||||
this.newGovernor = await Governor.new(name, this.token.address, 8, 32, this.timelock.address, 0);
|
||||
});
|
||||
|
||||
describe('using workflow', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.timelock.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.timelock.contract.methods.setPendingAdmin(this.newGovernor.address).encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 7 * 86400 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.timelock,
|
||||
'NewPendingAdmin',
|
||||
{ newPendingAdmin: this.newGovernor.address },
|
||||
);
|
||||
await this.newGovernor.__acceptAdmin();
|
||||
expect(await this.timelock.admin()).to.be.bignumber.equal(this.newGovernor.address);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
});
|
||||
369
test/governance/extensions/GovernorTimelockControl.test.js
Normal file
369
test/governance/extensions/GovernorTimelockControl.test.js
Normal file
@ -0,0 +1,369 @@
|
||||
const { constants, expectEvent, expectRevert } = require('@openzeppelin/test-helpers');
|
||||
const { expect } = require('chai');
|
||||
const Enums = require('../../helpers/enums');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('../GovernorWorkflow.behavior');
|
||||
|
||||
const {
|
||||
shouldSupportInterfaces,
|
||||
} = require('../../utils/introspection/SupportsInterface.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesMock');
|
||||
const Timelock = artifacts.require('TimelockController');
|
||||
const Governor = artifacts.require('GovernorTimelockControlMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
contract('GovernorTimelockControl', function (accounts) {
|
||||
const [ voter ] = accounts;
|
||||
|
||||
const name = 'OZ-Governor';
|
||||
// const version = '1';
|
||||
const tokenName = 'MockToken';
|
||||
const tokenSymbol = 'MTKN';
|
||||
const tokenSupply = web3.utils.toWei('100');
|
||||
|
||||
beforeEach(async function () {
|
||||
const [ deployer ] = await web3.eth.getAccounts();
|
||||
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
this.timelock = await Timelock.new(3600, [], []);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16, this.timelock.address, 0);
|
||||
this.receiver = await CallReceiver.new();
|
||||
// normal setup: governor is proposer, everyone is executor, timelock is its own admin
|
||||
await this.timelock.grantRole(await this.timelock.PROPOSER_ROLE(), this.mock.address);
|
||||
await this.timelock.grantRole(await this.timelock.EXECUTOR_ROLE(), constants.ZERO_ADDRESS);
|
||||
await this.timelock.revokeRole(await this.timelock.TIMELOCK_ADMIN_ROLE(), deployer);
|
||||
await this.token.mint(voter, tokenSupply);
|
||||
await this.token.delegate(voter, { from: voter });
|
||||
});
|
||||
|
||||
shouldSupportInterfaces([
|
||||
'ERC165',
|
||||
'Governor',
|
||||
'GovernorTimelock',
|
||||
]);
|
||||
|
||||
it('post 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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
|
||||
expect(await this.mock.timelock()).to.be.equal(this.timelock.address);
|
||||
});
|
||||
|
||||
describe('nominal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 3600 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
const timelockid = await this.timelock.hashOperationBatch(
|
||||
...this.settings.proposal.slice(0, 3),
|
||||
'0x0',
|
||||
this.descriptionHash,
|
||||
);
|
||||
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.queue,
|
||||
'ProposalQueued',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.queue.transactionHash,
|
||||
this.timelock,
|
||||
'CallScheduled',
|
||||
{ id: timelockid },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.timelock,
|
||||
'CallExecuted',
|
||||
{ id: timelockid },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.receiver,
|
||||
'MockFunctionCalled',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('executed by other proposer', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 3600 },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await this.timelock.executeBatch(
|
||||
...this.settings.proposal.slice(0, 3),
|
||||
'0x0',
|
||||
this.descriptionHash,
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Executed);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('not queued', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { enable: false },
|
||||
execute: { error: 'TimelockController: operation is not ready' },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Succeeded);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('to early', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'TimelockController: operation is not ready' },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Queued);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('re-queue / re-execute', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 3600 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Executed);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.queue(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('cancel before queue prevents scheduling', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { enable: false },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Succeeded);
|
||||
|
||||
expectEvent(
|
||||
await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'ProposalCanceled',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.queue(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('cancel after queue prevents execution', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 3600 },
|
||||
execute: { enable: false },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
const timelockid = await this.timelock.hashOperationBatch(
|
||||
...this.settings.proposal.slice(0, 3),
|
||||
'0x0',
|
||||
this.descriptionHash,
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Queued);
|
||||
|
||||
const receipt = await this.mock.cancel(...this.settings.proposal.slice(0, -1), this.descriptionHash);
|
||||
expectEvent(
|
||||
receipt,
|
||||
'ProposalCanceled',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
await expectEvent.inTransaction(
|
||||
receipt.receipt.transactionHash,
|
||||
this.timelock,
|
||||
'Cancelled',
|
||||
{ id: timelockid },
|
||||
);
|
||||
|
||||
expect(await this.mock.state(this.id)).to.be.bignumber.equal(Enums.ProposalState.Canceled);
|
||||
|
||||
await expectRevert(
|
||||
this.mock.execute(...this.settings.proposal.slice(0, -1), this.descriptionHash),
|
||||
'Governor: proposal not successful',
|
||||
);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('updateTimelock', function () {
|
||||
beforeEach(async function () {
|
||||
this.newTimelock = await Timelock.new(3600, [], []);
|
||||
});
|
||||
|
||||
it('protected', async function () {
|
||||
await expectRevert(
|
||||
this.mock.updateTimelock(this.newTimelock.address),
|
||||
'Governor: onlyGovernance',
|
||||
);
|
||||
});
|
||||
|
||||
describe('using workflow', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.mock.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.mock.contract.methods.updateTimelock(this.newTimelock.address).encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
voters: [
|
||||
{ voter: voter, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
queue: { delay: 3600 },
|
||||
},
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
expectEvent(
|
||||
this.receipts.propose,
|
||||
'ProposalCreated',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'ProposalExecuted',
|
||||
{ proposalId: this.id },
|
||||
);
|
||||
expectEvent(
|
||||
this.receipts.execute,
|
||||
'TimelockChange',
|
||||
{ oldTimelock: this.timelock.address, newTimelock: this.newTimelock.address },
|
||||
);
|
||||
expect(await this.mock.timelock()).to.be.bignumber.equal(this.newTimelock.address);
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
});
|
||||
122
test/governance/extensions/GovernorWeightQuorumFraction.test.js
Normal file
122
test/governance/extensions/GovernorWeightQuorumFraction.test.js
Normal file
@ -0,0 +1,122 @@
|
||||
const { BN, expectEvent, time } = require('@openzeppelin/test-helpers');
|
||||
const Enums = require('../../helpers/enums');
|
||||
|
||||
const {
|
||||
runGovernorWorkflow,
|
||||
} = require('./../GovernorWorkflow.behavior');
|
||||
|
||||
const Token = artifacts.require('ERC20VotesMock');
|
||||
const Governor = artifacts.require('GovernorMock');
|
||||
const CallReceiver = artifacts.require('CallReceiverMock');
|
||||
|
||||
contract('GovernorVotesQuorumFraction', function (accounts) {
|
||||
const [ owner, voter1, voter2, voter3, voter4 ] = accounts;
|
||||
|
||||
const name = 'OZ-Governor';
|
||||
// const version = '1';
|
||||
const tokenName = 'MockToken';
|
||||
const tokenSymbol = 'MTKN';
|
||||
const tokenSupply = new BN(web3.utils.toWei('100'));
|
||||
const ratio = new BN(8); // percents
|
||||
const newRatio = new BN(6); // percents
|
||||
|
||||
beforeEach(async function () {
|
||||
this.owner = owner;
|
||||
this.token = await Token.new(tokenName, tokenSymbol);
|
||||
this.mock = await Governor.new(name, this.token.address, 4, 16, ratio);
|
||||
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('4');
|
||||
expect(await this.mock.votingPeriod()).to.be.bignumber.equal('16');
|
||||
expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
|
||||
expect(await this.mock.quorumNumerator()).to.be.bignumber.equal(ratio);
|
||||
expect(await this.mock.quorumDenominator()).to.be.bignumber.equal('100');
|
||||
expect(await time.latestBlock().then(blockNumber => this.mock.quorum(blockNumber.subn(1))))
|
||||
.to.be.bignumber.equal(tokenSupply.mul(ratio).divn(100));
|
||||
});
|
||||
|
||||
describe('quroum not reached', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.receiver.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.receiver.contract.methods.mockFunction().encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: web3.utils.toWei('1'), support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'Governor: proposal not successful' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('update quorum ratio through proposal', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.mock.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.mock.contract.methods.updateQuorumNumerator(newRatio).encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: tokenSupply, support: Enums.VoteType.For },
|
||||
],
|
||||
};
|
||||
});
|
||||
afterEach(async function () {
|
||||
await expectEvent.inTransaction(
|
||||
this.receipts.execute.transactionHash,
|
||||
this.mock,
|
||||
'QuorumNumeratorUpdated',
|
||||
{
|
||||
oldQuorumNumerator: ratio,
|
||||
newQuorumNumerator: newRatio,
|
||||
},
|
||||
);
|
||||
|
||||
expect(await this.mock.quorumNumerator()).to.be.bignumber.equal(newRatio);
|
||||
expect(await this.mock.quorumDenominator()).to.be.bignumber.equal('100');
|
||||
expect(await time.latestBlock().then(blockNumber => this.mock.quorum(blockNumber.subn(1))))
|
||||
.to.be.bignumber.equal(tokenSupply.mul(newRatio).divn(100));
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
|
||||
describe('update quorum over the maximum', function () {
|
||||
beforeEach(async function () {
|
||||
this.settings = {
|
||||
proposal: [
|
||||
[ this.mock.address ],
|
||||
[ web3.utils.toWei('0') ],
|
||||
[ this.mock.contract.methods.updateQuorumNumerator(new BN(101)).encodeABI() ],
|
||||
'<proposal description>',
|
||||
],
|
||||
tokenHolder: owner,
|
||||
voters: [
|
||||
{ voter: voter1, weight: tokenSupply, support: Enums.VoteType.For },
|
||||
],
|
||||
steps: {
|
||||
execute: { error: 'GovernorVotesQuorumFraction: quorumNumerator over quorumDenominator' },
|
||||
},
|
||||
};
|
||||
});
|
||||
runGovernorWorkflow();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user