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 20 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
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
* `Governor`: shift vote start and end by one block to better match Compound's GovernorBravo and prevent voting at the Governor level if the voting snapshot is not ready. ([#2892](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2892))
* `PaymentSplitter`: now supports ERC20 assets in addition to Ether. ([#2858](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2858))
* `ECDSA`: add a variant of `toEthSignedMessageHash` for arbitrary length message hashing. ([#2865](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2865))
* `VestingWallet`: new contract that handles the vesting of Eth and ERC20 tokens following a customizable vesting schedule. ([#2748](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2748))
Amxx marked this conversation as resolved.
Show resolved Hide resolved

## 4.3.2 (2021-09-14)

Expand Down
14 changes: 13 additions & 1 deletion contracts/finance/README.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,20 @@
[.readme-notice]
NOTE: This document is better viewed at https://docs.openzeppelin.com/contracts/api/finance

This directory includes primitives for financial systems. We currently only offer the {PaymentSplitter} contract, but we want to grow this directory so we welcome ideas.
This directory includes primitives for financial systems:

- {PaymentSplitter} allows to split Eth and ERC20 payments among a group of accounts. The sender does not need to be
Amxx marked this conversation as resolved.
Show resolved Hide resolved
aware that the assets will be split in this way, since it is handled transparently by the contract. The split can be
in equal parts or in any other arbitrary proportion.

- {VestingWallet} handles the vesting of Eth and ERC20 tokens for a given beneficiary. Custody of multiple tokens can
Amxx marked this conversation as resolved.
Show resolved Hide resolved
be given to this contract, which will release the token to the beneficiary following a given, customizable, vesting
schedule.

== PaymentSplitter

{{PaymentSplitter}}

== VestingWallet

{{VestingWallet}}
Amxx marked this conversation as resolved.
Show resolved Hide resolved
130 changes: 130 additions & 0 deletions contracts/finance/VestingWallet.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "../token/ERC20/utils/SafeERC20.sol";
import "../utils/Address.sol";
import "../utils/Context.sol";
import "../utils/math/Math.sol";

/**
* @title VestingWallet
* @dev This contract handles the vesting of Eth and 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 schedule is customizable through the {vestedAmount} function.
*
* 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.
*/
contract VestingWallet is Context {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
event TokensReleased(uint256 amount);
event ERC20TokensReleased(address token, uint256 amount);
Amxx marked this conversation as resolved.
Show resolved Hide resolved

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

/**
* @dev Set the beneficiary, start timestamp and vesting duration of the vesting wallet.
*/
constructor(
address beneficiaryAddress,
uint256 startTimestamp,
uint256 durationSeconds
) {
require(beneficiaryAddress != address(0), "VestingWallet: beneficiary is zero address");
_beneficiary = beneficiaryAddress;
_start = startTimestamp;
_duration = durationSeconds;
}

/**
* @dev The contract should be able to receive Eth.
*/
receive() external payable {}
Amxx marked this conversation as resolved.
Show resolved Hide resolved

/**
* @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 Amont of eth already released
Amxx marked this conversation as resolved.
Show resolved Hide resolved
*/
function released() public view returns (uint256) {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
return _released;
}

/**
* @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 _erc20Released[token];
}

/**
* @dev Release the native token (ether) that have already vested.
*
* Emits a {TokensReleased} event.
*/
function release() public virtual {
uint256 releasable = vestedAmount(block.timestamp) - released();
_released += releasable;
emit TokensReleased(releasable);
Address.sendValue(payable(beneficiary()), releasable);
}

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

/**
* @dev Calculates the amount of ether that has already vested. Default implementation is a linear vesting curve.
*/
function vestedAmount(uint256 timestamp) public view virtual returns (uint256) {
Amxx marked this conversation as resolved.
Show resolved Hide resolved
if (timestamp < start()) {
return 0;
} else {
uint256 historicalBalance = address(this).balance + released();
return Math.min(historicalBalance, (historicalBalance * (timestamp - start())) / duration());
}
}

/**
* @dev Calculates the amount of tokens that has already vested. Default implementation is a linear vesting curve.
*/
function vestedAmount(address token, uint256 timestamp) public view virtual returns (uint256) {
frangio marked this conversation as resolved.
Show resolved Hide resolved
if (timestamp < start()) {
return 0;
} else {
uint256 historicalBalance = IERC20(token).balanceOf(address(this)) + released(token);
return Math.min(historicalBalance, (historicalBalance * (timestamp - start())) / duration());
}
}
}
70 changes: 70 additions & 0 deletions test/finance/VestingWallet.behavior.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
const { expectEvent } = require('@openzeppelin/test-helpers');
const { expect } = require('chai');

function shouldBehaveLikeVesting (beneficiary) {
it('check vesting schedule', async function () {
const args = this.token ? [ this.token.address ] : [];

for (const timestamp of this.schedule) {
expect(await this.mock.vestedAmount(...args, timestamp))
.to.be.bignumber.equal(this.vestingFn(timestamp));
}
});

it('execute vesting schedule', async function () {
const args = this.token ? [ this.token.address ] : [];

let released = web3.utils.toBN(0);
const before = await this.getBalance(beneficiary);

{
const receipt = await this.mock.release(...args);

await expectEvent.inTransaction(
receipt.tx,
this.mock,
this.token ? 'ERC20TokensReleased' : 'TokensReleased',
Object.fromEntries(Object.entries({
token: this.token && this.token.address,
amount: '0',
}).filter(x => x.every(Boolean))),
Amxx marked this conversation as resolved.
Show resolved Hide resolved
);

await this.checkRelease(receipt, beneficiary, '0');

expect(await this.getBalance(beneficiary)).to.be.bignumber.equal(before);
}

for (const timestamp of this.schedule) {
const vested = this.vestingFn(timestamp);

await new Promise(resolve => web3.currentProvider.send({
method: 'evm_setNextBlockTimestamp',
params: [ timestamp.toNumber() ],
}, resolve));

const receipt = await this.mock.release(...args);

await expectEvent.inTransaction(
receipt.tx,
this.mock,
this.token ? 'ERC20TokensReleased' : 'TokensReleased',
Object.fromEntries(Object.entries({
token: this.token && this.token.address,
amount: vested.sub(released),
}).filter(x => x.every(Boolean))),
);

await this.checkRelease(receipt, beneficiary, vested.sub(released));

expect(await this.getBalance(beneficiary))
.to.be.bignumber.equal(before.add(vested));

released = vested;
}
});
}

module.exports = {
shouldBehaveLikeVesting,
};
67 changes: 67 additions & 0 deletions test/finance/VestingWallet.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
const { constants, expectEvent, expectRevert, time } = require('@openzeppelin/test-helpers');
const { web3 } = require('@openzeppelin/test-helpers/src/setup');
const { expect } = require('chai');

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

const { shouldBehaveLikeVesting } = require('./VestingWallet.behavior');

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

contract('VestingWallet', function (accounts) {
const [ sender, beneficiary ] = 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.mock = await VestingWallet.new(beneficiary, this.start, duration);
});

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.mock.beneficiary()).to.be.equal(beneficiary);
expect(await this.mock.start()).to.be.bignumber.equal(this.start);
expect(await this.mock.duration()).to.be.bignumber.equal(duration);
});

describe('vesting schedule', function () {
beforeEach(async function () {
this.schedule = Array(64).fill().map((_, i) => web3.utils.toBN(i).mul(duration).divn(60).add(this.start));
this.vestingFn = timestamp => min(amount, amount.mul(timestamp.sub(this.start)).div(duration));
});

describe('Eth vesting', function () {
beforeEach(async function () {
await web3.eth.sendTransaction({ from: sender, to: this.mock.address, value: amount });
this.getBalance = account => web3.eth.getBalance(account).then(web3.utils.toBN);
this.checkRelease = () => {};
});

shouldBehaveLikeVesting(beneficiary);
});

describe('ERC20 vesting', function () {
beforeEach(async function () {
this.token = await ERC20Mock.new('Name', 'Symbol', this.mock.address, amount);
this.getBalance = (account) => this.token.balanceOf(account);
this.checkRelease = (receipt, to, value) => expectEvent.inTransaction(
receipt.tx,
this.token,
'Transfer',
{ from: this.mock.address, to, value },
);
});

shouldBehaveLikeVesting(beneficiary);
});
});
});