123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289 |
- const { ethers, predeploy } = require('hardhat');
- const { expect } = require('chai');
- const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
- const { packValidationData, UserOperation } = require('../../helpers/erc4337');
- const { MAX_UINT48 } = require('../../helpers/constants');
- const ADDRESS_ONE = '0x0000000000000000000000000000000000000001';
- const fixture = async () => {
- const [authorizer, sender, factory, paymaster] = await ethers.getSigners();
- const utils = await ethers.deployContract('$ERC4337Utils');
- const SIG_VALIDATION_SUCCESS = await utils.$SIG_VALIDATION_SUCCESS();
- const SIG_VALIDATION_FAILED = await utils.$SIG_VALIDATION_FAILED();
- return { utils, authorizer, sender, factory, paymaster, SIG_VALIDATION_SUCCESS, SIG_VALIDATION_FAILED };
- };
- describe('ERC4337Utils', function () {
- beforeEach(async function () {
- Object.assign(this, await loadFixture(fixture));
- });
- describe('entrypoint', function () {
- it('v0.7.0', async function () {
- await expect(this.utils.$ENTRYPOINT_V07()).to.eventually.equal(predeploy.entrypoint.v07);
- });
- it('v0.8.0', async function () {
- await expect(this.utils.$ENTRYPOINT_V08()).to.eventually.equal(predeploy.entrypoint.v08);
- });
- });
- describe('parseValidationData', function () {
- it('parses the validation data', async function () {
- const authorizer = this.authorizer;
- const validUntil = 0x12345678n;
- const validAfter = 0x9abcdef0n;
- const validationData = packValidationData(validAfter, validUntil, authorizer);
- await expect(this.utils.$parseValidationData(validationData)).to.eventually.deep.equal([
- authorizer.address,
- validAfter,
- validUntil,
- ]);
- });
- it('returns an type(uint48).max if until is 0', async function () {
- const authorizer = this.authorizer;
- const validAfter = 0x12345678n;
- const validationData = packValidationData(validAfter, 0, authorizer);
- await expect(this.utils.$parseValidationData(validationData)).to.eventually.deep.equal([
- authorizer.address,
- validAfter,
- MAX_UINT48,
- ]);
- });
- it('parse canonical values', async function () {
- await expect(this.utils.$parseValidationData(this.SIG_VALIDATION_SUCCESS)).to.eventually.deep.equal([
- ethers.ZeroAddress,
- 0n,
- MAX_UINT48,
- ]);
- await expect(this.utils.$parseValidationData(this.SIG_VALIDATION_FAILED)).to.eventually.deep.equal([
- ADDRESS_ONE,
- 0n,
- MAX_UINT48,
- ]);
- });
- });
- describe('packValidationData', function () {
- it('packs the validation data', async function () {
- const authorizer = this.authorizer;
- const validUntil = 0x12345678n;
- const validAfter = 0x9abcdef0n;
- const validationData = packValidationData(validAfter, validUntil, authorizer);
- await expect(
- this.utils.$packValidationData(ethers.Typed.address(authorizer), validAfter, validUntil),
- ).to.eventually.equal(validationData);
- });
- it('packs the validation data (bool)', async function () {
- const success = false;
- const validUntil = 0x12345678n;
- const validAfter = 0x9abcdef0n;
- const validationData = packValidationData(validAfter, validUntil, false);
- await expect(
- this.utils.$packValidationData(ethers.Typed.bool(success), validAfter, validUntil),
- ).to.eventually.equal(validationData);
- });
- it('packing reproduced canonical values', async function () {
- await expect(
- this.utils.$packValidationData(ethers.Typed.address(ethers.ZeroAddress), 0n, 0n),
- ).to.eventually.equal(this.SIG_VALIDATION_SUCCESS);
- await expect(this.utils.$packValidationData(ethers.Typed.bool(true), 0n, 0n)).to.eventually.equal(
- this.SIG_VALIDATION_SUCCESS,
- );
- await expect(this.utils.$packValidationData(ethers.Typed.address(ADDRESS_ONE), 0n, 0n)).to.eventually.equal(
- this.SIG_VALIDATION_FAILED,
- );
- await expect(this.utils.$packValidationData(ethers.Typed.bool(false), 0n, 0n)).to.eventually.equal(
- this.SIG_VALIDATION_FAILED,
- );
- });
- });
- describe('combineValidationData', function () {
- const validUntil1 = 0x12345678n;
- const validAfter1 = 0x9abcdef0n;
- const validUntil2 = 0x87654321n;
- const validAfter2 = 0xabcdef90n;
- it('combines the validation data', async function () {
- const validationData1 = packValidationData(validAfter1, validUntil1, ethers.ZeroAddress);
- const validationData2 = packValidationData(validAfter2, validUntil2, ethers.ZeroAddress);
- const expected = packValidationData(validAfter2, validUntil1, true);
- // check symmetry
- await expect(this.utils.$combineValidationData(validationData1, validationData2)).to.eventually.equal(expected);
- await expect(this.utils.$combineValidationData(validationData2, validationData1)).to.eventually.equal(expected);
- });
- for (const [authorizer1, authorizer2] of [
- [ethers.ZeroAddress, '0xbf023313b891fd6000544b79e353323aa94a4f29'],
- ['0xbf023313b891fd6000544b79e353323aa94a4f29', ethers.ZeroAddress],
- ]) {
- it('returns SIG_VALIDATION_FAILURE if one of the authorizers is not address(0)', async function () {
- const validationData1 = packValidationData(validAfter1, validUntil1, authorizer1);
- const validationData2 = packValidationData(validAfter2, validUntil2, authorizer2);
- const expected = packValidationData(validAfter2, validUntil1, false);
- // check symmetry
- await expect(this.utils.$combineValidationData(validationData1, validationData2)).to.eventually.equal(expected);
- await expect(this.utils.$combineValidationData(validationData2, validationData1)).to.eventually.equal(expected);
- });
- }
- });
- describe('getValidationData', function () {
- it('returns the validation data with valid validity range', async function () {
- const aggregator = this.authorizer;
- const validAfter = 0;
- const validUntil = MAX_UINT48;
- const validationData = packValidationData(validAfter, validUntil, aggregator);
- await expect(this.utils.$getValidationData(validationData)).to.eventually.deep.equal([aggregator.address, false]);
- });
- it('returns the validation data with invalid validity range (expired)', async function () {
- const aggregator = this.authorizer;
- const validAfter = 0;
- const validUntil = 1;
- const validationData = packValidationData(validAfter, validUntil, aggregator);
- await expect(this.utils.$getValidationData(validationData)).to.eventually.deep.equal([aggregator.address, true]);
- });
- it('returns the validation data with invalid validity range (not yet valid)', async function () {
- const aggregator = this.authorizer;
- const validAfter = MAX_UINT48;
- const validUntil = MAX_UINT48;
- const validationData = packValidationData(validAfter, validUntil, aggregator);
- await expect(this.utils.$getValidationData(validationData)).to.eventually.deep.equal([aggregator.address, true]);
- });
- it('returns address(0) and false for validationData = 0', async function () {
- await expect(this.utils.$getValidationData(0n)).to.eventually.deep.equal([ethers.ZeroAddress, false]);
- });
- });
- describe('hash', function () {
- for (const [version, instance] of Object.entries(predeploy.entrypoint)) {
- it(`returns the operation hash for entrypoint ${version}`, async function () {
- const userOp = new UserOperation({ sender: this.sender, nonce: 1 });
- const expected = await userOp.hash(instance);
- await expect(this.utils.$hash(userOp.packed, instance)).to.eventually.equal(expected);
- });
- }
- });
- describe('userOp values', function () {
- describe('intiCode', function () {
- beforeEach(async function () {
- this.userOp = new UserOperation({
- sender: this.sender,
- nonce: 1,
- verificationGas: 0x12345678n,
- factory: this.factory,
- factoryData: '0x123456',
- });
- this.emptyUserOp = new UserOperation({
- sender: this.sender,
- nonce: 1,
- });
- });
- it('returns factory', async function () {
- await expect(this.utils.$factory(this.userOp.packed)).to.eventually.equal(this.factory);
- await expect(this.utils.$factory(this.emptyUserOp.packed)).to.eventually.equal(ethers.ZeroAddress);
- });
- it('returns factoryData', async function () {
- await expect(this.utils.$factoryData(this.userOp.packed)).to.eventually.equal('0x123456');
- await expect(this.utils.$factoryData(this.emptyUserOp.packed)).to.eventually.equal('0x');
- });
- });
- it('returns verificationGasLimit', async function () {
- const userOp = new UserOperation({ sender: this.sender, nonce: 1, verificationGas: 0x12345678n });
- await expect(this.utils.$verificationGasLimit(userOp.packed)).to.eventually.equal(userOp.verificationGas);
- });
- it('returns callGasLimit', async function () {
- const userOp = new UserOperation({ sender: this.sender, nonce: 1, callGas: 0x12345678n });
- await expect(this.utils.$callGasLimit(userOp.packed)).to.eventually.equal(userOp.callGas);
- });
- it('returns maxPriorityFeePerGas', async function () {
- const userOp = new UserOperation({ sender: this.sender, nonce: 1, maxPriorityFee: 0x12345678n });
- await expect(this.utils.$maxPriorityFeePerGas(userOp.packed)).to.eventually.equal(userOp.maxPriorityFee);
- });
- it('returns maxFeePerGas', async function () {
- const userOp = new UserOperation({ sender: this.sender, nonce: 1, maxFeePerGas: 0x12345678n });
- await expect(this.utils.$maxFeePerGas(userOp.packed)).to.eventually.equal(userOp.maxFeePerGas);
- });
- it('returns gasPrice', async function () {
- const userOp = new UserOperation({
- sender: this.sender,
- nonce: 1,
- maxPriorityFee: 0x12345678n,
- maxFeePerGas: 0x87654321n,
- });
- await expect(this.utils.$gasPrice(userOp.packed)).to.eventually.equal(userOp.maxPriorityFee);
- });
- describe('paymasterAndData', function () {
- beforeEach(async function () {
- this.userOp = new UserOperation({
- sender: this.sender,
- nonce: 1,
- paymaster: this.paymaster,
- paymasterVerificationGasLimit: 0x12345678n,
- paymasterPostOpGasLimit: 0x87654321n,
- paymasterData: '0xbeefcafe',
- });
- this.emptyUserOp = new UserOperation({
- sender: this.sender,
- nonce: 1,
- });
- });
- it('returns paymaster', async function () {
- await expect(this.utils.$paymaster(this.userOp.packed)).to.eventually.equal(this.userOp.paymaster);
- await expect(this.utils.$paymaster(this.emptyUserOp.packed)).to.eventually.equal(ethers.ZeroAddress);
- });
- it('returns verificationGasLimit', async function () {
- await expect(this.utils.$paymasterVerificationGasLimit(this.userOp.packed)).to.eventually.equal(
- this.userOp.paymasterVerificationGasLimit,
- );
- await expect(this.utils.$paymasterVerificationGasLimit(this.emptyUserOp.packed)).to.eventually.equal(0n);
- });
- it('returns postOpGasLimit', async function () {
- await expect(this.utils.$paymasterPostOpGasLimit(this.userOp.packed)).to.eventually.equal(
- this.userOp.paymasterPostOpGasLimit,
- );
- await expect(this.utils.$paymasterPostOpGasLimit(this.emptyUserOp.packed)).to.eventually.equal(0n);
- });
- it('returns data', async function () {
- await expect(this.utils.$paymasterData(this.userOp.packed)).to.eventually.equal(this.userOp.paymasterData);
- await expect(this.utils.$paymasterData(this.emptyUserOp.packed)).to.eventually.equal('0x');
- });
- });
- });
- });
|