PaymentSplitter.test.js 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. const { balance, constants, ether, expectEvent, send, expectRevert } = require('@openzeppelin/test-helpers');
  2. const { ZERO_ADDRESS } = constants;
  3. const { expect } = require('chai');
  4. const PaymentSplitter = artifacts.require('PaymentSplitter');
  5. contract('PaymentSplitter', function (accounts) {
  6. const [ owner, payee1, payee2, payee3, nonpayee1, payer1 ] = accounts;
  7. const amount = ether('1');
  8. it('rejects an empty set of payees', async function () {
  9. await expectRevert(PaymentSplitter.new([], []), 'PaymentSplitter: no payees');
  10. });
  11. it('rejects more payees than shares', async function () {
  12. await expectRevert(PaymentSplitter.new([payee1, payee2, payee3], [20, 30]),
  13. 'PaymentSplitter: payees and shares length mismatch',
  14. );
  15. });
  16. it('rejects more shares than payees', async function () {
  17. await expectRevert(PaymentSplitter.new([payee1, payee2], [20, 30, 40]),
  18. 'PaymentSplitter: payees and shares length mismatch',
  19. );
  20. });
  21. it('rejects null payees', async function () {
  22. await expectRevert(PaymentSplitter.new([payee1, ZERO_ADDRESS], [20, 30]),
  23. 'PaymentSplitter: account is the zero address',
  24. );
  25. });
  26. it('rejects zero-valued shares', async function () {
  27. await expectRevert(PaymentSplitter.new([payee1, payee2], [20, 0]),
  28. 'PaymentSplitter: shares are 0',
  29. );
  30. });
  31. it('rejects repeated payees', async function () {
  32. await expectRevert(PaymentSplitter.new([payee1, payee1], [20, 30]),
  33. 'PaymentSplitter: account already has shares',
  34. );
  35. });
  36. context('once deployed', function () {
  37. beforeEach(async function () {
  38. this.payees = [payee1, payee2, payee3];
  39. this.shares = [20, 10, 70];
  40. this.contract = await PaymentSplitter.new(this.payees, this.shares);
  41. });
  42. it('has total shares', async function () {
  43. expect(await this.contract.totalShares()).to.be.bignumber.equal('100');
  44. });
  45. it('has payees', async function () {
  46. await Promise.all(this.payees.map(async (payee, index) => {
  47. expect(await this.contract.payee(index)).to.equal(payee);
  48. expect(await this.contract.released(payee)).to.be.bignumber.equal('0');
  49. }));
  50. });
  51. it('accepts payments', async function () {
  52. await send.ether(owner, this.contract.address, amount);
  53. expect(await balance.current(this.contract.address)).to.be.bignumber.equal(amount);
  54. });
  55. describe('shares', async function () {
  56. it('stores shares if address is payee', async function () {
  57. expect(await this.contract.shares(payee1)).to.be.bignumber.not.equal('0');
  58. });
  59. it('does not store shares if address is not payee', async function () {
  60. expect(await this.contract.shares(nonpayee1)).to.be.bignumber.equal('0');
  61. });
  62. });
  63. describe('release', async function () {
  64. it('reverts if no funds to claim', async function () {
  65. await expectRevert(this.contract.release(payee1),
  66. 'PaymentSplitter: account is not due payment',
  67. );
  68. });
  69. it('reverts if non-payee want to claim', async function () {
  70. await send.ether(payer1, this.contract.address, amount);
  71. await expectRevert(this.contract.release(nonpayee1),
  72. 'PaymentSplitter: account has no shares',
  73. );
  74. });
  75. });
  76. it('distributes funds to payees', async function () {
  77. await send.ether(payer1, this.contract.address, amount);
  78. // receive funds
  79. const initBalance = await balance.current(this.contract.address);
  80. expect(initBalance).to.be.bignumber.equal(amount);
  81. // distribute to payees
  82. const initAmount1 = await balance.current(payee1);
  83. const { logs: logs1 } = await this.contract.release(payee1, { gasPrice: 0 });
  84. const profit1 = (await balance.current(payee1)).sub(initAmount1);
  85. expect(profit1).to.be.bignumber.equal(ether('0.20'));
  86. expectEvent.inLogs(logs1, 'PaymentReleased', { to: payee1, amount: profit1 });
  87. const initAmount2 = await balance.current(payee2);
  88. const { logs: logs2 } = await this.contract.release(payee2, { gasPrice: 0 });
  89. const profit2 = (await balance.current(payee2)).sub(initAmount2);
  90. expect(profit2).to.be.bignumber.equal(ether('0.10'));
  91. expectEvent.inLogs(logs2, 'PaymentReleased', { to: payee2, amount: profit2 });
  92. const initAmount3 = await balance.current(payee3);
  93. const { logs: logs3 } = await this.contract.release(payee3, { gasPrice: 0 });
  94. const profit3 = (await balance.current(payee3)).sub(initAmount3);
  95. expect(profit3).to.be.bignumber.equal(ether('0.70'));
  96. expectEvent.inLogs(logs3, 'PaymentReleased', { to: payee3, amount: profit3 });
  97. // end balance should be zero
  98. expect(await balance.current(this.contract.address)).to.be.bignumber.equal('0');
  99. // check correct funds released accounting
  100. expect(await this.contract.totalReleased()).to.be.bignumber.equal(initBalance);
  101. });
  102. });
  103. });