From c5dbb372dc0514fa4009d7f3d2eb9bdc76a8369f Mon Sep 17 00:00:00 2001 From: Matthew Walsh Date: Fri, 1 Mar 2024 10:43:23 +0000 Subject: [PATCH] Normalize transaction data length (#3990) Normalize transaction data by padding to even length. Export `normalizeTransactionParams` method. --- .../transaction-controller/jest.config.js | 8 +- .../src/TransactionController.test.ts | 2 +- .../src/TransactionController.ts | 11 +- packages/transaction-controller/src/index.ts | 5 +- .../src/utils/utils.test.ts | 109 +++++++++++------- .../transaction-controller/src/utils/utils.ts | 19 ++- 6 files changed, 103 insertions(+), 51 deletions(-) diff --git a/packages/transaction-controller/jest.config.js b/packages/transaction-controller/jest.config.js index 10fa492868..1a339c1e35 100644 --- a/packages/transaction-controller/jest.config.js +++ b/packages/transaction-controller/jest.config.js @@ -17,10 +17,10 @@ module.exports = merge(baseConfig, { // An object that configures minimum threshold enforcement for coverage results coverageThreshold: { global: { - branches: 91.79, - functions: 98.56, - lines: 98.9, - statements: 98.91, + branches: 91.89, + functions: 98.86, + lines: 98.96, + statements: 98.97, }, }, diff --git a/packages/transaction-controller/src/TransactionController.test.ts b/packages/transaction-controller/src/TransactionController.test.ts index eb2f5af802..681dfe467b 100644 --- a/packages/transaction-controller/src/TransactionController.test.ts +++ b/packages/transaction-controller/src/TransactionController.test.ts @@ -5023,7 +5023,7 @@ describe('TransactionController', () => { describe('updateEditableParams', () => { const transactionId = '1'; const params = { - data: '0x0', + data: '0x12', from: ACCOUNT_2_MOCK, gas: '0x0', gasPrice: '0x50fd51da', diff --git a/packages/transaction-controller/src/TransactionController.ts b/packages/transaction-controller/src/TransactionController.ts index 2b1efd48df..b4fb6573f4 100644 --- a/packages/transaction-controller/src/TransactionController.ts +++ b/packages/transaction-controller/src/TransactionController.ts @@ -92,7 +92,7 @@ import { import { determineTransactionType } from './utils/transaction-type'; import { getIncreasedPriceFromExisting, - normalizeTxParams, + normalizeTransactionParams, isEIP1559Transaction, isFeeMarketEIP1559Values, isGasPriceValue, @@ -699,7 +699,7 @@ export class TransactionController extends BaseControllerV1< ): Promise { log('Adding transaction', txParams); - txParams = normalizeTxParams(txParams); + txParams = normalizeTransactionParams(txParams); if ( networkClientId && !this.#multichainTrackingHelper.has(networkClientId) @@ -1932,7 +1932,8 @@ export class TransactionController extends BaseControllerV1< throw new Error('No sign method defined.'); } - const normalizedTransactionParams = normalizeTxParams(transactionParams); + const normalizedTransactionParams = + normalizeTransactionParams(transactionParams); const type = isEIP1559Transaction(normalizedTransactionParams) ? TransactionEnvelopeType.feeMarket : TransactionEnvelopeType.legacy; @@ -3083,7 +3084,9 @@ export class TransactionController extends BaseControllerV1< ) { const { transactions } = this.state; - transactionMeta.txParams = normalizeTxParams(transactionMeta.txParams); + transactionMeta.txParams = normalizeTransactionParams( + transactionMeta.txParams, + ); validateTxParams(transactionMeta.txParams); diff --git a/packages/transaction-controller/src/index.ts b/packages/transaction-controller/src/index.ts index ea4f0a914c..c434c29b9f 100644 --- a/packages/transaction-controller/src/index.ts +++ b/packages/transaction-controller/src/index.ts @@ -1,6 +1,9 @@ export * from './TransactionController'; export type { EtherscanTransactionMeta } from './utils/etherscan'; -export { isEIP1559Transaction } from './utils/utils'; +export { + isEIP1559Transaction, + normalizeTransactionParams, +} from './utils/utils'; export * from './types'; export { determineTransactionType } from './utils/transaction-type'; export { mergeGasFeeEstimates } from './utils/gas-flow'; diff --git a/packages/transaction-controller/src/utils/utils.test.ts b/packages/transaction-controller/src/utils/utils.test.ts index 3377855b75..14e3374ac4 100644 --- a/packages/transaction-controller/src/utils/utils.test.ts +++ b/packages/transaction-controller/src/utils/utils.test.ts @@ -11,29 +11,30 @@ const GAS_PRICE = 'gasPrice'; const FAIL = 'lol'; const PASS = '0x1'; +const TRANSACTION_PARAMS_MOCK: TransactionParams = { + data: 'data', + from: 'FROM', + gas: 'gas', + gasPrice: 'gasPrice', + nonce: 'nonce', + to: 'TO', + value: 'value', + maxFeePerGas: 'maxFeePerGas', + maxPriorityFeePerGas: 'maxPriorityFeePerGas', + estimatedBaseFee: 'estimatedBaseFee', +}; + describe('utils', () => { afterEach(() => { jest.clearAllMocks(); }); - describe('normalizeTxParams', () => { - const commonInput = { - data: 'data', - from: 'FROM', - gas: 'gas', - gasPrice: 'gasPrice', - nonce: 'nonce', - to: 'TO', - value: 'value', - maxFeePerGas: 'maxFeePerGas', - maxPriorityFeePerGas: 'maxPriorityFeePerGas', - estimatedBaseFee: 'estimatedBaseFee', - }; + describe('normalizeTransactionParams', () => { + it('normalizes properties', () => { + const normalized = util.normalizeTransactionParams( + TRANSACTION_PARAMS_MOCK, + ); - it('normalizeTransaction', () => { - const normalized = util.normalizeTxParams({ - ...commonInput, - }); expect(normalized).toStrictEqual({ data: '0xdata', from: '0xfrom', @@ -48,31 +49,35 @@ describe('utils', () => { }); }); - it('normalizeTransaction if type is zero', () => { - const normalized = util.normalizeTxParams({ - ...commonInput, - type: '0x0', - }); - expect(normalized).toStrictEqual({ - data: '0xdata', - from: '0xfrom', - gas: '0xgas', - gasPrice: '0xgasPrice', - nonce: '0xnonce', - to: '0xto', - value: '0xvalue', - maxFeePerGas: '0xmaxFeePerGas', - maxPriorityFeePerGas: '0xmaxPriorityFeePerGas', - estimatedBaseFee: '0xestimatedBaseFee', - type: '0x0', - }); + it('retains legacy type if specified', () => { + expect( + util.normalizeTransactionParams({ + ...TRANSACTION_PARAMS_MOCK, + type: '0x0', + }), + ).toStrictEqual( + expect.objectContaining({ + type: '0x0', + }), + ); }); it('sets value if not specified', () => { - expect(util.normalizeTxParams({ from: '0xfrom' })).toStrictEqual({ - from: '0xfrom', - value: '0x0', - }); + expect( + util.normalizeTransactionParams({ + ...TRANSACTION_PARAMS_MOCK, + value: undefined, + }), + ).toStrictEqual(expect.objectContaining({ value: '0x0' })); + }); + + it('ensures data is even length prefixed hex string', () => { + expect( + util.normalizeTransactionParams({ + ...TRANSACTION_PARAMS_MOCK, + data: '123', + }), + ).toStrictEqual(expect.objectContaining({ data: '0x0123' })); }); }); @@ -248,4 +253,30 @@ describe('utils', () => { }); }); }); + + describe('padHexToEvenLength', () => { + it('returns same value if already even length and has prefix', () => { + expect(util.padHexToEvenLength('0x1234')).toBe('0x1234'); + }); + + it('returns same value if already even length and no prefix', () => { + expect(util.padHexToEvenLength('1234')).toBe('1234'); + }); + + it('returns padded value if not even length and has prefix', () => { + expect(util.padHexToEvenLength('0x123')).toBe('0x0123'); + }); + + it('returns padded value if not even length and no prefix', () => { + expect(util.padHexToEvenLength('123')).toBe('0123'); + }); + + it('returns same value if prefix only', () => { + expect(util.padHexToEvenLength('0x')).toBe('0x'); + }); + + it('returns padded value if zero', () => { + expect(util.padHexToEvenLength('0x0')).toBe('0x00'); + }); + }); }); diff --git a/packages/transaction-controller/src/utils/utils.ts b/packages/transaction-controller/src/utils/utils.ts index b4382414bf..7b14900295 100644 --- a/packages/transaction-controller/src/utils/utils.ts +++ b/packages/transaction-controller/src/utils/utils.ts @@ -21,7 +21,7 @@ export const ESTIMATE_GAS_ERROR = 'eth_estimateGas rpc method error'; // TODO: Replace `any` with type // eslint-disable-next-line @typescript-eslint/no-explicit-any const NORMALIZERS: { [param in keyof TransactionParams]: any } = { - data: (data: string) => add0x(data), + data: (data: string) => add0x(padHexToEvenLength(data)), from: (from: string) => add0x(from).toLowerCase(), gas: (gas: string) => add0x(gas), gasLimit: (gas: string) => add0x(gas), @@ -43,7 +43,7 @@ const NORMALIZERS: { [param in keyof TransactionParams]: any } = { * @param txParams - The transaction params to normalize. * @returns Normalized transaction params. */ -export function normalizeTxParams(txParams: TransactionParams) { +export function normalizeTransactionParams(txParams: TransactionParams) { const normalizedTxParams: TransactionParams = { from: '' }; for (const key of getKnownPropertyNames(NORMALIZERS)) { @@ -191,3 +191,18 @@ export function normalizeGasFeeValues( maxPriorityFeePerGas: normalize(gasFeeValues.maxPriorityFeePerGas), }; } + +/** + * Ensure a hex string is of even length by adding a leading 0 if necessary. + * Any existing `0x` prefix is preserved but is not added if missing. + * + * @param hex - The hex string to ensure is even. + * @returns The hex string with an even length. + */ +export function padHexToEvenLength(hex: string) { + const prefix = hex.toLowerCase().startsWith('0x') ? hex.slice(0, 2) : ''; + const data = prefix ? hex.slice(2) : hex; + const evenData = data.length % 2 === 0 ? data : `0${data}`; + + return prefix + evenData; +}