Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a VestingWallet #2748

Merged
merged 26 commits into from
Oct 18, 2021
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
112 changes: 112 additions & 0 deletions contracts/finance/VestingWallet.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "../token/ERC20/extensions/ERC20Votes.sol";
import "../token/ERC20/utils/SafeERC20.sol";
import "../utils/Context.sol";

/**
* @title VestingWallet
* @dev This contract handles the vesting of ERC20 tokens for a given beneficiary. Custody of multiple tokens can be
* given to this contract, which will release the token to the beneficiary following a given vesting schedule. The
* vesting scheduled is customizable through the `vestedAmount(address,uint256)` function.
Amxx marked this conversation as resolved.
Show resolved Hide resolved
*
* Any token transferred to this contract will follow the vesting schedule as if they were locked from the beginning.
* Consequently, if the vesting has already started, any amount of tokens sent to this contract will (at least partly)
* be immediately releasable.
*
* While tokens are locked, the beneficiary still has the ability to delegate the voting power potentially associated
* with these tokens.
Amxx marked this conversation as resolved.
Show resolved Hide resolved
*/
contract VestingWallet is Context {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
event TokensReleased(address token, uint256 amount);

mapping(address => uint256) private _released;
address private immutable _beneficiary;
uint256 private immutable _start;
uint256 private immutable _duration;
Amxx marked this conversation as resolved.
Show resolved Hide resolved

modifier onlyBeneficiary() {
require(beneficiary() == _msgSender(), "VestingWallet: access restricted to beneficiary");
_;
}

/**
* @dev Set the beneficiary, start timestamp and vesting duration of the vesting wallet.
*/
constructor(
address initialBeneficiary,
uint256 initialStart,
uint256 initialDuration
Amxx marked this conversation as resolved.
Show resolved Hide resolved
) {
require(initialBeneficiary != address(0), "VestingWallet: beneficiary is zero address");
_beneficiary = initialBeneficiary;
_start = initialStart;
_duration = initialDuration;
}

/**
* @dev Getter for the beneficiary address.
*/
function beneficiary() public view virtual returns (address) {
return _beneficiary;
}

/**
* @dev Getter for the start timestamp.
*/
function start() public view virtual returns (uint256) {
return _start;
}

/**
* @dev Getter for the vesting duration.
*/
function duration() public view virtual returns (uint256) {
return _duration;
}

/**
* @dev Delegate the voting right of tokens currently vesting
*/
function delegate(address token, address delegatee) public virtual onlyBeneficiary() {
ERC20Votes(token).delegate(delegatee);
}

/**
* @dev Amont of token already released
Amxx marked this conversation as resolved.
Show resolved Hide resolved
*/
function released(address token) public view returns (uint256) {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
return _released[token];
}

/**
* @dev Release the tokens that have already vested.
Amxx marked this conversation as resolved.
Show resolved Hide resolved
*/
function release(address token) public virtual {
uint256 releasable = vestedAmount(token, block.timestamp) - released(token);
_released[token] += releasable;
emit TokensReleased(token, releasable);
SafeERC20.safeTransfer(IERC20(token), beneficiary(), releasable);
}

/**
* @dev Calculates the amount that has already vested. Default implementation is a linear vesting curve.
*/
function vestedAmount(address token, uint256 timestamp) public view virtual returns (uint256) {
if (timestamp < start()) {
return 0;
} else if (timestamp >= start() + duration()) {
return _historicalBalance(token);
} else {
return (_historicalBalance(token) * (timestamp - start())) / duration();
}
}

/**
* @dev Calculates the historical balance (current balance + already released balance).
*/
function _historicalBalance(address token) internal view returns (uint256) {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
return IERC20(token).balanceOf(address(this)) + released(token);
}
}
118 changes: 118 additions & 0 deletions test/finance/VestingWallet.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
const { constants, expectEvent, expectRevert, time } = require('@openzeppelin/test-helpers');
const { expect } = require('chai');

const ERC20VotesMock = artifacts.require('ERC20VotesMock');
const VestingWallet = artifacts.require('VestingWallet');

const min = (...args) => args.slice(1).reduce((x, y) => x.lt(y) ? x : y, args[0]);

