Migrate metatx tests to ethers.js (#4737)
Co-authored-by: ernestognw <ernestognw@gmail.com>
This commit is contained in:
@ -1,134 +1,117 @@
|
||||
const ethSigUtil = require('eth-sig-util');
|
||||
const Wallet = require('ethereumjs-wallet').default;
|
||||
const { getDomain, domainType } = require('../helpers/eip712');
|
||||
const { MAX_UINT48 } = require('../helpers/constants');
|
||||
|
||||
const { expectEvent } = require('@openzeppelin/test-helpers');
|
||||
const { ethers } = require('hardhat');
|
||||
const { expect } = require('chai');
|
||||
const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers');
|
||||
|
||||
const ERC2771ContextMock = artifacts.require('ERC2771ContextMock');
|
||||
const ERC2771Forwarder = artifacts.require('ERC2771Forwarder');
|
||||
const ContextMockCaller = artifacts.require('ContextMockCaller');
|
||||
const { impersonate } = require('../helpers/account');
|
||||
const { getDomain } = require('../helpers/eip712');
|
||||
const { MAX_UINT48 } = require('../helpers/constants');
|
||||
|
||||
const { shouldBehaveLikeRegularContext } = require('../utils/Context.behavior');
|
||||
|
||||
contract('ERC2771Context', function (accounts) {
|
||||
const [, trustedForwarder] = accounts;
|
||||
async function fixture() {
|
||||
const [sender] = await ethers.getSigners();
|
||||
|
||||
const forwarder = await ethers.deployContract('ERC2771Forwarder', []);
|
||||
const forwarderAsSigner = await impersonate(forwarder.target);
|
||||
const context = await ethers.deployContract('ERC2771ContextMock', [forwarder]);
|
||||
const domain = await getDomain(forwarder);
|
||||
const types = {
|
||||
ForwardRequest: [
|
||||
{ name: 'from', type: 'address' },
|
||||
{ name: 'to', type: 'address' },
|
||||
{ name: 'value', type: 'uint256' },
|
||||
{ name: 'gas', type: 'uint256' },
|
||||
{ name: 'nonce', type: 'uint256' },
|
||||
{ name: 'deadline', type: 'uint48' },
|
||||
{ name: 'data', type: 'bytes' },
|
||||
],
|
||||
};
|
||||
|
||||
return { sender, forwarder, forwarderAsSigner, context, domain, types };
|
||||
}
|
||||
|
||||
describe('ERC2771Context', function () {
|
||||
beforeEach(async function () {
|
||||
this.forwarder = await ERC2771Forwarder.new('ERC2771Forwarder');
|
||||
this.recipient = await ERC2771ContextMock.new(this.forwarder.address);
|
||||
|
||||
this.domain = await getDomain(this.forwarder);
|
||||
this.types = {
|
||||
EIP712Domain: domainType(this.domain),
|
||||
ForwardRequest: [
|
||||
{ name: 'from', type: 'address' },
|
||||
{ name: 'to', type: 'address' },
|
||||
{ name: 'value', type: 'uint256' },
|
||||
{ name: 'gas', type: 'uint256' },
|
||||
{ name: 'nonce', type: 'uint256' },
|
||||
{ name: 'deadline', type: 'uint48' },
|
||||
{ name: 'data', type: 'bytes' },
|
||||
],
|
||||
};
|
||||
Object.assign(this, await loadFixture(fixture));
|
||||
});
|
||||
|
||||
it('recognize trusted forwarder', async function () {
|
||||
expect(await this.recipient.isTrustedForwarder(this.forwarder.address)).to.equal(true);
|
||||
expect(await this.context.isTrustedForwarder(this.forwarder)).to.equal(true);
|
||||
});
|
||||
|
||||
it('returns the trusted forwarder', async function () {
|
||||
expect(await this.recipient.trustedForwarder()).to.equal(this.forwarder.address);
|
||||
expect(await this.context.trustedForwarder()).to.equal(this.forwarder.target);
|
||||
});
|
||||
|
||||
context('when called directly', function () {
|
||||
beforeEach(async function () {
|
||||
this.context = this.recipient; // The Context behavior expects the contract in this.context
|
||||
this.caller = await ContextMockCaller.new();
|
||||
});
|
||||
|
||||
shouldBehaveLikeRegularContext(...accounts);
|
||||
describe('when called directly', function () {
|
||||
shouldBehaveLikeRegularContext();
|
||||
});
|
||||
|
||||
context('when receiving a relayed call', function () {
|
||||
beforeEach(async function () {
|
||||
this.wallet = Wallet.generate();
|
||||
this.sender = web3.utils.toChecksumAddress(this.wallet.getAddressString());
|
||||
this.data = {
|
||||
types: this.types,
|
||||
domain: this.domain,
|
||||
primaryType: 'ForwardRequest',
|
||||
};
|
||||
});
|
||||
|
||||
describe('when receiving a relayed call', function () {
|
||||
describe('msgSender', function () {
|
||||
it('returns the relayed transaction original sender', async function () {
|
||||
const data = this.recipient.contract.methods.msgSender().encodeABI();
|
||||
const nonce = await this.forwarder.nonces(this.sender);
|
||||
const data = this.context.interface.encodeFunctionData('msgSender');
|
||||
|
||||
const req = {
|
||||
from: this.sender,
|
||||
to: this.recipient.address,
|
||||
value: '0',
|
||||
gas: '100000',
|
||||
nonce: (await this.forwarder.nonces(this.sender)).toString(),
|
||||
deadline: MAX_UINT48,
|
||||
from: await this.sender.getAddress(),
|
||||
to: await this.context.getAddress(),
|
||||
value: 0n,
|
||||
data,
|
||||
gas: 100000n,
|
||||
nonce,
|
||||
deadline: MAX_UINT48,
|
||||
};
|
||||
|
||||
req.signature = ethSigUtil.signTypedMessage(this.wallet.getPrivateKey(), {
|
||||
data: { ...this.data, message: req },
|
||||
});
|
||||
req.signature = await this.sender.signTypedData(this.domain, this.types, req);
|
||||
|
||||
expect(await this.forwarder.verify(req)).to.equal(true);
|
||||
|
||||
const { tx } = await this.forwarder.execute(req);
|
||||
await expectEvent.inTransaction(tx, ERC2771ContextMock, 'Sender', { sender: this.sender });
|
||||
await expect(this.forwarder.execute(req)).to.emit(this.context, 'Sender').withArgs(this.sender.address);
|
||||
});
|
||||
|
||||
it('returns the original sender when calldata length is less than 20 bytes (address length)', async function () {
|
||||
// The forwarder doesn't produce calls with calldata length less than 20 bytes
|
||||
const recipient = await ERC2771ContextMock.new(trustedForwarder);
|
||||
|
||||
const { receipt } = await recipient.msgSender({ from: trustedForwarder });
|
||||
|
||||
await expectEvent(receipt, 'Sender', { sender: trustedForwarder });
|
||||
// The forwarder doesn't produce calls with calldata length less than 20 bytes so `this.forwarderAsSigner` is used instead.
|
||||
await expect(this.context.connect(this.forwarderAsSigner).msgSender())
|
||||
.to.emit(this.context, 'Sender')
|
||||
.withArgs(this.forwarder.target);
|
||||
});
|
||||
});
|
||||
|
||||
describe('msgData', function () {
|
||||
it('returns the relayed transaction original data', async function () {
|
||||
const integerValue = '42';
|
||||
const stringValue = 'OpenZeppelin';
|
||||
const data = this.recipient.contract.methods.msgData(integerValue, stringValue).encodeABI();
|
||||
const args = [42n, 'OpenZeppelin'];
|
||||
|
||||
const nonce = await this.forwarder.nonces(this.sender);
|
||||
const data = this.context.interface.encodeFunctionData('msgData', args);
|
||||
|
||||
const req = {
|
||||
from: this.sender,
|
||||
to: this.recipient.address,
|
||||
value: '0',
|
||||
gas: '100000',
|
||||
nonce: (await this.forwarder.nonces(this.sender)).toString(),
|
||||
deadline: MAX_UINT48,
|
||||
from: await this.sender.getAddress(),
|
||||
to: await this.context.getAddress(),
|
||||
value: 0n,
|
||||
data,
|
||||
gas: 100000n,
|
||||
nonce,
|
||||
deadline: MAX_UINT48,
|
||||
};
|
||||
|
||||
req.signature = ethSigUtil.signTypedMessage(this.wallet.getPrivateKey(), {
|
||||
data: { ...this.data, message: req },
|
||||
});
|
||||
req.signature = this.sender.signTypedData(this.domain, this.types, req);
|
||||
|
||||
expect(await this.forwarder.verify(req)).to.equal(true);
|
||||
|
||||
const { tx } = await this.forwarder.execute(req);
|
||||
await expectEvent.inTransaction(tx, ERC2771ContextMock, 'Data', { data, integerValue, stringValue });
|
||||
await expect(this.forwarder.execute(req))
|
||||
.to.emit(this.context, 'Data')
|
||||
.withArgs(data, ...args);
|
||||
});
|
||||
});
|
||||
|
||||
it('returns the full original data when calldata length is less than 20 bytes (address length)', async function () {
|
||||
// The forwarder doesn't produce calls with calldata length less than 20 bytes
|
||||
const recipient = await ERC2771ContextMock.new(trustedForwarder);
|
||||
const data = this.context.interface.encodeFunctionData('msgDataShort');
|
||||
|
||||
const { receipt } = await recipient.msgDataShort({ from: trustedForwarder });
|
||||
|
||||
const data = recipient.contract.methods.msgDataShort().encodeABI();
|
||||
await expectEvent(receipt, 'DataShort', { data });
|
||||
// The forwarder doesn't produce calls with calldata length less than 20 bytes so `this.forwarderAsSigner` is used instead.
|
||||
await expect(await this.context.connect(this.forwarderAsSigner).msgDataShort())
|
||||
.to.emit(this.context, 'DataShort')
|
||||
.withArgs(data);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user