145 lines
5.2 KiB
JavaScript
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 };
|