add a TokenTimelock contract
This commit is contained in:
41
contracts/token/TokenTimelock.sol
Normal file
41
contracts/token/TokenTimelock.sol
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
pragma solidity ^0.4.11;
|
||||||
|
|
||||||
|
|
||||||
|
import './ERC20Basic.sol';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title TokenTimelock
|
||||||
|
* @dev TokenTimelock is a token holder contract that will allow a
|
||||||
|
* beneficiary to extract the tokens after a time has passed
|
||||||
|
*/
|
||||||
|
contract TokenTimelock {
|
||||||
|
|
||||||
|
// ERC20 basic token contract being held
|
||||||
|
ERC20Basic token;
|
||||||
|
|
||||||
|
// beneficiary of tokens after they are released
|
||||||
|
address beneficiary;
|
||||||
|
|
||||||
|
// timestamp where token release is enabled
|
||||||
|
uint releaseTime;
|
||||||
|
|
||||||
|
function TokenTimelock(ERC20Basic _token, address _beneficiary, uint _releaseTime) {
|
||||||
|
require(_releaseTime > now);
|
||||||
|
token = _token;
|
||||||
|
beneficiary = _beneficiary;
|
||||||
|
releaseTime = _releaseTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev beneficiary claims tokens held by time lock
|
||||||
|
*/
|
||||||
|
function claim() {
|
||||||
|
require(msg.sender == beneficiary);
|
||||||
|
require(now >= releaseTime);
|
||||||
|
|
||||||
|
uint amount = token.balanceOf(this);
|
||||||
|
require(amount > 0);
|
||||||
|
|
||||||
|
token.transfer(beneficiary, amount);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -35,9 +35,13 @@
|
|||||||
"babel-preset-stage-2": "^6.18.0",
|
"babel-preset-stage-2": "^6.18.0",
|
||||||
"babel-preset-stage-3": "^6.17.0",
|
"babel-preset-stage-3": "^6.17.0",
|
||||||
"babel-register": "^6.23.0",
|
"babel-register": "^6.23.0",
|
||||||
|
"chai": "^4.0.2",
|
||||||
|
"chai-as-promised": "^7.0.0",
|
||||||
|
"chai-bignumber": "^2.0.0",
|
||||||
"coveralls": "^2.13.1",
|
"coveralls": "^2.13.1",
|
||||||
"ethereumjs-testrpc": "^3.0.2",
|
"ethereumjs-testrpc": "^3.0.2",
|
||||||
"mocha-lcov-reporter": "^1.3.0",
|
"mocha-lcov-reporter": "^1.3.0",
|
||||||
|
"moment": "^2.18.1",
|
||||||
"solidity-coverage": "^0.1.0",
|
"solidity-coverage": "^0.1.0",
|
||||||
"truffle": "3.2.2"
|
"truffle": "3.2.2"
|
||||||
}
|
}
|
||||||
|
|||||||
58
test/TokenTimelock.js
Normal file
58
test/TokenTimelock.js
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
const BigNumber = web3.BigNumber
|
||||||
|
|
||||||
|
require('chai')
|
||||||
|
.use(require('chai-as-promised'))
|
||||||
|
.use(require('chai-bignumber')(BigNumber))
|
||||||
|
.should()
|
||||||
|
|
||||||
|
import moment from 'moment'
|
||||||
|
|
||||||
|
import latestTime from './helpers/latestTime'
|
||||||
|
import increaseTime from './helpers/increaseTime'
|
||||||
|
|
||||||
|
const MintableToken = artifacts.require('MintableToken')
|
||||||
|
const TokenTimelock = artifacts.require('TokenTimelock')
|
||||||
|
|
||||||
|
contract('TokenTimelock', function ([_, owner, beneficiary]) {
|
||||||
|
|
||||||
|
const amount = new BigNumber(100)
|
||||||
|
|
||||||
|
beforeEach(async function () {
|
||||||
|
this.token = await MintableToken.new({from: owner})
|
||||||
|
this.releaseTime = latestTime().add(1, 'year').unix()
|
||||||
|
this.timelock = await TokenTimelock.new(this.token.address, beneficiary, this.releaseTime)
|
||||||
|
await this.token.mint(this.timelock.address, amount, {from: owner})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('cannot be claimed before time limit', async function () {
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.rejected
|
||||||
|
})
|
||||||
|
|
||||||
|
it('cannot be claimed just before time limit', async function () {
|
||||||
|
await increaseTime(moment.duration(0.99, 'year'))
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.rejected
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can be claimed just after limit', async function () {
|
||||||
|
await increaseTime(moment.duration(1.01, 'year'))
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.fulfilled
|
||||||
|
const balance = await this.token.balanceOf(beneficiary)
|
||||||
|
balance.should.be.bignumber.equal(amount)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can be claimed after time limit', async function () {
|
||||||
|
await increaseTime(moment.duration(2, 'year'))
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.fulfilled
|
||||||
|
const balance = await this.token.balanceOf(beneficiary)
|
||||||
|
balance.should.be.bignumber.equal(amount)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('cannot be claimed twice', async function () {
|
||||||
|
await increaseTime(moment.duration(2, 'year'))
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.fulfilled
|
||||||
|
await this.timelock.claim({from: beneficiary}).should.be.rejected
|
||||||
|
const balance = await this.token.balanceOf(beneficiary)
|
||||||
|
balance.should.be.bignumber.equal(amount)
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
23
test/helpers/increaseTime.js
Normal file
23
test/helpers/increaseTime.js
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
// Increases testrpc time by the passed duration (a moment.js instance)
|
||||||
|
export default function increaseTime(duration) {
|
||||||
|
const id = Date.now()
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
web3.currentProvider.sendAsync({
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
method: 'evm_increaseTime',
|
||||||
|
params: [duration.asSeconds()],
|
||||||
|
id: id,
|
||||||
|
}, err1 => {
|
||||||
|
if (err1) return reject(err1)
|
||||||
|
|
||||||
|
web3.currentProvider.sendAsync({
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
method: 'evm_mine',
|
||||||
|
id: id+1,
|
||||||
|
}, (err2, res) => {
|
||||||
|
return err2 ? reject(err2) : resolve(res)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
6
test/helpers/latestTime.js
Normal file
6
test/helpers/latestTime.js
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
import moment from 'moment'
|
||||||
|
|
||||||
|
// Returns a moment.js instance representing the time of the last mined block
|
||||||
|
export default function latestTime() {
|
||||||
|
return moment.unix(web3.eth.getBlock('latest').timestamp)
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user