From e1c49fc90520b601c11fae0e9070c3e4d57ca09e Mon Sep 17 00:00:00 2001 From: ernestognw Date: Thu, 26 Dec 2024 19:12:37 -0600 Subject: [PATCH 1/4] Await `.eventually` test matchers --- contracts/utils/cryptography/ERC7739Utils.sol | 22 ++++++-- test/account/Account.behavior.js | 29 +++++----- test/crosschain/axelar/AxelarGateway.test.js | 12 ++-- test/helpers/erc7739.js | 2 +- .../cryptography/ERC7739Signer.behavior.js | 17 +++--- test/utils/cryptography/ERC7739Signer.test.js | 4 +- test/utils/cryptography/ERC7739Utils.test.js | 56 +++++++++---------- 7 files changed, 76 insertions(+), 66 deletions(-) diff --git a/contracts/utils/cryptography/ERC7739Utils.sol b/contracts/utils/cryptography/ERC7739Utils.sol index eed6acd2..36f2f7ad 100644 --- a/contracts/utils/cryptography/ERC7739Utils.sol +++ b/contracts/utils/cryptography/ERC7739Utils.sol @@ -156,8 +156,8 @@ library ERC7739Utils { * @dev Parse the type name out of the ERC-7739 contents type description. Supports both the implicit and explicit * modes. * - * Following ERC-7739 specifications, a `contentsTypeName` is considered invalid if it's empty or it contains - * any of the following bytes , )\x00 + * Following ERC-7739 specifications, a `contentsTypeName` is considered invalid if it's empty, it contains + * any of the following bytes (`, )\x00`) or it starts with a forbidden character (a-z or `(`). * * If the `contentsType` is invalid, this returns an empty string. Otherwise, the return string has non-zero * length. @@ -173,10 +173,12 @@ library ERC7739Utils { for (uint256 i = 0; i < buffer.length; ++i) { bytes1 current = buffer[i]; if (current == bytes1("(")) { - // if name is empty - passthrough (fail) - if (i == 0) break; + if (_isForbiddenFirstChar(buffer[0])) { + // we found an invalid first character (forbidden) - passthrough (fail) + break; + } // we found the end of the contentsTypeName - return (string(buffer[:i]), contentsDescr); + return (string(buffer[:i]), string(buffer)); } else if (_isForbiddenChar(current)) { // we found an invalid character (forbidden) - passthrough (fail) break; @@ -187,6 +189,10 @@ library ERC7739Utils { for (uint256 i = buffer.length; i > 0; --i) { bytes1 current = buffer[i - 1]; if (current == bytes1(")")) { + if (i >= buffer.length || _isForbiddenFirstChar(buffer[i])) { + // we found an invalid first character (forbidden) - passthrough (fail) + break; + } // we found the end of the contentsTypeName return (string(buffer[i:]), string(buffer[:i])); } else if (_isForbiddenChar(current)) { @@ -215,6 +221,10 @@ library ERC7739Utils { } function _isForbiddenChar(bytes1 char) private pure returns (bool) { - return char == 0x00 || char == bytes1(" ") || char == bytes1(",") || char == bytes1("(") || char == bytes1(")"); + return char == 0x00 || char == bytes1(" ") || char == bytes1(",") || char == bytes1(")"); + } + + function _isForbiddenFirstChar(bytes1 char) private pure returns (bool) { + return (char > 0x60 && char < 0x7b) || char == bytes1("("); } } diff --git a/test/account/Account.behavior.js b/test/account/Account.behavior.js index 9e000155..f61a45a0 100644 --- a/test/account/Account.behavior.js +++ b/test/account/Account.behavior.js @@ -15,7 +15,7 @@ function shouldBehaveLikeAccountCore() { describe('entryPoint', function () { it('should return the canonical entrypoint', async function () { await this.mock.deploy(); - expect(this.mock.entryPoint()).to.eventually.equal(entrypoint); + await expect(this.mock.entryPoint()).to.eventually.equal(entrypoint); }); }); @@ -106,7 +106,7 @@ function shouldBehaveLikeAccountHolder() { 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); - expect( + await expect( this.token.balanceOfBatch( ids.map(() => this.mock), ids, @@ -115,7 +115,7 @@ function shouldBehaveLikeAccountHolder() { }); it('receives ERC1155 tokens from a multiple IDs', async function () { - expect( + await expect( this.token.balanceOfBatch( ids.map(() => this.mock), ids, @@ -123,7 +123,7 @@ function shouldBehaveLikeAccountHolder() { ).to.eventually.deep.equal(ids.map(() => 0n)); await this.token.connect(this.other).safeBatchTransferFrom(this.other, this.mock, ids, values, data); - expect( + await expect( this.token.balanceOfBatch( ids.map(() => this.mock), ids, @@ -143,7 +143,7 @@ function shouldBehaveLikeAccountHolder() { it('receives an ERC721 token', async function () { await this.token.connect(this.other).safeTransferFrom(this.other, this.mock, tokenId); - expect(this.token.ownerOf(tokenId)).to.eventually.equal(this.mock); + await expect(this.token.ownerOf(tokenId)).to.eventually.equal(this.mock); }); }); }); @@ -156,7 +156,7 @@ function shouldBehaveLikeAccountERC7821({ deployable = true } = {}) { await setBalance(this.mock.target, ethers.parseEther('1')); // account is not initially deployed - expect(ethers.provider.getCode(this.mock)).to.eventually.equal('0x'); + await expect(ethers.provider.getCode(this.mock)).to.eventually.equal('0x'); this.encodeUserOpCalldata = (...calls) => this.mock.interface.encodeFunctionData('execute', [ @@ -195,13 +195,14 @@ function shouldBehaveLikeAccountERC7821({ deployable = true } = {}) { .then(op => op.addInitCode()) .then(op => this.signUserOp(op)); - expect(this.mock.getNonce()).to.eventually.equal(0); + // Can't call the account to get its nonce before it's deployed + await expect(entrypoint.getNonce(this.mock.target, 0)).to.eventually.equal(0); await expect(entrypoint.handleOps([operation.packed], this.beneficiary)) .to.emit(entrypoint, 'AccountDeployed') .withArgs(operation.hash(), this.mock, this.factory, ethers.ZeroAddress) .to.emit(this.target, 'MockFunctionCalledExtra') .withArgs(this.mock, 17); - expect(this.mock.getNonce()).to.eventually.equal(1); + await expect(this.mock.getNonce()).to.eventually.equal(1); }); it('should revert if the signature is invalid', async function () { @@ -238,11 +239,11 @@ function shouldBehaveLikeAccountERC7821({ deployable = true } = {}) { }) .then(op => this.signUserOp(op)); - expect(this.mock.getNonce()).to.eventually.equal(0); + await expect(this.mock.getNonce()).to.eventually.equal(0); await expect(entrypoint.handleOps([operation.packed], this.beneficiary)) .to.emit(this.target, 'MockFunctionCalledExtra') .withArgs(this.mock, 42); - expect(this.mock.getNonce()).to.eventually.equal(1); + await expect(this.mock.getNonce()).to.eventually.equal(1); }); it('should support sending eth to an EOA', async function () { @@ -250,12 +251,12 @@ function shouldBehaveLikeAccountERC7821({ deployable = true } = {}) { .createUserOp({ callData: this.encodeUserOpCalldata({ target: this.other, value }) }) .then(op => this.signUserOp(op)); - expect(this.mock.getNonce()).to.eventually.equal(0); + await expect(this.mock.getNonce()).to.eventually.equal(0); await expect(entrypoint.handleOps([operation.packed], this.beneficiary)).to.changeEtherBalance( this.other, value, ); - expect(this.mock.getNonce()).to.eventually.equal(1); + await expect(this.mock.getNonce()).to.eventually.equal(1); }); it('should support batch execution', async function () { @@ -275,11 +276,11 @@ function shouldBehaveLikeAccountERC7821({ deployable = true } = {}) { }) .then(op => this.signUserOp(op)); - expect(this.mock.getNonce()).to.eventually.equal(0); + await expect(this.mock.getNonce()).to.eventually.equal(0); const tx = entrypoint.handleOps([operation.packed], this.beneficiary); await expect(tx).to.changeEtherBalances([this.other, this.target], [value1, value2]); await expect(tx).to.emit(this.target, 'MockFunctionCalledExtra').withArgs(this.mock, value2); - expect(this.mock.getNonce()).to.eventually.equal(1); + await expect(this.mock.getNonce()).to.eventually.equal(1); }); }); }); diff --git a/test/crosschain/axelar/AxelarGateway.test.js b/test/crosschain/axelar/AxelarGateway.test.js index 0de9d6ae..b2802f07 100644 --- a/test/crosschain/axelar/AxelarGateway.test.js +++ b/test/crosschain/axelar/AxelarGateway.test.js @@ -32,13 +32,13 @@ describe('AxelarGateway', function () { }); it('initial setup', async function () { - expect(this.srcGateway.localGateway()).to.eventually.equal(this.axelar); - expect(this.srcGateway.getEquivalentChain(this.CAIP2)).to.eventually.equal('local'); - expect(this.srcGateway.getRemoteGateway(this.CAIP2)).to.eventually.equal(getAddress(this.dstGateway)); + await expect(this.srcGateway.localGateway()).to.eventually.equal(this.axelar); + await expect(this.srcGateway.getEquivalentChain(this.CAIP2)).to.eventually.equal('local'); + await expect(this.srcGateway.getRemoteGateway(this.CAIP2)).to.eventually.equal(getAddress(this.dstGateway)); - expect(this.dstGateway.localGateway()).to.eventually.equal(this.axelar); - expect(this.dstGateway.getEquivalentChain(this.CAIP2)).to.eventually.equal('local'); - expect(this.dstGateway.getRemoteGateway(this.CAIP2)).to.eventually.equal(getAddress(this.srcGateway)); + await expect(this.dstGateway.localGateway()).to.eventually.equal(this.axelar); + await expect(this.dstGateway.getEquivalentChain(this.CAIP2)).to.eventually.equal('local'); + await expect(this.dstGateway.getRemoteGateway(this.CAIP2)).to.eventually.equal(getAddress(this.srcGateway)); }); it('workflow', async function () { diff --git a/test/helpers/erc7739.js b/test/helpers/erc7739.js index f27618c3..63b873af 100644 --- a/test/helpers/erc7739.js +++ b/test/helpers/erc7739.js @@ -77,7 +77,7 @@ class TypedDataSignHelper { ethers.concat([ signature, ethers.TypedDataEncoder.hashDomain(domain), // appDomainSeparator - ethers.TypedDataEncoder.hashStruct(this.contentsTypeName, this.allTypes, message.contents), // contentsHash + this.hashStruct(this.contentsTypeName, message.contents), // contentsHash ethers.toUtf8Bytes(this.contentDescr), ethers.toBeHex(this.contentDescr.length, 2), ]), diff --git a/test/utils/cryptography/ERC7739Signer.behavior.js b/test/utils/cryptography/ERC7739Signer.behavior.js index b6b4055c..4923f561 100644 --- a/test/utils/cryptography/ERC7739Signer.behavior.js +++ b/test/utils/cryptography/ERC7739Signer.behavior.js @@ -19,14 +19,14 @@ function shouldBehaveLikeERC7739Signer() { const hash = PersonalSignHelper.hash(text); const signature = await PersonalSignHelper.sign(this.signTypedData, text, this.domain); - expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); + await expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); }); it('returns false for an invalid personal signature', async function () { const hash = PersonalSignHelper.hash('Message the app expects'); const signature = await PersonalSignHelper.sign(this.signTypedData, 'Message signed is different', this.domain); - expect(this.mock.isValidSignature(hash, signature)).to.eventually.not.equal(MAGIC_VALUE); + await expect(this.mock.isValidSignature(hash, signature)).to.eventually.not.equal(MAGIC_VALUE); }); }); @@ -56,7 +56,7 @@ function shouldBehaveLikeERC7739Signer() { const hash = ethers.TypedDataEncoder.hash(this.appDomain, { Permit }, message.contents); const signature = await TypedDataSignHelper.sign(this.signTypedData, this.appDomain, { Permit }, message); - expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); + await expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); }); it('returns true for valid typed data signature (nested types)', async function () { @@ -72,7 +72,7 @@ function shouldBehaveLikeERC7739Signer() { const hash = TypedDataSignHelper.hash(this.appDomain, contentsTypes, message.contents); const signature = await TypedDataSignHelper.sign(this.signTypedData, this.appDomain, contentsTypes, message); - expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); + await expect(this.mock.isValidSignature(hash, signature)).to.eventually.equal(MAGIC_VALUE); }); it('returns false for an invalid typed data signature', async function () { @@ -89,14 +89,13 @@ function shouldBehaveLikeERC7739Signer() { const hash = ethers.TypedDataEncoder.hash(this.appDomain, { Permit }, appContents); const signature = await TypedDataSignHelper.sign(this.signTypedData, this.appDomain, { Permit }, message); - expect(this.mock.isValidSignature(hash, signature)).to.eventually.not.equal(MAGIC_VALUE); + await expect(this.mock.isValidSignature(hash, signature)).to.eventually.not.equal(MAGIC_VALUE); }); }); - it('support detection', function () { - expect( - this.mock.isValidSignature('0x7739773977397739773977397739773977397739773977397739773977397739', ''), - ).to.eventually.equal('0x77390001'); + it('support detection', async function () { + const hash = '0x7739773977397739773977397739773977397739773977397739773977397739'; + await expect(this.mock.isValidSignature(hash, '0x')).to.eventually.equal('0x77390001'); }); }); } diff --git a/test/utils/cryptography/ERC7739Signer.test.js b/test/utils/cryptography/ERC7739Signer.test.js index 8fe0b711..05337e34 100644 --- a/test/utils/cryptography/ERC7739Signer.test.js +++ b/test/utils/cryptography/ERC7739Signer.test.js @@ -1,6 +1,6 @@ const { ethers } = require('hardhat'); const { shouldBehaveLikeERC7739Signer } = require('./ERC7739Signer.behavior'); -const { NonNativeSigner, P256SigningKey, RSASigningKey } = require('../../helpers/signers'); +const { NonNativeSigner, P256SigningKey, RSASHA256SigningKey } = require('../../helpers/signers'); describe('ERC7739Signer', function () { describe('for an ECDSA signer', function () { @@ -26,7 +26,7 @@ describe('ERC7739Signer', function () { describe('for an RSA signer', function () { before(async function () { - this.signer = new NonNativeSigner(RSASigningKey.random()); + this.signer = new NonNativeSigner(RSASHA256SigningKey.random()); this.mock = await ethers.deployContract('ERC7739SignerRSAMock', [ this.signer.signingKey.publicKey.e, this.signer.signingKey.publicKey.n, diff --git a/test/utils/cryptography/ERC7739Utils.test.js b/test/utils/cryptography/ERC7739Utils.test.js index 91c17ee6..1a959e31 100644 --- a/test/utils/cryptography/ERC7739Utils.test.js +++ b/test/utils/cryptography/ERC7739Utils.test.js @@ -2,26 +2,12 @@ const { expect } = require('chai'); const { ethers } = require('hardhat'); const { loadFixture } = require('@nomicfoundation/hardhat-network-helpers'); -const { Permit, domainType } = require('@openzeppelin/contracts/test/helpers/eip712'); +const { Permit } = require('@openzeppelin/contracts/test/helpers/eip712'); const { PersonalSignHelper, TypedDataSignHelper } = require('../../helpers/erc7739'); // Helper for ERC20Permit applications const helper = TypedDataSignHelper.from({ Permit }); -function domainComponentsType(domain) { - return ethers.TypedDataEncoder.from({ EIP712Domain: domainType(domain) }) - .encodeType('EIP712Domain') - .replace(/EIP712Domain\((.*)\)/, (_, s) => s); -} - -function domainComponentsBytes(domain) { - return ethers.hexlify( - ethers - .getBytes(ethers.TypedDataEncoder.from({ EIP712Domain: domainType(domain) }).encodeData('EIP712Domain', domain)) - .slice(32), - ); -} - const fixture = async () => { const mock = await ethers.deployContract('$ERC7739Utils'); const domain = { @@ -65,9 +51,9 @@ describe('ERC7739Utils', function () { ethers.toBeHex(contentDescr.length, 2), ]); - expect(this.mock.$encodeTypedDataSig(signature, appSeparator, contentsHash, contentDescr)).to.eventually.equal( - encoded, - ); + await expect( + this.mock.$encodeTypedDataSig(signature, appSeparator, contentsHash, contentDescr), + ).to.eventually.equal(encoded); }); }); @@ -85,7 +71,7 @@ describe('ERC7739Utils', function () { ethers.toBeHex(contentDescr.length, 2), ]); - expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ + await expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ ethers.hexlify(signature), appSeparator, contentsHash, @@ -95,7 +81,7 @@ describe('ERC7739Utils', function () { it('returns default empty values if the signature is too short', async function () { const encoded = ethers.randomBytes(65); // DOMAIN_SEPARATOR (32 bytes) + CONTENTS (32 bytes) + CONTENTS_TYPE_LENGTH (2 bytes) - 1 - expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ + await expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ '0x', ethers.ZeroHash, ethers.ZeroHash, @@ -105,7 +91,7 @@ describe('ERC7739Utils', function () { it('returns default empty values if the length is invalid', async function () { const encoded = ethers.concat([ethers.randomBytes(64), '0x3f']); // Can't be less than 64 bytes - expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ + await expect(this.mock.$decodeTypedDataSig(encoded)).to.eventually.deep.equal([ '0x', ethers.ZeroHash, ethers.ZeroHash, @@ -118,7 +104,7 @@ describe('ERC7739Utils', function () { it('should produce a personal signature EIP-712 nested type', async function () { const text = 'Hello, world!'; - expect(this.mock.$personalSignStructHash(ethers.hashMessage(text))).to.eventually.equal( + await expect(this.mock.$personalSignStructHash(ethers.hashMessage(text))).to.eventually.equal( ethers.TypedDataEncoder.hashStruct('PersonalSign', PersonalSignHelper.types, PersonalSignHelper.prepare(text)), ); }); @@ -131,14 +117,28 @@ describe('ERC7739Utils', function () { const contentsHash = helper.hashStruct('Permit', message.contents); const hash = helper.hashStruct('TypedDataSign', message); - expect( + const domainBytes = ethers.AbiCoder.defaultAbiCoder().encode( + ['bytes32', 'bytes32', 'uint256', 'address', 'bytes32'], + [ + ethers.id(this.domain.name), + ethers.id(this.domain.version), + this.domain.chainId, + this.domain.verifyingContract, + ethers.ZeroHash, + ], + ); + + await expect( this.mock.$typedDataSignStructHash( - helper.contentDescr, + helper.contentsTypeName, + ethers.Typed.string(helper.contentDescr), contentsHash, - domainComponentsType(this.domain), - domainComponentsBytes(this.domain), + domainBytes, ), ).to.eventually.equal(hash); + await expect( + this.mock.$typedDataSignStructHash(helper.contentDescr, contentsHash, domainBytes), + ).to.eventually.equal(hash); }); }); @@ -146,7 +146,7 @@ describe('ERC7739Utils', function () { it('should match', async function () { const typedDataSignType = ethers.TypedDataEncoder.from(helper.allTypes).encodeType('TypedDataSign'); - expect( + await expect( this.mock.$typedDataSignTypehash( helper.contentsTypeName, typedDataSignType.slice(typedDataSignType.indexOf(')') + 1), @@ -203,7 +203,7 @@ describe('ERC7739Utils', function () { })), )) { it(descr, async function () { - expect(this.mock.$decodeContentsDescr(contentDescr)).to.eventually.deep.equal([ + await expect(this.mock.$decodeContentsDescr(contentDescr)).to.eventually.deep.equal([ contentTypeName ?? '', contentTypeName ? contentType ?? contentDescr : '', ]); From 8aab665f197656c58a27ba42490f3a8ac8d0536f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ernesto=20Garc=C3=ADa?= Date: Fri, 27 Dec 2024 10:34:33 -0600 Subject: [PATCH 2/4] Update contracts/utils/cryptography/ERC7739Utils.sol --- contracts/utils/cryptography/ERC7739Utils.sol | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/utils/cryptography/ERC7739Utils.sol b/contracts/utils/cryptography/ERC7739Utils.sol index 36f2f7ad..cfad0b42 100644 --- a/contracts/utils/cryptography/ERC7739Utils.sol +++ b/contracts/utils/cryptography/ERC7739Utils.sol @@ -178,7 +178,7 @@ library ERC7739Utils { break; } // we found the end of the contentsTypeName - return (string(buffer[:i]), string(buffer)); + return (string(buffer[:i]), contentsDescr); } else if (_isForbiddenChar(current)) { // we found an invalid character (forbidden) - passthrough (fail) break; From 78801f5d0808a23a1d8f55dae653ae10199fb25b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ernesto=20Garc=C3=ADa?= Date: Fri, 27 Dec 2024 14:19:02 -0600 Subject: [PATCH 3/4] Update contracts/utils/cryptography/ERC7739Utils.sol Co-authored-by: Arr00 <13561405+arr00@users.noreply.github.com> --- contracts/utils/cryptography/ERC7739Utils.sol | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/utils/cryptography/ERC7739Utils.sol b/contracts/utils/cryptography/ERC7739Utils.sol index cfad0b42..8072cde8 100644 --- a/contracts/utils/cryptography/ERC7739Utils.sol +++ b/contracts/utils/cryptography/ERC7739Utils.sol @@ -157,7 +157,7 @@ library ERC7739Utils { * modes. * * Following ERC-7739 specifications, a `contentsTypeName` is considered invalid if it's empty, it contains - * any of the following bytes (`, )\x00`) or it starts with a forbidden character (a-z or `(`). + * any of the following bytes (`, )\x00`), or it starts with a forbidden character (a-z or `(`). * * If the `contentsType` is invalid, this returns an empty string. Otherwise, the return string has non-zero * length. From d1b12895028089ebcca19a27a031f702ddeca3c2 Mon Sep 17 00:00:00 2001 From: ernestognw Date: Sat, 28 Dec 2024 12:21:48 -0600 Subject: [PATCH 4/4] Revert ERC7739Utils changes --- contracts/utils/cryptography/ERC7739Utils.sol | 20 +++++-------------- test/utils/cryptography/ERC7739Utils.test.js | 17 ++++++++-------- 2 files changed, 13 insertions(+), 24 deletions(-) diff --git a/contracts/utils/cryptography/ERC7739Utils.sol b/contracts/utils/cryptography/ERC7739Utils.sol index 8072cde8..eed6acd2 100644 --- a/contracts/utils/cryptography/ERC7739Utils.sol +++ b/contracts/utils/cryptography/ERC7739Utils.sol @@ -156,8 +156,8 @@ library ERC7739Utils { * @dev Parse the type name out of the ERC-7739 contents type description. Supports both the implicit and explicit * modes. * - * Following ERC-7739 specifications, a `contentsTypeName` is considered invalid if it's empty, it contains - * any of the following bytes (`, )\x00`), or it starts with a forbidden character (a-z or `(`). + * Following ERC-7739 specifications, a `contentsTypeName` is considered invalid if it's empty or it contains + * any of the following bytes , )\x00 * * If the `contentsType` is invalid, this returns an empty string. Otherwise, the return string has non-zero * length. @@ -173,10 +173,8 @@ library ERC7739Utils { for (uint256 i = 0; i < buffer.length; ++i) { bytes1 current = buffer[i]; if (current == bytes1("(")) { - if (_isForbiddenFirstChar(buffer[0])) { - // we found an invalid first character (forbidden) - passthrough (fail) - break; - } + // if name is empty - passthrough (fail) + if (i == 0) break; // we found the end of the contentsTypeName return (string(buffer[:i]), contentsDescr); } else if (_isForbiddenChar(current)) { @@ -189,10 +187,6 @@ library ERC7739Utils { for (uint256 i = buffer.length; i > 0; --i) { bytes1 current = buffer[i - 1]; if (current == bytes1(")")) { - if (i >= buffer.length || _isForbiddenFirstChar(buffer[i])) { - // we found an invalid first character (forbidden) - passthrough (fail) - break; - } // we found the end of the contentsTypeName return (string(buffer[i:]), string(buffer[:i])); } else if (_isForbiddenChar(current)) { @@ -221,10 +215,6 @@ library ERC7739Utils { } function _isForbiddenChar(bytes1 char) private pure returns (bool) { - return char == 0x00 || char == bytes1(" ") || char == bytes1(",") || char == bytes1(")"); - } - - function _isForbiddenFirstChar(bytes1 char) private pure returns (bool) { - return (char > 0x60 && char < 0x7b) || char == bytes1("("); + return char == 0x00 || char == bytes1(" ") || char == bytes1(",") || char == bytes1("(") || char == bytes1(")"); } } diff --git a/test/utils/cryptography/ERC7739Utils.test.js b/test/utils/cryptography/ERC7739Utils.test.js index 1a959e31..cec843f4 100644 --- a/test/utils/cryptography/ERC7739Utils.test.js +++ b/test/utils/cryptography/ERC7739Utils.test.js @@ -156,7 +156,6 @@ describe('ERC7739Utils', function () { }); describe('decodeContentsDescr', function () { - const forbiddenFirstChars = 'abcdefghijklmnopqrstuvwxyz('; const forbiddenChars = ', )\x00'; for (const { descr, contentDescr, contentTypeName, contentType } of [].concat( @@ -181,16 +180,16 @@ describe('ERC7739Utils', function () { contentDescr: 'SomeType', contentTypeName: null, }, - forbiddenFirstChars.split('').map(char => ({ - descr: `should return nothing if starts with [${char}] (implicit)`, - contentDescr: `${char}SomeType()`, + { + descr: 'should return nothing if starts with [(] (implicit)', + contentDescr: '(SomeType(address foo,uint256 bar)', contentTypeName: null, - })), - forbiddenFirstChars.split('').map(char => ({ - descr: `should return nothing if starts with [${char}] (explicit)`, - contentDescr: `${char}SomeType()${char}SomeType`, + }, + { + descr: 'should return nothing if starts with [(] (explicit)', + contentDescr: '(SomeType(address foo,uint256 bar)(SomeType', contentTypeName: null, - })), + }, forbiddenChars.split('').map(char => ({ descr: `should return nothing if contains [${char}] (implicit)`, contentDescr: `SomeType${char}(address foo,uint256 bar)`,