| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315 | 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 = artifacts.require('ERC20Mock');const ERC20DecimalsMock = artifacts.require('ERC20DecimalsMock');contract('ERC20', function (accounts) {  const [ initialHolder, recipient, anotherAccount ] = accounts;  const name = 'My Token';  const symbol = 'MTKN';  const initialSupply = new BN(100);  beforeEach(async function () {    this.token = await ERC20Mock.new(name, symbol, initialHolder, initialSupply);  });  it('has a name', async function () {    expect(await this.token.name()).to.equal(name);  });  it('has a symbol', async function () {    expect(await this.token.symbol()).to.equal(symbol);  });  it('has 18 decimals', async function () {    expect(await this.token.decimals()).to.be.bignumber.equal('18');  });  describe('set decimals', function () {    const decimals = new BN(6);    it('can set decimals during construction', async function () {      const token = await ERC20DecimalsMock.new(name, symbol, decimals);      expect(await token.decimals()).to.be.bignumber.equal(decimals);    });  });  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('_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',        );      });    });  });});
 |