123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107 |
- 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();
- });
- });
- });
|