123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420 |
- const { expectEvent } = require('@openzeppelin/test-helpers');
- const { expect } = require('chai');
- const Enums = require('../../helpers/enums');
- const { GovernorHelper, proposalStatesToBitMap } = require('../../helpers/governance');
- const { expectRevertCustomError } = require('../../helpers/customError');
- const { clockFromReceipt } = require('../../helpers/time');
- const { selector } = require('../../helpers/methods');
- const AccessManager = artifacts.require('$AccessManager');
- const Governor = artifacts.require('$GovernorTimelockAccessMock');
- const AccessManagedTarget = artifacts.require('$AccessManagedTarget');
- const TOKENS = [
- // { Token: artifacts.require('$ERC20Votes'), mode: 'blocknumber' },
- { Token: artifacts.require('$ERC20VotesTimestampMock'), mode: 'timestamp' },
- ];
- const hashOperation = (caller, target, data) =>
- web3.utils.keccak256(web3.eth.abi.encodeParameters(['address', 'address', 'bytes'], [caller, target, data]));
- contract('GovernorTimelockAccess', function (accounts) {
- const [admin, 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 = web3.utils.toBN(4);
- const votingPeriod = web3.utils.toBN(16);
- const value = web3.utils.toWei('1');
- for (const { mode, Token } of TOKENS) {
- describe(`using ${Token._json.contractName}`, function () {
- beforeEach(async function () {
- this.token = await Token.new(tokenName, tokenSymbol, tokenName, version);
- this.manager = await AccessManager.new(admin);
- this.mock = await Governor.new(
- name,
- votingDelay,
- votingPeriod,
- 0, // proposal threshold
- this.manager.address,
- 0, // base delay
- this.token.address,
- 0, // quorum
- );
- this.receiver = await AccessManagedTarget.new(this.manager.address);
- this.helper = new GovernorHelper(this.mock, mode);
- await web3.eth.sendTransaction({ from: admin, to: this.mock.address, value });
- await this.token.$_mint(admin, tokenSupply);
- await this.helper.delegate({ token: this.token, to: voter1, value: web3.utils.toWei('10') }, { from: admin });
- await this.helper.delegate({ token: this.token, to: voter2, value: web3.utils.toWei('7') }, { from: admin });
- await this.helper.delegate({ token: this.token, to: voter3, value: web3.utils.toWei('5') }, { from: admin });
- await this.helper.delegate({ token: this.token, to: voter4, value: web3.utils.toWei('2') }, { from: admin });
- // default proposals
- this.restricted = {};
- this.restricted.selector = this.receiver.contract.methods.fnRestricted().encodeABI();
- this.restricted.operation = {
- target: this.receiver.address,
- value: '0',
- data: this.restricted.selector,
- };
- this.restricted.operationId = hashOperation(
- this.mock.address,
- this.restricted.operation.target,
- this.restricted.operation.data,
- );
- this.unrestricted = {};
- this.unrestricted.selector = this.receiver.contract.methods.fnUnrestricted().encodeABI();
- this.unrestricted.operation = {
- target: this.receiver.address,
- value: '0',
- data: this.unrestricted.selector,
- };
- this.unrestricted.operationId = hashOperation(
- this.mock.address,
- this.unrestricted.operation.target,
- this.unrestricted.operation.data,
- );
- this.fallback = {};
- this.fallback.operation = {
- target: this.receiver.address,
- value: '0',
- data: '0x1234',
- };
- this.fallback.operationId = hashOperation(
- this.mock.address,
- this.fallback.operation.target,
- this.fallback.operation.data,
- );
- });
- it('accepts ether transfers', async function () {
- await web3.eth.sendTransaction({ from: admin, to: this.mock.address, value: 1 });
- });
- 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(votingDelay);
- expect(await this.mock.votingPeriod()).to.be.bignumber.equal(votingPeriod);
- expect(await this.mock.quorum(0)).to.be.bignumber.equal('0');
- expect(await this.mock.accessManager()).to.be.equal(this.manager.address);
- });
- describe('base delay only', function () {
- for (const [delay, queue] of [
- [0, true],
- [0, false],
- [1000, true],
- ]) {
- it(`delay ${delay}, ${queue ? 'with' : 'without'} queuing`, async function () {
- await this.mock.$_setBaseDelaySeconds(delay);
- this.proposal = await this.helper.setProposal([this.unrestricted.operation], 'descr');
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- if (queue) {
- const txQueue = await this.helper.queue();
- expectEvent(txQueue, 'ProposalQueued', { proposalId: this.proposal.id });
- }
- if (delay > 0) {
- await this.helper.waitForEta();
- }
- const txExecute = await this.helper.execute();
- expectEvent(txExecute, 'ProposalExecuted', { proposalId: this.proposal.id });
- expectEvent.inTransaction(txExecute, this.receiver, 'CalledUnrestricted');
- });
- }
- });
- it('single operation with access manager delay', async function () {
- const delay = 1000;
- const roleId = '1';
- await this.manager.setTargetFunctionRole(this.receiver.address, [this.restricted.selector], roleId, {
- from: admin,
- });
- await this.manager.grantRole(roleId, this.mock.address, delay, { from: admin });
- this.proposal = await this.helper.setProposal([this.restricted.operation], 'descr');
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- const txQueue = await this.helper.queue();
- await this.helper.waitForEta();
- const txExecute = await this.helper.execute();
- expectEvent(txQueue, 'ProposalQueued', { proposalId: this.proposal.id });
- await expectEvent.inTransaction(txQueue.tx, this.manager, 'OperationScheduled', {
- operationId: this.restricted.operationId,
- nonce: '1',
- schedule: web3.utils.toBN(await clockFromReceipt.timestamp(txQueue.receipt)).addn(delay),
- caller: this.mock.address,
- target: this.restricted.operation.target,
- data: this.restricted.operation.data,
- });
- expectEvent(txExecute, 'ProposalExecuted', { proposalId: this.proposal.id });
- await expectEvent.inTransaction(txExecute.tx, this.manager, 'OperationExecuted', {
- operationId: this.restricted.operationId,
- nonce: '1',
- });
- await expectEvent.inTransaction(txExecute.tx, this.receiver, 'CalledRestricted');
- });
- it('bundle of varied operations', async function () {
- const managerDelay = 1000;
- const roleId = '1';
- const baseDelay = managerDelay * 2;
- await this.mock.$_setBaseDelaySeconds(baseDelay);
- await this.manager.setTargetFunctionRole(this.receiver.address, [this.restricted.selector], roleId, {
- from: admin,
- });
- await this.manager.grantRole(roleId, this.mock.address, managerDelay, { from: admin });
- this.proposal = await this.helper.setProposal(
- [this.restricted.operation, this.unrestricted.operation, this.fallback.operation],
- 'descr',
- );
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- const txQueue = await this.helper.queue();
- await this.helper.waitForEta();
- const txExecute = await this.helper.execute();
- expectEvent(txQueue, 'ProposalQueued', { proposalId: this.proposal.id });
- await expectEvent.inTransaction(txQueue.tx, this.manager, 'OperationScheduled', {
- operationId: this.restricted.operationId,
- nonce: '1',
- schedule: web3.utils.toBN(await clockFromReceipt.timestamp(txQueue.receipt)).addn(baseDelay),
- caller: this.mock.address,
- target: this.restricted.operation.target,
- data: this.restricted.operation.data,
- });
- expectEvent(txExecute, 'ProposalExecuted', { proposalId: this.proposal.id });
- await expectEvent.inTransaction(txExecute.tx, this.manager, 'OperationExecuted', {
- operationId: this.restricted.operationId,
- nonce: '1',
- });
- await expectEvent.inTransaction(txExecute.tx, this.receiver, 'CalledRestricted');
- await expectEvent.inTransaction(txExecute.tx, this.receiver, 'CalledUnrestricted');
- await expectEvent.inTransaction(txExecute.tx, this.receiver, 'CalledFallback');
- });
- describe('cancel', function () {
- const delay = 1000;
- const roleId = '1';
- beforeEach(async function () {
- await this.manager.setTargetFunctionRole(this.receiver.address, [this.restricted.selector], roleId, {
- from: admin,
- });
- await this.manager.grantRole(roleId, this.mock.address, delay, { from: admin });
- });
- it('cancellation after queue (internal)', async function () {
- this.proposal = await this.helper.setProposal([this.restricted.operation], 'descr');
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- await this.helper.queue();
- const txCancel = await this.helper.cancel('internal');
- expectEvent(txCancel, 'ProposalCanceled', { proposalId: this.proposal.id });
- await expectEvent.inTransaction(txCancel.tx, this.manager, 'OperationCanceled', {
- operationId: this.restricted.operationId,
- nonce: '1',
- });
- await this.helper.waitForEta();
- await expectRevertCustomError(this.helper.execute(), 'GovernorUnexpectedProposalState', [
- this.proposal.id,
- Enums.ProposalState.Canceled,
- proposalStatesToBitMap([Enums.ProposalState.Succeeded, Enums.ProposalState.Queued]),
- ]);
- });
- it('cancel calls already canceled by guardian', async function () {
- const operationA = { target: this.receiver.address, data: this.restricted.selector + '00' };
- const operationB = { target: this.receiver.address, data: this.restricted.selector + '01' };
- const operationC = { target: this.receiver.address, data: this.restricted.selector + '02' };
- const operationAId = hashOperation(this.mock.address, operationA.target, operationA.data);
- const operationBId = hashOperation(this.mock.address, operationB.target, operationB.data);
- const proposal1 = new GovernorHelper(this.mock, mode);
- const proposal2 = new GovernorHelper(this.mock, mode);
- proposal1.setProposal([operationA, operationB], 'proposal A+B');
- proposal2.setProposal([operationA, operationC], 'proposal A+C');
- for (const p of [proposal1, proposal2]) {
- await p.propose();
- await p.waitForSnapshot();
- await p.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await p.waitForDeadline();
- }
- // Can queue the first proposal
- await proposal1.queue();
- // Cannot queue the second proposal: operation A already scheduled with delay
- await expectRevertCustomError(proposal2.queue(), 'AccessManagerAlreadyScheduled', [operationAId]);
- // Admin cancels operation B on the manager
- await this.manager.cancel(this.mock.address, operationB.target, operationB.data, { from: admin });
- // Still cannot queue the second proposal: operation A already scheduled with delay
- await expectRevertCustomError(proposal2.queue(), 'AccessManagerAlreadyScheduled', [operationAId]);
- await proposal1.waitForEta();
- // Cannot execute first proposal: operation B has been canceled
- await expectRevertCustomError(proposal1.execute(), 'AccessManagerNotScheduled', [operationBId]);
- // Cancel the first proposal to release operation A
- await proposal1.cancel('internal');
- // can finally queue the second proposal
- await proposal2.queue();
- await proposal2.waitForEta();
- // Can execute second proposal
- await proposal2.execute();
- });
- });
- describe('ignore AccessManager', function () {
- it('defaults', async function () {
- expect(await this.mock.isAccessManagerIgnored(this.receiver.address, this.restricted.selector)).to.equal(
- false,
- );
- expect(await this.mock.isAccessManagerIgnored(this.mock.address, '0x12341234')).to.equal(true);
- });
- it('internal setter', async function () {
- const p1 = { target: this.receiver.address, selector: this.restricted.selector, ignored: true };
- const tx1 = await this.mock.$_setAccessManagerIgnored(p1.target, p1.selector, p1.ignored);
- expect(await this.mock.isAccessManagerIgnored(p1.target, p1.selector)).to.equal(p1.ignored);
- expectEvent(tx1, 'AccessManagerIgnoredSet', p1);
- const p2 = { target: this.mock.address, selector: '0x12341234', ignored: false };
- const tx2 = await this.mock.$_setAccessManagerIgnored(p2.target, p2.selector, p2.ignored);
- expect(await this.mock.isAccessManagerIgnored(p2.target, p2.selector)).to.equal(p2.ignored);
- expectEvent(tx2, 'AccessManagerIgnoredSet', p2);
- });
- it('external setter', async function () {
- const setAccessManagerIgnored = (...args) =>
- this.mock.contract.methods.setAccessManagerIgnored(...args).encodeABI();
- await this.helper.setProposal(
- [
- {
- target: this.mock.address,
- data: setAccessManagerIgnored(
- this.receiver.address,
- [this.restricted.selector, this.unrestricted.selector],
- true,
- ),
- value: '0',
- },
- {
- target: this.mock.address,
- data: setAccessManagerIgnored(this.mock.address, ['0x12341234', '0x67896789'], false),
- value: '0',
- },
- ],
- 'descr',
- );
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- const tx = await this.helper.execute();
- expectEvent(tx, 'AccessManagerIgnoredSet');
- expect(await this.mock.isAccessManagerIgnored(this.receiver.address, this.restricted.selector)).to.equal(
- true,
- );
- expect(await this.mock.isAccessManagerIgnored(this.receiver.address, this.unrestricted.selector)).to.equal(
- true,
- );
- expect(await this.mock.isAccessManagerIgnored(this.mock.address, '0x12341234')).to.equal(false);
- expect(await this.mock.isAccessManagerIgnored(this.mock.address, '0x67896789')).to.equal(false);
- });
- it('locked function', async function () {
- const setAccessManagerIgnored = selector('setAccessManagerIgnored(address,bytes4[],bool)');
- await expectRevertCustomError(
- this.mock.$_setAccessManagerIgnored(this.mock.address, setAccessManagerIgnored, true),
- 'GovernorLockedIgnore',
- [],
- );
- await this.mock.$_setAccessManagerIgnored(this.receiver.address, setAccessManagerIgnored, true);
- });
- it('ignores access manager', async function () {
- const amount = 100;
- const target = this.token.address;
- const data = this.token.contract.methods.transfer(voter4, amount).encodeABI();
- const selector = data.slice(0, 10);
- await this.token.$_mint(this.mock.address, amount);
- const roleId = '1';
- await this.manager.setTargetFunctionRole(target, [selector], roleId, { from: admin });
- await this.manager.grantRole(roleId, this.mock.address, 0, { from: admin });
- await this.helper.setProposal([{ target, data, value: '0' }], '1');
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- await expectRevertCustomError(this.helper.execute(), 'ERC20InsufficientBalance', [
- this.manager.address,
- 0,
- amount,
- ]);
- await this.mock.$_setAccessManagerIgnored(target, selector, true);
- await this.helper.setProposal([{ target, data, value: '0' }], '2');
- await this.helper.propose();
- await this.helper.waitForSnapshot();
- await this.helper.vote({ support: Enums.VoteType.For }, { from: voter1 });
- await this.helper.waitForDeadline();
- const tx = await this.helper.execute();
- expectEvent.inTransaction(tx, this.token, 'Transfer', { from: this.mock.address });
- });
- });
- });
- }
- });
|