From 8ce40071342b6c2c565f58861c53546c3239b455 Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Wed, 29 Aug 2018 23:11:22 -0300 Subject: [PATCH 01/11] Refactor TokenEscrow tests * Extracted behavior to a different file --- test/payment/TokenEscrow.behavior.js | 122 +++++++++++++++++++++++++++ test/payment/TokenEscrow.test.js | 115 +------------------------ 2 files changed, 126 insertions(+), 111 deletions(-) create mode 100644 test/payment/TokenEscrow.behavior.js diff --git a/test/payment/TokenEscrow.behavior.js b/test/payment/TokenEscrow.behavior.js new file mode 100644 index 00000000000..76e5732bbb6 --- /dev/null +++ b/test/payment/TokenEscrow.behavior.js @@ -0,0 +1,122 @@ +const { expectThrow } = require('../helpers/expectThrow'); +const { EVMRevert } = require('../helpers/EVMRevert'); +const expectEvent = require('../helpers/expectEvent'); + +const BigNumber = web3.BigNumber; + +require('chai') + .use(require('chai-bignumber')(BigNumber)) + .should(); + +function shouldBehaveLikeTokenEscrow (owner, [payee1, payee2]) { + const amount = new BigNumber(100); + const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); + + it('stores the token\'s address', async function () { + const address = await this.escrow.token(); + address.should.be.equal(this.token.address); + }); + + context('when not approved by payer', function () { + it('reverts on deposits', async function () { + await expectThrow( + this.escrow.deposit(payee1, amount, { from: owner }), + EVMRevert + ); + }); + }); + + context('when approved by payer', function () { + beforeEach(async function () { + this.token.approve(this.escrow.address, MAX_UINT256, { from: owner }); + }); + + describe('deposits', function () { + it('accepts a single deposit', async function () { + await this.escrow.deposit(payee1, amount, { from: owner }); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(amount); + + (await this.escrow.depositsOf(payee1)).should.be.bignumber.equal(amount); + }); + + it('accepts an empty deposit', async function () { + await this.escrow.deposit(payee1, new BigNumber(0), { from: owner }); + }); + + it('reverts when non-owners deposit', async function () { + await expectThrow(this.escrow.deposit(payee1, amount, { from: payee2 }), EVMRevert); + }); + + it('emits a deposited event', async function () { + const receipt = await this.escrow.deposit(payee1, amount, { from: owner }); + + const event = expectEvent.inLogs(receipt.logs, 'Deposited', { payee: payee1 }); + event.args.tokenAmount.should.be.bignumber.equal(amount); + }); + + it('adds multiple deposits on a single account', async function () { + await this.escrow.deposit(payee1, amount, { from: owner }); + await this.escrow.deposit(payee1, amount * 2, { from: owner }); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(amount * 3); + + (await this.escrow.depositsOf(payee1)).should.be.bignumber.equal(amount * 3); + }); + + it('tracks deposits to multiple accounts', async function () { + await this.escrow.deposit(payee1, amount, { from: owner }); + await this.escrow.deposit(payee2, amount * 2, { from: owner }); + + (await this.escrow.depositsOf(payee1)).should.be.bignumber.equal(amount); + (await this.escrow.depositsOf(payee2)).should.be.bignumber.equal(amount * 2); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(amount * 3); + }); + }); + + context('with deposit', function () { + beforeEach(async function () { + await this.escrow.deposit(payee1, amount, { from: owner }); + }); + + describe('withdrawals', function () { + it('withdraws payments', async function () { + const payeeInitialBalance = await this.token.balanceOf(payee1); + + await this.escrow.withdraw(payee1, { from: owner }); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(0); + + (await this.escrow.depositsOf(payee1)).should.be.bignumber.equal(0); + + const payeeFinalBalance = await this.token.balanceOf(payee1); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); + + it('accepts empty withdrawal', async function () { + await this.escrow.withdraw(payee1, { from: owner }); + + (await this.escrow.depositsOf(payee1)).should.be.bignumber.equal(0); + + await this.escrow.withdraw(payee1, { from: owner }); + }); + + it('reverts when non-owners withdraw', async function () { + await expectThrow(this.escrow.withdraw(payee1, { from: payee1 }), EVMRevert); + }); + + it('emits a withdrawn event', async function () { + const receipt = await this.escrow.withdraw(payee1, { from: owner }); + + const event = expectEvent.inLogs(receipt.logs, 'Withdrawn', { payee: payee1 }); + event.args.tokenAmount.should.be.bignumber.equal(amount); + }); + }); + }); + }); +} + +module.exports = { + shouldBehaveLikeTokenEscrow, +}; diff --git a/test/payment/TokenEscrow.test.js b/test/payment/TokenEscrow.test.js index 43738e93ad4..656a9935bf9 100644 --- a/test/payment/TokenEscrow.test.js +++ b/test/payment/TokenEscrow.test.js @@ -1,18 +1,13 @@ +const { shouldBehaveLikeTokenEscrow } = require('./TokenEscrow.behavior'); const { expectThrow } = require('../helpers/expectThrow'); const { EVMRevert } = require('../helpers/EVMRevert'); -const expectEvent = require('../helpers/expectEvent'); const BigNumber = web3.BigNumber; -require('chai') - .use(require('chai-bignumber')(BigNumber)) - .should(); - const TokenEscrow = artifacts.require('TokenEscrow'); const StandardToken = artifacts.require('StandardTokenMock'); -contract('TokenEscrow', function ([_, owner, payee1, payee2]) { - const amount = new BigNumber(100); +contract('TokenEscrow', function ([_, owner, ...otherAccounts]) { const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'; const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); @@ -25,111 +20,9 @@ contract('TokenEscrow', function ([_, owner, payee1, payee2]) { context('with token', function () { beforeEach(async function () { this.token = await StandardToken.new(owner, MAX_UINT256); - this.tokenEscrow = await TokenEscrow.new(this.token.address, { from: owner }); - }); - - it('stores the token\'s address', async function () { - const address = await this.tokenEscrow.token(); - address.should.be.equal(this.token.address); - }); - - context('when not approved by payer', function () { - it('reverts on deposits', async function () { - await expectThrow( - this.tokenEscrow.deposit(payee1, amount, { from: owner }), - EVMRevert - ); - }); + this.escrow = await TokenEscrow.new(this.token.address, { from: owner }); }); - context('when approved by payer', function () { - beforeEach(async function () { - this.token.approve(this.tokenEscrow.address, MAX_UINT256, { from: owner }); - }); - - describe('deposits', function () { - it('accepts a single deposit', async function () { - await this.tokenEscrow.deposit(payee1, amount, { from: owner }); - - (await this.token.balanceOf(this.tokenEscrow.address)).should.be.bignumber.equal(amount); - - (await this.tokenEscrow.depositsOf(payee1)).should.be.bignumber.equal(amount); - }); - - it('accepts an empty deposit', async function () { - await this.tokenEscrow.deposit(payee1, new BigNumber(0), { from: owner }); - }); - - it('reverts when non-owners deposit', async function () { - await expectThrow(this.tokenEscrow.deposit(payee1, amount, { from: payee2 }), EVMRevert); - }); - - it('emits a deposited event', async function () { - const receipt = await this.tokenEscrow.deposit(payee1, amount, { from: owner }); - - const event = expectEvent.inLogs(receipt.logs, 'Deposited', { payee: payee1 }); - event.args.tokenAmount.should.be.bignumber.equal(amount); - }); - - it('adds multiple deposits on a single account', async function () { - await this.tokenEscrow.deposit(payee1, amount, { from: owner }); - await this.tokenEscrow.deposit(payee1, amount * 2, { from: owner }); - - (await this.token.balanceOf(this.tokenEscrow.address)).should.be.bignumber.equal(amount * 3); - - (await this.tokenEscrow.depositsOf(payee1)).should.be.bignumber.equal(amount * 3); - }); - - it('tracks deposits to multiple accounts', async function () { - await this.tokenEscrow.deposit(payee1, amount, { from: owner }); - await this.tokenEscrow.deposit(payee2, amount * 2, { from: owner }); - - (await this.tokenEscrow.depositsOf(payee1)).should.be.bignumber.equal(amount); - (await this.tokenEscrow.depositsOf(payee2)).should.be.bignumber.equal(amount * 2); - - (await this.token.balanceOf(this.tokenEscrow.address)).should.be.bignumber.equal(amount * 3); - }); - }); - - context('with deposit', function () { - beforeEach(async function () { - await this.tokenEscrow.deposit(payee1, amount, { from: owner }); - }); - - describe('withdrawals', function () { - it('withdraws payments', async function () { - const payeeInitialBalance = await this.token.balanceOf(payee1); - - await this.tokenEscrow.withdraw(payee1, { from: owner }); - - (await this.token.balanceOf(this.tokenEscrow.address)).should.be.bignumber.equal(0); - - (await this.tokenEscrow.depositsOf(payee1)).should.be.bignumber.equal(0); - - const payeeFinalBalance = await this.token.balanceOf(payee1); - payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); - }); - - it('accepts empty withdrawal', async function () { - await this.tokenEscrow.withdraw(payee1, { from: owner }); - - (await this.tokenEscrow.depositsOf(payee1)).should.be.bignumber.equal(0); - - await this.tokenEscrow.withdraw(payee1, { from: owner }); - }); - - it('reverts when non-owners withdraw', async function () { - await expectThrow(this.tokenEscrow.withdraw(payee1, { from: payee1 }), EVMRevert); - }); - - it('emits a withdrawn event', async function () { - const receipt = await this.tokenEscrow.withdraw(payee1, { from: owner }); - - const event = expectEvent.inLogs(receipt.logs, 'Withdrawn', { payee: payee1 }); - event.args.tokenAmount.should.be.bignumber.equal(amount); - }); - }); - }); - }); + shouldBehaveLikeTokenEscrow(owner, otherAccounts); }); }); From 051daeac07b83c15529e8b936e55f601c24ec11b Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Wed, 29 Aug 2018 23:13:42 -0300 Subject: [PATCH 02/11] Add ConditionalTokenEscrow --- .../mocks/ConditionalTokenEscrowMock.sol | 20 +++++++ contracts/payment/ConditionalTokenEscrow.sol | 22 ++++++++ test/payment/ConditionalTokenEscrow.test.js | 55 +++++++++++++++++++ 3 files changed, 97 insertions(+) create mode 100644 contracts/mocks/ConditionalTokenEscrowMock.sol create mode 100644 contracts/payment/ConditionalTokenEscrow.sol create mode 100644 test/payment/ConditionalTokenEscrow.test.js diff --git a/contracts/mocks/ConditionalTokenEscrowMock.sol b/contracts/mocks/ConditionalTokenEscrowMock.sol new file mode 100644 index 00000000000..e7a0c83b2ce --- /dev/null +++ b/contracts/mocks/ConditionalTokenEscrowMock.sol @@ -0,0 +1,20 @@ +pragma solidity ^0.4.24; + +import "../payment/ConditionalTokenEscrow.sol"; +import "../token/ERC20/ERC20.sol"; + + +// mock class using ConditionalTokenEscrow +contract ConditionalTokenEscrowMock is ConditionalTokenEscrow { + mapping(address => bool) public allowed; + + constructor (ERC20 _token) public TokenEscrow(_token) { } + + function setAllowed(address _payee, bool _allowed) public { + allowed[_payee] = _allowed; + } + + function withdrawalAllowed(address _payee) public view returns (bool) { + return allowed[_payee]; + } +} diff --git a/contracts/payment/ConditionalTokenEscrow.sol b/contracts/payment/ConditionalTokenEscrow.sol new file mode 100644 index 00000000000..3fcbeb84e2a --- /dev/null +++ b/contracts/payment/ConditionalTokenEscrow.sol @@ -0,0 +1,22 @@ +pragma solidity ^0.4.24; + +import "./TokenEscrow.sol"; + + +/** + * @title ConditionalTokenEscrow + * @dev Base abstract escrow to only allow withdrawal if a condition is met. + */ +contract ConditionalTokenEscrow is TokenEscrow { + /** + * @dev Returns whether an address is allowed to withdraw their tokens. To be + * implemented by derived contracts. + * @param _payee The destination address of the tokens. + */ + function withdrawalAllowed(address _payee) public view returns (bool); + + function withdraw(address _payee) public { + require(withdrawalAllowed(_payee)); + super.withdraw(_payee); + } +} diff --git a/test/payment/ConditionalTokenEscrow.test.js b/test/payment/ConditionalTokenEscrow.test.js new file mode 100644 index 00000000000..0db0e2d7480 --- /dev/null +++ b/test/payment/ConditionalTokenEscrow.test.js @@ -0,0 +1,55 @@ +const { shouldBehaveLikeTokenEscrow } = require('./TokenEscrow.behavior'); +const { expectThrow } = require('../helpers/expectThrow'); +const { EVMRevert } = require('../helpers/EVMRevert'); + +const BigNumber = web3.BigNumber; + +require('chai') + .use(require('chai-bignumber')(BigNumber)) + .should(); + +const ConditionalTokenEscrow = artifacts.require('ConditionalTokenEscrowMock'); +const StandardToken = artifacts.require('StandardTokenMock'); + +contract('ConditionalTokenEscrow', function ([_, owner, payee, ...otherAccounts]) { + const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'; + const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); + + it('reverts when deployed with a null token address', async function () { + await expectThrow( + ConditionalTokenEscrow.new(ZERO_ADDRESS, { from: owner }), EVMRevert + ); + }); + + context('with token', function () { + beforeEach(async function () { + this.token = await StandardToken.new(owner, MAX_UINT256); + this.escrow = await ConditionalTokenEscrow.new(this.token.address, { from: owner }); + }); + + context('when withdrawal is allowed', function () { + beforeEach(async function () { + await Promise.all(otherAccounts.map( + payee => this.escrow.setAllowed(payee, true)) + ); + }); + + shouldBehaveLikeTokenEscrow(owner, otherAccounts); + }); + + context('when withdrawal is disallowed', function () { + const amount = web3.toWei(23.0, 'ether'); + + beforeEach(async function () { + await this.token.approve(this.escrow.address, MAX_UINT256, { from: owner }); + await this.escrow.setAllowed(payee, false); + }); + + it('reverts on withdrawals', async function () { + await this.escrow.deposit(payee, amount, { from: owner }); + + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); + }); + }); +}); From 0fe956bf09b58b52eab02507a866065580da4236 Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Wed, 29 Aug 2018 23:14:10 -0300 Subject: [PATCH 03/11] Add RefundTokenEscrow --- contracts/payment/RefundTokenEscrow.sol | 76 +++++++++++ test/payment/RefundTokenEscrow.test.js | 160 ++++++++++++++++++++++++ 2 files changed, 236 insertions(+) create mode 100644 contracts/payment/RefundTokenEscrow.sol create mode 100644 test/payment/RefundTokenEscrow.test.js diff --git a/contracts/payment/RefundTokenEscrow.sol b/contracts/payment/RefundTokenEscrow.sol new file mode 100644 index 00000000000..b9a026ae422 --- /dev/null +++ b/contracts/payment/RefundTokenEscrow.sol @@ -0,0 +1,76 @@ +pragma solidity ^0.4.24; + +import "./ConditionalTokenEscrow.sol"; +import "../token/ERC20/ERC20.sol"; + + +/** + * @title RefundEscrow + * @dev Escrow that holds funds for a beneficiary, deposited from multiple parties. + * The contract owner may close the deposit period, and allow for either withdrawal + * by the beneficiary, or refunds to the depositors. + */ +contract RefundTokenEscrow is ConditionalTokenEscrow { + + enum State { Active, Refunding, Closed } + + event Closed(); + event RefundsEnabled(); + + State public state; + address public beneficiary; + + /** + * @dev Constructor. + * @param _beneficiary The beneficiary of the deposits. + */ + constructor(ERC20 _token, address _beneficiary) public TokenEscrow(_token) { + require(_beneficiary != address(0)); + beneficiary = _beneficiary; + state = State.Active; + } + + /** + * @dev Stores funds that may later be refunded. + * @param _refundee The address funds will be sent to if a refund occurs. + */ + function deposit(address _refundee, uint256 _amount) public { + require(state == State.Active); + super.deposit(_refundee, _amount); + } + + /** + * @dev Allows for the beneficiary to withdraw their funds, rejecting + * further deposits. + */ + function close() public onlyOwner { + require(state == State.Active); + state = State.Closed; + emit Closed(); + } + + /** + * @dev Allows for refunds to take place, rejecting further deposits. + */ + function enableRefunds() public onlyOwner { + require(state == State.Active); + state = State.Refunding; + emit RefundsEnabled(); + } + + /** + * @dev Withdraws the beneficiary's funds. + */ + function beneficiaryWithdraw() public { + require(state == State.Closed); + uint256 amount = token.balanceOf(address(this)); + token.safeTransfer(beneficiary, amount); + } + + /** + * @dev Returns whether refundees can withdraw their deposits (be refunded). + */ + function withdrawalAllowed(address _payee) public view returns (bool) { + return state == State.Refunding; + } +} diff --git a/test/payment/RefundTokenEscrow.test.js b/test/payment/RefundTokenEscrow.test.js new file mode 100644 index 00000000000..cc904172ed9 --- /dev/null +++ b/test/payment/RefundTokenEscrow.test.js @@ -0,0 +1,160 @@ +const { expectThrow } = require('../helpers/expectThrow'); +const { EVMRevert } = require('../helpers/EVMRevert'); +const expectEvent = require('../helpers/expectEvent'); + +const BigNumber = web3.BigNumber; + +require('chai') + .use(require('chai-bignumber')(BigNumber)) + .should(); + +const RefundTokenEscrow = artifacts.require('RefundTokenEscrow'); +const StandardToken = artifacts.require('StandardTokenMock'); + +contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refundee2]) { + const amount = web3.toWei(54.0, 'ether'); + const refundees = [refundee1, refundee2]; + const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'; + const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); + + it('reverts when deployed with a null token address', async function () { + await expectThrow( + RefundTokenEscrow.new(ZERO_ADDRESS, beneficiary, { from: owner }) + ); + }); + + context('with token', function () { + beforeEach(async function () { + this.token = await StandardToken.new(owner, MAX_UINT256); + }); + + it('reverts when deployed with a null beneficiary', async function () { + await expectThrow( + RefundTokenEscrow.new(this.token.address, ZERO_ADDRESS, { from: owner }) + ); + }); + + context('once deployed', function () { + beforeEach(async function () { + this.escrow = await RefundTokenEscrow.new(this.token.address, beneficiary, { from: owner }); + this.token.approve(this.escrow.address, MAX_UINT256, { from: owner }); + }); + + context('active state', function () { + it('accepts deposits', async function () { + await this.escrow.deposit(refundee1, amount, { from: owner }); + + (await this.escrow.depositsOf(refundee1)).should.be.bignumber.equal(amount); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(amount); + }); + + it('reverts on refund attempts', async function () { + await this.escrow.deposit(refundee1, amount, { from: owner }); + await expectThrow(this.escrow.withdraw(refundee1), EVMRevert); + }); + + it('reverts on beneficiary withdrawal', async function () { + await this.escrow.deposit(refundee1, amount, { from: owner }); + await expectThrow(this.escrow.beneficiaryWithdraw(), EVMRevert); + }); + }); + + it('reverts when non-owners enter close state', async function () { + await expectThrow(this.escrow.close({ from: beneficiary }), EVMRevert); + }); + + it('emits Closed event when owner enters close state', async function () { + const receipt = await this.escrow.close({ from: owner }); + + expectEvent.inLogs(receipt.logs, 'Closed'); + }); + + context('closed state', function () { + beforeEach(async function () { + await Promise.all(refundees.map( + refundee => this.escrow.deposit(refundee, amount, { from: owner })) + ); + + await this.escrow.close({ from: owner }); + }); + + it('rejects deposits', async function () { + await expectThrow(this.escrow.deposit(refundee1, amount, { from: owner }), EVMRevert); + }); + + it('reverts on refund attempts', async function () { + await expectThrow(this.escrow.withdraw(refundee1), EVMRevert); + }); + + it('allows beneficiary withdrawal', async function () { + const beneficiaryInitialBalance = await this.token.balanceOf(beneficiary); + await this.escrow.beneficiaryWithdraw(); + const beneficiaryFinalBalance = await this.token.balanceOf(beneficiary); + + beneficiaryFinalBalance + .sub(beneficiaryInitialBalance) + .should.be.bignumber.equal(amount * refundees.length); + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(0); + }); + + it('reverts on entering the refund state', async function () { + await expectThrow(this.escrow.enableRefunds({ from: owner }), EVMRevert); + }); + + it('reverts on re-entering the closed state', async function () { + await expectThrow(this.escrow.close({ from: owner }), EVMRevert); + }); + }); + + it('reverts when non-owners enter refund state', async function () { + await expectThrow(this.escrow.enableRefunds({ from: beneficiary }), EVMRevert); + }); + + it('emits RefundsEnabled event when owner enters refund state', async function () { + const receipt = await this.escrow.enableRefunds({ from: owner }); + + expectEvent.inLogs(receipt.logs, 'RefundsEnabled'); + }); + + context('refund state', function () { + beforeEach(async function () { + await Promise.all(refundees.map( + refundee => this.escrow.deposit(refundee, amount, { from: owner })) + ); + + await this.escrow.enableRefunds({ from: owner }); + }); + + it('rejects deposits', async function () { + await expectThrow(this.escrow.deposit(refundee1, amount, { from: owner }), EVMRevert); + }); + + it('refunds refundees', async function () { + for (const refundee of [refundee1, refundee2]) { + const refundeeInitialBalance = await this.token.balanceOf(refundee); + await this.escrow.withdraw(refundee, { from: owner }); + const refundeeFinalBalance = await this.token.balanceOf(refundee); + + refundeeFinalBalance.sub(refundeeInitialBalance).should.be.bignumber.equal(amount); + } + + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(0); + }); + + it('reverts on beneficiary withdrawal', async function () { + await expectThrow(this.escrow.beneficiaryWithdraw(), EVMRevert); + }); + + it('reverts on entering the closed state', async function () { + await expectThrow(this.escrow.close({ from: owner }), EVMRevert); + }); + + it('reverts on re-entering the refund state', async function () { + await expectThrow(this.escrow.enableRefunds({ from: owner }), EVMRevert); + }); + }); + }); + }); +}); From 5cdfda9f0b254f113d22e658633958bdb24a8efb Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Thu, 30 Aug 2018 20:39:36 -0300 Subject: [PATCH 04/11] Add TimelockedEscrow --- contracts/payment/TimelockedEscrow.sol | 29 +++++++++++ test/payment/TimelockedEscrow.test.js | 66 ++++++++++++++++++++++++++ 2 files changed, 95 insertions(+) create mode 100644 contracts/payment/TimelockedEscrow.sol create mode 100644 test/payment/TimelockedEscrow.test.js diff --git a/contracts/payment/TimelockedEscrow.sol b/contracts/payment/TimelockedEscrow.sol new file mode 100644 index 00000000000..4ad16eccfe8 --- /dev/null +++ b/contracts/payment/TimelockedEscrow.sol @@ -0,0 +1,29 @@ +pragma solidity ^0.4.24; + +import "./ConditionalEscrow.sol"; + + +/** + * @title TimelockedEscrow + * @dev . + */ +contract TimelockedEscrow is ConditionalEscrow { + + uint256 public releaseTime; + + constructor (uint256 _releaseTime) public { + // solium-disable-next-line security/no-block-members + require(_releaseTime > block.timestamp); + releaseTime = _releaseTime; + } + + /** + * @dev Returns whether an address is allowed to withdraw their funds. + * @param _payee The destination address of the funds. + */ + function withdrawalAllowed(address _payee) public view returns (bool) { + // solium-disable-next-line security/no-block-members + require(block.timestamp >= releaseTime); + return true; + } +} diff --git a/test/payment/TimelockedEscrow.test.js b/test/payment/TimelockedEscrow.test.js new file mode 100644 index 00000000000..5082e763299 --- /dev/null +++ b/test/payment/TimelockedEscrow.test.js @@ -0,0 +1,66 @@ +const { expectThrow } = require('../helpers/expectThrow'); +const { EVMRevert } = require('../helpers/EVMRevert'); +const { latestTime } = require('../helpers/latestTime'); +const { increaseTimeTo, duration } = require('../helpers/increaseTime'); +const { ethGetBalance } = require('../helpers/web3'); + +const BigNumber = web3.BigNumber; + +require('chai') + .use(require('chai-bignumber')(BigNumber)) + .should(); + +const TimelockedEscrow = artifacts.require('TimelockedEscrow'); + +contract('TimelockedEscrow', function ([_, owner, payee, ...otherAccounts]) { + it('reverts when release time is in the past', async function () { + const releaseTime = (await latestTime()) - duration.minutes(1); + await expectThrow(TimelockedEscrow.new(releaseTime, { from: owner }), EVMRevert); + }); + + context('once deployed', function () { + const amount = web3.toWei(10.0, 'ether'); + + beforeEach(async function () { + this.releaseTime = (await latestTime()) + duration.years(1); + this.escrow = await TimelockedEscrow.new(this.releaseTime, { from: owner }); + await this.escrow.deposit(payee, { from: owner, value: amount }); + }); + + it('stores the release time', async function () { + const releaseTime = await this.escrow.releaseTime(); + releaseTime.should.be.bignumber.equal(this.releaseTime); + }); + + it('rejects withdrawals before release time', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); + + it('rejects withdrawals right before release time', async function () { + await increaseTimeTo(this.releaseTime - duration.seconds(5)); + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); + + it('allows withdrawals right after release time', async function () { + await increaseTimeTo(this.releaseTime + duration.seconds(5)); + + const payeeInitialBalance = await ethGetBalance(payee); + + await this.escrow.withdraw(payee, { from: owner }); + + const payeeFinalBalance = await ethGetBalance(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); + + it('allows withdrawals after release time', async function () { + await increaseTimeTo(this.releaseTime + duration.years(1)); + + const payeeInitialBalance = await ethGetBalance(payee); + + await this.escrow.withdraw(payee, { from: owner }); + + const payeeFinalBalance = await ethGetBalance(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); + }); +}); From 38fb51de1a2833e1038b05fa6801534093b5165d Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Thu, 30 Aug 2018 22:45:15 -0300 Subject: [PATCH 05/11] Add TimelockedTokenEscrow --- contracts/payment/TimelockedTokenEscrow.sol | 29 ++++++++ test/payment/TimelockedTokenEscrow.test.js | 80 +++++++++++++++++++++ 2 files changed, 109 insertions(+) create mode 100644 contracts/payment/TimelockedTokenEscrow.sol create mode 100644 test/payment/TimelockedTokenEscrow.test.js diff --git a/contracts/payment/TimelockedTokenEscrow.sol b/contracts/payment/TimelockedTokenEscrow.sol new file mode 100644 index 00000000000..af94ff86eb4 --- /dev/null +++ b/contracts/payment/TimelockedTokenEscrow.sol @@ -0,0 +1,29 @@ +pragma solidity ^0.4.24; + +import "./ConditionalTokenEscrow.sol"; + + +/** + * @title TimelockedTokenEscrow + * @dev . + */ +contract TimelockedTokenEscrow is ConditionalTokenEscrow { + + uint256 public releaseTime; + + constructor (ERC20 _token, uint256 _releaseTime) public TokenEscrow(_token) { + // solium-disable-next-line security/no-block-members + require(_releaseTime > block.timestamp); + releaseTime = _releaseTime; + } + + /** + * @dev Returns whether an address is allowed to withdraw their funds. + * @param _payee The destination address of the tokens. + */ + function withdrawalAllowed(address _payee) public view returns (bool) { + // solium-disable-next-line security/no-block-members + require(block.timestamp >= releaseTime); + return true; + } +} diff --git a/test/payment/TimelockedTokenEscrow.test.js b/test/payment/TimelockedTokenEscrow.test.js new file mode 100644 index 00000000000..6c24680cb17 --- /dev/null +++ b/test/payment/TimelockedTokenEscrow.test.js @@ -0,0 +1,80 @@ +const { expectThrow } = require('../helpers/expectThrow'); +const { EVMRevert } = require('../helpers/EVMRevert'); +const { latestTime } = require('../helpers/latestTime'); +const { increaseTimeTo, duration } = require('../helpers/increaseTime'); + +const BigNumber = web3.BigNumber; + +require('chai') + .use(require('chai-bignumber')(BigNumber)) + .should(); + +const TimelockedTokenEscrow = artifacts.require('TimelockedTokenEscrow'); +const StandardToken = artifacts.require('StandardTokenMock'); + +contract('TimelockedTokenEscrow', function ([_, owner, payee, ...otherAccounts]) { + const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); + + beforeEach(async function () { + this.token = await StandardToken.new(owner, MAX_UINT256); + }); + + it('reverts when release time is in the past', async function () { + const releaseTime = (await latestTime()) - duration.minutes(1); + await expectThrow( + TimelockedTokenEscrow.new(this.token.address, releaseTime, { from: owner }), + EVMRevert + ); + }); + + context('once deployed', function () { + const amount = new BigNumber(100); + + beforeEach(async function () { + this.releaseTime = (await latestTime()) + duration.years(1); + this.escrow = await TimelockedTokenEscrow.new( + this.token.address, + this.releaseTime, + { from: owner } + ); + await this.token.approve(this.escrow.address, MAX_UINT256, { from: owner }); + await this.escrow.deposit(payee, amount, { from: owner }); + }); + + it('stores the release time', async function () { + const releaseTime = await this.escrow.releaseTime(); + releaseTime.should.be.bignumber.equal(this.releaseTime); + }); + + it('rejects withdrawals before release time', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); + + it('rejects withdrawals right before release time', async function () { + await increaseTimeTo(this.releaseTime - duration.seconds(5)); + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); + + it('allows withdrawals right after release time', async function () { + await increaseTimeTo(this.releaseTime + duration.seconds(5)); + + const payeeInitialBalance = await this.token.balanceOf(payee); + + await this.escrow.withdraw(payee, { from: owner }); + + const payeeFinalBalance = await this.token.balanceOf(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); + + it('allows withdrawals after release time', async function () { + await increaseTimeTo(this.releaseTime + duration.years(1)); + + const payeeInitialBalance = await this.token.balanceOf(payee); + + await this.escrow.withdraw(payee, { from: owner }); + + const payeeFinalBalance = await this.token.balanceOf(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); + }); +}); From 47af090b3af01a3d6eeb7e44e7fcec424bc4045a Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Thu, 30 Aug 2018 23:11:43 -0300 Subject: [PATCH 06/11] Improve documentation of contracts * Updated docs of ConditionalTokenEscrow, RefundTokenEscrow, TimelockedEscrow and TimelockedTokenEscrow contracts --- contracts/payment/ConditionalTokenEscrow.sol | 7 ++++--- contracts/payment/RefundTokenEscrow.sol | 14 ++++++++------ contracts/payment/TimelockedEscrow.sol | 8 ++++++-- contracts/payment/TimelockedTokenEscrow.sol | 9 +++++++-- 4 files changed, 25 insertions(+), 13 deletions(-) diff --git a/contracts/payment/ConditionalTokenEscrow.sol b/contracts/payment/ConditionalTokenEscrow.sol index 3fcbeb84e2a..2f7ef29f446 100644 --- a/contracts/payment/ConditionalTokenEscrow.sol +++ b/contracts/payment/ConditionalTokenEscrow.sol @@ -5,12 +5,13 @@ import "./TokenEscrow.sol"; /** * @title ConditionalTokenEscrow - * @dev Base abstract escrow to only allow withdrawal if a condition is met. + * @dev Base abstract escrow to only allow withdrawal of tokens + * if a condition is met. */ contract ConditionalTokenEscrow is TokenEscrow { /** - * @dev Returns whether an address is allowed to withdraw their tokens. To be - * implemented by derived contracts. + * @dev Returns whether an address is allowed to withdraw their tokens. + * To be implemented by derived contracts. * @param _payee The destination address of the tokens. */ function withdrawalAllowed(address _payee) public view returns (bool); diff --git a/contracts/payment/RefundTokenEscrow.sol b/contracts/payment/RefundTokenEscrow.sol index b9a026ae422..3385dcfa964 100644 --- a/contracts/payment/RefundTokenEscrow.sol +++ b/contracts/payment/RefundTokenEscrow.sol @@ -5,8 +5,8 @@ import "../token/ERC20/ERC20.sol"; /** - * @title RefundEscrow - * @dev Escrow that holds funds for a beneficiary, deposited from multiple parties. + * @title RefundTokenEscrow + * @dev Escrow that holds tokens for a beneficiary, deposited from multiple parties. * The contract owner may close the deposit period, and allow for either withdrawal * by the beneficiary, or refunds to the depositors. */ @@ -22,6 +22,7 @@ contract RefundTokenEscrow is ConditionalTokenEscrow { /** * @dev Constructor. + * @param _token Address of the ERC20 token that will be put in escrow. * @param _beneficiary The beneficiary of the deposits. */ constructor(ERC20 _token, address _beneficiary) public TokenEscrow(_token) { @@ -31,8 +32,9 @@ contract RefundTokenEscrow is ConditionalTokenEscrow { } /** - * @dev Stores funds that may later be refunded. - * @param _refundee The address funds will be sent to if a refund occurs. + * @dev Stores tokens that may later be refunded. + * @param _refundee The address tokens will be sent to if a refund occurs. + * @param _amount The amount of tokens to store. */ function deposit(address _refundee, uint256 _amount) public { require(state == State.Active); @@ -40,7 +42,7 @@ contract RefundTokenEscrow is ConditionalTokenEscrow { } /** - * @dev Allows for the beneficiary to withdraw their funds, rejecting + * @dev Allows for the beneficiary to withdraw their tokens, rejecting * further deposits. */ function close() public onlyOwner { @@ -59,7 +61,7 @@ contract RefundTokenEscrow is ConditionalTokenEscrow { } /** - * @dev Withdraws the beneficiary's funds. + * @dev Withdraws the beneficiary's tokens. */ function beneficiaryWithdraw() public { require(state == State.Closed); diff --git a/contracts/payment/TimelockedEscrow.sol b/contracts/payment/TimelockedEscrow.sol index 4ad16eccfe8..4f296189f08 100644 --- a/contracts/payment/TimelockedEscrow.sol +++ b/contracts/payment/TimelockedEscrow.sol @@ -5,12 +5,17 @@ import "./ConditionalEscrow.sol"; /** * @title TimelockedEscrow - * @dev . + * @dev Escrow that holds funds for given amount of time, + * preventing their withdrawal until the time has passed. */ contract TimelockedEscrow is ConditionalEscrow { uint256 public releaseTime; + /** + * @dev Constructor. + * @param _releaseTime Time when the funds will be available for withdrawal. + */ constructor (uint256 _releaseTime) public { // solium-disable-next-line security/no-block-members require(_releaseTime > block.timestamp); @@ -19,7 +24,6 @@ contract TimelockedEscrow is ConditionalEscrow { /** * @dev Returns whether an address is allowed to withdraw their funds. - * @param _payee The destination address of the funds. */ function withdrawalAllowed(address _payee) public view returns (bool) { // solium-disable-next-line security/no-block-members diff --git a/contracts/payment/TimelockedTokenEscrow.sol b/contracts/payment/TimelockedTokenEscrow.sol index af94ff86eb4..69ad4e8eac6 100644 --- a/contracts/payment/TimelockedTokenEscrow.sol +++ b/contracts/payment/TimelockedTokenEscrow.sol @@ -5,12 +5,18 @@ import "./ConditionalTokenEscrow.sol"; /** * @title TimelockedTokenEscrow - * @dev . + * @dev Escrow that holds tokens for given amount of time, + * preventing their whithdrawal until the time has passed. */ contract TimelockedTokenEscrow is ConditionalTokenEscrow { uint256 public releaseTime; + /** + * @dev Constructor. + * @param _token Address of the ERC20 token that will be put in escrow. + * @param _releaseTime Time when the tokens will be available for withdrawal. + */ constructor (ERC20 _token, uint256 _releaseTime) public TokenEscrow(_token) { // solium-disable-next-line security/no-block-members require(_releaseTime > block.timestamp); @@ -19,7 +25,6 @@ contract TimelockedTokenEscrow is ConditionalTokenEscrow { /** * @dev Returns whether an address is allowed to withdraw their funds. - * @param _payee The destination address of the tokens. */ function withdrawalAllowed(address _payee) public view returns (bool) { // solium-disable-next-line security/no-block-members From 113d243156ae27787235e2d7ee9cf3520823c02d Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Sun, 2 Sep 2018 22:47:37 -0300 Subject: [PATCH 07/11] Update TimelockedEscrow and tests * Improved withdrawalAllowed to return the condition (removed require) * Specified units of releaseTime in a comment * Improved tests readability with context blocks --- contracts/payment/TimelockedEscrow.sol | 7 ++-- test/payment/TimelockedEscrow.test.js | 49 +++++++++++++++++--------- 2 files changed, 35 insertions(+), 21 deletions(-) diff --git a/contracts/payment/TimelockedEscrow.sol b/contracts/payment/TimelockedEscrow.sol index 4f296189f08..234a827c9a9 100644 --- a/contracts/payment/TimelockedEscrow.sol +++ b/contracts/payment/TimelockedEscrow.sol @@ -9,14 +9,14 @@ import "./ConditionalEscrow.sol"; * preventing their withdrawal until the time has passed. */ contract TimelockedEscrow is ConditionalEscrow { - + // In seconds since unix epoch uint256 public releaseTime; /** * @dev Constructor. * @param _releaseTime Time when the funds will be available for withdrawal. */ - constructor (uint256 _releaseTime) public { + constructor(uint256 _releaseTime) public { // solium-disable-next-line security/no-block-members require(_releaseTime > block.timestamp); releaseTime = _releaseTime; @@ -27,7 +27,6 @@ contract TimelockedEscrow is ConditionalEscrow { */ function withdrawalAllowed(address _payee) public view returns (bool) { // solium-disable-next-line security/no-block-members - require(block.timestamp >= releaseTime); - return true; + return block.timestamp >= releaseTime; } } diff --git a/test/payment/TimelockedEscrow.test.js b/test/payment/TimelockedEscrow.test.js index 5082e763299..72c7c685297 100644 --- a/test/payment/TimelockedEscrow.test.js +++ b/test/payment/TimelockedEscrow.test.js @@ -32,35 +32,50 @@ contract('TimelockedEscrow', function ([_, owner, payee, ...otherAccounts]) { releaseTime.should.be.bignumber.equal(this.releaseTime); }); - it('rejects withdrawals before release time', async function () { - await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + context('before release time', function () { + it('rejects withdrawals', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); }); - it('rejects withdrawals right before release time', async function () { - await increaseTimeTo(this.releaseTime - duration.seconds(5)); - await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + context('right before release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime - duration.seconds(5)); + }); + + it('rejects withdrawals', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); }); - it('allows withdrawals right after release time', async function () { - await increaseTimeTo(this.releaseTime + duration.seconds(5)); + context('right after release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime + duration.seconds(5)); + }); - const payeeInitialBalance = await ethGetBalance(payee); + it('allows withdrawals', async function () { + const payeeInitialBalance = await ethGetBalance(payee); - await this.escrow.withdraw(payee, { from: owner }); + await this.escrow.withdraw(payee, { from: owner }); - const payeeFinalBalance = await ethGetBalance(payee); - payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + const payeeFinalBalance = await ethGetBalance(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); }); - it('allows withdrawals after release time', async function () { - await increaseTimeTo(this.releaseTime + duration.years(1)); + context('after release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime + duration.years(1)); + }); - const payeeInitialBalance = await ethGetBalance(payee); + it('allows withdrawals', async function () { + const payeeInitialBalance = await ethGetBalance(payee); - await this.escrow.withdraw(payee, { from: owner }); + await this.escrow.withdraw(payee, { from: owner }); - const payeeFinalBalance = await ethGetBalance(payee); - payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + const payeeFinalBalance = await ethGetBalance(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); }); }); }); From 34be9ebeafc153dfb7ea6ab05e367a320ea1e35b Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Sun, 2 Sep 2018 22:51:12 -0300 Subject: [PATCH 08/11] Update TimelockedTokenEscrow and tests * Specified units in releaseTime as a comment * Improved tests readability with context blocks --- contracts/payment/TimelockedTokenEscrow.sol | 2 +- test/payment/TimelockedTokenEscrow.test.js | 49 ++++++++++++++------- 2 files changed, 33 insertions(+), 18 deletions(-) diff --git a/contracts/payment/TimelockedTokenEscrow.sol b/contracts/payment/TimelockedTokenEscrow.sol index 69ad4e8eac6..e586ddfa7a3 100644 --- a/contracts/payment/TimelockedTokenEscrow.sol +++ b/contracts/payment/TimelockedTokenEscrow.sol @@ -9,7 +9,7 @@ import "./ConditionalTokenEscrow.sol"; * preventing their whithdrawal until the time has passed. */ contract TimelockedTokenEscrow is ConditionalTokenEscrow { - + // In seconds since unix epoch uint256 public releaseTime; /** diff --git a/test/payment/TimelockedTokenEscrow.test.js b/test/payment/TimelockedTokenEscrow.test.js index 6c24680cb17..734519ae992 100644 --- a/test/payment/TimelockedTokenEscrow.test.js +++ b/test/payment/TimelockedTokenEscrow.test.js @@ -46,35 +46,50 @@ contract('TimelockedTokenEscrow', function ([_, owner, payee, ...otherAccounts]) releaseTime.should.be.bignumber.equal(this.releaseTime); }); - it('rejects withdrawals before release time', async function () { - await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + context('before release time', function () { + it('rejects withdrawals', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); }); - it('rejects withdrawals right before release time', async function () { - await increaseTimeTo(this.releaseTime - duration.seconds(5)); - await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + context('right before release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime - duration.seconds(5)); + }); + + it('rejects withdrawals', async function () { + await expectThrow(this.escrow.withdraw(payee, { from: owner }), EVMRevert); + }); }); - it('allows withdrawals right after release time', async function () { - await increaseTimeTo(this.releaseTime + duration.seconds(5)); + context('right after release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime + duration.seconds(5)); + }); - const payeeInitialBalance = await this.token.balanceOf(payee); + it('allows withdrawals', async function () { + const payeeInitialBalance = await this.token.balanceOf(payee); - await this.escrow.withdraw(payee, { from: owner }); + await this.escrow.withdraw(payee, { from: owner }); - const payeeFinalBalance = await this.token.balanceOf(payee); - payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + const payeeFinalBalance = await this.token.balanceOf(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); }); - it('allows withdrawals after release time', async function () { - await increaseTimeTo(this.releaseTime + duration.years(1)); + context('after release time', function () { + beforeEach(async function () { + await increaseTimeTo(this.releaseTime + duration.years(1)); + }); - const payeeInitialBalance = await this.token.balanceOf(payee); + it('allows withdrawals', async function () { + const payeeInitialBalance = await this.token.balanceOf(payee); - await this.escrow.withdraw(payee, { from: owner }); + await this.escrow.withdraw(payee, { from: owner }); - const payeeFinalBalance = await this.token.balanceOf(payee); - payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + const payeeFinalBalance = await this.token.balanceOf(payee); + payeeFinalBalance.sub(payeeInitialBalance).should.be.bignumber.equal(amount); + }); }); }); }); From 2350782c61add05066f50818ff33601897c5d68b Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Sun, 2 Sep 2018 22:53:13 -0300 Subject: [PATCH 09/11] Update RefundTokenEscrow and tests * Fixed contract formatting * Improved tests readability with describe blocks * Added tests for the state variable --- contracts/payment/RefundTokenEscrow.sol | 1 - test/payment/RefundTokenEscrow.test.js | 52 ++++++++++++++++++------- 2 files changed, 38 insertions(+), 15 deletions(-) diff --git a/contracts/payment/RefundTokenEscrow.sol b/contracts/payment/RefundTokenEscrow.sol index 3385dcfa964..f09bf6502e0 100644 --- a/contracts/payment/RefundTokenEscrow.sol +++ b/contracts/payment/RefundTokenEscrow.sol @@ -11,7 +11,6 @@ import "../token/ERC20/ERC20.sol"; * by the beneficiary, or refunds to the depositors. */ contract RefundTokenEscrow is ConditionalTokenEscrow { - enum State { Active, Refunding, Closed } event Closed(); diff --git a/test/payment/RefundTokenEscrow.test.js b/test/payment/RefundTokenEscrow.test.js index cc904172ed9..cfd3988f5dd 100644 --- a/test/payment/RefundTokenEscrow.test.js +++ b/test/payment/RefundTokenEscrow.test.js @@ -17,6 +17,12 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'; const MAX_UINT256 = new BigNumber(2).pow(256).minus(1); + const RefundStates = { + 'Active': 0, + 'Refunding': 1, + 'Closed': 2, + }; + it('reverts when deployed with a null token address', async function () { await expectThrow( RefundTokenEscrow.new(ZERO_ADDRESS, beneficiary, { from: owner }) @@ -40,6 +46,14 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun this.token.approve(this.escrow.address, MAX_UINT256, { from: owner }); }); + it('is in active state', async function () { + (await this.escrow.state()).should.be.bignumber.equal(RefundStates.Active); + }); + + it('stores the beneficiary', async function () { + (await this.escrow.beneficiary()).should.be.equal(beneficiary); + }); + context('active state', function () { it('accepts deposits', async function () { await this.escrow.deposit(refundee1, amount, { from: owner }); @@ -51,7 +65,7 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun it('reverts on refund attempts', async function () { await this.escrow.deposit(refundee1, amount, { from: owner }); - await expectThrow(this.escrow.withdraw(refundee1), EVMRevert); + await expectThrow(this.escrow.withdraw(refundee1, { from: owner }), EVMRevert); }); it('reverts on beneficiary withdrawal', async function () { @@ -60,14 +74,18 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun }); }); - it('reverts when non-owners enter close state', async function () { - await expectThrow(this.escrow.close({ from: beneficiary }), EVMRevert); - }); + describe('entering close state', function () { + it('reverts when non-owners enter close state', async function () { + await expectThrow(this.escrow.close({ from: beneficiary }), EVMRevert); + }); - it('emits Closed event when owner enters close state', async function () { - const receipt = await this.escrow.close({ from: owner }); + it('emits Closed event when owner enters close state', async function () { + const receipt = await this.escrow.close({ from: owner }); - expectEvent.inLogs(receipt.logs, 'Closed'); + (await this.escrow.state()).should.be.bignumber.equal(RefundStates.Closed); + + expectEvent.inLogs(receipt.logs, 'Closed'); + }); }); context('closed state', function () { @@ -108,14 +126,18 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun }); }); - it('reverts when non-owners enter refund state', async function () { - await expectThrow(this.escrow.enableRefunds({ from: beneficiary }), EVMRevert); - }); + describe('entering refund state', function () { + it('reverts when non-owners enter refund state', async function () { + await expectThrow(this.escrow.enableRefunds({ from: beneficiary }), EVMRevert); + }); - it('emits RefundsEnabled event when owner enters refund state', async function () { - const receipt = await this.escrow.enableRefunds({ from: owner }); + it('emits RefundsEnabled event when owner enters refund state', async function () { + const receipt = await this.escrow.enableRefunds({ from: owner }); - expectEvent.inLogs(receipt.logs, 'RefundsEnabled'); + (await this.escrow.state()).should.be.bignumber.equal(RefundStates.Refunding); + + expectEvent.inLogs(receipt.logs, 'RefundsEnabled'); + }); }); context('refund state', function () { @@ -132,7 +154,7 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun }); it('refunds refundees', async function () { - for (const refundee of [refundee1, refundee2]) { + for (const refundee of refundees) { const refundeeInitialBalance = await this.token.balanceOf(refundee); await this.escrow.withdraw(refundee, { from: owner }); const refundeeFinalBalance = await this.token.balanceOf(refundee); @@ -140,6 +162,8 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun refundeeFinalBalance.sub(refundeeInitialBalance).should.be.bignumber.equal(amount); } + await this.escrow.withdraw(refundee1, { from: owner }); + (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(0); }); From 5191abe2ccbbd2b14cceacbe720c94687a82234c Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Sun, 2 Sep 2018 22:56:37 -0300 Subject: [PATCH 10/11] Improved naming of contract mock variable --- test/payment/ConditionalTokenEscrow.test.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/test/payment/ConditionalTokenEscrow.test.js b/test/payment/ConditionalTokenEscrow.test.js index 0db0e2d7480..11268e1b318 100644 --- a/test/payment/ConditionalTokenEscrow.test.js +++ b/test/payment/ConditionalTokenEscrow.test.js @@ -8,7 +8,7 @@ require('chai') .use(require('chai-bignumber')(BigNumber)) .should(); -const ConditionalTokenEscrow = artifacts.require('ConditionalTokenEscrowMock'); +const ConditionalTokenEscrowMock = artifacts.require('ConditionalTokenEscrowMock'); const StandardToken = artifacts.require('StandardTokenMock'); contract('ConditionalTokenEscrow', function ([_, owner, payee, ...otherAccounts]) { @@ -17,14 +17,14 @@ contract('ConditionalTokenEscrow', function ([_, owner, payee, ...otherAccounts] it('reverts when deployed with a null token address', async function () { await expectThrow( - ConditionalTokenEscrow.new(ZERO_ADDRESS, { from: owner }), EVMRevert + ConditionalTokenEscrowMock.new(ZERO_ADDRESS, { from: owner }), EVMRevert ); }); context('with token', function () { beforeEach(async function () { this.token = await StandardToken.new(owner, MAX_UINT256); - this.escrow = await ConditionalTokenEscrow.new(this.token.address, { from: owner }); + this.escrow = await ConditionalTokenEscrowMock.new(this.token.address, { from: owner }); }); context('when withdrawal is allowed', function () { From e67847f7693b6c338edd26ae2fdac1ef1459714b Mon Sep 17 00:00:00 2001 From: Martin Abbatemarco Date: Mon, 3 Sep 2018 11:09:38 -0300 Subject: [PATCH 11/11] Remove extra withdrawal in RefundTokenEscrow test --- test/payment/RefundTokenEscrow.test.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/test/payment/RefundTokenEscrow.test.js b/test/payment/RefundTokenEscrow.test.js index cfd3988f5dd..4078ce33866 100644 --- a/test/payment/RefundTokenEscrow.test.js +++ b/test/payment/RefundTokenEscrow.test.js @@ -162,8 +162,6 @@ contract('RefundTokenEscrow', function ([_, owner, beneficiary, refundee1, refun refundeeFinalBalance.sub(refundeeInitialBalance).should.be.bignumber.equal(amount); } - await this.escrow.withdraw(refundee1, { from: owner }); - (await this.token.balanceOf(this.escrow.address)).should.be.bignumber.equal(0); });