| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 | const { constants, expectRevert } = require('@openzeppelin/test-helpers');const ERC20ReturnFalseMock = artifacts.require('ERC20ReturnFalseMock');const ERC20ReturnTrueMock = artifacts.require('ERC20ReturnTrueMock');const ERC20NoReturnMock = artifacts.require('ERC20NoReturnMock');const ERC20PermitNoRevertMock = artifacts.require('ERC20PermitNoRevertMock');const SafeERC20Wrapper = artifacts.require('SafeERC20Wrapper');const { EIP712Domain, Permit } = require('../../../helpers/eip712');const { fromRpcSig } = require('ethereumjs-util');const ethSigUtil = require('eth-sig-util');const Wallet = require('ethereumjs-wallet').default;contract('SafeERC20', function (accounts) {  const [ hasNoCode ] = accounts;  describe('with address that has no contract code', function () {    beforeEach(async function () {      this.wrapper = await SafeERC20Wrapper.new(hasNoCode);    });    shouldRevertOnAllCalls('Address: call to non-contract');  });  describe('with token that returns false on all calls', function () {    beforeEach(async function () {      this.wrapper = await SafeERC20Wrapper.new((await ERC20ReturnFalseMock.new()).address);    });    shouldRevertOnAllCalls('SafeERC20: ERC20 operation did not succeed');  });  describe('with token that returns true on all calls', function () {    beforeEach(async function () {      this.wrapper = await SafeERC20Wrapper.new((await ERC20ReturnTrueMock.new()).address);    });    shouldOnlyRevertOnErrors();  });  describe('with token that returns no boolean values', function () {    beforeEach(async function () {      this.wrapper = await SafeERC20Wrapper.new((await ERC20NoReturnMock.new()).address);    });    shouldOnlyRevertOnErrors();  });  describe('with token that doesn\'t revert on invalid permit', function () {    const wallet = Wallet.generate();    const owner = wallet.getAddressString();    const spender = hasNoCode;    beforeEach(async function () {      this.token = await ERC20PermitNoRevertMock.new();      this.wrapper = await SafeERC20Wrapper.new(this.token.address);      const chainId = await this.token.getChainId();      this.data = {        primaryType: 'Permit',        types: { EIP712Domain, Permit },        domain: { name: 'ERC20PermitNoRevertMock', version: '1', chainId, verifyingContract: this.token.address },        message: { owner, spender, value: '42', nonce: '0', deadline: constants.MAX_UINT256 },      };      this.signature = fromRpcSig(ethSigUtil.signTypedMessage(wallet.getPrivateKey(), { data: this.data }));    });    it('accepts owner signature', async function () {      expect(await this.token.nonces(owner)).to.be.bignumber.equal('0');      expect(await this.token.allowance(owner, spender)).to.be.bignumber.equal('0');      await this.wrapper.permit(        this.data.message.owner,        this.data.message.spender,        this.data.message.value,        this.data.message.deadline,        this.signature.v,        this.signature.r,        this.signature.s,      );      expect(await this.token.nonces(owner)).to.be.bignumber.equal('1');      expect(await this.token.allowance(owner, spender)).to.be.bignumber.equal(this.data.message.value);    });    it('revert on reused signature', async function () {      expect(await this.token.nonces(owner)).to.be.bignumber.equal('0');      // use valid signature and consume nounce      await this.wrapper.permit(        this.data.message.owner,        this.data.message.spender,        this.data.message.value,        this.data.message.deadline,        this.signature.v,        this.signature.r,        this.signature.s,      );      expect(await this.token.nonces(owner)).to.be.bignumber.equal('1');      // invalid call does not revert for this token implementation      await this.token.permit(        this.data.message.owner,        this.data.message.spender,        this.data.message.value,        this.data.message.deadline,        this.signature.v,        this.signature.r,        this.signature.s,      );      expect(await this.token.nonces(owner)).to.be.bignumber.equal('1');      // invalid call revert when called through the SafeERC20 library      await expectRevert(        this.wrapper.permit(          this.data.message.owner,          this.data.message.spender,          this.data.message.value,          this.data.message.deadline,          this.signature.v,          this.signature.r,          this.signature.s,        ),        'SafeERC20: permit did not succeed',      );      expect(await this.token.nonces(owner)).to.be.bignumber.equal('1');    });    it('revert on invalid signature', async function () {      // signature that is not valid for owner      const invalidSignature = {        v: 27,        r: '0x71753dc5ecb5b4bfc0e3bc530d79ce5988760ed3f3a234c86a5546491f540775',        s: '0x0049cedee5aed990aabed5ad6a9f6e3c565b63379894b5fa8b512eb2b79e485d',      };      // invalid call does not revert for this token implementation      await this.token.permit(        this.data.message.owner,        this.data.message.spender,        this.data.message.value,        this.data.message.deadline,        invalidSignature.v,        invalidSignature.r,        invalidSignature.s,      );      // invalid call revert when called through the SafeERC20 library      await expectRevert(        this.wrapper.permit(          this.data.message.owner,          this.data.message.spender,          this.data.message.value,          this.data.message.deadline,          invalidSignature.v,          invalidSignature.r,          invalidSignature.s,        ),        'SafeERC20: permit did not succeed',      );    });  });});function shouldRevertOnAllCalls (reason) {  it('reverts on transfer', async function () {    await expectRevert(this.wrapper.transfer(), reason);  });  it('reverts on transferFrom', async function () {    await expectRevert(this.wrapper.transferFrom(), reason);  });  it('reverts on approve', async function () {    await expectRevert(this.wrapper.approve(0), reason);  });  it('reverts on increaseAllowance', async function () {    // [TODO] make sure it's reverting for the right reason    await expectRevert.unspecified(this.wrapper.increaseAllowance(0));  });  it('reverts on decreaseAllowance', async function () {    // [TODO] make sure it's reverting for the right reason    await expectRevert.unspecified(this.wrapper.decreaseAllowance(0));  });}function shouldOnlyRevertOnErrors () {  it('doesn\'t revert on transfer', async function () {    await this.wrapper.transfer();  });  it('doesn\'t revert on transferFrom', async function () {    await this.wrapper.transferFrom();  });  describe('approvals', function () {    context('with zero allowance', function () {      beforeEach(async function () {        await this.wrapper.setAllowance(0);      });      it('doesn\'t revert when approving a non-zero allowance', async function () {        await this.wrapper.approve(100);      });      it('doesn\'t revert when approving a zero allowance', async function () {        await this.wrapper.approve(0);      });      it('doesn\'t revert when increasing the allowance', async function () {        await this.wrapper.increaseAllowance(10);      });      it('reverts when decreasing the allowance', async function () {        await expectRevert(          this.wrapper.decreaseAllowance(10),          'SafeERC20: decreased allowance below zero',        );      });    });    context('with non-zero allowance', function () {      beforeEach(async function () {        await this.wrapper.setAllowance(100);      });      it('reverts when approving a non-zero allowance', async function () {        await expectRevert(          this.wrapper.approve(20),          'SafeERC20: approve from non-zero to non-zero allowance',        );      });      it('doesn\'t revert when approving a zero allowance', async function () {        await this.wrapper.approve(0);      });      it('doesn\'t revert when increasing the allowance', async function () {        await this.wrapper.increaseAllowance(10);      });      it('doesn\'t revert when decreasing the allowance to a positive value', async function () {        await this.wrapper.decreaseAllowance(50);      });      it('reverts when decreasing the allowance to a negative value', async function () {        await expectRevert(          this.wrapper.decreaseAllowance(200),          'SafeERC20: decreased allowance below zero',        );      });    });  });}
 |