123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366 |
- const { accounts, contract } = require('@openzeppelin/test-environment');
- const { BN, constants, expectEvent, expectRevert } = require('@openzeppelin/test-helpers');
- const { expect } = require('chai');
- const { ZERO_ADDRESS } = constants;
- const {
- shouldBehaveLikeERC20,
- shouldBehaveLikeERC20Transfer,
- shouldBehaveLikeERC20Approve,
- } = require('./ERC20.behavior');
- const ERC20Mock = contract.fromArtifact('ERC20Mock');
- describe('ERC20', function () {
- const [ initialHolder, recipient, anotherAccount ] = accounts;
- const initialSupply = new BN(100);
- beforeEach(async function () {
- this.token = await ERC20Mock.new(initialHolder, initialSupply);
- });
- shouldBehaveLikeERC20('ERC20', initialSupply, initialHolder, recipient, anotherAccount);
- describe('decrease allowance', function () {
- describe('when the spender is not the zero address', function () {
- const spender = recipient;
- function shouldDecreaseApproval (amount) {
- describe('when there was no approved amount before', function () {
- it('reverts', async function () {
- await expectRevert(this.token.decreaseAllowance(
- spender, amount, { from: initialHolder }), 'ERC20: decreased allowance below zero'
- );
- });
- });
- describe('when the spender had an approved amount', function () {
- const approvedAmount = amount;
- beforeEach(async function () {
- ({ logs: this.logs } = await this.token.approve(spender, approvedAmount, { from: initialHolder }));
- });
- it('emits an approval event', async function () {
- const { logs } = await this.token.decreaseAllowance(spender, approvedAmount, { from: initialHolder });
- expectEvent.inLogs(logs, 'Approval', {
- owner: initialHolder,
- spender: spender,
- value: new BN(0),
- });
- });
- it('decreases the spender allowance subtracting the requested amount', async function () {
- await this.token.decreaseAllowance(spender, approvedAmount.subn(1), { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal('1');
- });
- it('sets the allowance to zero when all allowance is removed', async function () {
- await this.token.decreaseAllowance(spender, approvedAmount, { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal('0');
- });
- it('reverts when more than the full allowance is removed', async function () {
- await expectRevert(
- this.token.decreaseAllowance(spender, approvedAmount.addn(1), { from: initialHolder }),
- 'ERC20: decreased allowance below zero'
- );
- });
- });
- }
- describe('when the sender has enough balance', function () {
- const amount = initialSupply;
- shouldDecreaseApproval(amount);
- });
- describe('when the sender does not have enough balance', function () {
- const amount = initialSupply.addn(1);
- shouldDecreaseApproval(amount);
- });
- });
- describe('when the spender is the zero address', function () {
- const amount = initialSupply;
- const spender = ZERO_ADDRESS;
- it('reverts', async function () {
- await expectRevert(this.token.decreaseAllowance(
- spender, amount, { from: initialHolder }), 'ERC20: decreased allowance below zero'
- );
- });
- });
- });
- describe('increase allowance', function () {
- const amount = initialSupply;
- describe('when the spender is not the zero address', function () {
- const spender = recipient;
- describe('when the sender has enough balance', function () {
- it('emits an approval event', async function () {
- const { logs } = await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expectEvent.inLogs(logs, 'Approval', {
- owner: initialHolder,
- spender: spender,
- value: amount,
- });
- });
- describe('when there was no approved amount before', function () {
- it('approves the requested amount', async function () {
- await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal(amount);
- });
- });
- describe('when the spender had an approved amount', function () {
- beforeEach(async function () {
- await this.token.approve(spender, new BN(1), { from: initialHolder });
- });
- it('increases the spender allowance adding the requested amount', async function () {
- await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal(amount.addn(1));
- });
- });
- });
- describe('when the sender does not have enough balance', function () {
- const amount = initialSupply.addn(1);
- it('emits an approval event', async function () {
- const { logs } = await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expectEvent.inLogs(logs, 'Approval', {
- owner: initialHolder,
- spender: spender,
- value: amount,
- });
- });
- describe('when there was no approved amount before', function () {
- it('approves the requested amount', async function () {
- await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal(amount);
- });
- });
- describe('when the spender had an approved amount', function () {
- beforeEach(async function () {
- await this.token.approve(spender, new BN(1), { from: initialHolder });
- });
- it('increases the spender allowance adding the requested amount', async function () {
- await this.token.increaseAllowance(spender, amount, { from: initialHolder });
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal(amount.addn(1));
- });
- });
- });
- });
- describe('when the spender is the zero address', function () {
- const spender = ZERO_ADDRESS;
- it('reverts', async function () {
- await expectRevert(
- this.token.increaseAllowance(spender, amount, { from: initialHolder }), 'ERC20: approve to the zero address'
- );
- });
- });
- });
- describe('_mint', function () {
- const amount = new BN(50);
- it('rejects a null account', async function () {
- await expectRevert(
- this.token.mint(ZERO_ADDRESS, amount), 'ERC20: mint to the zero address'
- );
- });
- describe('for a non zero account', function () {
- beforeEach('minting', async function () {
- const { logs } = await this.token.mint(recipient, amount);
- this.logs = logs;
- });
- it('increments totalSupply', async function () {
- const expectedSupply = initialSupply.add(amount);
- expect(await this.token.totalSupply()).to.be.bignumber.equal(expectedSupply);
- });
- it('increments recipient balance', async function () {
- expect(await this.token.balanceOf(recipient)).to.be.bignumber.equal(amount);
- });
- it('emits Transfer event', async function () {
- const event = expectEvent.inLogs(this.logs, 'Transfer', {
- from: ZERO_ADDRESS,
- to: recipient,
- });
- expect(event.args.value).to.be.bignumber.equal(amount);
- });
- });
- });
- describe('_burn', function () {
- it('rejects a null account', async function () {
- await expectRevert(this.token.burn(ZERO_ADDRESS, new BN(1)),
- 'ERC20: burn from the zero address');
- });
- describe('for a non zero account', function () {
- it('rejects burning more than balance', async function () {
- await expectRevert(this.token.burn(
- initialHolder, initialSupply.addn(1)), 'ERC20: burn amount exceeds balance'
- );
- });
- const describeBurn = function (description, amount) {
- describe(description, function () {
- beforeEach('burning', async function () {
- const { logs } = await this.token.burn(initialHolder, amount);
- this.logs = logs;
- });
- it('decrements totalSupply', async function () {
- const expectedSupply = initialSupply.sub(amount);
- expect(await this.token.totalSupply()).to.be.bignumber.equal(expectedSupply);
- });
- it('decrements initialHolder balance', async function () {
- const expectedBalance = initialSupply.sub(amount);
- expect(await this.token.balanceOf(initialHolder)).to.be.bignumber.equal(expectedBalance);
- });
- it('emits Transfer event', async function () {
- const event = expectEvent.inLogs(this.logs, 'Transfer', {
- from: initialHolder,
- to: ZERO_ADDRESS,
- });
- expect(event.args.value).to.be.bignumber.equal(amount);
- });
- });
- };
- describeBurn('for entire balance', initialSupply);
- describeBurn('for less amount than balance', initialSupply.subn(1));
- });
- });
- describe('_burnFrom', function () {
- const allowance = new BN(70);
- const spender = anotherAccount;
- beforeEach('approving', async function () {
- await this.token.approve(spender, allowance, { from: initialHolder });
- });
- it('rejects a null account', async function () {
- await expectRevert(this.token.burnFrom(ZERO_ADDRESS, new BN(1)),
- 'ERC20: burn from the zero address'
- );
- });
- describe('for a non zero account', function () {
- it('rejects burning more than allowance', async function () {
- await expectRevert(this.token.burnFrom(initialHolder, allowance.addn(1)),
- 'ERC20: burn amount exceeds allowance'
- );
- });
- it('rejects burning more than balance', async function () {
- await expectRevert(this.token.burnFrom(initialHolder, initialSupply.addn(1)),
- 'ERC20: burn amount exceeds balance'
- );
- });
- const describeBurnFrom = function (description, amount) {
- describe(description, function () {
- beforeEach('burning', async function () {
- const { logs } = await this.token.burnFrom(initialHolder, amount, { from: spender });
- this.logs = logs;
- });
- it('decrements totalSupply', async function () {
- const expectedSupply = initialSupply.sub(amount);
- expect(await this.token.totalSupply()).to.be.bignumber.equal(expectedSupply);
- });
- it('decrements initialHolder balance', async function () {
- const expectedBalance = initialSupply.sub(amount);
- expect(await this.token.balanceOf(initialHolder)).to.be.bignumber.equal(expectedBalance);
- });
- it('decrements spender allowance', async function () {
- const expectedAllowance = allowance.sub(amount);
- expect(await this.token.allowance(initialHolder, spender)).to.be.bignumber.equal(expectedAllowance);
- });
- it('emits a Transfer event', async function () {
- const event = expectEvent.inLogs(this.logs, 'Transfer', {
- from: initialHolder,
- to: ZERO_ADDRESS,
- });
- expect(event.args.value).to.be.bignumber.equal(amount);
- });
- it('emits an Approval event', async function () {
- expectEvent.inLogs(this.logs, 'Approval', {
- owner: initialHolder,
- spender: spender,
- value: await this.token.allowance(initialHolder, spender),
- });
- });
- });
- };
- describeBurnFrom('for entire allowance', allowance);
- describeBurnFrom('for less amount than allowance', allowance.subn(1));
- });
- });
- describe('_transfer', function () {
- shouldBehaveLikeERC20Transfer('ERC20', initialHolder, recipient, initialSupply, function (from, to, amount) {
- return this.token.transferInternal(from, to, amount);
- });
- describe('when the sender is the zero address', function () {
- it('reverts', async function () {
- await expectRevert(this.token.transferInternal(ZERO_ADDRESS, recipient, initialSupply),
- 'ERC20: transfer from the zero address'
- );
- });
- });
- });
- describe('_approve', function () {
- shouldBehaveLikeERC20Approve('ERC20', initialHolder, recipient, initialSupply, function (owner, spender, amount) {
- return this.token.approveInternal(owner, spender, amount);
- });
- describe('when the owner is the zero address', function () {
- it('reverts', async function () {
- await expectRevert(this.token.approveInternal(ZERO_ADDRESS, recipient, initialSupply),
- 'ERC20: approve from the zero address'
- );
- });
- });
- });
- });
|