123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272 |
- const { ethers } = require('hardhat');
- const { expect } = require('chai');
- const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
- const DUMMY_ID = '0xdeadbeef';
- const DUMMY_ID_2 = '0xcafebabe';
- const DUMMY_ID_3 = '0xdecafbad';
- const DUMMY_UNSUPPORTED_ID = '0xbaddcafe';
- const DUMMY_UNSUPPORTED_ID_2 = '0xbaadcafe';
- const DUMMY_ACCOUNT = '0x1111111111111111111111111111111111111111';
- async function fixture() {
- return { mock: await ethers.deployContract('$ERC165Checker') };
- }
- describe('ERC165Checker', function () {
- beforeEach(async function () {
- Object.assign(this, await loadFixture(fixture));
- });
- describe('ERC165 missing return data', function () {
- before(async function () {
- this.target = await ethers.deployContract('ERC165MissingData');
- });
- it('does not support ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('does not support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- });
- describe('ERC165 malicious return data', function () {
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165MaliciousData');
- });
- it('does not support ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('does not support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.true;
- });
- });
- describe('ERC165 revert on invalid interface', function () {
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165RevertInvalid', [[DUMMY_ID]]);
- });
- it('does not support ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, '0xffffffff')).to.eventually.be.false;
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.true;
- });
- });
- describe('ERC165 not supported', function () {
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165NotSupported');
- });
- it('does not support ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('does not support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- });
- describe('ERC165 supported', function () {
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165InterfacesSupported', [[]]);
- });
- it('supports ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.true;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('does not support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.false;
- });
- });
- describe('ERC165 and single interface supported', function () {
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165InterfacesSupported', [[DUMMY_ID]]);
- });
- it('supports ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.true;
- });
- it('supports mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(this.target, DUMMY_ID)).to.eventually.be.true;
- });
- it('supports mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, [DUMMY_ID])).to.eventually.be.true;
- });
- it('supports mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, [DUMMY_ID])).to.eventually.deep.equal([true]);
- });
- it('supports mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, DUMMY_ID)).to.eventually.be.true;
- });
- });
- describe('ERC165 and many interfaces supported', function () {
- const supportedInterfaces = [DUMMY_ID, DUMMY_ID_2, DUMMY_ID_3];
- beforeEach(async function () {
- this.target = await ethers.deployContract('ERC165InterfacesSupported', [supportedInterfaces]);
- });
- it('supports ERC165', async function () {
- await expect(this.mock.$supportsERC165(this.target)).to.eventually.be.true;
- });
- it('supports each interfaceId via supportsInterface', async function () {
- for (const interfaceId of supportedInterfaces) {
- await expect(this.mock.$supportsInterface(this.target, interfaceId)).to.eventually.be.true;
- }
- });
- it('supports all interfaceIds via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(this.target, supportedInterfaces)).to.eventually.be.true;
- });
- it('supports none of the interfaces queried via supportsAllInterfaces', async function () {
- const interfaceIdsToTest = [DUMMY_UNSUPPORTED_ID, DUMMY_UNSUPPORTED_ID_2];
- await expect(this.mock.$supportsAllInterfaces(this.target, interfaceIdsToTest)).to.eventually.be.false;
- });
- it('supports not all of the interfaces queried via supportsAllInterfaces', async function () {
- const interfaceIdsToTest = [...supportedInterfaces, DUMMY_UNSUPPORTED_ID];
- await expect(this.mock.$supportsAllInterfaces(this.target, interfaceIdsToTest)).to.eventually.be.false;
- });
- it('supports all interfaceIds via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(this.target, supportedInterfaces)).to.eventually.deep.equal(
- supportedInterfaces.map(i => supportedInterfaces.includes(i)),
- );
- });
- it('supports none of the interfaces queried via getSupportedInterfaces', async function () {
- const interfaceIdsToTest = [DUMMY_UNSUPPORTED_ID, DUMMY_UNSUPPORTED_ID_2];
- await expect(this.mock.$getSupportedInterfaces(this.target, interfaceIdsToTest)).to.eventually.deep.equal(
- interfaceIdsToTest.map(i => supportedInterfaces.includes(i)),
- );
- });
- it('supports not all of the interfaces queried via getSupportedInterfaces', async function () {
- const interfaceIdsToTest = [...supportedInterfaces, DUMMY_UNSUPPORTED_ID];
- await expect(this.mock.$getSupportedInterfaces(this.target, interfaceIdsToTest)).to.eventually.deep.equal(
- interfaceIdsToTest.map(i => supportedInterfaces.includes(i)),
- );
- });
- it('supports each interfaceId via supportsERC165InterfaceUnchecked', async function () {
- for (const interfaceId of supportedInterfaces) {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(this.target, interfaceId)).to.eventually.be.true;
- }
- });
- });
- describe('account address does not support ERC165', function () {
- it('does not support ERC165', async function () {
- await expect(this.mock.$supportsERC165(DUMMY_ACCOUNT)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsInterface', async function () {
- await expect(this.mock.$supportsInterface(DUMMY_ACCOUNT, DUMMY_ID)).to.eventually.be.false;
- });
- it('does not support mock interface via supportsAllInterfaces', async function () {
- await expect(this.mock.$supportsAllInterfaces(DUMMY_ACCOUNT, [DUMMY_ID])).to.eventually.be.false;
- });
- it('does not support mock interface via getSupportedInterfaces', async function () {
- await expect(this.mock.$getSupportedInterfaces(DUMMY_ACCOUNT, [DUMMY_ID])).to.eventually.deep.equal([false]);
- });
- it('does not support mock interface via supportsERC165InterfaceUnchecked', async function () {
- await expect(this.mock.$supportsERC165InterfaceUnchecked(DUMMY_ACCOUNT, DUMMY_ID)).to.eventually.be.false;
- });
- });
- it('Return bomb resistance', async function () {
- this.target = await ethers.deployContract('ERC165ReturnBombMock');
- const { gasUsed: gasUsed1 } = await this.mock.$supportsInterface.send(this.target, DUMMY_ID).then(tx => tx.wait());
- expect(gasUsed1).to.be.lessThan(120_000n); // 3*30k + 21k + some margin
- const { gasUsed: gasUsed2 } = await this.mock.$getSupportedInterfaces
- .send(this.target, [DUMMY_ID, DUMMY_ID_2, DUMMY_ID_3, DUMMY_UNSUPPORTED_ID, DUMMY_UNSUPPORTED_ID_2])
- .then(tx => tx.wait());
- expect(gasUsed2).to.be.lessThan(250_000n); // (2+5)*30k + 21k + some margin
- });
- });
|