123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- const { ethers } = require('ethers');
- const { expect } = require('chai');
- const { selector } = require('../../helpers/methods');
- const INVALID_ID = '0xffffffff';
- const SIGNATURES = {
- ERC165: ['supportsInterface(bytes4)'],
- ERC721: [
- 'balanceOf(address)',
- 'ownerOf(uint256)',
- 'approve(address,uint256)',
- 'getApproved(uint256)',
- 'setApprovalForAll(address,bool)',
- 'isApprovedForAll(address,address)',
- 'transferFrom(address,address,uint256)',
- 'safeTransferFrom(address,address,uint256)',
- 'safeTransferFrom(address,address,uint256,bytes)',
- ],
- ERC721Enumerable: ['totalSupply()', 'tokenOfOwnerByIndex(address,uint256)', 'tokenByIndex(uint256)'],
- ERC721Metadata: ['name()', 'symbol()', 'tokenURI(uint256)'],
- ERC1155: [
- 'balanceOf(address,uint256)',
- 'balanceOfBatch(address[],uint256[])',
- 'setApprovalForAll(address,bool)',
- 'isApprovedForAll(address,address)',
- 'safeTransferFrom(address,address,uint256,uint256,bytes)',
- 'safeBatchTransferFrom(address,address,uint256[],uint256[],bytes)',
- ],
- ERC1155Receiver: [
- 'onERC1155Received(address,address,uint256,uint256,bytes)',
- 'onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)',
- ],
- AccessControl: [
- 'hasRole(bytes32,address)',
- 'getRoleAdmin(bytes32)',
- 'grantRole(bytes32,address)',
- 'revokeRole(bytes32,address)',
- 'renounceRole(bytes32,address)',
- ],
- AccessControlEnumerable: ['getRoleMember(bytes32,uint256)', 'getRoleMemberCount(bytes32)'],
- AccessControlDefaultAdminRules: [
- 'defaultAdminDelay()',
- 'pendingDefaultAdminDelay()',
- 'defaultAdmin()',
- 'pendingDefaultAdmin()',
- 'defaultAdminDelayIncreaseWait()',
- 'changeDefaultAdminDelay(uint48)',
- 'rollbackDefaultAdminDelay()',
- 'beginDefaultAdminTransfer(address)',
- 'acceptDefaultAdminTransfer()',
- 'cancelDefaultAdminTransfer()',
- ],
- Governor: [
- 'name()',
- 'version()',
- 'COUNTING_MODE()',
- 'hashProposal(address[],uint256[],bytes[],bytes32)',
- 'state(uint256)',
- 'proposalThreshold()',
- 'proposalSnapshot(uint256)',
- 'proposalDeadline(uint256)',
- 'proposalProposer(uint256)',
- 'proposalEta(uint256)',
- 'proposalNeedsQueuing(uint256)',
- 'votingDelay()',
- 'votingPeriod()',
- 'quorum(uint256)',
- 'getVotes(address,uint256)',
- 'getVotesWithParams(address,uint256,bytes)',
- 'hasVoted(uint256,address)',
- 'propose(address[],uint256[],bytes[],string)',
- 'queue(address[],uint256[],bytes[],bytes32)',
- 'execute(address[],uint256[],bytes[],bytes32)',
- 'cancel(address[],uint256[],bytes[],bytes32)',
- 'castVote(uint256,uint8)',
- 'castVoteWithReason(uint256,uint8,string)',
- 'castVoteWithReasonAndParams(uint256,uint8,string,bytes)',
- 'castVoteBySig(uint256,uint8,address,bytes)',
- 'castVoteWithReasonAndParamsBySig(uint256,uint8,address,string,bytes,bytes)',
- ],
- ERC2981: ['royaltyInfo(uint256,uint256)'],
- };
- const INTERFACE_IDS = Object.fromEntries(
- Object.entries(SIGNATURES).map(([name, signatures]) => [
- name,
- ethers.toBeHex(
- signatures.reduce((id, fnSig) => id ^ BigInt(selector(fnSig)), 0n),
- 4,
- ),
- ]),
- );
- function shouldSupportInterfaces(interfaces = []) {
- describe('ERC165', function () {
- beforeEach(function () {
- this.contractUnderTest = this.mock || this.token || this.holder;
- });
- describe('when the interfaceId is supported', function () {
- it('uses less than 30k gas', async function () {
- for (const k of interfaces) {
- const interface = INTERFACE_IDS[k] ?? k;
- expect(await this.contractUnderTest.supportsInterface.estimateGas(interface)).to.be.lte(30000);
- }
- });
- it('returns true', async function () {
- for (const k of interfaces) {
- const interfaceId = INTERFACE_IDS[k] ?? k;
- expect(await this.contractUnderTest.supportsInterface(interfaceId)).to.equal(true, `does not support ${k}`);
- }
- });
- });
- describe('when the interfaceId is not supported', function () {
- it('uses less than 30k', async function () {
- expect(await this.contractUnderTest.supportsInterface.estimateGas(INVALID_ID)).to.be.lte(30000);
- });
- it('returns false', async function () {
- expect(await this.contractUnderTest.supportsInterface(INVALID_ID)).to.be.equal(false, `supports ${INVALID_ID}`);
- });
- });
- it('all interface functions are in ABI', async function () {
- for (const k of interfaces) {
- // skip interfaces for which we don't have a function list
- if (SIGNATURES[k] === undefined) continue;
- for (const fnSig of SIGNATURES[k]) {
- // TODO: Remove Truffle case when ethersjs migration is done
- if (this.contractUnderTest.abi) {
- const fnSelector = selector(fnSig);
- return expect(this.contractUnderTest.abi.filter(fn => fn.signature === fnSelector).length).to.equal(
- 1,
- `did not find ${fnSig}`,
- );
- }
- expect(!!this.contractUnderTest.interface.getFunction(fnSig), `did not find ${fnSig}`).to.be.true;
- }
- }
- });
- });
- }
- module.exports = {
- shouldSupportInterfaces,
- };
|