Files
openzeppelin-contracts/test/account/Account.behavior.js
2025-06-02 08:22:57 -06:00

145 lines
5.2 KiB
JavaScript

const { ethers, entrypoint } = require('hardhat');
const { expect } = require('chai');
const { impersonate } = require('../helpers/account');
const { SIG_VALIDATION_SUCCESS, SIG_VALIDATION_FAILURE } = require('../helpers/erc4337');
const { shouldSupportInterfaces } = require('../utils/introspection/SupportsInterface.behavior');
function shouldBehaveLikeAccountCore() {
describe('entryPoint', function () {
it('should return the canonical entrypoint', async function () {
await this.mock.deploy();
await expect(this.mock.entryPoint()).to.eventually.equal(entrypoint.v08);
});
});
describe('validateUserOp', function () {
beforeEach(async function () {
await this.other.sendTransaction({ to: this.mock.target, value: ethers.parseEther('1') });
await this.mock.deploy();
this.userOp ??= {};
});
it('should revert if the caller is not the canonical entrypoint', async function () {
// empty operation (does nothing)
const operation = await this.mock.createUserOp(this.userOp).then(op => this.signUserOp(op));
await expect(this.mock.connect(this.other).validateUserOp(operation.packed, operation.hash(), 0))
.to.be.revertedWithCustomError(this.mock, 'AccountUnauthorized')
.withArgs(this.other);
});
describe('when the caller is the canonical entrypoint', function () {
beforeEach(async function () {
this.mockFromEntrypoint = this.mock.connect(await impersonate(entrypoint.v08.target));
});
it('should return SIG_VALIDATION_SUCCESS if the signature is valid', async function () {
// empty operation (does nothing)
const operation = await this.mock.createUserOp(this.userOp).then(op => this.signUserOp(op));
expect(await this.mockFromEntrypoint.validateUserOp.staticCall(operation.packed, operation.hash(), 0)).to.eq(
SIG_VALIDATION_SUCCESS,
);
});
it('should return SIG_VALIDATION_FAILURE if the signature is invalid', async function () {
// empty operation (does nothing)
const operation = await this.mock.createUserOp(this.userOp);
operation.signature = (await this.invalidSig?.()) ?? '0x00';
expect(await this.mockFromEntrypoint.validateUserOp.staticCall(operation.packed, operation.hash(), 0)).to.eq(
SIG_VALIDATION_FAILURE,
);
});
it('should pay missing account funds for execution', async function () {
// empty operation (does nothing)
const operation = await this.mock.createUserOp(this.userOp).then(op => this.signUserOp(op));
const value = 42n;
await expect(
this.mockFromEntrypoint.validateUserOp(operation.packed, operation.hash(), value),
).to.changeEtherBalances([this.mock, entrypoint.v08], [-value, value]);
});
});
});
describe('fallback', function () {
it('should receive ether', async function () {
await this.mock.deploy();
const value = 42n;
await expect(this.other.sendTransaction({ to: this.mock, value })).to.changeEtherBalances(
[this.other, this.mock],
[-value, value],
);
});
});
}
function shouldBehaveLikeAccountHolder() {
describe('onReceived', function () {
beforeEach(async function () {
await this.mock.deploy();
});
shouldSupportInterfaces(['ERC1155Receiver']);
describe('onERC1155Received', function () {
const ids = [1n, 2n, 3n];
const values = [1000n, 2000n, 3000n];
const data = '0x12345678';
beforeEach(async function () {
this.token = await ethers.deployContract('$ERC1155', ['https://somedomain.com/{id}.json']);
await this.token.$_mintBatch(this.other, ids, values, '0x');
});
it('receives ERC1155 tokens from a single ID', async function () {
await this.token.connect(this.other).safeTransferFrom(this.other, this.mock, ids[0], values[0], data);
await expect(
this.token.balanceOfBatch(
ids.map(() => this.mock),
ids,
),
).to.eventually.deep.equal(values.map((v, i) => (i == 0 ? v : 0n)));
});
it('receives ERC1155 tokens from a multiple IDs', async function () {
await expect(
this.token.balanceOfBatch(
ids.map(() => this.mock),
ids,
),
).to.eventually.deep.equal(ids.map(() => 0n));
await this.token.connect(this.other).safeBatchTransferFrom(this.other, this.mock, ids, values, data);
await expect(
this.token.balanceOfBatch(
ids.map(() => this.mock),
ids,
),
).to.eventually.deep.equal(values);
});
});
describe('onERC721Received', function () {
const tokenId = 1n;
beforeEach(async function () {
this.token = await ethers.deployContract('$ERC721', ['Some NFT', 'SNFT']);
await this.token.$_mint(this.other, tokenId);
});
it('receives an ERC721 token', async function () {
await this.token.connect(this.other).safeTransferFrom(this.other, this.mock, tokenId);
await expect(this.token.ownerOf(tokenId)).to.eventually.equal(this.mock);
});
});
});
}
module.exports = { shouldBehaveLikeAccountCore, shouldBehaveLikeAccountHolder };