|
@@ -0,0 +1,107 @@
|
|
|
+const { ethers } = require('hardhat');
|
|
|
+const { expect } = require('chai');
|
|
|
+const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
|
|
|
+
|
|
|
+const { min } = require('../helpers/math');
|
|
|
+const time = require('../helpers/time');
|
|
|
+
|
|
|
+const { shouldBehaveLikeVesting } = require('./VestingWallet.behavior');
|
|
|
+
|
|
|
+async function fixture() {
|
|
|
+ const amount = ethers.parseEther('100');
|
|
|
+ const duration = time.duration.years(4);
|
|
|
+ const start = (await time.clock.timestamp()) + time.duration.hours(1);
|
|
|
+ const cliffDuration = time.duration.years(1);
|
|
|
+ const cliff = start + cliffDuration;
|
|
|
+
|
|
|
+ const [sender, beneficiary] = await ethers.getSigners();
|
|
|
+ const mock = await ethers.deployContract('$VestingWalletCliff', [beneficiary, start, duration, cliffDuration]);
|
|
|
+
|
|
|
+ const token = await ethers.deployContract('$ERC20', ['Name', 'Symbol']);
|
|
|
+ await token.$_mint(mock, amount);
|
|
|
+ await sender.sendTransaction({ to: mock, value: amount });
|
|
|
+
|
|
|
+ const pausableToken = await ethers.deployContract('$ERC20Pausable', ['Name', 'Symbol']);
|
|
|
+ const beneficiaryMock = await ethers.deployContract('EtherReceiverMock');
|
|
|
+
|
|
|
+ const env = {
|
|
|
+ eth: {
|
|
|
+ checkRelease: async (tx, amount) => {
|
|
|
+ await expect(tx).to.emit(mock, 'EtherReleased').withArgs(amount);
|
|
|
+ await expect(tx).to.changeEtherBalances([mock, beneficiary], [-amount, amount]);
|
|
|
+ },
|
|
|
+ setupFailure: async () => {
|
|
|
+ await beneficiaryMock.setAcceptEther(false);
|
|
|
+ await mock.connect(beneficiary).transferOwnership(beneficiaryMock);
|
|
|
+ return { args: [], error: [mock, 'FailedInnerCall'] };
|
|
|
+ },
|
|
|
+ releasedEvent: 'EtherReleased',
|
|
|
+ argsVerify: [],
|
|
|
+ args: [],
|
|
|
+ },
|
|
|
+ token: {
|
|
|
+ checkRelease: async (tx, amount) => {
|
|
|
+ await expect(tx).to.emit(token, 'Transfer').withArgs(mock, beneficiary, amount);
|
|
|
+ await expect(tx).to.changeTokenBalances(token, [mock, beneficiary], [-amount, amount]);
|
|
|
+ },
|
|
|
+ setupFailure: async () => {
|
|
|
+ await pausableToken.$_pause();
|
|
|
+ return {
|
|
|
+ args: [ethers.Typed.address(pausableToken)],
|
|
|
+ error: [pausableToken, 'EnforcedPause'],
|
|
|
+ };
|
|
|
+ },
|
|
|
+ releasedEvent: 'ERC20Released',
|
|
|
+ argsVerify: [token],
|
|
|
+ args: [ethers.Typed.address(token)],
|
|
|
+ },
|
|
|
+ };
|
|
|
+
|
|
|
+ const schedule = Array(64)
|
|
|
+ .fill()
|
|
|
+ .map((_, i) => (BigInt(i) * duration) / 60n + start);
|
|
|
+
|
|
|
+ const vestingFn = timestamp => min(amount, timestamp < cliff ? 0n : (amount * (timestamp - start)) / duration);
|
|
|
+
|
|
|
+ return { mock, duration, start, beneficiary, cliff, schedule, vestingFn, env };
|
|
|
+}
|
|
|
+
|
|
|
+describe('VestingWalletCliff', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ Object.assign(this, await loadFixture(fixture));
|
|
|
+ });
|
|
|
+
|
|
|
+ it('rejects a larger cliff than vesting duration', async function () {
|
|
|
+ await expect(
|
|
|
+ ethers.deployContract('$VestingWalletCliff', [this.beneficiary, this.start, this.duration, this.duration + 1n]),
|
|
|
+ )
|
|
|
+ .revertedWithCustomError(this.mock, 'InvalidCliffDuration')
|
|
|
+ .withArgs(this.duration + 1n, this.duration);
|
|
|
+ });
|
|
|
+
|
|
|
+ it('check vesting contract', async function () {
|
|
|
+ expect(await this.mock.owner()).to.equal(this.beneficiary);
|
|
|
+ expect(await this.mock.start()).to.equal(this.start);
|
|
|
+ expect(await this.mock.duration()).to.equal(this.duration);
|
|
|
+ expect(await this.mock.end()).to.equal(this.start + this.duration);
|
|
|
+ expect(await this.mock.cliff()).to.equal(this.cliff);
|
|
|
+ });
|
|
|
+
|
|
|
+ describe('vesting schedule', function () {
|
|
|
+ describe('Eth vesting', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ Object.assign(this, this.env.eth);
|
|
|
+ });
|
|
|
+
|
|
|
+ shouldBehaveLikeVesting();
|
|
|
+ });
|
|
|
+
|
|
|
+ describe('ERC20 vesting', function () {
|
|
|
+ beforeEach(async function () {
|
|
|
+ Object.assign(this, this.env.token);
|
|
|
+ });
|
|
|
+
|
|
|
+ shouldBehaveLikeVesting();
|
|
|
+ });
|
|
|
+ });
|
|
|
+});
|