Add a VestingWallet (#2748)
Co-authored-by: Francisco Giordano <frangio.1@gmail.com>
This commit is contained in:
@ -11,6 +11,7 @@
|
|||||||
* `PaymentSplitter`: now supports ERC20 assets in addition to Ether. ([#2858](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2858))
|
* `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))
|
* `ECDSA`: add a variant of `toEthSignedMessageHash` for arbitrary length message hashing. ([#2865](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2865))
|
||||||
* `MerkleProof`: add a `processProof` function that returns the rebuilt root hash given a leaf and a proof. ([#2841](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/2841))
|
* `MerkleProof`: add a `processProof` function that returns the rebuilt root hash given a leaf and a proof. ([#2841](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/2841))
|
||||||
|
* `VestingWallet`: new contract that handles the vesting of Ether and ERC20 tokens following a customizable vesting schedule. ([#2748](https://github.com/OpenZeppelin/openzeppelin-contracts/pull/#2748))
|
||||||
|
|
||||||
## 4.3.2 (2021-09-14)
|
## 4.3.2 (2021-09-14)
|
||||||
|
|
||||||
|
|||||||
@ -3,8 +3,18 @@
|
|||||||
[.readme-notice]
|
[.readme-notice]
|
||||||
NOTE: This document is better viewed at https://docs.openzeppelin.com/contracts/api/finance
|
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
|
- {PaymentSplitter} allows to split Ether and ERC20 payments among a group of accounts. The sender does not need to be
|
||||||
|
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 Ether 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, customizable, vesting
|
||||||
|
schedule.
|
||||||
|
|
||||||
|
== Contracts
|
||||||
|
|
||||||
{{PaymentSplitter}}
|
{{PaymentSplitter}}
|
||||||
|
|
||||||
|
{{VestingWallet}}
|
||||||
|
|||||||
134
contracts/finance/VestingWallet.sol
Normal file
134
contracts/finance/VestingWallet.sol
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
// 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 {
|
||||||
|
event EtherReleased(uint256 amount);
|
||||||
|
event ERC20Released(address token, uint256 amount);
|
||||||
|
|
||||||
|
uint256 private _released;
|
||||||
|
mapping(address => uint256) private _erc20Released;
|
||||||
|
address private immutable _beneficiary;
|
||||||
|
uint64 private immutable _start;
|
||||||
|
uint64 private immutable _duration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Set the beneficiary, start timestamp and vesting duration of the vesting wallet.
|
||||||
|
*/
|
||||||
|
constructor(
|
||||||
|
address beneficiaryAddress,
|
||||||
|
uint64 startTimestamp,
|
||||||
|
uint64 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 virtual {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @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 Amount of eth already released
|
||||||
|
*/
|
||||||
|
function released() public view virtual returns (uint256) {
|
||||||
|
return _released;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Amount of token already released
|
||||||
|
*/
|
||||||
|
function released(address token) public view virtual returns (uint256) {
|
||||||
|
return _erc20Released[token];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Release the native token (ether) that have already vested.
|
||||||
|
*
|
||||||
|
* Emits a {TokensReleased} event.
|
||||||
|
*/
|
||||||
|
function release() public virtual {
|
||||||
|
uint256 releasable = vestedAmount(uint64(block.timestamp)) - released();
|
||||||
|
_released += releasable;
|
||||||
|
emit EtherReleased(releasable);
|
||||||
|
Address.sendValue(payable(beneficiary()), releasable);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Release the tokens that have already vested.
|
||||||
|
*
|
||||||
|
* Emits a {TokensReleased} event.
|
||||||
|
*/
|
||||||
|
function release(address token) public virtual {
|
||||||
|
uint256 releasable = vestedAmount(token, uint64(block.timestamp)) - released(token);
|
||||||
|
_erc20Released[token] += releasable;
|
||||||
|
emit ERC20Released(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(uint64 timestamp) public view virtual returns (uint256) {
|
||||||
|
return _vestingSchedule(address(this).balance + released(), timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Calculates the amount of tokens that has already vested. Default implementation is a linear vesting curve.
|
||||||
|
*/
|
||||||
|
function vestedAmount(address token, uint64 timestamp) public view virtual returns (uint256) {
|
||||||
|
return _vestingSchedule(IERC20(token).balanceOf(address(this)) + released(token), timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Virtual implementation of the vesting formula. This returns the amout vested, as a function of time, for
|
||||||
|
* an asset given its total historical allocation.
|
||||||
|
*/
|
||||||
|
function _vestingSchedule(uint256 totalAllocation, uint64 timestamp) internal view virtual returns (uint256) {
|
||||||
|
if (timestamp < start()) {
|
||||||
|
return 0;
|
||||||
|
} else if (timestamp > start() + duration()) {
|
||||||
|
return totalAllocation;
|
||||||
|
} else {
|
||||||
|
return (totalAllocation * (timestamp - start())) / duration();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
72
test/finance/VestingWallet.behavior.js
Normal file
72
test/finance/VestingWallet.behavior.js
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
const { expectEvent } = require('@openzeppelin/test-helpers');
|
||||||
|
const { expect } = require('chai');
|
||||||
|
|
||||||
|
function releasedEvent (token, amount) {
|
||||||
|
return token
|
||||||
|
? [ 'ERC20Released', { token: token.address, amount } ]
|
||||||
|
: [ 'EtherReleased', { amount } ];
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldBehaveLikeVesting (beneficiary) {
|
||||||
|
it('check vesting schedule', async function () {
|
||||||
|
const [ method, ...args ] = this.token
|
||||||
|
? [ 'vestedAmount(address,uint64)', this.token.address ]
|
||||||
|
: [ 'vestedAmount(uint64)' ];
|
||||||
|
|
||||||
|
for (const timestamp of this.schedule) {
|
||||||
|
expect(await this.mock.methods[method](...args, timestamp))
|
||||||
|
.to.be.bignumber.equal(this.vestingFn(timestamp));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('execute vesting schedule', async function () {
|
||||||
|
const [ method, ...args ] = this.token
|
||||||
|
? [ 'release(address)', this.token.address ]
|
||||||
|
: [ 'release()' ];
|
||||||
|
|
||||||
|
let released = web3.utils.toBN(0);
|
||||||
|
const before = await this.getBalance(beneficiary);
|
||||||
|
|
||||||
|
{
|
||||||
|
const receipt = await this.mock.methods[method](...args);
|
||||||
|
|
||||||
|
await expectEvent.inTransaction(
|
||||||
|
receipt.tx,
|
||||||
|
this.mock,
|
||||||
|
...releasedEvent(this.token, '0'),
|
||||||
|
);
|
||||||
|
|
||||||
|
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.methods[method](...args);
|
||||||
|
|
||||||
|
await expectEvent.inTransaction(
|
||||||
|
receipt.tx,
|
||||||
|
this.mock,
|
||||||
|
...releasedEvent(this.token, vested.sub(released)),
|
||||||
|
);
|
||||||
|
|
||||||
|
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
test/finance/VestingWallet.test.js
Normal file
67
test/finance/VestingWallet.test.js
Normal file
@ -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);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user