@metamask/transaction-controller 56.0.0 → 56.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +32 -1
- package/dist/TransactionController.cjs +64 -18
- package/dist/TransactionController.cjs.map +1 -1
- package/dist/TransactionController.d.cts +10 -2
- package/dist/TransactionController.d.cts.map +1 -1
- package/dist/TransactionController.d.mts +10 -2
- package/dist/TransactionController.d.mts.map +1 -1
- package/dist/TransactionController.mjs +66 -20
- package/dist/TransactionController.mjs.map +1 -1
- package/dist/api/accounts-api.cjs +3 -2
- package/dist/api/accounts-api.cjs.map +1 -1
- package/dist/api/accounts-api.d.cts +1 -0
- package/dist/api/accounts-api.d.cts.map +1 -1
- package/dist/api/accounts-api.d.mts +1 -0
- package/dist/api/accounts-api.d.mts.map +1 -1
- package/dist/api/accounts-api.mjs +3 -2
- package/dist/api/accounts-api.mjs.map +1 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.cjs +2 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.cjs.map +1 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.d.cts.map +1 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.d.mts.map +1 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.mjs +2 -1
- package/dist/helpers/AccountsApiRemoteTransactionSource.mjs.map +1 -1
- package/dist/helpers/GasFeePoller.cjs +76 -48
- package/dist/helpers/GasFeePoller.cjs.map +1 -1
- package/dist/helpers/GasFeePoller.d.cts +15 -3
- package/dist/helpers/GasFeePoller.d.cts.map +1 -1
- package/dist/helpers/GasFeePoller.d.mts +15 -3
- package/dist/helpers/GasFeePoller.d.mts.map +1 -1
- package/dist/helpers/GasFeePoller.mjs +73 -46
- package/dist/helpers/GasFeePoller.mjs.map +1 -1
- package/dist/helpers/IncomingTransactionHelper.cjs +22 -4
- package/dist/helpers/IncomingTransactionHelper.cjs.map +1 -1
- package/dist/helpers/IncomingTransactionHelper.d.cts +5 -2
- package/dist/helpers/IncomingTransactionHelper.d.cts.map +1 -1
- package/dist/helpers/IncomingTransactionHelper.d.mts +5 -2
- package/dist/helpers/IncomingTransactionHelper.d.mts.map +1 -1
- package/dist/helpers/IncomingTransactionHelper.mjs +22 -4
- package/dist/helpers/IncomingTransactionHelper.mjs.map +1 -1
- package/dist/helpers/PendingTransactionTracker.cjs +34 -3
- package/dist/helpers/PendingTransactionTracker.cjs.map +1 -1
- package/dist/helpers/PendingTransactionTracker.d.cts +12 -0
- package/dist/helpers/PendingTransactionTracker.d.cts.map +1 -1
- package/dist/helpers/PendingTransactionTracker.d.mts +12 -0
- package/dist/helpers/PendingTransactionTracker.d.mts.map +1 -1
- package/dist/helpers/PendingTransactionTracker.mjs +34 -3
- package/dist/helpers/PendingTransactionTracker.mjs.map +1 -1
- package/dist/hooks/SequentialPublishBatchHook.cjs +114 -0
- package/dist/hooks/SequentialPublishBatchHook.cjs.map +1 -0
- package/dist/hooks/SequentialPublishBatchHook.d.cts +24 -0
- package/dist/hooks/SequentialPublishBatchHook.d.cts.map +1 -0
- package/dist/hooks/SequentialPublishBatchHook.d.mts +24 -0
- package/dist/hooks/SequentialPublishBatchHook.d.mts.map +1 -0
- package/dist/hooks/SequentialPublishBatchHook.mjs +110 -0
- package/dist/hooks/SequentialPublishBatchHook.mjs.map +1 -0
- package/dist/types.cjs.map +1 -1
- package/dist/types.d.cts +4 -0
- package/dist/types.d.cts.map +1 -1
- package/dist/types.d.mts +4 -0
- package/dist/types.d.mts.map +1 -1
- package/dist/types.mjs.map +1 -1
- package/dist/utils/batch.cjs +12 -6
- package/dist/utils/batch.cjs.map +1 -1
- package/dist/utils/batch.d.cts +3 -0
- package/dist/utils/batch.d.cts.map +1 -1
- package/dist/utils/batch.d.mts +3 -0
- package/dist/utils/batch.d.mts.map +1 -1
- package/dist/utils/batch.mjs +12 -6
- package/dist/utils/batch.mjs.map +1 -1
- package/dist/utils/gas-fees.cjs +3 -7
- package/dist/utils/gas-fees.cjs.map +1 -1
- package/dist/utils/gas-fees.d.cts.map +1 -1
- package/dist/utils/gas-fees.d.mts.map +1 -1
- package/dist/utils/gas-fees.mjs +3 -7
- package/dist/utils/gas-fees.mjs.map +1 -1
- package/dist/utils/transaction-type.cjs +4 -1
- package/dist/utils/transaction-type.cjs.map +1 -1
- package/dist/utils/transaction-type.d.cts.map +1 -1
- package/dist/utils/transaction-type.d.mts.map +1 -1
- package/dist/utils/transaction-type.mjs +4 -1
- package/dist/utils/transaction-type.mjs.map +1 -1
- package/dist/utils/validation.cjs +1 -0
- package/dist/utils/validation.cjs.map +1 -1
- package/dist/utils/validation.d.cts +2 -1
- package/dist/utils/validation.d.cts.map +1 -1
- package/dist/utils/validation.d.mts +2 -1
- package/dist/utils/validation.d.mts.map +1 -1
- package/dist/utils/validation.mjs +1 -0
- package/dist/utils/validation.mjs.map +1 -1
- package/package.json +3 -3
|
@@ -9,7 +9,7 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
|
|
|
9
9
|
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
10
10
|
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
11
11
|
};
|
|
12
|
-
var _PendingTransactionTracker_instances, _PendingTransactionTracker_droppedBlockCountByHash, _PendingTransactionTracker_getChainId, _PendingTransactionTracker_getEthQuery, _PendingTransactionTracker_getNetworkClientId, _PendingTransactionTracker_getTransactions, _PendingTransactionTracker_isResubmitEnabled, _PendingTransactionTracker_listener, _PendingTransactionTracker_log, _PendingTransactionTracker_getGlobalLock, _PendingTransactionTracker_publishTransaction, _PendingTransactionTracker_running, _PendingTransactionTracker_transactionPoller, _PendingTransactionTracker_beforeCheckPendingTransaction, _PendingTransactionTracker_start, _PendingTransactionTracker_onLatestBlock, _PendingTransactionTracker_checkTransactions, _PendingTransactionTracker_resubmitTransactions, _PendingTransactionTracker_isKnownTransactionError, _PendingTransactionTracker_resubmitTransaction, _PendingTransactionTracker_isResubmitDue, _PendingTransactionTracker_checkTransaction, _PendingTransactionTracker_onTransactionConfirmed, _PendingTransactionTracker_isTransactionDropped, _PendingTransactionTracker_isNonceTaken, _PendingTransactionTracker_getPendingTransactions, _PendingTransactionTracker_warnTransaction, _PendingTransactionTracker_failTransaction, _PendingTransactionTracker_dropTransaction, _PendingTransactionTracker_updateTransaction, _PendingTransactionTracker_getTransactionReceipt, _PendingTransactionTracker_getBlockByHash, _PendingTransactionTracker_getNetworkTransactionCount, _PendingTransactionTracker_getChainTransactions, _PendingTransactionTracker_getNetworkClientTransactions;
|
|
12
|
+
var _PendingTransactionTracker_instances, _PendingTransactionTracker_droppedBlockCountByHash, _PendingTransactionTracker_getChainId, _PendingTransactionTracker_getEthQuery, _PendingTransactionTracker_getNetworkClientId, _PendingTransactionTracker_getTransactions, _PendingTransactionTracker_isResubmitEnabled, _PendingTransactionTracker_listener, _PendingTransactionTracker_log, _PendingTransactionTracker_getGlobalLock, _PendingTransactionTracker_publishTransaction, _PendingTransactionTracker_running, _PendingTransactionTracker_transactionPoller, _PendingTransactionTracker_transactionToForcePoll, _PendingTransactionTracker_beforeCheckPendingTransaction, _PendingTransactionTracker_start, _PendingTransactionTracker_onLatestBlock, _PendingTransactionTracker_checkTransactions, _PendingTransactionTracker_resubmitTransactions, _PendingTransactionTracker_isKnownTransactionError, _PendingTransactionTracker_resubmitTransaction, _PendingTransactionTracker_isResubmitDue, _PendingTransactionTracker_cleanTransactionToForcePoll, _PendingTransactionTracker_checkTransaction, _PendingTransactionTracker_onTransactionConfirmed, _PendingTransactionTracker_isTransactionDropped, _PendingTransactionTracker_isNonceTaken, _PendingTransactionTracker_getPendingTransactions, _PendingTransactionTracker_warnTransaction, _PendingTransactionTracker_failTransaction, _PendingTransactionTracker_dropTransaction, _PendingTransactionTracker_updateTransaction, _PendingTransactionTracker_getTransactionReceipt, _PendingTransactionTracker_getBlockByHash, _PendingTransactionTracker_getNetworkTransactionCount, _PendingTransactionTracker_getChainTransactions, _PendingTransactionTracker_getNetworkClientTransactions;
|
|
13
13
|
import { query } from "@metamask/controller-utils";
|
|
14
14
|
// This package purposefully relies on Node's EventEmitter module.
|
|
15
15
|
// eslint-disable-next-line import-x/no-nodejs-modules
|
|
@@ -53,6 +53,7 @@ export class PendingTransactionTracker {
|
|
|
53
53
|
_PendingTransactionTracker_publishTransaction.set(this, void 0);
|
|
54
54
|
_PendingTransactionTracker_running.set(this, void 0);
|
|
55
55
|
_PendingTransactionTracker_transactionPoller.set(this, void 0);
|
|
56
|
+
_PendingTransactionTracker_transactionToForcePoll.set(this, void 0);
|
|
56
57
|
_PendingTransactionTracker_beforeCheckPendingTransaction.set(this, void 0);
|
|
57
58
|
this.startIfPendingTransactions = () => {
|
|
58
59
|
const pendingTransactions = __classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_getPendingTransactions).call(this);
|
|
@@ -74,6 +75,7 @@ export class PendingTransactionTracker {
|
|
|
74
75
|
__classPrivateFieldSet(this, _PendingTransactionTracker_getGlobalLock, getGlobalLock, "f");
|
|
75
76
|
__classPrivateFieldSet(this, _PendingTransactionTracker_publishTransaction, publishTransaction, "f");
|
|
76
77
|
__classPrivateFieldSet(this, _PendingTransactionTracker_running, false, "f");
|
|
78
|
+
__classPrivateFieldSet(this, _PendingTransactionTracker_transactionToForcePoll, undefined, "f");
|
|
77
79
|
__classPrivateFieldSet(this, _PendingTransactionTracker_transactionPoller, new TransactionPoller({
|
|
78
80
|
blockTracker,
|
|
79
81
|
chainId: getChainId(),
|
|
@@ -84,6 +86,21 @@ export class PendingTransactionTracker {
|
|
|
84
86
|
(() => Promise.resolve(true)), "f");
|
|
85
87
|
__classPrivateFieldSet(this, _PendingTransactionTracker_log, createModuleLogger(log, `${getChainId()}:${getNetworkClientId()}`), "f");
|
|
86
88
|
}
|
|
89
|
+
/**
|
|
90
|
+
* Adds a transaction to the polling mechanism for monitoring its status.
|
|
91
|
+
*
|
|
92
|
+
* This method forcefully adds a single transaction to the list of transactions
|
|
93
|
+
* being polled, ensuring that its status is checked, event emitted but no update is performed.
|
|
94
|
+
* It overrides the default behavior by prioritizing the given transaction for polling.
|
|
95
|
+
*
|
|
96
|
+
* @param transactionMeta - The transaction metadata to be added for polling.
|
|
97
|
+
*
|
|
98
|
+
* The transaction will now be monitored for updates, such as confirmation or failure.
|
|
99
|
+
*/
|
|
100
|
+
addTransactionToPoll(transactionMeta) {
|
|
101
|
+
__classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_start).call(this, [transactionMeta]);
|
|
102
|
+
__classPrivateFieldSet(this, _PendingTransactionTracker_transactionToForcePoll, transactionMeta, "f");
|
|
103
|
+
}
|
|
87
104
|
/**
|
|
88
105
|
* Force checks the network if the given transaction is confirmed and updates it's status.
|
|
89
106
|
*
|
|
@@ -111,7 +128,7 @@ export class PendingTransactionTracker {
|
|
|
111
128
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'Stopped polling');
|
|
112
129
|
}
|
|
113
130
|
}
|
|
114
|
-
_PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTransactionTracker_getChainId = new WeakMap(), _PendingTransactionTracker_getEthQuery = new WeakMap(), _PendingTransactionTracker_getNetworkClientId = new WeakMap(), _PendingTransactionTracker_getTransactions = new WeakMap(), _PendingTransactionTracker_isResubmitEnabled = new WeakMap(), _PendingTransactionTracker_listener = new WeakMap(), _PendingTransactionTracker_log = new WeakMap(), _PendingTransactionTracker_getGlobalLock = new WeakMap(), _PendingTransactionTracker_publishTransaction = new WeakMap(), _PendingTransactionTracker_running = new WeakMap(), _PendingTransactionTracker_transactionPoller = new WeakMap(), _PendingTransactionTracker_beforeCheckPendingTransaction = new WeakMap(), _PendingTransactionTracker_instances = new WeakSet(), _PendingTransactionTracker_start = function _PendingTransactionTracker_start(pendingTransactions) {
|
|
131
|
+
_PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTransactionTracker_getChainId = new WeakMap(), _PendingTransactionTracker_getEthQuery = new WeakMap(), _PendingTransactionTracker_getNetworkClientId = new WeakMap(), _PendingTransactionTracker_getTransactions = new WeakMap(), _PendingTransactionTracker_isResubmitEnabled = new WeakMap(), _PendingTransactionTracker_listener = new WeakMap(), _PendingTransactionTracker_log = new WeakMap(), _PendingTransactionTracker_getGlobalLock = new WeakMap(), _PendingTransactionTracker_publishTransaction = new WeakMap(), _PendingTransactionTracker_running = new WeakMap(), _PendingTransactionTracker_transactionPoller = new WeakMap(), _PendingTransactionTracker_transactionToForcePoll = new WeakMap(), _PendingTransactionTracker_beforeCheckPendingTransaction = new WeakMap(), _PendingTransactionTracker_instances = new WeakSet(), _PendingTransactionTracker_start = function _PendingTransactionTracker_start(pendingTransactions) {
|
|
115
132
|
__classPrivateFieldGet(this, _PendingTransactionTracker_transactionPoller, "f").setPendingTransactions(pendingTransactions);
|
|
116
133
|
if (__classPrivateFieldGet(this, _PendingTransactionTracker_running, "f")) {
|
|
117
134
|
return;
|
|
@@ -140,7 +157,10 @@ _PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTran
|
|
|
140
157
|
}
|
|
141
158
|
}, _PendingTransactionTracker_checkTransactions = async function _PendingTransactionTracker_checkTransactions() {
|
|
142
159
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'Checking transactions');
|
|
143
|
-
const pendingTransactions =
|
|
160
|
+
const pendingTransactions = [
|
|
161
|
+
...__classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_getPendingTransactions).call(this),
|
|
162
|
+
...(__classPrivateFieldGet(this, _PendingTransactionTracker_transactionToForcePoll, "f") ? [__classPrivateFieldGet(this, _PendingTransactionTracker_transactionToForcePoll, "f")] : []),
|
|
163
|
+
];
|
|
144
164
|
if (!pendingTransactions.length) {
|
|
145
165
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'No pending transactions to check');
|
|
146
166
|
return;
|
|
@@ -209,6 +229,10 @@ _PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTran
|
|
|
209
229
|
// Capped at ~15 minutes between retries
|
|
210
230
|
const requiredBlocksSinceFirstRetry = Math.min(MAX_RETRY_BLOCK_DISTANCE, Math.pow(2, retryCount));
|
|
211
231
|
return blocksSinceFirstRetry >= requiredBlocksSinceFirstRetry;
|
|
232
|
+
}, _PendingTransactionTracker_cleanTransactionToForcePoll = function _PendingTransactionTracker_cleanTransactionToForcePoll(transactionId) {
|
|
233
|
+
if (__classPrivateFieldGet(this, _PendingTransactionTracker_transactionToForcePoll, "f")?.id === transactionId) {
|
|
234
|
+
__classPrivateFieldSet(this, _PendingTransactionTracker_transactionToForcePoll, undefined, "f");
|
|
235
|
+
}
|
|
212
236
|
}, _PendingTransactionTracker_checkTransaction = async function _PendingTransactionTracker_checkTransaction(txMeta) {
|
|
213
237
|
const { hash, id } = txMeta;
|
|
214
238
|
if (!hash && (await __classPrivateFieldGet(this, _PendingTransactionTracker_beforeCheckPendingTransaction, "f").call(this, txMeta))) {
|
|
@@ -255,6 +279,11 @@ _PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTran
|
|
|
255
279
|
const { id } = txMeta;
|
|
256
280
|
const { blockHash } = receipt;
|
|
257
281
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'Transaction confirmed', id);
|
|
282
|
+
if (__classPrivateFieldGet(this, _PendingTransactionTracker_transactionToForcePoll, "f")) {
|
|
283
|
+
__classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_cleanTransactionToForcePoll).call(this, txMeta.id);
|
|
284
|
+
this.hub.emit('transaction-confirmed', txMeta);
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
258
287
|
const { baseFeePerGas, timestamp: blockTimestamp } = await __classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_getBlockByHash).call(this, blockHash, false);
|
|
259
288
|
const updatedTxMeta = cloneDeep(txMeta);
|
|
260
289
|
updatedTxMeta.baseFeePerGas = baseFeePerGas;
|
|
@@ -312,9 +341,11 @@ _PendingTransactionTracker_droppedBlockCountByHash = new WeakMap(), _PendingTran
|
|
|
312
341
|
}, 'PendingTransactionTracker:#warnTransaction - Warning added');
|
|
313
342
|
}, _PendingTransactionTracker_failTransaction = function _PendingTransactionTracker_failTransaction(txMeta, error) {
|
|
314
343
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'Transaction failed', txMeta.id, error);
|
|
344
|
+
__classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_cleanTransactionToForcePoll).call(this, txMeta.id);
|
|
315
345
|
this.hub.emit('transaction-failed', txMeta, error);
|
|
316
346
|
}, _PendingTransactionTracker_dropTransaction = function _PendingTransactionTracker_dropTransaction(txMeta) {
|
|
317
347
|
__classPrivateFieldGet(this, _PendingTransactionTracker_log, "f").call(this, 'Transaction dropped', txMeta.id);
|
|
348
|
+
__classPrivateFieldGet(this, _PendingTransactionTracker_instances, "m", _PendingTransactionTracker_cleanTransactionToForcePoll).call(this, txMeta.id);
|
|
318
349
|
this.hub.emit('transaction-dropped', txMeta);
|
|
319
350
|
}, _PendingTransactionTracker_updateTransaction = function _PendingTransactionTracker_updateTransaction(txMeta, note) {
|
|
320
351
|
this.hub.emit('transaction-updated', txMeta, note);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"PendingTransactionTracker.mjs","sourceRoot":"","sources":["../../src/helpers/PendingTransactionTracker.ts"],"names":[],"mappings":"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","sourcesContent":["import { query } from '@metamask/controller-utils';\nimport type EthQuery from '@metamask/eth-query';\nimport type {\n BlockTracker,\n NetworkClientId,\n} from '@metamask/network-controller';\nimport type { Hex } from '@metamask/utils';\n// This package purposefully relies on Node's EventEmitter module.\n// eslint-disable-next-line import-x/no-nodejs-modules\nimport EventEmitter from 'events';\nimport { cloneDeep, merge } from 'lodash';\n\nimport { TransactionPoller } from './TransactionPoller';\nimport { createModuleLogger, projectLogger } from '../logger';\nimport type { TransactionControllerMessenger } from '../TransactionController';\nimport type { TransactionMeta, TransactionReceipt } from '../types';\nimport { TransactionStatus, TransactionType } from '../types';\n\n/**\n * We wait this many blocks before emitting a 'transaction-dropped' event\n * This is because we could be talking to a node that is out of sync\n */\nconst DROPPED_BLOCK_COUNT = 3;\n\nconst RECEIPT_STATUS_SUCCESS = '0x1';\nconst RECEIPT_STATUS_FAILURE = '0x0';\nconst MAX_RETRY_BLOCK_DISTANCE = 50;\n\nconst KNOWN_TRANSACTION_ERRORS = [\n 'replacement transaction underpriced',\n 'known transaction',\n 'gas price too low to replace',\n 'transaction with the same hash was already imported',\n 'gateway timeout',\n 'nonce too low',\n];\n\nconst log = createModuleLogger(projectLogger, 'pending-transactions');\n\ntype SuccessfulTransactionReceipt = TransactionReceipt & {\n blockNumber: string;\n blockHash: string;\n};\n\ntype Events = {\n 'transaction-confirmed': [txMeta: TransactionMeta];\n 'transaction-dropped': [txMeta: TransactionMeta];\n 'transaction-failed': [txMeta: TransactionMeta, error: Error];\n 'transaction-updated': [txMeta: TransactionMeta, note: string];\n};\n\n// This interface was created before this ESLint rule was added.\n// Convert to a `type` in a future major version.\n// eslint-disable-next-line @typescript-eslint/consistent-type-definitions\nexport interface PendingTransactionTrackerEventEmitter extends EventEmitter {\n on<T extends keyof Events>(\n eventName: T,\n listener: (...args: Events[T]) => void,\n ): this;\n\n emit<T extends keyof Events>(eventName: T, ...args: Events[T]): boolean;\n}\n\nexport class PendingTransactionTracker {\n hub: PendingTransactionTrackerEventEmitter;\n\n readonly #droppedBlockCountByHash: Map<string, number>;\n\n readonly #getChainId: () => string;\n\n readonly #getEthQuery: (networkClientId?: NetworkClientId) => EthQuery;\n\n readonly #getNetworkClientId: () => NetworkClientId;\n\n readonly #getTransactions: () => TransactionMeta[];\n\n readonly #isResubmitEnabled: () => boolean;\n\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n readonly #listener: any;\n\n readonly #log: debug.Debugger;\n\n readonly #getGlobalLock: () => Promise<() => void>;\n\n readonly #publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<string>;\n\n #running: boolean;\n\n readonly #transactionPoller: TransactionPoller;\n\n readonly #beforeCheckPendingTransaction: (\n transactionMeta: TransactionMeta,\n ) => Promise<boolean>;\n\n constructor({\n blockTracker,\n getChainId,\n getEthQuery,\n getGlobalLock,\n getNetworkClientId,\n getTransactions,\n hooks,\n isResubmitEnabled,\n messenger,\n publishTransaction,\n }: {\n blockTracker: BlockTracker;\n getChainId: () => Hex;\n getEthQuery: (networkClientId?: NetworkClientId) => EthQuery;\n getNetworkClientId: () => string;\n getTransactions: () => TransactionMeta[];\n isResubmitEnabled?: () => boolean;\n getGlobalLock: () => Promise<() => void>;\n publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<string>;\n hooks?: {\n beforeCheckPendingTransaction?: (\n transactionMeta: TransactionMeta,\n ) => Promise<boolean>;\n };\n messenger: TransactionControllerMessenger;\n }) {\n this.hub = new EventEmitter() as PendingTransactionTrackerEventEmitter;\n\n this.#droppedBlockCountByHash = new Map();\n this.#getChainId = getChainId;\n this.#getEthQuery = getEthQuery;\n this.#getNetworkClientId = getNetworkClientId;\n this.#getTransactions = getTransactions;\n this.#isResubmitEnabled = isResubmitEnabled ?? (() => true);\n this.#listener = this.#onLatestBlock.bind(this);\n this.#getGlobalLock = getGlobalLock;\n this.#publishTransaction = publishTransaction;\n this.#running = false;\n\n this.#transactionPoller = new TransactionPoller({\n blockTracker,\n chainId: getChainId(),\n messenger,\n });\n\n this.#beforeCheckPendingTransaction =\n hooks?.beforeCheckPendingTransaction ??\n /* istanbul ignore next */\n (() => Promise.resolve(true));\n\n this.#log = createModuleLogger(\n log,\n `${getChainId()}:${getNetworkClientId()}`,\n );\n }\n\n startIfPendingTransactions = () => {\n const pendingTransactions = this.#getPendingTransactions();\n\n if (pendingTransactions.length) {\n this.#start(pendingTransactions);\n } else {\n this.stop();\n }\n };\n\n /**\n * Force checks the network if the given transaction is confirmed and updates it's status.\n *\n * @param txMeta - The transaction to check\n */\n async forceCheckTransaction(txMeta: TransactionMeta) {\n const releaseLock = await this.#getGlobalLock();\n\n try {\n await this.#checkTransaction(txMeta);\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to check transaction', error);\n } finally {\n releaseLock();\n }\n }\n\n #start(pendingTransactions: TransactionMeta[]) {\n this.#transactionPoller.setPendingTransactions(pendingTransactions);\n\n if (this.#running) {\n return;\n }\n\n this.#transactionPoller.start(this.#listener);\n this.#running = true;\n\n this.#log('Started polling');\n }\n\n stop() {\n if (!this.#running) {\n return;\n }\n\n this.#transactionPoller.stop();\n this.#running = false;\n\n this.#log('Stopped polling');\n }\n\n async #onLatestBlock(latestBlockNumber: string) {\n const releaseLock = await this.#getGlobalLock();\n\n try {\n await this.#checkTransactions();\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to check transactions', error);\n } finally {\n releaseLock();\n }\n\n try {\n await this.#resubmitTransactions(latestBlockNumber);\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to resubmit transactions', error);\n }\n }\n\n async #checkTransactions() {\n this.#log('Checking transactions');\n\n const pendingTransactions = this.#getPendingTransactions();\n\n if (!pendingTransactions.length) {\n this.#log('No pending transactions to check');\n return;\n }\n\n this.#log('Found pending transactions to check', {\n count: pendingTransactions.length,\n ids: pendingTransactions.map((tx) => tx.id),\n });\n\n await Promise.all(\n pendingTransactions.map((tx) => this.#checkTransaction(tx)),\n );\n }\n\n async #resubmitTransactions(latestBlockNumber: string) {\n if (!this.#isResubmitEnabled() || !this.#running) {\n return;\n }\n\n this.#log('Resubmitting transactions');\n\n const pendingTransactions = this.#getPendingTransactions();\n\n if (!pendingTransactions.length) {\n this.#log('No pending transactions to resubmit');\n return;\n }\n\n this.#log('Found pending transactions to resubmit', {\n count: pendingTransactions.length,\n ids: pendingTransactions.map((tx) => tx.id),\n });\n\n for (const txMeta of pendingTransactions) {\n try {\n await this.#resubmitTransaction(txMeta, latestBlockNumber);\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n } catch (error: any) {\n /* istanbul ignore next */\n const errorMessage =\n error.value?.message?.toLowerCase() ||\n error.message?.toLowerCase() ||\n String(error);\n\n if (this.#isKnownTransactionError(errorMessage)) {\n this.#log('Ignoring known transaction error', errorMessage);\n continue;\n }\n\n this.#warnTransaction(\n txMeta,\n error.message,\n 'There was an error when resubmitting this transaction.',\n );\n }\n }\n }\n\n #isKnownTransactionError(errorMessage: string) {\n return KNOWN_TRANSACTION_ERRORS.some((knownError) =>\n errorMessage.includes(knownError),\n );\n }\n\n async #resubmitTransaction(\n txMeta: TransactionMeta,\n latestBlockNumber: string,\n ) {\n if (!this.#isResubmitDue(txMeta, latestBlockNumber)) {\n return;\n }\n\n if (!(await this.#beforeCheckPendingTransaction(txMeta))) {\n return;\n }\n\n const ethQuery = this.#getEthQuery(txMeta.networkClientId);\n await this.#publishTransaction(ethQuery, txMeta);\n\n const retryCount = (txMeta.retryCount ?? 0) + 1;\n\n this.#updateTransaction(\n merge({}, txMeta, { retryCount }),\n 'PendingTransactionTracker:transaction-retry - Retry count increased',\n );\n }\n\n #isResubmitDue(txMeta: TransactionMeta, latestBlockNumber: string): boolean {\n const txMetaWithFirstRetryBlockNumber = cloneDeep(txMeta);\n\n if (!txMetaWithFirstRetryBlockNumber.firstRetryBlockNumber) {\n txMetaWithFirstRetryBlockNumber.firstRetryBlockNumber = latestBlockNumber;\n\n this.#updateTransaction(\n txMetaWithFirstRetryBlockNumber,\n 'PendingTransactionTracker:#isResubmitDue - First retry block number set',\n );\n }\n\n const { firstRetryBlockNumber } = txMetaWithFirstRetryBlockNumber;\n\n const blocksSinceFirstRetry =\n Number.parseInt(latestBlockNumber, 16) -\n Number.parseInt(firstRetryBlockNumber, 16);\n\n const retryCount = txMeta.retryCount || 0;\n\n // Exponential backoff to limit retries at publishing\n // Capped at ~15 minutes between retries\n const requiredBlocksSinceFirstRetry = Math.min(\n MAX_RETRY_BLOCK_DISTANCE,\n Math.pow(2, retryCount),\n );\n\n return blocksSinceFirstRetry >= requiredBlocksSinceFirstRetry;\n }\n\n async #checkTransaction(txMeta: TransactionMeta) {\n const { hash, id } = txMeta;\n\n if (!hash && (await this.#beforeCheckPendingTransaction(txMeta))) {\n const error = new Error(\n 'We had an error while submitting this transaction, please try again.',\n );\n\n error.name = 'NoTxHashError';\n\n this.#failTransaction(txMeta, error);\n\n return;\n }\n\n if (this.#isNonceTaken(txMeta)) {\n this.#log('Nonce already taken', id);\n this.#dropTransaction(txMeta);\n return;\n }\n\n try {\n const receipt = await this.#getTransactionReceipt(hash);\n const isSuccess = receipt?.status === RECEIPT_STATUS_SUCCESS;\n const isFailure = receipt?.status === RECEIPT_STATUS_FAILURE;\n\n if (isFailure) {\n this.#log('Transaction receipt has failed status');\n\n this.#failTransaction(\n txMeta,\n new Error('Transaction dropped or replaced'),\n );\n\n return;\n }\n\n const { blockNumber, blockHash } = receipt || {};\n\n if (isSuccess && blockNumber && blockHash) {\n await this.#onTransactionConfirmed(txMeta, {\n ...receipt,\n blockNumber,\n blockHash,\n });\n\n return;\n }\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n } catch (error: any) {\n this.#log('Failed to check transaction', id, error);\n\n this.#warnTransaction(\n txMeta,\n error.message,\n 'There was a problem loading this transaction.',\n );\n\n return;\n }\n\n if (await this.#isTransactionDropped(txMeta)) {\n this.#dropTransaction(txMeta);\n }\n }\n\n async #onTransactionConfirmed(\n txMeta: TransactionMeta,\n receipt: SuccessfulTransactionReceipt,\n ) {\n const { id } = txMeta;\n const { blockHash } = receipt;\n\n this.#log('Transaction confirmed', id);\n\n const { baseFeePerGas, timestamp: blockTimestamp } =\n await this.#getBlockByHash(blockHash, false);\n\n const updatedTxMeta = cloneDeep(txMeta);\n updatedTxMeta.baseFeePerGas = baseFeePerGas;\n updatedTxMeta.blockTimestamp = blockTimestamp;\n updatedTxMeta.status = TransactionStatus.confirmed;\n updatedTxMeta.txParams = {\n ...updatedTxMeta.txParams,\n gasUsed: receipt.gasUsed,\n };\n updatedTxMeta.txReceipt = receipt;\n updatedTxMeta.verifiedOnBlockchain = true;\n\n this.#updateTransaction(\n updatedTxMeta,\n 'PendingTransactionTracker:#onTransactionConfirmed - Transaction confirmed',\n );\n\n this.hub.emit('transaction-confirmed', updatedTxMeta);\n }\n\n async #isTransactionDropped(txMeta: TransactionMeta) {\n const {\n hash,\n id,\n txParams: { nonce, from },\n } = txMeta;\n\n /* istanbul ignore next */\n if (!nonce || !hash) {\n return false;\n }\n\n const networkNextNonceHex = await this.#getNetworkTransactionCount(from);\n const networkNextNonceNumber = parseInt(networkNextNonceHex, 16);\n const nonceNumber = parseInt(nonce, 16);\n\n if (nonceNumber >= networkNextNonceNumber) {\n return false;\n }\n\n let droppedBlockCount = this.#droppedBlockCountByHash.get(hash);\n\n if (droppedBlockCount === undefined) {\n droppedBlockCount = 0;\n this.#droppedBlockCountByHash.set(hash, droppedBlockCount);\n }\n\n if (droppedBlockCount < DROPPED_BLOCK_COUNT) {\n this.#log('Incrementing dropped block count', { id, droppedBlockCount });\n this.#droppedBlockCountByHash.set(hash, droppedBlockCount + 1);\n return false;\n }\n\n this.#log('Hit dropped block count', id);\n\n this.#droppedBlockCountByHash.delete(hash);\n return true;\n }\n\n #isNonceTaken(txMeta: TransactionMeta): boolean {\n const { id, txParams } = txMeta;\n\n return this.#getChainTransactions().some(\n (tx) =>\n tx.id !== id &&\n tx.txParams.from === txParams.from &&\n tx.status === TransactionStatus.confirmed &&\n tx.txParams.nonce &&\n tx.txParams.nonce === txParams.nonce &&\n tx.type !== TransactionType.incoming,\n );\n }\n\n #getPendingTransactions(): TransactionMeta[] {\n return this.#getNetworkClientTransactions().filter(\n (tx) =>\n tx.status === TransactionStatus.submitted &&\n !tx.verifiedOnBlockchain &&\n !tx.isUserOperation,\n );\n }\n\n #warnTransaction(txMeta: TransactionMeta, error: string, message: string) {\n this.#updateTransaction(\n {\n ...txMeta,\n warning: { error, message },\n },\n 'PendingTransactionTracker:#warnTransaction - Warning added',\n );\n }\n\n #failTransaction(txMeta: TransactionMeta, error: Error) {\n this.#log('Transaction failed', txMeta.id, error);\n this.hub.emit('transaction-failed', txMeta, error);\n }\n\n #dropTransaction(txMeta: TransactionMeta) {\n this.#log('Transaction dropped', txMeta.id);\n this.hub.emit('transaction-dropped', txMeta);\n }\n\n #updateTransaction(txMeta: TransactionMeta, note: string) {\n this.hub.emit('transaction-updated', txMeta, note);\n }\n\n async #getTransactionReceipt(\n txHash?: string,\n ): Promise<TransactionReceipt | undefined> {\n return await query(this.#getEthQuery(), 'getTransactionReceipt', [txHash]);\n }\n\n async #getBlockByHash(\n blockHash: string,\n includeTransactionDetails: boolean,\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ): Promise<any> {\n return await query(this.#getEthQuery(), 'getBlockByHash', [\n blockHash,\n includeTransactionDetails,\n ]);\n }\n\n async #getNetworkTransactionCount(address: string): Promise<string> {\n return await query(this.#getEthQuery(), 'getTransactionCount', [address]);\n }\n\n #getChainTransactions(): TransactionMeta[] {\n const chainId = this.#getChainId();\n return this.#getTransactions().filter((tx) => tx.chainId === chainId);\n }\n\n #getNetworkClientTransactions(): TransactionMeta[] {\n const networkClientId = this.#getNetworkClientId();\n return this.#getTransactions().filter(\n (tx) => tx.networkClientId === networkClientId,\n );\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"PendingTransactionTracker.mjs","sourceRoot":"","sources":["../../src/helpers/PendingTransactionTracker.ts"],"names":[],"mappings":"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","sourcesContent":["import { query } from '@metamask/controller-utils';\nimport type EthQuery from '@metamask/eth-query';\nimport type {\n BlockTracker,\n NetworkClientId,\n} from '@metamask/network-controller';\nimport type { Hex } from '@metamask/utils';\n// This package purposefully relies on Node's EventEmitter module.\n// eslint-disable-next-line import-x/no-nodejs-modules\nimport EventEmitter from 'events';\nimport { cloneDeep, merge } from 'lodash';\n\nimport { TransactionPoller } from './TransactionPoller';\nimport { createModuleLogger, projectLogger } from '../logger';\nimport type { TransactionControllerMessenger } from '../TransactionController';\nimport type { TransactionMeta, TransactionReceipt } from '../types';\nimport { TransactionStatus, TransactionType } from '../types';\n\n/**\n * We wait this many blocks before emitting a 'transaction-dropped' event\n * This is because we could be talking to a node that is out of sync\n */\nconst DROPPED_BLOCK_COUNT = 3;\n\nconst RECEIPT_STATUS_SUCCESS = '0x1';\nconst RECEIPT_STATUS_FAILURE = '0x0';\nconst MAX_RETRY_BLOCK_DISTANCE = 50;\n\nconst KNOWN_TRANSACTION_ERRORS = [\n 'replacement transaction underpriced',\n 'known transaction',\n 'gas price too low to replace',\n 'transaction with the same hash was already imported',\n 'gateway timeout',\n 'nonce too low',\n];\n\nconst log = createModuleLogger(projectLogger, 'pending-transactions');\n\ntype SuccessfulTransactionReceipt = TransactionReceipt & {\n blockNumber: string;\n blockHash: string;\n};\n\ntype Events = {\n 'transaction-confirmed': [txMeta: TransactionMeta];\n 'transaction-dropped': [txMeta: TransactionMeta];\n 'transaction-failed': [txMeta: TransactionMeta, error: Error];\n 'transaction-updated': [txMeta: TransactionMeta, note: string];\n};\n\n// This interface was created before this ESLint rule was added.\n// Convert to a `type` in a future major version.\n// eslint-disable-next-line @typescript-eslint/consistent-type-definitions\nexport interface PendingTransactionTrackerEventEmitter extends EventEmitter {\n on<T extends keyof Events>(\n eventName: T,\n listener: (...args: Events[T]) => void,\n ): this;\n\n emit<T extends keyof Events>(eventName: T, ...args: Events[T]): boolean;\n}\n\nexport class PendingTransactionTracker {\n hub: PendingTransactionTrackerEventEmitter;\n\n readonly #droppedBlockCountByHash: Map<string, number>;\n\n readonly #getChainId: () => string;\n\n readonly #getEthQuery: (networkClientId?: NetworkClientId) => EthQuery;\n\n readonly #getNetworkClientId: () => NetworkClientId;\n\n readonly #getTransactions: () => TransactionMeta[];\n\n readonly #isResubmitEnabled: () => boolean;\n\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n readonly #listener: any;\n\n readonly #log: debug.Debugger;\n\n readonly #getGlobalLock: () => Promise<() => void>;\n\n readonly #publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<string>;\n\n #running: boolean;\n\n readonly #transactionPoller: TransactionPoller;\n\n #transactionToForcePoll: TransactionMeta | undefined;\n\n readonly #beforeCheckPendingTransaction: (\n transactionMeta: TransactionMeta,\n ) => Promise<boolean>;\n\n constructor({\n blockTracker,\n getChainId,\n getEthQuery,\n getGlobalLock,\n getNetworkClientId,\n getTransactions,\n hooks,\n isResubmitEnabled,\n messenger,\n publishTransaction,\n }: {\n blockTracker: BlockTracker;\n getChainId: () => Hex;\n getEthQuery: (networkClientId?: NetworkClientId) => EthQuery;\n getNetworkClientId: () => string;\n getTransactions: () => TransactionMeta[];\n isResubmitEnabled?: () => boolean;\n getGlobalLock: () => Promise<() => void>;\n publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<string>;\n hooks?: {\n beforeCheckPendingTransaction?: (\n transactionMeta: TransactionMeta,\n ) => Promise<boolean>;\n };\n messenger: TransactionControllerMessenger;\n }) {\n this.hub = new EventEmitter() as PendingTransactionTrackerEventEmitter;\n\n this.#droppedBlockCountByHash = new Map();\n this.#getChainId = getChainId;\n this.#getEthQuery = getEthQuery;\n this.#getNetworkClientId = getNetworkClientId;\n this.#getTransactions = getTransactions;\n this.#isResubmitEnabled = isResubmitEnabled ?? (() => true);\n this.#listener = this.#onLatestBlock.bind(this);\n this.#getGlobalLock = getGlobalLock;\n this.#publishTransaction = publishTransaction;\n this.#running = false;\n this.#transactionToForcePoll = undefined;\n\n this.#transactionPoller = new TransactionPoller({\n blockTracker,\n chainId: getChainId(),\n messenger,\n });\n\n this.#beforeCheckPendingTransaction =\n hooks?.beforeCheckPendingTransaction ??\n /* istanbul ignore next */\n (() => Promise.resolve(true));\n\n this.#log = createModuleLogger(\n log,\n `${getChainId()}:${getNetworkClientId()}`,\n );\n }\n\n startIfPendingTransactions = () => {\n const pendingTransactions = this.#getPendingTransactions();\n\n if (pendingTransactions.length) {\n this.#start(pendingTransactions);\n } else {\n this.stop();\n }\n };\n\n /**\n * Adds a transaction to the polling mechanism for monitoring its status.\n *\n * This method forcefully adds a single transaction to the list of transactions\n * being polled, ensuring that its status is checked, event emitted but no update is performed.\n * It overrides the default behavior by prioritizing the given transaction for polling.\n *\n * @param transactionMeta - The transaction metadata to be added for polling.\n *\n * The transaction will now be monitored for updates, such as confirmation or failure.\n */\n addTransactionToPoll(transactionMeta: TransactionMeta): void {\n this.#start([transactionMeta]);\n this.#transactionToForcePoll = transactionMeta;\n }\n\n /**\n * Force checks the network if the given transaction is confirmed and updates it's status.\n *\n * @param txMeta - The transaction to check\n */\n async forceCheckTransaction(txMeta: TransactionMeta) {\n const releaseLock = await this.#getGlobalLock();\n\n try {\n await this.#checkTransaction(txMeta);\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to check transaction', error);\n } finally {\n releaseLock();\n }\n }\n\n #start(pendingTransactions: TransactionMeta[]) {\n this.#transactionPoller.setPendingTransactions(pendingTransactions);\n\n if (this.#running) {\n return;\n }\n\n this.#transactionPoller.start(this.#listener);\n this.#running = true;\n\n this.#log('Started polling');\n }\n\n stop() {\n if (!this.#running) {\n return;\n }\n\n this.#transactionPoller.stop();\n this.#running = false;\n\n this.#log('Stopped polling');\n }\n\n async #onLatestBlock(latestBlockNumber: string) {\n const releaseLock = await this.#getGlobalLock();\n\n try {\n await this.#checkTransactions();\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to check transactions', error);\n } finally {\n releaseLock();\n }\n\n try {\n await this.#resubmitTransactions(latestBlockNumber);\n } catch (error) {\n /* istanbul ignore next */\n this.#log('Failed to resubmit transactions', error);\n }\n }\n\n async #checkTransactions() {\n this.#log('Checking transactions');\n\n const pendingTransactions: TransactionMeta[] = [\n ...this.#getPendingTransactions(),\n ...(this.#transactionToForcePoll ? [this.#transactionToForcePoll] : []),\n ];\n\n if (!pendingTransactions.length) {\n this.#log('No pending transactions to check');\n return;\n }\n\n this.#log('Found pending transactions to check', {\n count: pendingTransactions.length,\n ids: pendingTransactions.map((tx) => tx.id),\n });\n\n await Promise.all(\n pendingTransactions.map((tx) => this.#checkTransaction(tx)),\n );\n }\n\n async #resubmitTransactions(latestBlockNumber: string) {\n if (!this.#isResubmitEnabled() || !this.#running) {\n return;\n }\n\n this.#log('Resubmitting transactions');\n\n const pendingTransactions = this.#getPendingTransactions();\n\n if (!pendingTransactions.length) {\n this.#log('No pending transactions to resubmit');\n return;\n }\n\n this.#log('Found pending transactions to resubmit', {\n count: pendingTransactions.length,\n ids: pendingTransactions.map((tx) => tx.id),\n });\n\n for (const txMeta of pendingTransactions) {\n try {\n await this.#resubmitTransaction(txMeta, latestBlockNumber);\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n } catch (error: any) {\n /* istanbul ignore next */\n const errorMessage =\n error.value?.message?.toLowerCase() ||\n error.message?.toLowerCase() ||\n String(error);\n\n if (this.#isKnownTransactionError(errorMessage)) {\n this.#log('Ignoring known transaction error', errorMessage);\n continue;\n }\n\n this.#warnTransaction(\n txMeta,\n error.message,\n 'There was an error when resubmitting this transaction.',\n );\n }\n }\n }\n\n #isKnownTransactionError(errorMessage: string) {\n return KNOWN_TRANSACTION_ERRORS.some((knownError) =>\n errorMessage.includes(knownError),\n );\n }\n\n async #resubmitTransaction(\n txMeta: TransactionMeta,\n latestBlockNumber: string,\n ) {\n if (!this.#isResubmitDue(txMeta, latestBlockNumber)) {\n return;\n }\n\n if (!(await this.#beforeCheckPendingTransaction(txMeta))) {\n return;\n }\n\n const ethQuery = this.#getEthQuery(txMeta.networkClientId);\n await this.#publishTransaction(ethQuery, txMeta);\n\n const retryCount = (txMeta.retryCount ?? 0) + 1;\n\n this.#updateTransaction(\n merge({}, txMeta, { retryCount }),\n 'PendingTransactionTracker:transaction-retry - Retry count increased',\n );\n }\n\n #isResubmitDue(txMeta: TransactionMeta, latestBlockNumber: string): boolean {\n const txMetaWithFirstRetryBlockNumber = cloneDeep(txMeta);\n\n if (!txMetaWithFirstRetryBlockNumber.firstRetryBlockNumber) {\n txMetaWithFirstRetryBlockNumber.firstRetryBlockNumber = latestBlockNumber;\n\n this.#updateTransaction(\n txMetaWithFirstRetryBlockNumber,\n 'PendingTransactionTracker:#isResubmitDue - First retry block number set',\n );\n }\n\n const { firstRetryBlockNumber } = txMetaWithFirstRetryBlockNumber;\n\n const blocksSinceFirstRetry =\n Number.parseInt(latestBlockNumber, 16) -\n Number.parseInt(firstRetryBlockNumber, 16);\n\n const retryCount = txMeta.retryCount || 0;\n\n // Exponential backoff to limit retries at publishing\n // Capped at ~15 minutes between retries\n const requiredBlocksSinceFirstRetry = Math.min(\n MAX_RETRY_BLOCK_DISTANCE,\n Math.pow(2, retryCount),\n );\n\n return blocksSinceFirstRetry >= requiredBlocksSinceFirstRetry;\n }\n\n #cleanTransactionToForcePoll(transactionId: string) {\n if (this.#transactionToForcePoll?.id === transactionId) {\n this.#transactionToForcePoll = undefined;\n }\n }\n\n async #checkTransaction(txMeta: TransactionMeta) {\n const { hash, id } = txMeta;\n\n if (!hash && (await this.#beforeCheckPendingTransaction(txMeta))) {\n const error = new Error(\n 'We had an error while submitting this transaction, please try again.',\n );\n\n error.name = 'NoTxHashError';\n\n this.#failTransaction(txMeta, error);\n\n return;\n }\n\n if (this.#isNonceTaken(txMeta)) {\n this.#log('Nonce already taken', id);\n this.#dropTransaction(txMeta);\n return;\n }\n\n try {\n const receipt = await this.#getTransactionReceipt(hash);\n const isSuccess = receipt?.status === RECEIPT_STATUS_SUCCESS;\n const isFailure = receipt?.status === RECEIPT_STATUS_FAILURE;\n\n if (isFailure) {\n this.#log('Transaction receipt has failed status');\n\n this.#failTransaction(\n txMeta,\n new Error('Transaction dropped or replaced'),\n );\n\n return;\n }\n\n const { blockNumber, blockHash } = receipt || {};\n\n if (isSuccess && blockNumber && blockHash) {\n await this.#onTransactionConfirmed(txMeta, {\n ...receipt,\n blockNumber,\n blockHash,\n });\n\n return;\n }\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n } catch (error: any) {\n this.#log('Failed to check transaction', id, error);\n\n this.#warnTransaction(\n txMeta,\n error.message,\n 'There was a problem loading this transaction.',\n );\n\n return;\n }\n\n if (await this.#isTransactionDropped(txMeta)) {\n this.#dropTransaction(txMeta);\n }\n }\n\n async #onTransactionConfirmed(\n txMeta: TransactionMeta,\n receipt: SuccessfulTransactionReceipt,\n ) {\n const { id } = txMeta;\n const { blockHash } = receipt;\n\n this.#log('Transaction confirmed', id);\n\n if (this.#transactionToForcePoll) {\n this.#cleanTransactionToForcePoll(txMeta.id);\n this.hub.emit('transaction-confirmed', txMeta);\n return;\n }\n\n const { baseFeePerGas, timestamp: blockTimestamp } =\n await this.#getBlockByHash(blockHash, false);\n\n const updatedTxMeta = cloneDeep(txMeta);\n updatedTxMeta.baseFeePerGas = baseFeePerGas;\n updatedTxMeta.blockTimestamp = blockTimestamp;\n updatedTxMeta.status = TransactionStatus.confirmed;\n updatedTxMeta.txParams = {\n ...updatedTxMeta.txParams,\n gasUsed: receipt.gasUsed,\n };\n updatedTxMeta.txReceipt = receipt;\n updatedTxMeta.verifiedOnBlockchain = true;\n\n this.#updateTransaction(\n updatedTxMeta,\n 'PendingTransactionTracker:#onTransactionConfirmed - Transaction confirmed',\n );\n\n this.hub.emit('transaction-confirmed', updatedTxMeta);\n }\n\n async #isTransactionDropped(txMeta: TransactionMeta) {\n const {\n hash,\n id,\n txParams: { nonce, from },\n } = txMeta;\n\n /* istanbul ignore next */\n if (!nonce || !hash) {\n return false;\n }\n\n const networkNextNonceHex = await this.#getNetworkTransactionCount(from);\n const networkNextNonceNumber = parseInt(networkNextNonceHex, 16);\n const nonceNumber = parseInt(nonce, 16);\n\n if (nonceNumber >= networkNextNonceNumber) {\n return false;\n }\n\n let droppedBlockCount = this.#droppedBlockCountByHash.get(hash);\n\n if (droppedBlockCount === undefined) {\n droppedBlockCount = 0;\n this.#droppedBlockCountByHash.set(hash, droppedBlockCount);\n }\n\n if (droppedBlockCount < DROPPED_BLOCK_COUNT) {\n this.#log('Incrementing dropped block count', { id, droppedBlockCount });\n this.#droppedBlockCountByHash.set(hash, droppedBlockCount + 1);\n return false;\n }\n\n this.#log('Hit dropped block count', id);\n\n this.#droppedBlockCountByHash.delete(hash);\n return true;\n }\n\n #isNonceTaken(txMeta: TransactionMeta): boolean {\n const { id, txParams } = txMeta;\n\n return this.#getChainTransactions().some(\n (tx) =>\n tx.id !== id &&\n tx.txParams.from === txParams.from &&\n tx.status === TransactionStatus.confirmed &&\n tx.txParams.nonce &&\n tx.txParams.nonce === txParams.nonce &&\n tx.type !== TransactionType.incoming,\n );\n }\n\n #getPendingTransactions(): TransactionMeta[] {\n return this.#getNetworkClientTransactions().filter(\n (tx) =>\n tx.status === TransactionStatus.submitted &&\n !tx.verifiedOnBlockchain &&\n !tx.isUserOperation,\n );\n }\n\n #warnTransaction(txMeta: TransactionMeta, error: string, message: string) {\n this.#updateTransaction(\n {\n ...txMeta,\n warning: { error, message },\n },\n 'PendingTransactionTracker:#warnTransaction - Warning added',\n );\n }\n\n #failTransaction(txMeta: TransactionMeta, error: Error) {\n this.#log('Transaction failed', txMeta.id, error);\n this.#cleanTransactionToForcePoll(txMeta.id);\n this.hub.emit('transaction-failed', txMeta, error);\n }\n\n #dropTransaction(txMeta: TransactionMeta) {\n this.#log('Transaction dropped', txMeta.id);\n this.#cleanTransactionToForcePoll(txMeta.id);\n this.hub.emit('transaction-dropped', txMeta);\n }\n\n #updateTransaction(txMeta: TransactionMeta, note: string) {\n this.hub.emit('transaction-updated', txMeta, note);\n }\n\n async #getTransactionReceipt(\n txHash?: string,\n ): Promise<TransactionReceipt | undefined> {\n return await query(this.#getEthQuery(), 'getTransactionReceipt', [txHash]);\n }\n\n async #getBlockByHash(\n blockHash: string,\n includeTransactionDetails: boolean,\n // TODO: Replace `any` with type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ): Promise<any> {\n return await query(this.#getEthQuery(), 'getBlockByHash', [\n blockHash,\n includeTransactionDetails,\n ]);\n }\n\n async #getNetworkTransactionCount(address: string): Promise<string> {\n return await query(this.#getEthQuery(), 'getTransactionCount', [address]);\n }\n\n #getChainTransactions(): TransactionMeta[] {\n const chainId = this.#getChainId();\n return this.#getTransactions().filter((tx) => tx.chainId === chainId);\n }\n\n #getNetworkClientTransactions(): TransactionMeta[] {\n const networkClientId = this.#getNetworkClientId();\n return this.#getTransactions().filter(\n (tx) => tx.networkClientId === networkClientId,\n );\n }\n}\n"]}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
|
3
|
+
if (kind === "m") throw new TypeError("Private method is not writable");
|
|
4
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
|
5
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
|
6
|
+
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
|
7
|
+
};
|
|
8
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
9
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
10
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
11
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
12
|
+
};
|
|
13
|
+
var _SequentialPublishBatchHook_instances, _SequentialPublishBatchHook_publishTransaction, _SequentialPublishBatchHook_getTransaction, _SequentialPublishBatchHook_getEthQuery, _SequentialPublishBatchHook_getPendingTransactionTracker, _SequentialPublishBatchHook_boundListeners, _SequentialPublishBatchHook_hook, _SequentialPublishBatchHook_waitForTransactionEvent, _SequentialPublishBatchHook_onConfirmed, _SequentialPublishBatchHook_onFailedOrDropped, _SequentialPublishBatchHook_removeListeners;
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
exports.SequentialPublishBatchHook = void 0;
|
|
16
|
+
const rpc_errors_1 = require("@metamask/rpc-errors");
|
|
17
|
+
const utils_1 = require("@metamask/utils");
|
|
18
|
+
const logger_1 = require("../logger.cjs");
|
|
19
|
+
const log = (0, utils_1.createModuleLogger)(logger_1.projectLogger, 'sequential-publish-batch-hook');
|
|
20
|
+
/**
|
|
21
|
+
* Custom publish logic that also publishes additional sequential transactions in a batch.
|
|
22
|
+
* Requires the batch to be successful to resolve.
|
|
23
|
+
*/
|
|
24
|
+
class SequentialPublishBatchHook {
|
|
25
|
+
constructor({ publishTransaction, getTransaction, getPendingTransactionTracker, getEthQuery, }) {
|
|
26
|
+
_SequentialPublishBatchHook_instances.add(this);
|
|
27
|
+
_SequentialPublishBatchHook_publishTransaction.set(this, void 0);
|
|
28
|
+
_SequentialPublishBatchHook_getTransaction.set(this, void 0);
|
|
29
|
+
_SequentialPublishBatchHook_getEthQuery.set(this, void 0);
|
|
30
|
+
_SequentialPublishBatchHook_getPendingTransactionTracker.set(this, void 0);
|
|
31
|
+
_SequentialPublishBatchHook_boundListeners.set(this, {});
|
|
32
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_publishTransaction, publishTransaction, "f");
|
|
33
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getTransaction, getTransaction, "f");
|
|
34
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getEthQuery, getEthQuery, "f");
|
|
35
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getPendingTransactionTracker, getPendingTransactionTracker, "f");
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* @returns The publish batch hook function.
|
|
39
|
+
*/
|
|
40
|
+
getHook() {
|
|
41
|
+
return __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_hook).bind(this);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
exports.SequentialPublishBatchHook = SequentialPublishBatchHook;
|
|
45
|
+
_SequentialPublishBatchHook_publishTransaction = new WeakMap(), _SequentialPublishBatchHook_getTransaction = new WeakMap(), _SequentialPublishBatchHook_getEthQuery = new WeakMap(), _SequentialPublishBatchHook_getPendingTransactionTracker = new WeakMap(), _SequentialPublishBatchHook_boundListeners = new WeakMap(), _SequentialPublishBatchHook_instances = new WeakSet(), _SequentialPublishBatchHook_hook = async function _SequentialPublishBatchHook_hook({ from, networkClientId, transactions, }) {
|
|
46
|
+
log('Starting sequential publish batch hook', { from, networkClientId });
|
|
47
|
+
const pendingTransactionTracker = __classPrivateFieldGet(this, _SequentialPublishBatchHook_getPendingTransactionTracker, "f").call(this, networkClientId);
|
|
48
|
+
const results = [];
|
|
49
|
+
for (const transaction of transactions) {
|
|
50
|
+
try {
|
|
51
|
+
const transactionMeta = __classPrivateFieldGet(this, _SequentialPublishBatchHook_getTransaction, "f").call(this, String(transaction.id));
|
|
52
|
+
const transactionHash = await __classPrivateFieldGet(this, _SequentialPublishBatchHook_publishTransaction, "f").call(this, __classPrivateFieldGet(this, _SequentialPublishBatchHook_getEthQuery, "f").call(this, networkClientId), transactionMeta);
|
|
53
|
+
log('Transaction published', { transactionHash });
|
|
54
|
+
const transactionUpdated = {
|
|
55
|
+
...transactionMeta,
|
|
56
|
+
hash: transactionHash,
|
|
57
|
+
};
|
|
58
|
+
const confirmationPromise = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_waitForTransactionEvent).call(this, pendingTransactionTracker, transactionUpdated.id, transactionUpdated.hash);
|
|
59
|
+
pendingTransactionTracker.addTransactionToPoll(transactionUpdated);
|
|
60
|
+
await confirmationPromise;
|
|
61
|
+
results.push({ transactionHash });
|
|
62
|
+
}
|
|
63
|
+
catch (error) {
|
|
64
|
+
log('Batch transaction failed', { transaction, error });
|
|
65
|
+
pendingTransactionTracker.stop();
|
|
66
|
+
throw rpc_errors_1.rpcErrors.internal(`Failed to publish batch transaction`);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
log('Sequential publish batch hook completed', { results });
|
|
70
|
+
pendingTransactionTracker.stop();
|
|
71
|
+
return { results };
|
|
72
|
+
}, _SequentialPublishBatchHook_waitForTransactionEvent =
|
|
73
|
+
/**
|
|
74
|
+
* Waits for a transaction event (confirmed, failed, or dropped) and resolves/rejects accordingly.
|
|
75
|
+
*
|
|
76
|
+
* @param pendingTransactionTracker - The tracker instance to subscribe to events.
|
|
77
|
+
* @param transactionId - The transaction ID.
|
|
78
|
+
* @param transactionHash - The hash of the transaction.
|
|
79
|
+
* @returns A promise that resolves when the transaction is confirmed or rejects if it fails or is dropped.
|
|
80
|
+
*/
|
|
81
|
+
async function _SequentialPublishBatchHook_waitForTransactionEvent(pendingTransactionTracker, transactionId, transactionHash) {
|
|
82
|
+
return new Promise((resolve, reject) => {
|
|
83
|
+
const onConfirmed = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_onConfirmed).bind(this, transactionId, transactionHash, resolve, pendingTransactionTracker);
|
|
84
|
+
const onFailedOrDropped = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_onFailedOrDropped).bind(this, transactionId, transactionHash, reject, pendingTransactionTracker);
|
|
85
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId] = {
|
|
86
|
+
onConfirmed,
|
|
87
|
+
onFailedOrDropped,
|
|
88
|
+
};
|
|
89
|
+
pendingTransactionTracker.hub.on('transaction-confirmed', onConfirmed);
|
|
90
|
+
pendingTransactionTracker.hub.on('transaction-failed', onFailedOrDropped);
|
|
91
|
+
pendingTransactionTracker.hub.on('transaction-dropped', onFailedOrDropped);
|
|
92
|
+
});
|
|
93
|
+
}, _SequentialPublishBatchHook_onConfirmed = function _SequentialPublishBatchHook_onConfirmed(transactionId, transactionHash, resolve, pendingTransactionTracker, txMeta) {
|
|
94
|
+
if (txMeta.id !== transactionId) {
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
log('Transaction confirmed', { transactionHash });
|
|
98
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_removeListeners).call(this, pendingTransactionTracker, transactionId);
|
|
99
|
+
resolve(txMeta);
|
|
100
|
+
}, _SequentialPublishBatchHook_onFailedOrDropped = function _SequentialPublishBatchHook_onFailedOrDropped(transactionId, transactionHash, reject, pendingTransactionTracker, txMeta, error) {
|
|
101
|
+
if (txMeta.id !== transactionId) {
|
|
102
|
+
return;
|
|
103
|
+
}
|
|
104
|
+
log('Transaction failed or dropped', { transactionHash, error });
|
|
105
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_removeListeners).call(this, pendingTransactionTracker, transactionId);
|
|
106
|
+
reject(new Error(`Transaction ${transactionHash} failed or dropped.`));
|
|
107
|
+
}, _SequentialPublishBatchHook_removeListeners = function _SequentialPublishBatchHook_removeListeners(pendingTransactionTracker, transactionId) {
|
|
108
|
+
const listeners = __classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId];
|
|
109
|
+
pendingTransactionTracker.hub.off('transaction-confirmed', listeners.onConfirmed);
|
|
110
|
+
pendingTransactionTracker.hub.off('transaction-failed', listeners.onFailedOrDropped);
|
|
111
|
+
pendingTransactionTracker.hub.off('transaction-dropped', listeners.onFailedOrDropped);
|
|
112
|
+
delete __classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId];
|
|
113
|
+
};
|
|
114
|
+
//# sourceMappingURL=SequentialPublishBatchHook.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SequentialPublishBatchHook.cjs","sourceRoot":"","sources":["../../src/hooks/SequentialPublishBatchHook.ts"],"names":[],"mappings":"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","sourcesContent":["import type EthQuery from '@metamask/eth-query';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { createModuleLogger } from '@metamask/utils';\nimport type { Hex } from '@metamask/utils';\n\nimport type { PendingTransactionTracker } from '../helpers/PendingTransactionTracker';\nimport { projectLogger } from '../logger';\nimport {\n type PublishBatchHook,\n type PublishBatchHookRequest,\n type PublishBatchHookResult,\n type TransactionMeta,\n} from '../types';\n\nconst log = createModuleLogger(projectLogger, 'sequential-publish-batch-hook');\n\ntype SequentialPublishBatchHookOptions = {\n publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<Hex>;\n getTransaction: (id: string) => TransactionMeta;\n getEthQuery: (networkClientId: string) => EthQuery;\n getPendingTransactionTracker: (\n networkClientId: string,\n ) => PendingTransactionTracker;\n};\n\n/**\n * Custom publish logic that also publishes additional sequential transactions in a batch.\n * Requires the batch to be successful to resolve.\n */\nexport class SequentialPublishBatchHook {\n readonly #publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<Hex>;\n\n readonly #getTransaction: (id: string) => TransactionMeta;\n\n readonly #getEthQuery: (networkClientId: string) => EthQuery;\n\n readonly #getPendingTransactionTracker: (\n networkClientId: string,\n ) => PendingTransactionTracker;\n\n #boundListeners: Record<\n string,\n {\n onConfirmed: (txMeta: TransactionMeta) => void;\n onFailedOrDropped: (txMeta: TransactionMeta, error?: Error) => void;\n }\n > = {};\n\n constructor({\n publishTransaction,\n getTransaction,\n getPendingTransactionTracker,\n getEthQuery,\n }: SequentialPublishBatchHookOptions) {\n this.#publishTransaction = publishTransaction;\n this.#getTransaction = getTransaction;\n this.#getEthQuery = getEthQuery;\n this.#getPendingTransactionTracker = getPendingTransactionTracker;\n }\n\n /**\n * @returns The publish batch hook function.\n */\n getHook(): PublishBatchHook {\n return this.#hook.bind(this);\n }\n\n async #hook({\n from,\n networkClientId,\n transactions,\n }: PublishBatchHookRequest): Promise<PublishBatchHookResult> {\n log('Starting sequential publish batch hook', { from, networkClientId });\n\n const pendingTransactionTracker =\n this.#getPendingTransactionTracker(networkClientId);\n const results = [];\n\n for (const transaction of transactions) {\n try {\n const transactionMeta = this.#getTransaction(String(transaction.id));\n\n const transactionHash = await this.#publishTransaction(\n this.#getEthQuery(networkClientId),\n transactionMeta,\n );\n log('Transaction published', { transactionHash });\n\n const transactionUpdated = {\n ...transactionMeta,\n hash: transactionHash,\n };\n\n const confirmationPromise = this.#waitForTransactionEvent(\n pendingTransactionTracker,\n transactionUpdated.id,\n transactionUpdated.hash,\n );\n\n pendingTransactionTracker.addTransactionToPoll(transactionUpdated);\n\n await confirmationPromise;\n results.push({ transactionHash });\n } catch (error) {\n log('Batch transaction failed', { transaction, error });\n pendingTransactionTracker.stop();\n throw rpcErrors.internal(`Failed to publish batch transaction`);\n }\n }\n\n log('Sequential publish batch hook completed', { results });\n pendingTransactionTracker.stop();\n\n return { results };\n }\n\n /**\n * Waits for a transaction event (confirmed, failed, or dropped) and resolves/rejects accordingly.\n *\n * @param pendingTransactionTracker - The tracker instance to subscribe to events.\n * @param transactionId - The transaction ID.\n * @param transactionHash - The hash of the transaction.\n * @returns A promise that resolves when the transaction is confirmed or rejects if it fails or is dropped.\n */\n async #waitForTransactionEvent(\n pendingTransactionTracker: PendingTransactionTracker,\n transactionId: string,\n transactionHash: string,\n ): Promise<TransactionMeta> {\n return new Promise((resolve, reject) => {\n const onConfirmed = this.#onConfirmed.bind(\n this,\n transactionId,\n transactionHash,\n resolve,\n pendingTransactionTracker,\n );\n\n const onFailedOrDropped = this.#onFailedOrDropped.bind(\n this,\n transactionId,\n transactionHash,\n reject,\n pendingTransactionTracker,\n );\n\n this.#boundListeners[transactionId] = {\n onConfirmed,\n onFailedOrDropped,\n };\n\n pendingTransactionTracker.hub.on('transaction-confirmed', onConfirmed);\n pendingTransactionTracker.hub.on('transaction-failed', onFailedOrDropped);\n pendingTransactionTracker.hub.on(\n 'transaction-dropped',\n onFailedOrDropped,\n );\n });\n }\n\n #onConfirmed(\n transactionId: string,\n transactionHash: string,\n resolve: (txMeta: TransactionMeta) => void,\n pendingTransactionTracker: PendingTransactionTracker,\n txMeta: TransactionMeta,\n ): void {\n if (txMeta.id !== transactionId) {\n return;\n }\n\n log('Transaction confirmed', { transactionHash });\n this.#removeListeners(pendingTransactionTracker, transactionId);\n resolve(txMeta);\n }\n\n #onFailedOrDropped(\n transactionId: string,\n transactionHash: string,\n reject: (error: Error) => void,\n pendingTransactionTracker: PendingTransactionTracker,\n txMeta: TransactionMeta,\n error?: Error,\n ): void {\n if (txMeta.id !== transactionId) {\n return;\n }\n\n log('Transaction failed or dropped', { transactionHash, error });\n this.#removeListeners(pendingTransactionTracker, transactionId);\n reject(new Error(`Transaction ${transactionHash} failed or dropped.`));\n }\n\n #removeListeners(\n pendingTransactionTracker: PendingTransactionTracker,\n transactionId: string,\n ): void {\n const listeners = this.#boundListeners[transactionId];\n\n pendingTransactionTracker.hub.off(\n 'transaction-confirmed',\n listeners.onConfirmed,\n );\n pendingTransactionTracker.hub.off(\n 'transaction-failed',\n listeners.onFailedOrDropped,\n );\n pendingTransactionTracker.hub.off(\n 'transaction-dropped',\n listeners.onFailedOrDropped,\n );\n\n delete this.#boundListeners[transactionId];\n }\n}\n"]}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import type EthQuery from "@metamask/eth-query";
|
|
2
|
+
import type { Hex } from "@metamask/utils";
|
|
3
|
+
import type { PendingTransactionTracker } from "../helpers/PendingTransactionTracker.cjs";
|
|
4
|
+
import { type PublishBatchHook, type TransactionMeta } from "../types.cjs";
|
|
5
|
+
type SequentialPublishBatchHookOptions = {
|
|
6
|
+
publishTransaction: (ethQuery: EthQuery, transactionMeta: TransactionMeta) => Promise<Hex>;
|
|
7
|
+
getTransaction: (id: string) => TransactionMeta;
|
|
8
|
+
getEthQuery: (networkClientId: string) => EthQuery;
|
|
9
|
+
getPendingTransactionTracker: (networkClientId: string) => PendingTransactionTracker;
|
|
10
|
+
};
|
|
11
|
+
/**
|
|
12
|
+
* Custom publish logic that also publishes additional sequential transactions in a batch.
|
|
13
|
+
* Requires the batch to be successful to resolve.
|
|
14
|
+
*/
|
|
15
|
+
export declare class SequentialPublishBatchHook {
|
|
16
|
+
#private;
|
|
17
|
+
constructor({ publishTransaction, getTransaction, getPendingTransactionTracker, getEthQuery, }: SequentialPublishBatchHookOptions);
|
|
18
|
+
/**
|
|
19
|
+
* @returns The publish batch hook function.
|
|
20
|
+
*/
|
|
21
|
+
getHook(): PublishBatchHook;
|
|
22
|
+
}
|
|
23
|
+
export {};
|
|
24
|
+
//# sourceMappingURL=SequentialPublishBatchHook.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SequentialPublishBatchHook.d.cts","sourceRoot":"","sources":["../../src/hooks/SequentialPublishBatchHook.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,QAAQ,4BAA4B;AAGhD,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAE3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,iDAA6C;AAEtF,OAAO,EACL,KAAK,gBAAgB,EAGrB,KAAK,eAAe,EACrB,qBAAiB;AAIlB,KAAK,iCAAiC,GAAG;IACvC,kBAAkB,EAAE,CAClB,QAAQ,EAAE,QAAQ,EAClB,eAAe,EAAE,eAAe,KAC7B,OAAO,CAAC,GAAG,CAAC,CAAC;IAClB,cAAc,EAAE,CAAC,EAAE,EAAE,MAAM,KAAK,eAAe,CAAC;IAChD,WAAW,EAAE,CAAC,eAAe,EAAE,MAAM,KAAK,QAAQ,CAAC;IACnD,4BAA4B,EAAE,CAC5B,eAAe,EAAE,MAAM,KACpB,yBAAyB,CAAC;CAChC,CAAC;AAEF;;;GAGG;AACH,qBAAa,0BAA0B;;gBAsBzB,EACV,kBAAkB,EAClB,cAAc,EACd,4BAA4B,EAC5B,WAAW,GACZ,EAAE,iCAAiC;IAOpC;;OAEG;IACH,OAAO,IAAI,gBAAgB;CAuJ5B"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import type EthQuery from "@metamask/eth-query";
|
|
2
|
+
import type { Hex } from "@metamask/utils";
|
|
3
|
+
import type { PendingTransactionTracker } from "../helpers/PendingTransactionTracker.mjs";
|
|
4
|
+
import { type PublishBatchHook, type TransactionMeta } from "../types.mjs";
|
|
5
|
+
type SequentialPublishBatchHookOptions = {
|
|
6
|
+
publishTransaction: (ethQuery: EthQuery, transactionMeta: TransactionMeta) => Promise<Hex>;
|
|
7
|
+
getTransaction: (id: string) => TransactionMeta;
|
|
8
|
+
getEthQuery: (networkClientId: string) => EthQuery;
|
|
9
|
+
getPendingTransactionTracker: (networkClientId: string) => PendingTransactionTracker;
|
|
10
|
+
};
|
|
11
|
+
/**
|
|
12
|
+
* Custom publish logic that also publishes additional sequential transactions in a batch.
|
|
13
|
+
* Requires the batch to be successful to resolve.
|
|
14
|
+
*/
|
|
15
|
+
export declare class SequentialPublishBatchHook {
|
|
16
|
+
#private;
|
|
17
|
+
constructor({ publishTransaction, getTransaction, getPendingTransactionTracker, getEthQuery, }: SequentialPublishBatchHookOptions);
|
|
18
|
+
/**
|
|
19
|
+
* @returns The publish batch hook function.
|
|
20
|
+
*/
|
|
21
|
+
getHook(): PublishBatchHook;
|
|
22
|
+
}
|
|
23
|
+
export {};
|
|
24
|
+
//# sourceMappingURL=SequentialPublishBatchHook.d.mts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SequentialPublishBatchHook.d.mts","sourceRoot":"","sources":["../../src/hooks/SequentialPublishBatchHook.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,QAAQ,4BAA4B;AAGhD,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAE3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,iDAA6C;AAEtF,OAAO,EACL,KAAK,gBAAgB,EAGrB,KAAK,eAAe,EACrB,qBAAiB;AAIlB,KAAK,iCAAiC,GAAG;IACvC,kBAAkB,EAAE,CAClB,QAAQ,EAAE,QAAQ,EAClB,eAAe,EAAE,eAAe,KAC7B,OAAO,CAAC,GAAG,CAAC,CAAC;IAClB,cAAc,EAAE,CAAC,EAAE,EAAE,MAAM,KAAK,eAAe,CAAC;IAChD,WAAW,EAAE,CAAC,eAAe,EAAE,MAAM,KAAK,QAAQ,CAAC;IACnD,4BAA4B,EAAE,CAC5B,eAAe,EAAE,MAAM,KACpB,yBAAyB,CAAC;CAChC,CAAC;AAEF;;;GAGG;AACH,qBAAa,0BAA0B;;gBAsBzB,EACV,kBAAkB,EAClB,cAAc,EACd,4BAA4B,EAC5B,WAAW,GACZ,EAAE,iCAAiC;IAOpC;;OAEG;IACH,OAAO,IAAI,gBAAgB;CAuJ5B"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
|
2
|
+
if (kind === "m") throw new TypeError("Private method is not writable");
|
|
3
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
|
4
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
|
5
|
+
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
|
6
|
+
};
|
|
7
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
8
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
9
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
10
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
11
|
+
};
|
|
12
|
+
var _SequentialPublishBatchHook_instances, _SequentialPublishBatchHook_publishTransaction, _SequentialPublishBatchHook_getTransaction, _SequentialPublishBatchHook_getEthQuery, _SequentialPublishBatchHook_getPendingTransactionTracker, _SequentialPublishBatchHook_boundListeners, _SequentialPublishBatchHook_hook, _SequentialPublishBatchHook_waitForTransactionEvent, _SequentialPublishBatchHook_onConfirmed, _SequentialPublishBatchHook_onFailedOrDropped, _SequentialPublishBatchHook_removeListeners;
|
|
13
|
+
import { rpcErrors } from "@metamask/rpc-errors";
|
|
14
|
+
import { createModuleLogger } from "@metamask/utils";
|
|
15
|
+
import { projectLogger } from "../logger.mjs";
|
|
16
|
+
const log = createModuleLogger(projectLogger, 'sequential-publish-batch-hook');
|
|
17
|
+
/**
|
|
18
|
+
* Custom publish logic that also publishes additional sequential transactions in a batch.
|
|
19
|
+
* Requires the batch to be successful to resolve.
|
|
20
|
+
*/
|
|
21
|
+
export class SequentialPublishBatchHook {
|
|
22
|
+
constructor({ publishTransaction, getTransaction, getPendingTransactionTracker, getEthQuery, }) {
|
|
23
|
+
_SequentialPublishBatchHook_instances.add(this);
|
|
24
|
+
_SequentialPublishBatchHook_publishTransaction.set(this, void 0);
|
|
25
|
+
_SequentialPublishBatchHook_getTransaction.set(this, void 0);
|
|
26
|
+
_SequentialPublishBatchHook_getEthQuery.set(this, void 0);
|
|
27
|
+
_SequentialPublishBatchHook_getPendingTransactionTracker.set(this, void 0);
|
|
28
|
+
_SequentialPublishBatchHook_boundListeners.set(this, {});
|
|
29
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_publishTransaction, publishTransaction, "f");
|
|
30
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getTransaction, getTransaction, "f");
|
|
31
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getEthQuery, getEthQuery, "f");
|
|
32
|
+
__classPrivateFieldSet(this, _SequentialPublishBatchHook_getPendingTransactionTracker, getPendingTransactionTracker, "f");
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* @returns The publish batch hook function.
|
|
36
|
+
*/
|
|
37
|
+
getHook() {
|
|
38
|
+
return __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_hook).bind(this);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
_SequentialPublishBatchHook_publishTransaction = new WeakMap(), _SequentialPublishBatchHook_getTransaction = new WeakMap(), _SequentialPublishBatchHook_getEthQuery = new WeakMap(), _SequentialPublishBatchHook_getPendingTransactionTracker = new WeakMap(), _SequentialPublishBatchHook_boundListeners = new WeakMap(), _SequentialPublishBatchHook_instances = new WeakSet(), _SequentialPublishBatchHook_hook = async function _SequentialPublishBatchHook_hook({ from, networkClientId, transactions, }) {
|
|
42
|
+
log('Starting sequential publish batch hook', { from, networkClientId });
|
|
43
|
+
const pendingTransactionTracker = __classPrivateFieldGet(this, _SequentialPublishBatchHook_getPendingTransactionTracker, "f").call(this, networkClientId);
|
|
44
|
+
const results = [];
|
|
45
|
+
for (const transaction of transactions) {
|
|
46
|
+
try {
|
|
47
|
+
const transactionMeta = __classPrivateFieldGet(this, _SequentialPublishBatchHook_getTransaction, "f").call(this, String(transaction.id));
|
|
48
|
+
const transactionHash = await __classPrivateFieldGet(this, _SequentialPublishBatchHook_publishTransaction, "f").call(this, __classPrivateFieldGet(this, _SequentialPublishBatchHook_getEthQuery, "f").call(this, networkClientId), transactionMeta);
|
|
49
|
+
log('Transaction published', { transactionHash });
|
|
50
|
+
const transactionUpdated = {
|
|
51
|
+
...transactionMeta,
|
|
52
|
+
hash: transactionHash,
|
|
53
|
+
};
|
|
54
|
+
const confirmationPromise = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_waitForTransactionEvent).call(this, pendingTransactionTracker, transactionUpdated.id, transactionUpdated.hash);
|
|
55
|
+
pendingTransactionTracker.addTransactionToPoll(transactionUpdated);
|
|
56
|
+
await confirmationPromise;
|
|
57
|
+
results.push({ transactionHash });
|
|
58
|
+
}
|
|
59
|
+
catch (error) {
|
|
60
|
+
log('Batch transaction failed', { transaction, error });
|
|
61
|
+
pendingTransactionTracker.stop();
|
|
62
|
+
throw rpcErrors.internal(`Failed to publish batch transaction`);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
log('Sequential publish batch hook completed', { results });
|
|
66
|
+
pendingTransactionTracker.stop();
|
|
67
|
+
return { results };
|
|
68
|
+
}, _SequentialPublishBatchHook_waitForTransactionEvent =
|
|
69
|
+
/**
|
|
70
|
+
* Waits for a transaction event (confirmed, failed, or dropped) and resolves/rejects accordingly.
|
|
71
|
+
*
|
|
72
|
+
* @param pendingTransactionTracker - The tracker instance to subscribe to events.
|
|
73
|
+
* @param transactionId - The transaction ID.
|
|
74
|
+
* @param transactionHash - The hash of the transaction.
|
|
75
|
+
* @returns A promise that resolves when the transaction is confirmed or rejects if it fails or is dropped.
|
|
76
|
+
*/
|
|
77
|
+
async function _SequentialPublishBatchHook_waitForTransactionEvent(pendingTransactionTracker, transactionId, transactionHash) {
|
|
78
|
+
return new Promise((resolve, reject) => {
|
|
79
|
+
const onConfirmed = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_onConfirmed).bind(this, transactionId, transactionHash, resolve, pendingTransactionTracker);
|
|
80
|
+
const onFailedOrDropped = __classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_onFailedOrDropped).bind(this, transactionId, transactionHash, reject, pendingTransactionTracker);
|
|
81
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId] = {
|
|
82
|
+
onConfirmed,
|
|
83
|
+
onFailedOrDropped,
|
|
84
|
+
};
|
|
85
|
+
pendingTransactionTracker.hub.on('transaction-confirmed', onConfirmed);
|
|
86
|
+
pendingTransactionTracker.hub.on('transaction-failed', onFailedOrDropped);
|
|
87
|
+
pendingTransactionTracker.hub.on('transaction-dropped', onFailedOrDropped);
|
|
88
|
+
});
|
|
89
|
+
}, _SequentialPublishBatchHook_onConfirmed = function _SequentialPublishBatchHook_onConfirmed(transactionId, transactionHash, resolve, pendingTransactionTracker, txMeta) {
|
|
90
|
+
if (txMeta.id !== transactionId) {
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
log('Transaction confirmed', { transactionHash });
|
|
94
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_removeListeners).call(this, pendingTransactionTracker, transactionId);
|
|
95
|
+
resolve(txMeta);
|
|
96
|
+
}, _SequentialPublishBatchHook_onFailedOrDropped = function _SequentialPublishBatchHook_onFailedOrDropped(transactionId, transactionHash, reject, pendingTransactionTracker, txMeta, error) {
|
|
97
|
+
if (txMeta.id !== transactionId) {
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
log('Transaction failed or dropped', { transactionHash, error });
|
|
101
|
+
__classPrivateFieldGet(this, _SequentialPublishBatchHook_instances, "m", _SequentialPublishBatchHook_removeListeners).call(this, pendingTransactionTracker, transactionId);
|
|
102
|
+
reject(new Error(`Transaction ${transactionHash} failed or dropped.`));
|
|
103
|
+
}, _SequentialPublishBatchHook_removeListeners = function _SequentialPublishBatchHook_removeListeners(pendingTransactionTracker, transactionId) {
|
|
104
|
+
const listeners = __classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId];
|
|
105
|
+
pendingTransactionTracker.hub.off('transaction-confirmed', listeners.onConfirmed);
|
|
106
|
+
pendingTransactionTracker.hub.off('transaction-failed', listeners.onFailedOrDropped);
|
|
107
|
+
pendingTransactionTracker.hub.off('transaction-dropped', listeners.onFailedOrDropped);
|
|
108
|
+
delete __classPrivateFieldGet(this, _SequentialPublishBatchHook_boundListeners, "f")[transactionId];
|
|
109
|
+
};
|
|
110
|
+
//# sourceMappingURL=SequentialPublishBatchHook.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SequentialPublishBatchHook.mjs","sourceRoot":"","sources":["../../src/hooks/SequentialPublishBatchHook.ts"],"names":[],"mappings":"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","sourcesContent":["import type EthQuery from '@metamask/eth-query';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { createModuleLogger } from '@metamask/utils';\nimport type { Hex } from '@metamask/utils';\n\nimport type { PendingTransactionTracker } from '../helpers/PendingTransactionTracker';\nimport { projectLogger } from '../logger';\nimport {\n type PublishBatchHook,\n type PublishBatchHookRequest,\n type PublishBatchHookResult,\n type TransactionMeta,\n} from '../types';\n\nconst log = createModuleLogger(projectLogger, 'sequential-publish-batch-hook');\n\ntype SequentialPublishBatchHookOptions = {\n publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<Hex>;\n getTransaction: (id: string) => TransactionMeta;\n getEthQuery: (networkClientId: string) => EthQuery;\n getPendingTransactionTracker: (\n networkClientId: string,\n ) => PendingTransactionTracker;\n};\n\n/**\n * Custom publish logic that also publishes additional sequential transactions in a batch.\n * Requires the batch to be successful to resolve.\n */\nexport class SequentialPublishBatchHook {\n readonly #publishTransaction: (\n ethQuery: EthQuery,\n transactionMeta: TransactionMeta,\n ) => Promise<Hex>;\n\n readonly #getTransaction: (id: string) => TransactionMeta;\n\n readonly #getEthQuery: (networkClientId: string) => EthQuery;\n\n readonly #getPendingTransactionTracker: (\n networkClientId: string,\n ) => PendingTransactionTracker;\n\n #boundListeners: Record<\n string,\n {\n onConfirmed: (txMeta: TransactionMeta) => void;\n onFailedOrDropped: (txMeta: TransactionMeta, error?: Error) => void;\n }\n > = {};\n\n constructor({\n publishTransaction,\n getTransaction,\n getPendingTransactionTracker,\n getEthQuery,\n }: SequentialPublishBatchHookOptions) {\n this.#publishTransaction = publishTransaction;\n this.#getTransaction = getTransaction;\n this.#getEthQuery = getEthQuery;\n this.#getPendingTransactionTracker = getPendingTransactionTracker;\n }\n\n /**\n * @returns The publish batch hook function.\n */\n getHook(): PublishBatchHook {\n return this.#hook.bind(this);\n }\n\n async #hook({\n from,\n networkClientId,\n transactions,\n }: PublishBatchHookRequest): Promise<PublishBatchHookResult> {\n log('Starting sequential publish batch hook', { from, networkClientId });\n\n const pendingTransactionTracker =\n this.#getPendingTransactionTracker(networkClientId);\n const results = [];\n\n for (const transaction of transactions) {\n try {\n const transactionMeta = this.#getTransaction(String(transaction.id));\n\n const transactionHash = await this.#publishTransaction(\n this.#getEthQuery(networkClientId),\n transactionMeta,\n );\n log('Transaction published', { transactionHash });\n\n const transactionUpdated = {\n ...transactionMeta,\n hash: transactionHash,\n };\n\n const confirmationPromise = this.#waitForTransactionEvent(\n pendingTransactionTracker,\n transactionUpdated.id,\n transactionUpdated.hash,\n );\n\n pendingTransactionTracker.addTransactionToPoll(transactionUpdated);\n\n await confirmationPromise;\n results.push({ transactionHash });\n } catch (error) {\n log('Batch transaction failed', { transaction, error });\n pendingTransactionTracker.stop();\n throw rpcErrors.internal(`Failed to publish batch transaction`);\n }\n }\n\n log('Sequential publish batch hook completed', { results });\n pendingTransactionTracker.stop();\n\n return { results };\n }\n\n /**\n * Waits for a transaction event (confirmed, failed, or dropped) and resolves/rejects accordingly.\n *\n * @param pendingTransactionTracker - The tracker instance to subscribe to events.\n * @param transactionId - The transaction ID.\n * @param transactionHash - The hash of the transaction.\n * @returns A promise that resolves when the transaction is confirmed or rejects if it fails or is dropped.\n */\n async #waitForTransactionEvent(\n pendingTransactionTracker: PendingTransactionTracker,\n transactionId: string,\n transactionHash: string,\n ): Promise<TransactionMeta> {\n return new Promise((resolve, reject) => {\n const onConfirmed = this.#onConfirmed.bind(\n this,\n transactionId,\n transactionHash,\n resolve,\n pendingTransactionTracker,\n );\n\n const onFailedOrDropped = this.#onFailedOrDropped.bind(\n this,\n transactionId,\n transactionHash,\n reject,\n pendingTransactionTracker,\n );\n\n this.#boundListeners[transactionId] = {\n onConfirmed,\n onFailedOrDropped,\n };\n\n pendingTransactionTracker.hub.on('transaction-confirmed', onConfirmed);\n pendingTransactionTracker.hub.on('transaction-failed', onFailedOrDropped);\n pendingTransactionTracker.hub.on(\n 'transaction-dropped',\n onFailedOrDropped,\n );\n });\n }\n\n #onConfirmed(\n transactionId: string,\n transactionHash: string,\n resolve: (txMeta: TransactionMeta) => void,\n pendingTransactionTracker: PendingTransactionTracker,\n txMeta: TransactionMeta,\n ): void {\n if (txMeta.id !== transactionId) {\n return;\n }\n\n log('Transaction confirmed', { transactionHash });\n this.#removeListeners(pendingTransactionTracker, transactionId);\n resolve(txMeta);\n }\n\n #onFailedOrDropped(\n transactionId: string,\n transactionHash: string,\n reject: (error: Error) => void,\n pendingTransactionTracker: PendingTransactionTracker,\n txMeta: TransactionMeta,\n error?: Error,\n ): void {\n if (txMeta.id !== transactionId) {\n return;\n }\n\n log('Transaction failed or dropped', { transactionHash, error });\n this.#removeListeners(pendingTransactionTracker, transactionId);\n reject(new Error(`Transaction ${transactionHash} failed or dropped.`));\n }\n\n #removeListeners(\n pendingTransactionTracker: PendingTransactionTracker,\n transactionId: string,\n ): void {\n const listeners = this.#boundListeners[transactionId];\n\n pendingTransactionTracker.hub.off(\n 'transaction-confirmed',\n listeners.onConfirmed,\n );\n pendingTransactionTracker.hub.off(\n 'transaction-failed',\n listeners.onFailedOrDropped,\n );\n pendingTransactionTracker.hub.off(\n 'transaction-dropped',\n listeners.onFailedOrDropped,\n );\n\n delete this.#boundListeners[transactionId];\n }\n}\n"]}
|