123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 |
- const { ethers } = require('hardhat');
- const { expect } = require('chai');
- const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
- const { zip } = require('../../helpers/iterate');
- const { shouldBehaveLikeERC1155 } = require('./ERC1155.behavior');
- const initialURI = 'https://token-cdn-domain/{id}.json';
- async function fixture() {
- const [operator, holder, ...otherAccounts] = await ethers.getSigners();
- const token = await ethers.deployContract('$ERC1155', [initialURI]);
- return { token, operator, holder, otherAccounts };
- }
- describe('ERC1155', function () {
- beforeEach(async function () {
- Object.assign(this, await loadFixture(fixture));
- });
- shouldBehaveLikeERC1155();
- describe('internal functions', function () {
- const tokenId = 1990n;
- const mintValue = 9001n;
- const burnValue = 3000n;
- const tokenBatchIds = [2000n, 2010n, 2020n];
- const mintValues = [5000n, 10000n, 42195n];
- const burnValues = [5000n, 9001n, 195n];
- const data = '0x12345678';
- describe('_mint', function () {
- it('reverts with a zero destination address', async function () {
- await expect(this.token.$_mint(ethers.ZeroAddress, tokenId, mintValue, data))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidReceiver')
- .withArgs(ethers.ZeroAddress);
- });
- describe('with minted tokens', function () {
- beforeEach(async function () {
- this.tx = await this.token.connect(this.operator).$_mint(this.holder, tokenId, mintValue, data);
- });
- it('emits a TransferSingle event', async function () {
- await expect(this.tx)
- .to.emit(this.token, 'TransferSingle')
- .withArgs(this.operator, ethers.ZeroAddress, this.holder, tokenId, mintValue);
- });
- it('credits the minted token value', async function () {
- expect(await this.token.balanceOf(this.holder, tokenId)).to.equal(mintValue);
- });
- });
- });
- describe('_mintBatch', function () {
- it('reverts with a zero destination address', async function () {
- await expect(this.token.$_mintBatch(ethers.ZeroAddress, tokenBatchIds, mintValues, data))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidReceiver')
- .withArgs(ethers.ZeroAddress);
- });
- it('reverts if length of inputs do not match', async function () {
- await expect(this.token.$_mintBatch(this.holder, tokenBatchIds, mintValues.slice(1), data))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidArrayLength')
- .withArgs(tokenBatchIds.length, mintValues.length - 1);
- await expect(this.token.$_mintBatch(this.holder, tokenBatchIds.slice(1), mintValues, data))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidArrayLength')
- .withArgs(tokenBatchIds.length - 1, mintValues.length);
- });
- describe('with minted batch of tokens', function () {
- beforeEach(async function () {
- this.tx = await this.token.connect(this.operator).$_mintBatch(this.holder, tokenBatchIds, mintValues, data);
- });
- it('emits a TransferBatch event', async function () {
- await expect(this.tx)
- .to.emit(this.token, 'TransferBatch')
- .withArgs(this.operator, ethers.ZeroAddress, this.holder, tokenBatchIds, mintValues);
- });
- it('credits the minted batch of tokens', async function () {
- const holderBatchBalances = await this.token.balanceOfBatch(
- tokenBatchIds.map(() => this.holder),
- tokenBatchIds,
- );
- expect(holderBatchBalances).to.deep.equal(mintValues);
- });
- });
- });
- describe('_burn', function () {
- it("reverts when burning the zero account's tokens", async function () {
- await expect(this.token.$_burn(ethers.ZeroAddress, tokenId, mintValue))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidSender')
- .withArgs(ethers.ZeroAddress);
- });
- it('reverts when burning a non-existent token id', async function () {
- await expect(this.token.$_burn(this.holder, tokenId, mintValue))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InsufficientBalance')
- .withArgs(this.holder, 0, mintValue, tokenId);
- });
- it('reverts when burning more than available tokens', async function () {
- await this.token.connect(this.operator).$_mint(this.holder, tokenId, mintValue, data);
- await expect(this.token.$_burn(this.holder, tokenId, mintValue + 1n))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InsufficientBalance')
- .withArgs(this.holder, mintValue, mintValue + 1n, tokenId);
- });
- describe('with minted-then-burnt tokens', function () {
- beforeEach(async function () {
- await this.token.$_mint(this.holder, tokenId, mintValue, data);
- this.tx = await this.token.connect(this.operator).$_burn(this.holder, tokenId, burnValue);
- });
- it('emits a TransferSingle event', async function () {
- await expect(this.tx)
- .to.emit(this.token, 'TransferSingle')
- .withArgs(this.operator, this.holder, ethers.ZeroAddress, tokenId, burnValue);
- });
- it('accounts for both minting and burning', async function () {
- expect(await this.token.balanceOf(this.holder, tokenId)).to.equal(mintValue - burnValue);
- });
- });
- });
- describe('_burnBatch', function () {
- it("reverts when burning the zero account's tokens", async function () {
- await expect(this.token.$_burnBatch(ethers.ZeroAddress, tokenBatchIds, burnValues))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidSender')
- .withArgs(ethers.ZeroAddress);
- });
- it('reverts if length of inputs do not match', async function () {
- await expect(this.token.$_burnBatch(this.holder, tokenBatchIds, burnValues.slice(1)))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidArrayLength')
- .withArgs(tokenBatchIds.length, burnValues.length - 1);
- await expect(this.token.$_burnBatch(this.holder, tokenBatchIds.slice(1), burnValues))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InvalidArrayLength')
- .withArgs(tokenBatchIds.length - 1, burnValues.length);
- });
- it('reverts when burning a non-existent token id', async function () {
- await expect(this.token.$_burnBatch(this.holder, tokenBatchIds, burnValues))
- .to.be.revertedWithCustomError(this.token, 'ERC1155InsufficientBalance')
- .withArgs(this.holder, 0, burnValues[0], tokenBatchIds[0]);
- });
- describe('with minted-then-burnt tokens', function () {
- beforeEach(async function () {
- await this.token.$_mintBatch(this.holder, tokenBatchIds, mintValues, data);
- this.tx = await this.token.connect(this.operator).$_burnBatch(this.holder, tokenBatchIds, burnValues);
- });
- it('emits a TransferBatch event', async function () {
- await expect(this.tx)
- .to.emit(this.token, 'TransferBatch')
- .withArgs(this.operator, this.holder, ethers.ZeroAddress, tokenBatchIds, burnValues);
- });
- it('accounts for both minting and burning', async function () {
- const holderBatchBalances = await this.token.balanceOfBatch(
- tokenBatchIds.map(() => this.holder),
- tokenBatchIds,
- );
- expect(holderBatchBalances).to.deep.equal(
- zip(mintValues, burnValues).map(([mintValue, burnValue]) => mintValue - burnValue),
- );
- });
- });
- });
- });
- describe('ERC1155MetadataURI', function () {
- const firstTokenID = 42n;
- const secondTokenID = 1337n;
- it('emits no URI event in constructor', async function () {
- await expect(this.token.deploymentTransaction()).to.not.emit(this.token, 'URI');
- });
- it('sets the initial URI for all token types', async function () {
- expect(await this.token.uri(firstTokenID)).to.equal(initialURI);
- expect(await this.token.uri(secondTokenID)).to.equal(initialURI);
- });
- describe('_setURI', function () {
- const newURI = 'https://token-cdn-domain/{locale}/{id}.json';
- it('emits no URI event', async function () {
- await expect(this.token.$_setURI(newURI)).to.not.emit(this.token, 'URI');
- });
- it('sets the new URI for all token types', async function () {
- await this.token.$_setURI(newURI);
- expect(await this.token.uri(firstTokenID)).to.equal(newURI);
- expect(await this.token.uri(secondTokenID)).to.equal(newURI);
- });
- });
- });
- });
|