contract('VestingWallet', function (accounts) {
const [ beneficiary, other ] = accounts;

const amount = web3.utils.toBN(web3.utils.toWei('100'));
const duration = web3.utils.toBN(4 * 365 * 86400); // 4 years

beforeEach(async function () {
this.start = (await time.latest()).addn(3600); // in 1 hour
this.token = await ERC20VotesMock.new('Name', 'Symbol');
this.vesting = await VestingWallet.new(beneficiary, this.start, duration);
await this.token.mint(this.vesting.address, amount);

this.schedule = Array(256).fill()
.map((_, i) => web3.utils.toBN(i).mul(duration).divn(224).add(this.start))
.map(timestamp => ({
timestamp,
vested: min(amount.mul(timestamp.sub(this.start)).div(duration), amount),
Amxx marked this conversation as resolved.
Show resolved Hide resolved
}));
});

it('rejects zero address for beneficiary', async function () {
await expectRevert(
VestingWallet.new(constants.ZERO_ADDRESS, this.start, duration),
'VestingWallet: beneficiary is zero address',
);
});

it('check vesting contract', async function () {
expect(await this.vesting.beneficiary()).to.be.equal(beneficiary);
expect(await this.vesting.start()).to.be.bignumber.equal(this.start);
expect(await this.vesting.duration()).to.be.bignumber.equal(duration);
});

describe('vesting schedule', function () {
it('check vesting schedule', async function () {
for (const { timestamp, vested } of this.schedule) {
expect(await this.vesting.vestedAmount(this.token.address, timestamp)).to.be.bignumber.equal(vested);
}
});

it('execute vesting schedule', async function () {
const { tx } = await this.vesting.release(this.token.address);
await expectEvent.inTransaction(tx, this.vesting, 'TokensReleased', {
token: this.token.address,
amount: '0',
});
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: this.vesting.address,
to: beneficiary,
value: '0',
});

// on schedule
let released = web3.utils.toBN(0);
for (const { timestamp, vested } of this.schedule) {
await new Promise(resolve => web3.currentProvider.send({
method: 'evm_setNextBlockTimestamp',
params: [ timestamp.toNumber() ],
}, resolve));

const { tx } = await this.vesting.release(this.token.address);
await expectEvent.inTransaction(tx, this.vesting, 'TokensReleased', {
token: this.token.address,
amount: vested.sub(released),
});
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: this.vesting.address,
to: beneficiary,
value: vested.sub(released),
});

released = vested;

expect(await this.token.balanceOf(this.vesting.address)).to.be.bignumber.equal(amount.sub(vested));
expect(await this.token.balanceOf(beneficiary)).to.be.bignumber.equal(vested);
}
});
});

describe('delegate vote', function () {
it('wrong caller', async function () {
expect(await this.token.delegates(this.vesting.address)).to.be.equal(constants.ZERO_ADDRESS);

await expectRevert(
this.vesting.delegate(this.token.address, other, { from: other }),
'VestingWallet: access restricted to beneficiary',
);

expect(await this.token.delegates(this.vesting.address)).to.be.equal(constants.ZERO_ADDRESS);
});

it('authorized call', async function () {
expect(await this.token.delegates(this.vesting.address)).to.be.equal(constants.ZERO_ADDRESS);

const { tx } = await this.vesting.delegate(this.token.address, other, { from: beneficiary });
await expectEvent.inTransaction(tx, this.token, 'DelegateChanged', {
delegator: this.vesting.address,
fromDelegate: constants.ZERO_ADDRESS,
toDelegate: other,
});
await expectEvent.inTransaction(tx, this.token, 'DelegateVotesChanged', {
delegate: other,
previousBalance: '0',
newBalance: amount,
});

expect(await this.token.delegates(this.vesting.address)).to.be.equal(other);
});
});
});
24 changes: 12 additions & 12 deletions test/token/ERC20/extensions/ERC20Wrapper.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -44,12 +44,12 @@ contract('ERC20', function (accounts) {
it('valid', async function () {
await this.underlying.approve(this.token.address, initialSupply, { from: initialHolder });
const { tx } = await this.token.depositFor(initialHolder, initialSupply, { from: initialHolder });
expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
await expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
from: initialHolder,
to: this.token.address,
value: initialSupply,
});
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: ZERO_ADDRESS,
to: initialHolder,
value: initialSupply,
Expand All @@ -74,12 +74,12 @@ contract('ERC20', function (accounts) {
it('to other account', async function () {
await this.underlying.approve(this.token.address, initialSupply, { from: initialHolder });
const { tx } = await this.token.depositFor(anotherAccount, initialSupply, { from: initialHolder });
expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
await expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
from: initialHolder,
to: this.token.address,
value: initialSupply,
});
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: ZERO_ADDRESS,
to: anotherAccount,
value: initialSupply,
Expand All @@ -104,12 +104,12 @@ contract('ERC20', function (accounts) {
const value = new BN(42);

const { tx } = await this.token.withdrawTo(initialHolder, value, { from: initialHolder });
expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
await expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
from: this.token.address,
to: initialHolder,
value: value,
});
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: initialHolder,
to: ZERO_ADDRESS,
value: value,
Expand All @@ -118,12 +118,12 @@ contract('ERC20', function (accounts) {

it('entire balance', async function () {
const { tx } = await this.token.withdrawTo(initialHolder, initialSupply, { from: initialHolder });
expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
await expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
from: this.token.address,
to: initialHolder,
value: initialSupply,
});
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: initialHolder,
to: ZERO_ADDRESS,
value: initialSupply,
Expand All @@ -132,12 +132,12 @@ contract('ERC20', function (accounts) {

it('to other account', async function () {
const { tx } = await this.token.withdrawTo(anotherAccount, initialSupply, { from: initialHolder });
expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
await expectEvent.inTransaction(tx, this.underlying, 'Transfer', {
from: this.token.address,
to: anotherAccount,
value: initialSupply,
});
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: initialHolder,
to: ZERO_ADDRESS,
value: initialSupply,
Expand All @@ -151,7 +151,7 @@ contract('ERC20', function (accounts) {
await this.token.depositFor(initialHolder, initialSupply, { from: initialHolder });

const { tx } = await this.token.recover(anotherAccount);
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: ZERO_ADDRESS,
to: anotherAccount,
value: '0',
Expand All @@ -162,7 +162,7 @@ contract('ERC20', function (accounts) {
await this.underlying.transfer(this.token.address, initialSupply, { from: initialHolder });

const { tx } = await this.token.recover(anotherAccount);
expectEvent.inTransaction(tx, this.token, 'Transfer', {
await expectEvent.inTransaction(tx, this.token, 'Transfer', {
from: ZERO_ADDRESS,
to: anotherAccount,
value: initialSupply,
Expand Down