@bitgo-beta/abstract-eth 1.2.3-alpha.49 → 1.2.3-alpha.491

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.
Files changed (157) hide show
  1. package/dist/src/abstractEthLikeCoin.d.ts +18 -9
  2. package/dist/src/abstractEthLikeCoin.d.ts.map +1 -1
  3. package/dist/src/abstractEthLikeCoin.js +39 -15
  4. package/dist/src/abstractEthLikeNewCoins.d.ts +847 -0
  5. package/dist/src/abstractEthLikeNewCoins.d.ts.map +1 -0
  6. package/dist/src/abstractEthLikeNewCoins.js +2531 -0
  7. package/dist/src/ethLikeToken.d.ts +36 -6
  8. package/dist/src/ethLikeToken.d.ts.map +1 -1
  9. package/dist/src/ethLikeToken.js +285 -10
  10. package/dist/src/index.d.ts +2 -0
  11. package/dist/src/index.d.ts.map +1 -1
  12. package/dist/src/index.js +8 -2
  13. package/dist/src/lib/constants.d.ts +4 -0
  14. package/dist/src/lib/constants.d.ts.map +1 -0
  15. package/dist/src/lib/constants.js +11 -0
  16. package/dist/src/lib/contractCall.d.ts +8 -0
  17. package/dist/src/lib/contractCall.d.ts.map +1 -0
  18. package/dist/src/lib/contractCall.js +17 -0
  19. package/dist/src/lib/iface.d.ts +133 -0
  20. package/dist/src/lib/iface.d.ts.map +1 -0
  21. package/dist/src/lib/iface.js +8 -0
  22. package/dist/src/lib/index.d.ts +17 -0
  23. package/dist/src/lib/index.d.ts.map +1 -0
  24. package/dist/src/lib/index.js +58 -0
  25. package/dist/src/lib/keyPair.d.ts +26 -0
  26. package/dist/src/lib/keyPair.d.ts.map +1 -0
  27. package/dist/src/lib/keyPair.js +65 -0
  28. package/dist/src/lib/messages/eip191/eip191Message.d.ts +12 -0
  29. package/dist/src/lib/messages/eip191/eip191Message.d.ts.map +1 -0
  30. package/dist/src/lib/messages/eip191/eip191Message.js +25 -0
  31. package/dist/src/lib/messages/eip191/eip191MessageBuilder.d.ts +19 -0
  32. package/dist/src/lib/messages/eip191/eip191MessageBuilder.d.ts.map +1 -0
  33. package/dist/src/lib/messages/eip191/eip191MessageBuilder.js +27 -0
  34. package/dist/src/lib/messages/eip191/index.d.ts +3 -0
  35. package/dist/src/lib/messages/eip191/index.d.ts.map +1 -0
  36. package/dist/src/lib/messages/eip191/index.js +19 -0
  37. package/dist/src/lib/messages/eip712/eip712Message.d.ts +6 -0
  38. package/dist/src/lib/messages/eip712/eip712Message.d.ts.map +1 -0
  39. package/dist/src/lib/messages/eip712/eip712Message.js +27 -0
  40. package/dist/src/lib/messages/eip712/eip712MessageBuilder.d.ts +7 -0
  41. package/dist/src/lib/messages/eip712/eip712MessageBuilder.d.ts.map +1 -0
  42. package/dist/src/lib/messages/eip712/eip712MessageBuilder.js +15 -0
  43. package/dist/src/lib/messages/eip712/index.d.ts +3 -0
  44. package/dist/src/lib/messages/eip712/index.d.ts.map +1 -0
  45. package/dist/src/lib/messages/eip712/index.js +19 -0
  46. package/dist/src/lib/messages/index.d.ts +4 -0
  47. package/dist/src/lib/messages/index.d.ts.map +1 -0
  48. package/dist/src/lib/messages/index.js +20 -0
  49. package/dist/src/lib/messages/messageBuilderFactory.d.ts +7 -0
  50. package/dist/src/lib/messages/messageBuilderFactory.d.ts.map +1 -0
  51. package/dist/src/lib/messages/messageBuilderFactory.js +23 -0
  52. package/dist/src/lib/transaction.d.ts +67 -0
  53. package/dist/src/lib/transaction.d.ts.map +1 -0
  54. package/dist/src/lib/transaction.js +142 -0
  55. package/dist/src/lib/transactionBuilder.d.ts +270 -0
  56. package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
  57. package/dist/src/lib/transactionBuilder.js +827 -0
  58. package/dist/src/lib/transferBuilder.d.ts +76 -0
  59. package/dist/src/lib/transferBuilder.d.ts.map +1 -0
  60. package/dist/src/lib/transferBuilder.js +307 -0
  61. package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.d.ts +54 -0
  62. package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.d.ts.map +1 -0
  63. package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.js +120 -0
  64. package/dist/src/lib/transferBuilders/index.d.ts +4 -0
  65. package/dist/src/lib/transferBuilders/index.d.ts.map +1 -0
  66. package/dist/src/lib/transferBuilders/index.js +20 -0
  67. package/dist/src/lib/transferBuilders/transferBuilderERC1155.d.ts +17 -0
  68. package/dist/src/lib/transferBuilders/transferBuilderERC1155.d.ts.map +1 -0
  69. package/dist/src/lib/transferBuilders/transferBuilderERC1155.js +96 -0
  70. package/dist/src/lib/transferBuilders/transferBuilderERC721.d.ts +16 -0
  71. package/dist/src/lib/transferBuilders/transferBuilderERC721.d.ts.map +1 -0
  72. package/dist/src/lib/transferBuilders/transferBuilderERC721.js +81 -0
  73. package/dist/src/lib/types.d.ts +39 -0
  74. package/dist/src/lib/types.d.ts.map +1 -0
  75. package/dist/src/lib/types.js +137 -0
  76. package/dist/src/lib/utils.d.ts +310 -0
  77. package/dist/src/lib/utils.d.ts.map +1 -0
  78. package/dist/src/lib/utils.js +829 -0
  79. package/dist/src/lib/walletUtil.d.ts +40 -0
  80. package/dist/src/lib/walletUtil.d.ts.map +1 -0
  81. package/dist/src/lib/walletUtil.js +43 -0
  82. package/dist/src/types.d.ts +9 -0
  83. package/dist/src/types.d.ts.map +1 -0
  84. package/dist/src/types.js +3 -0
  85. package/dist/test/index.d.ts +2 -0
  86. package/dist/test/index.d.ts.map +1 -0
  87. package/dist/test/index.js +18 -0
  88. package/dist/test/unit/coin.d.ts +8 -0
  89. package/dist/test/unit/coin.d.ts.map +1 -0
  90. package/dist/test/unit/coin.js +577 -0
  91. package/dist/test/unit/index.d.ts +6 -0
  92. package/dist/test/unit/index.d.ts.map +1 -0
  93. package/dist/test/unit/index.js +22 -0
  94. package/dist/test/unit/messages/abstractEthMessageBuilderTests.d.ts +3 -0
  95. package/dist/test/unit/messages/abstractEthMessageBuilderTests.d.ts.map +1 -0
  96. package/dist/test/unit/messages/abstractEthMessageBuilderTests.js +110 -0
  97. package/dist/test/unit/messages/abstractEthMessageTestTypes.d.ts +43 -0
  98. package/dist/test/unit/messages/abstractEthMessageTestTypes.d.ts.map +1 -0
  99. package/dist/test/unit/messages/abstractEthMessageTestTypes.js +3 -0
  100. package/dist/test/unit/messages/abstractEthMessagesTests.d.ts +3 -0
  101. package/dist/test/unit/messages/abstractEthMessagesTests.d.ts.map +1 -0
  102. package/dist/test/unit/messages/abstractEthMessagesTests.js +129 -0
  103. package/dist/test/unit/messages/eip191/eip191Message.d.ts +2 -0
  104. package/dist/test/unit/messages/eip191/eip191Message.d.ts.map +1 -0
  105. package/dist/test/unit/messages/eip191/eip191Message.js +15 -0
  106. package/dist/test/unit/messages/eip191/eip191MessageBuilder.d.ts +2 -0
  107. package/dist/test/unit/messages/eip191/eip191MessageBuilder.d.ts.map +1 -0
  108. package/dist/test/unit/messages/eip191/eip191MessageBuilder.js +16 -0
  109. package/dist/test/unit/messages/eip191/fixtures.d.ts +109 -0
  110. package/dist/test/unit/messages/eip191/fixtures.d.ts.map +1 -0
  111. package/dist/test/unit/messages/eip191/fixtures.js +63 -0
  112. package/dist/test/unit/messages/eip712/eip712Message.d.ts +2 -0
  113. package/dist/test/unit/messages/eip712/eip712Message.d.ts.map +1 -0
  114. package/dist/test/unit/messages/eip712/eip712Message.js +15 -0
  115. package/dist/test/unit/messages/eip712/eip712MessageBuilder.d.ts +2 -0
  116. package/dist/test/unit/messages/eip712/eip712MessageBuilder.d.ts.map +1 -0
  117. package/dist/test/unit/messages/eip712/eip712MessageBuilder.js +16 -0
  118. package/dist/test/unit/messages/eip712/fixtures.d.ts +76 -0
  119. package/dist/test/unit/messages/eip712/fixtures.d.ts.map +1 -0
  120. package/dist/test/unit/messages/eip712/fixtures.js +120 -0
  121. package/dist/test/unit/messages/index.d.ts +4 -0
  122. package/dist/test/unit/messages/index.d.ts.map +1 -0
  123. package/dist/test/unit/messages/index.js +20 -0
  124. package/dist/test/unit/messages/messageBuilderFactory.d.ts +2 -0
  125. package/dist/test/unit/messages/messageBuilderFactory.d.ts.map +1 -0
  126. package/dist/test/unit/messages/messageBuilderFactory.js +52 -0
  127. package/dist/test/unit/token.d.ts +2 -0
  128. package/dist/test/unit/token.d.ts.map +1 -0
  129. package/dist/test/unit/token.js +37 -0
  130. package/dist/test/unit/transaction.d.ts +3 -0
  131. package/dist/test/unit/transaction.d.ts.map +1 -0
  132. package/dist/test/unit/transaction.js +60 -0
  133. package/dist/test/unit/transactionBuilder/addressInitialization.d.ts +8 -0
  134. package/dist/test/unit/transactionBuilder/addressInitialization.d.ts.map +1 -0
  135. package/dist/test/unit/transactionBuilder/addressInitialization.js +95 -0
  136. package/dist/test/unit/transactionBuilder/flushNft.d.ts +2 -0
  137. package/dist/test/unit/transactionBuilder/flushNft.d.ts.map +1 -0
  138. package/dist/test/unit/transactionBuilder/flushNft.js +381 -0
  139. package/dist/test/unit/transactionBuilder/index.d.ts +5 -0
  140. package/dist/test/unit/transactionBuilder/index.d.ts.map +1 -0
  141. package/dist/test/unit/transactionBuilder/index.js +21 -0
  142. package/dist/test/unit/transactionBuilder/send.d.ts +3 -0
  143. package/dist/test/unit/transactionBuilder/send.d.ts.map +1 -0
  144. package/dist/test/unit/transactionBuilder/send.js +197 -0
  145. package/dist/test/unit/transactionBuilder/walletInitialization.d.ts +11 -0
  146. package/dist/test/unit/transactionBuilder/walletInitialization.d.ts.map +1 -0
  147. package/dist/test/unit/transactionBuilder/walletInitialization.js +137 -0
  148. package/dist/test/unit/transferBuilder.d.ts +2 -0
  149. package/dist/test/unit/transferBuilder.d.ts.map +1 -0
  150. package/dist/test/unit/transferBuilder.js +76 -0
  151. package/dist/test/unit/utils.d.ts +2 -0
  152. package/dist/test/unit/utils.d.ts.map +1 -0
  153. package/dist/test/unit/utils.js +184 -0
  154. package/dist/tsconfig.tsbuildinfo +1 -8564
  155. package/package.json +34 -10
  156. package/.eslintignore +0 -5
  157. package/CHANGELOG.md +0 -194
@@ -0,0 +1,2531 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.AbstractEthLikeNewCoins = exports.optionalDeps = exports.DEFAULT_SCAN_FACTOR = void 0;
7
+ exports.isVerifyContractBaseAddressOptions = isVerifyContractBaseAddressOptions;
8
+ /**
9
+ * @prettier
10
+ */
11
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
12
+ const sdk_lib_mpc_1 = require("@bitgo-beta/sdk-lib-mpc");
13
+ const secp256k1_1 = require("@bitgo-beta/secp256k1");
14
+ const statics_1 = require("@bitgo-beta/statics");
15
+ const tx_1 = require("@ethereumjs/tx");
16
+ const rlp_1 = require("@ethereumjs/rlp");
17
+ const eth_sig_util_1 = require("@metamask/eth-sig-util");
18
+ const bignumber_js_1 = require("bignumber.js");
19
+ const bn_js_1 = __importDefault(require("bn.js"));
20
+ const crypto_1 = require("crypto");
21
+ const debug_1 = __importDefault(require("debug"));
22
+ const ethereumjs_util_1 = require("ethereumjs-util");
23
+ const keccak_1 = __importDefault(require("keccak"));
24
+ const lodash_1 = __importDefault(require("lodash"));
25
+ const secp256k1_2 = __importDefault(require("secp256k1"));
26
+ const abstractEthLikeCoin_1 = require("./abstractEthLikeCoin");
27
+ const ethLikeToken_1 = require("./ethLikeToken");
28
+ const lib_1 = require("./lib");
29
+ exports.DEFAULT_SCAN_FACTOR = 20;
30
+ /**
31
+ * Type guard to check if params are for BIP32 base address verification (V1, V2, V4)
32
+ * These wallet versions use ethAddress for address derivation
33
+ */
34
+ function isVerifyContractBaseAddressOptions(params) {
35
+ return ((params.walletVersion === 1 || params.walletVersion === 2 || params.walletVersion === 4) &&
36
+ 'keychains' in params &&
37
+ Array.isArray(params.keychains) &&
38
+ params.keychains.length === 3 &&
39
+ params.keychains.every((kc) => 'ethAddress' in kc && typeof kc.ethAddress === 'string'));
40
+ }
41
+ const debug = (0, debug_1.default)('bitgo:v2:ethlike');
42
+ exports.optionalDeps = {
43
+ get ethAbi() {
44
+ try {
45
+ return require('ethereumjs-abi');
46
+ }
47
+ catch (e) {
48
+ debug('unable to load ethereumjs-abi:');
49
+ debug(e.stack);
50
+ throw new sdk_core_1.EthereumLibraryUnavailableError(`ethereumjs-abi`);
51
+ }
52
+ },
53
+ get ethUtil() {
54
+ try {
55
+ return require('ethereumjs-util');
56
+ }
57
+ catch (e) {
58
+ debug('unable to load ethereumjs-util:');
59
+ debug(e.stack);
60
+ throw new sdk_core_1.EthereumLibraryUnavailableError(`ethereumjs-util`);
61
+ }
62
+ },
63
+ get EthTx() {
64
+ try {
65
+ return require('@ethereumjs/tx');
66
+ }
67
+ catch (e) {
68
+ debug('unable to load @ethereumjs/tx');
69
+ debug(e.stack);
70
+ throw new sdk_core_1.EthereumLibraryUnavailableError(`@ethereumjs/tx`);
71
+ }
72
+ },
73
+ get EthCommon() {
74
+ try {
75
+ return require('@ethereumjs/common');
76
+ }
77
+ catch (e) {
78
+ debug('unable to load @ethereumjs/common:');
79
+ debug(e.stack);
80
+ throw new sdk_core_1.EthereumLibraryUnavailableError(`@ethereumjs/common`);
81
+ }
82
+ },
83
+ };
84
+ class AbstractEthLikeNewCoins extends abstractEthLikeCoin_1.AbstractEthLikeCoin {
85
+ constructor(bitgo, staticsCoin) {
86
+ super(bitgo, staticsCoin);
87
+ /**
88
+ * Get the data required to make an ETH function call defined by the given types and values
89
+ *
90
+ * @param {string} functionName - The name of the function being called, e.g. transfer
91
+ * @param types The types of the function call in order
92
+ * @param values The values of the function call in order
93
+ * @return {Buffer} The combined data for the function call
94
+ */
95
+ this.getMethodCallData = (functionName, types, values) => {
96
+ return Buffer.concat([
97
+ // function signature
98
+ exports.optionalDeps.ethAbi.methodID(functionName, types),
99
+ // function arguments
100
+ exports.optionalDeps.ethAbi.rawEncode(types, values),
101
+ ]);
102
+ };
103
+ if (!staticsCoin) {
104
+ throw new Error('missing required constructor parameter staticsCoin');
105
+ }
106
+ this.staticsCoin = staticsCoin;
107
+ this.sendMethodName = 'sendMultiSig';
108
+ }
109
+ /**
110
+ * Method to return the coin's network object
111
+ * @returns {EthLikeNetwork | undefined}
112
+ */
113
+ getNetwork() {
114
+ return this.staticsCoin?.network;
115
+ }
116
+ /**
117
+ * Evaluates whether an address string is valid for this coin
118
+ * @param {string} address
119
+ * @returns {boolean} True if address is the valid ethlike adderss
120
+ */
121
+ isValidAddress(address) {
122
+ return exports.optionalDeps.ethUtil.isValidAddress(exports.optionalDeps.ethUtil.addHexPrefix(address));
123
+ }
124
+ /**
125
+ * Flag for sending data along with transactions
126
+ * @returns {boolean} True if okay to send tx data (ETH), false otherwise
127
+ */
128
+ transactionDataAllowed() {
129
+ return true;
130
+ }
131
+ /** @inheritDoc */
132
+ supportsMessageSigning() {
133
+ return true;
134
+ }
135
+ /** @inheritDoc */
136
+ supportsSigningTypedData() {
137
+ return true;
138
+ }
139
+ /**
140
+ * Default expire time for a contract call (1 week)
141
+ * @returns {number} Time in seconds
142
+ */
143
+ getDefaultExpireTime() {
144
+ return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
145
+ }
146
+ /**
147
+ * Method to get the custom chain common object based on params from recovery
148
+ * @param {number} chainId - the chain id of the custom chain
149
+ * @returns {EthLikeCommon.default}
150
+ */
151
+ static getCustomChainCommon(chainId) {
152
+ const coinName = statics_1.coins.coinNameFromChainId(chainId);
153
+ if (!coinName) {
154
+ throw new statics_1.ChainIdNotFoundError(chainId);
155
+ }
156
+ const coin = statics_1.coins.get(coinName);
157
+ const ethLikeCommon = (0, lib_1.getCommon)(coin.network);
158
+ return ethLikeCommon;
159
+ }
160
+ /**
161
+ * Gets correct Eth Common object based on params from either recovery or tx building
162
+ * @param {EIP1559} eip1559 - configs that specify whether we should construct an eip1559 tx
163
+ * @param {ReplayProtectionOptions} replayProtectionOptions - check if chain id supports replay protection
164
+ * @returns {EthLikeCommon.default}
165
+ */
166
+ static getEthLikeCommon(eip1559, replayProtectionOptions) {
167
+ // if eip1559 params are specified, default to london hardfork, otherwise,
168
+ // default to petersburg to avoid replay protection issues
169
+ const defaultHardfork = !!eip1559 ? 'london' : exports.optionalDeps.EthCommon.Hardfork.Petersburg;
170
+ const ethLikeCommon = AbstractEthLikeNewCoins.getCustomChainCommon(replayProtectionOptions?.chain);
171
+ ethLikeCommon.setHardfork(replayProtectionOptions?.hardfork ?? defaultHardfork);
172
+ return ethLikeCommon;
173
+ }
174
+ /**
175
+ * Method to build the tx object
176
+ * @param {BuildTransactionParams} params - params to build transaction
177
+ * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}
178
+ */
179
+ static buildTransaction(params) {
180
+ // if eip1559 params are specified, default to london hardfork, otherwise,
181
+ // default to tangerine whistle to avoid replay protection issues
182
+ const ethLikeCommon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);
183
+ const baseParams = {
184
+ to: params.to,
185
+ nonce: params.nonce,
186
+ value: params.value,
187
+ data: params.data,
188
+ gasLimit: new exports.optionalDeps.ethUtil.BN(params.gasLimit),
189
+ };
190
+ const unsignedEthTx = !!params.eip1559
191
+ ? exports.optionalDeps.EthTx.FeeMarketEIP1559Transaction.fromTxData({
192
+ ...baseParams,
193
+ maxFeePerGas: new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas),
194
+ maxPriorityFeePerGas: new exports.optionalDeps.ethUtil.BN(params.eip1559.maxPriorityFeePerGas),
195
+ }, { common: ethLikeCommon })
196
+ : exports.optionalDeps.EthTx.Transaction.fromTxData({
197
+ ...baseParams,
198
+ gasPrice: new exports.optionalDeps.ethUtil.BN(params.gasPrice),
199
+ }, { common: ethLikeCommon });
200
+ return unsignedEthTx;
201
+ }
202
+ /**
203
+ * Query explorer for the balance of an address
204
+ * @param {String} address - the ETHLike address
205
+ * @param {String} apiKey - optional API key to use instead of the one from the environment
206
+ * @returns {BigNumber} address balance
207
+ */
208
+ async queryAddressBalance(address, apiKey) {
209
+ const result = await this.recoveryBlockchainExplorerQuery({
210
+ chainid: this.getChainId().toString(),
211
+ module: 'account',
212
+ action: 'balance',
213
+ address: address,
214
+ }, apiKey);
215
+ // throw if the result does not exist or the result is not a valid number
216
+ if (!result || !result.result || isNaN(result.result)) {
217
+ throw new Error(`Could not obtain address balance for ${address} from the explorer, got: ${result.result}`);
218
+ }
219
+ return new exports.optionalDeps.ethUtil.BN(result.result, 10);
220
+ }
221
+ /**
222
+ * @param {Recipient[]} recipients - the recipients of the transaction
223
+ * @param {number} expireTime - the expire time of the transaction
224
+ * @param {number} contractSequenceId - the contract sequence id of the transaction
225
+ * @returns {string}
226
+ */
227
+ getOperationSha3ForExecuteAndConfirm(recipients, expireTime, contractSequenceId) {
228
+ if (!recipients || !Array.isArray(recipients)) {
229
+ throw new Error('expecting array of recipients');
230
+ }
231
+ // Right now we only support 1 recipient
232
+ if (recipients.length !== 1) {
233
+ throw new Error('must send to exactly 1 recipient');
234
+ }
235
+ if (!lodash_1.default.isNumber(expireTime)) {
236
+ throw new Error('expireTime must be number of seconds since epoch');
237
+ }
238
+ if (!lodash_1.default.isNumber(contractSequenceId)) {
239
+ throw new Error('contractSequenceId must be number');
240
+ }
241
+ // Check inputs
242
+ recipients.forEach(function (recipient) {
243
+ if (!lodash_1.default.isString(recipient.address) ||
244
+ !exports.optionalDeps.ethUtil.isValidAddress(exports.optionalDeps.ethUtil.addHexPrefix(recipient.address))) {
245
+ throw new Error('Invalid address: ' + recipient.address);
246
+ }
247
+ let amount;
248
+ try {
249
+ amount = new bignumber_js_1.BigNumber(recipient.amount);
250
+ }
251
+ catch (e) {
252
+ throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');
253
+ }
254
+ recipient.amount = amount.toFixed(0);
255
+ if (recipient.data && !lodash_1.default.isString(recipient.data)) {
256
+ throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');
257
+ }
258
+ });
259
+ const recipient = recipients[0];
260
+ return exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId)));
261
+ }
262
+ /**
263
+ * Get transfer operation for coin
264
+ * @param {Recipient} recipient - recipient info
265
+ * @param {number} expireTime - expiry time
266
+ * @param {number} contractSequenceId - sequence id
267
+ * @returns {Array} operation array
268
+ */
269
+ getOperation(recipient, expireTime, contractSequenceId) {
270
+ const network = this.getNetwork();
271
+ return [
272
+ ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
273
+ [
274
+ network.nativeCoinOperationHashPrefix,
275
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
276
+ recipient.amount,
277
+ Buffer.from(exports.optionalDeps.ethUtil.stripHexPrefix(exports.optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),
278
+ expireTime,
279
+ contractSequenceId,
280
+ ],
281
+ ];
282
+ }
283
+ /**
284
+ * Queries the contract (via explorer API) for the next sequence ID
285
+ * @param {String} address - address of the contract
286
+ * @param {String} apiKey - optional API key to use instead of the one from the environment
287
+ * @returns {Promise<Number>} sequence ID
288
+ */
289
+ async querySequenceId(address, apiKey) {
290
+ // Get sequence ID using contract call
291
+ const sequenceIdMethodSignature = exports.optionalDeps.ethAbi.methodID('getNextSequenceId', []);
292
+ const sequenceIdArgs = exports.optionalDeps.ethAbi.rawEncode([], []);
293
+ const sequenceIdData = Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');
294
+ const result = await this.recoveryBlockchainExplorerQuery({
295
+ chainid: this.getChainId().toString(),
296
+ module: 'proxy',
297
+ action: 'eth_call',
298
+ to: address,
299
+ data: sequenceIdData,
300
+ tag: 'latest',
301
+ }, apiKey);
302
+ if (!result || !result.result) {
303
+ throw new Error('Could not obtain sequence ID from explorer, got: ' + result.result);
304
+ }
305
+ const sequenceIdHex = result.result;
306
+ return new exports.optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();
307
+ }
308
+ /**
309
+ * Recover an unsupported token from a BitGo multisig wallet
310
+ * This builds a half-signed transaction, for which there will be an admin route to co-sign and broadcast. Optionally
311
+ * the user can set params.broadcast = true and the half-signed tx will be sent to BitGo for cosigning and broadcasting
312
+ * @param {RecoverTokenOptions} params
313
+ * @param {Wallet} params.wallet - the wallet to recover the token from
314
+ * @param {string} params.tokenContractAddress - the contract address of the unsupported token
315
+ * @param {string} params.recipient - the destination address recovered tokens should be sent to
316
+ * @param {string} params.walletPassphrase - the wallet passphrase
317
+ * @param {string} params.prv - the xprv
318
+ * @param {boolean} params.broadcast - if true, we will automatically submit the half-signed tx to BitGo for cosigning and broadcasting
319
+ * @returns {Promise<RecoverTokenTransaction>}
320
+ */
321
+ async recoverToken(params) {
322
+ const network = this.getNetwork();
323
+ if (!lodash_1.default.isObject(params)) {
324
+ throw new Error(`recoverToken must be passed a params object. Got ${params} (type ${typeof params})`);
325
+ }
326
+ if (lodash_1.default.isUndefined(params.tokenContractAddress) || !lodash_1.default.isString(params.tokenContractAddress)) {
327
+ throw new Error(`tokenContractAddress must be a string, got ${params.tokenContractAddress} (type ${typeof params.tokenContractAddress})`);
328
+ }
329
+ if (!this.isValidAddress(params.tokenContractAddress)) {
330
+ throw new Error('tokenContractAddress not a valid address');
331
+ }
332
+ if (lodash_1.default.isUndefined(params.wallet) || !(params.wallet instanceof sdk_core_1.Wallet)) {
333
+ throw new Error(`wallet must be a wallet instance, got ${params.wallet} (type ${typeof params.wallet})`);
334
+ }
335
+ if (lodash_1.default.isUndefined(params.recipient) || !lodash_1.default.isString(params.recipient)) {
336
+ throw new Error(`recipient must be a string, got ${params.recipient} (type ${typeof params.recipient})`);
337
+ }
338
+ if (!this.isValidAddress(params.recipient)) {
339
+ throw new Error('recipient not a valid address');
340
+ }
341
+ if (!exports.optionalDeps.ethUtil.bufferToHex || !exports.optionalDeps.ethAbi.soliditySHA3) {
342
+ throw new Error('ethereum not fully supported in this environment');
343
+ }
344
+ // Get token balance from external API
345
+ const coinSpecific = params.wallet.coinSpecific();
346
+ if (!coinSpecific || !lodash_1.default.isString(coinSpecific.baseAddress)) {
347
+ throw new Error('missing required coin specific property baseAddress');
348
+ }
349
+ const recoveryAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, coinSpecific.baseAddress);
350
+ if (params.broadcast) {
351
+ // We're going to create a normal ETH transaction that sends an amount of 0 ETH to the
352
+ // tokenContractAddress and encode the unsupported-token-send data in the data field
353
+ // #tricksy
354
+ const sendMethodArgs = [
355
+ {
356
+ name: '_to',
357
+ type: 'address',
358
+ value: params.recipient,
359
+ },
360
+ {
361
+ name: '_value',
362
+ type: 'uint256',
363
+ value: recoveryAmount.toString(10),
364
+ },
365
+ ];
366
+ const methodSignature = exports.optionalDeps.ethAbi.methodID('transfer', lodash_1.default.map(sendMethodArgs, 'type'));
367
+ const encodedArgs = exports.optionalDeps.ethAbi.rawEncode(lodash_1.default.map(sendMethodArgs, 'type'), lodash_1.default.map(sendMethodArgs, 'value'));
368
+ const sendData = Buffer.concat([methodSignature, encodedArgs]);
369
+ const broadcastParams = {
370
+ address: params.tokenContractAddress,
371
+ amount: '0',
372
+ data: sendData.toString('hex'),
373
+ };
374
+ if (params.walletPassphrase) {
375
+ broadcastParams.walletPassphrase = params.walletPassphrase;
376
+ }
377
+ else if (params.prv) {
378
+ broadcastParams.prv = params.prv;
379
+ }
380
+ return await params.wallet.send(broadcastParams);
381
+ }
382
+ const recipient = {
383
+ address: params.recipient,
384
+ amount: recoveryAmount.toString(10),
385
+ };
386
+ // This signature will be valid for one week
387
+ const expireTime = Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
388
+ // Get sequence ID. We do this by building a 'fake' eth transaction, so the platform will increment and return us the new sequence id
389
+ // This _does_ require the user to have a non-zero wallet balance
390
+ const { nextContractSequenceId, gasPrice, gasLimit } = (await params.wallet.prebuildTransaction({
391
+ recipients: [
392
+ {
393
+ address: params.recipient,
394
+ amount: '1',
395
+ },
396
+ ],
397
+ }));
398
+ // these recoveries need to be processed by support, but if the customer sends any transactions before recovery is
399
+ // complete the sequence ID will be invalid. artificially inflate the sequence ID to allow more time for processing
400
+ const safeSequenceId = nextContractSequenceId + 1000;
401
+ // Build sendData for ethereum tx
402
+ const operationTypes = ['string', 'address', 'uint', 'address', 'uint', 'uint'];
403
+ const operationArgs = [
404
+ // Token operation has prefix has been added here so that ether operation hashes, signatures cannot be re-used for tokenSending
405
+ network.tokenOperationHashPrefix,
406
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
407
+ recipient.amount,
408
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(params.tokenContractAddress), 16),
409
+ expireTime,
410
+ safeSequenceId,
411
+ ];
412
+ const operationHash = exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(operationTypes, operationArgs));
413
+ const userPrv = await params.wallet.getPrv({
414
+ prv: params.prv,
415
+ walletPassphrase: params.walletPassphrase,
416
+ });
417
+ const signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userPrv));
418
+ return {
419
+ halfSigned: {
420
+ recipient: recipient,
421
+ expireTime: expireTime,
422
+ contractSequenceId: safeSequenceId,
423
+ operationHash: operationHash,
424
+ signature: signature,
425
+ gasLimit: gasLimit,
426
+ gasPrice: gasPrice,
427
+ tokenContractAddress: params.tokenContractAddress,
428
+ walletId: params.wallet.id(),
429
+ },
430
+ };
431
+ }
432
+ /**
433
+ * Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key
434
+ * @param {PrecreateBitGoOptions} params
435
+ * @param {string} params.enterprise {String} the enterprise id to associate with this key
436
+ * @param {string} params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)
437
+ * @returns {void}
438
+ */
439
+ preCreateBitGo(params) {
440
+ // We always need params object, since either enterprise or newFeeAddress is required
441
+ if (!lodash_1.default.isObject(params)) {
442
+ throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);
443
+ }
444
+ if (lodash_1.default.isUndefined(params.enterprise) && lodash_1.default.isUndefined(params.newFeeAddress)) {
445
+ throw new Error('expecting enterprise when adding BitGo key. If you want to create a new ETH bitgo key, set the newFeeAddress parameter to true.');
446
+ }
447
+ // Check whether key should be an enterprise key or a BitGo key for a new fee address
448
+ if (!lodash_1.default.isUndefined(params.enterprise) && !lodash_1.default.isUndefined(params.newFeeAddress)) {
449
+ throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);
450
+ }
451
+ if (!lodash_1.default.isUndefined(params.enterprise) && !lodash_1.default.isString(params.enterprise)) {
452
+ throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);
453
+ }
454
+ if (!lodash_1.default.isUndefined(params.newFeeAddress) && !lodash_1.default.isBoolean(params.newFeeAddress)) {
455
+ throw new Error(`newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`);
456
+ }
457
+ }
458
+ /**
459
+ * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address
460
+ * @param {string} address
461
+ * @param {string} apiKey - optional API key to use instead of the one from the environment
462
+ * @returns {Promise<number>}
463
+ */
464
+ async getAddressNonce(address, apiKey) {
465
+ // Get nonce for backup key (should be 0)
466
+ let nonce = 0;
467
+ const result = await this.recoveryBlockchainExplorerQuery({
468
+ chainid: this.getChainId().toString(),
469
+ module: 'account',
470
+ action: 'txlist',
471
+ address,
472
+ }, apiKey);
473
+ if (result && typeof result?.nonce === 'number') {
474
+ return Number(result.nonce);
475
+ }
476
+ if (!result || !Array.isArray(result.result)) {
477
+ throw new Error('Unable to find next nonce from Etherscan, got: ' + JSON.stringify(result));
478
+ }
479
+ const backupKeyTxList = result.result;
480
+ if (backupKeyTxList.length > 0) {
481
+ // Calculate last nonce used
482
+ const outgoingTxs = backupKeyTxList.filter((tx) => tx.from === address);
483
+ nonce = outgoingTxs.length;
484
+ }
485
+ return nonce;
486
+ }
487
+ /**
488
+ * Helper function for recover()
489
+ * This transforms the unsigned transaction information into a format the BitGo offline vault expects
490
+ * @param {UnformattedTxInfo} txInfo - tx info
491
+ * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object
492
+ * @param {string} userKey - the user's key
493
+ * @param {string} backupKey - the backup key
494
+ * @param {Buffer} gasPrice - gas price for the tx
495
+ * @param {number} gasLimit - gas limit for the tx
496
+ * @param {EIP1559} eip1559 - eip1559 params
497
+ * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options
498
+ * @param {apiKey} apiKey - optional apiKey to use when retrieving block chain data
499
+ * @returns {Promise<OfflineVaultTxInfo>}
500
+ */
501
+ async formatForOfflineVault(txInfo, ethTx, userKey, backupKey, gasPrice, gasLimit, eip1559, replayProtectionOptions, apiKey) {
502
+ if (!ethTx.to) {
503
+ throw new Error('Eth tx must have a `to` address');
504
+ }
505
+ const backupHDNode = secp256k1_1.bip32.fromBase58(backupKey);
506
+ const backupSigningKey = backupHDNode.publicKey;
507
+ const response = {
508
+ tx: ethTx.serialize().toString('hex'),
509
+ userKey,
510
+ backupKey,
511
+ coin: this.getChain(),
512
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
513
+ gasLimit,
514
+ recipients: [txInfo.recipient],
515
+ walletContractAddress: ethTx.to.toString(),
516
+ amount: txInfo.recipient.amount,
517
+ backupKeyNonce: await this.getAddressNonce(`0x${exports.optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`, apiKey),
518
+ eip1559,
519
+ replayProtectionOptions,
520
+ };
521
+ lodash_1.default.extend(response, txInfo);
522
+ response.nextContractSequenceId = response.contractSequenceId;
523
+ return response;
524
+ }
525
+ /**
526
+ * Helper function for recover()
527
+ * This transforms the unsigned transaction information into a format the BitGo offline vault expects
528
+ * @param {UnformattedTxInfo} txInfo - tx info
529
+ * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object
530
+ * @param {string} userKey - the user's key
531
+ * @param {string} backupKey - the backup key
532
+ * @param {Buffer} gasPrice - gas price for the tx
533
+ * @param {number} gasLimit - gas limit for the tx
534
+ * @param {number} backupKeyNonce - the nonce of the backup key address
535
+ * @param {EIP1559} eip1559 - eip1559 params
536
+ * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options
537
+ * @returns {Promise<OfflineVaultTxInfo>}
538
+ */
539
+ formatForOfflineVaultTSS(txInfo, ethTx, userKey, backupKey, gasPrice, gasLimit, backupKeyNonce, eip1559, replayProtectionOptions) {
540
+ if (!ethTx.to) {
541
+ throw new Error('Eth tx must have a `to` address');
542
+ }
543
+ const response = {
544
+ tx: ethTx.serialize().toString('hex'),
545
+ txHex: ethTx.getMessageToSign(false).toString(),
546
+ userKey,
547
+ backupKey,
548
+ coin: this.getChain(),
549
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
550
+ gasLimit,
551
+ recipients: [txInfo.recipient],
552
+ walletContractAddress: ethTx.to.toString(),
553
+ amount: txInfo.recipient.amount,
554
+ backupKeyNonce: backupKeyNonce,
555
+ eip1559,
556
+ replayProtectionOptions,
557
+ };
558
+ lodash_1.default.extend(response, txInfo);
559
+ return response;
560
+ }
561
+ /**
562
+ * Check whether the gas price passed in by user are within our max and min bounds
563
+ * If they are not set, set them to the defaults
564
+ * @param {number} userGasPrice - user defined gas price
565
+ * @returns {number} the gas price to use for this transaction
566
+ */
567
+ setGasPrice(userGasPrice) {
568
+ if (!userGasPrice) {
569
+ return statics_1.ethGasConfigs.defaultGasPrice;
570
+ }
571
+ const gasPriceMax = statics_1.ethGasConfigs.maximumGasPrice;
572
+ const gasPriceMin = statics_1.ethGasConfigs.minimumGasPrice;
573
+ if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {
574
+ throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);
575
+ }
576
+ return userGasPrice;
577
+ }
578
+ /**
579
+ * Check whether gas limit passed in by user are within our max and min bounds
580
+ * If they are not set, set them to the defaults
581
+ * @param {number} userGasLimit user defined gas limit
582
+ * @returns {number} the gas limit to use for this transaction
583
+ */
584
+ setGasLimit(userGasLimit) {
585
+ if (!userGasLimit) {
586
+ return statics_1.ethGasConfigs.defaultGasLimit;
587
+ }
588
+ const gasLimitMax = statics_1.ethGasConfigs.maximumGasLimit;
589
+ const gasLimitMin = statics_1.ethGasConfigs.minimumGasLimit;
590
+ if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {
591
+ throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);
592
+ }
593
+ return userGasLimit;
594
+ }
595
+ /**
596
+ * Helper function for signTransaction for the rare case that SDK is doing the second signature
597
+ * Note: we are expecting this to be called from the offline vault
598
+ * @param {SignFinalOptions.txPrebuild} params.txPrebuild
599
+ * @param {string} params.prv
600
+ * @returns {{txHex: string}}
601
+ */
602
+ async signFinalEthLike(params) {
603
+ const signingKey = new lib_1.KeyPair({ prv: params.prv }).getKeys().prv;
604
+ if (lodash_1.default.isUndefined(signingKey)) {
605
+ throw new Error('missing private key');
606
+ }
607
+ const txBuilder = this.getTransactionBuilder(params.common);
608
+ try {
609
+ txBuilder.from(params.txPrebuild.halfSigned?.txHex);
610
+ }
611
+ catch (e) {
612
+ throw new Error('invalid half-signed transaction');
613
+ }
614
+ txBuilder.sign({ key: signingKey });
615
+ const tx = await txBuilder.build();
616
+ return {
617
+ txHex: tx.toBroadcastFormat(),
618
+ };
619
+ }
620
+ /**
621
+ * Assemble half-sign prebuilt transaction
622
+ * @param {SignTransactionOptions} params
623
+ */
624
+ async signTransaction(params) {
625
+ // Normally the SDK provides the first signature for an EthLike tx, but occasionally it provides the second and final one.
626
+ if (params.isLastSignature) {
627
+ // In this case when we're doing the second (final) signature, the logic is different.
628
+ return await this.signFinalEthLike(params);
629
+ }
630
+ const txBuilder = this.getTransactionBuilder(params.common);
631
+ txBuilder.from(params.txPrebuild.txHex);
632
+ txBuilder
633
+ .transfer()
634
+ .coin(this.staticsCoin?.name)
635
+ .key(new lib_1.KeyPair({ prv: params.prv }).getKeys().prv);
636
+ if (params.walletVersion) {
637
+ txBuilder.walletVersion(params.walletVersion);
638
+ }
639
+ const transaction = await txBuilder.build();
640
+ // In case of tx with contract data from a custodial wallet, we are running into an issue
641
+ // as halfSigned is not having the data field. So, we are adding the data field to the halfSigned tx
642
+ let recipients = params.txPrebuild.recipients || params.recipients;
643
+ if (recipients === undefined) {
644
+ recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));
645
+ }
646
+ const txParams = {
647
+ eip1559: params.txPrebuild.eip1559,
648
+ txHex: transaction.toBroadcastFormat(),
649
+ recipients: recipients,
650
+ expiration: params.txPrebuild.expireTime,
651
+ hopTransaction: params.txPrebuild.hopTransaction,
652
+ custodianTransactionId: params.custodianTransactionId,
653
+ expireTime: params.expireTime,
654
+ contractSequenceId: params.txPrebuild.nextContractSequenceId,
655
+ sequenceId: params.sequenceId,
656
+ ...(params.txPrebuild.isBatch ? { isBatch: params.txPrebuild.isBatch } : {}),
657
+ };
658
+ return { halfSigned: txParams };
659
+ }
660
+ /**
661
+ * Method to validate recovery params
662
+ * @param {RecoverOptions} params
663
+ * @returns {void}
664
+ */
665
+ validateRecoveryParams(params) {
666
+ if (params.userKey === undefined) {
667
+ throw new Error('missing userKey');
668
+ }
669
+ if (params.backupKey === undefined) {
670
+ throw new Error('missing backupKey');
671
+ }
672
+ if (!params.isUnsignedSweep &&
673
+ params.walletPassphrase === undefined &&
674
+ !params.userKey.startsWith('xpub') &&
675
+ !params.isTss) {
676
+ throw new Error('missing wallet passphrase');
677
+ }
678
+ if (params.walletContractAddress === undefined || !this.isValidAddress(params.walletContractAddress)) {
679
+ throw new Error('invalid walletContractAddress');
680
+ }
681
+ if (params.recoveryDestination === undefined || !this.isValidAddress(params.recoveryDestination)) {
682
+ throw new Error('invalid recoveryDestination');
683
+ }
684
+ if (!this.staticsCoin?.features.includes(statics_1.CoinFeature.EIP1559)) {
685
+ if (params.eip1559) {
686
+ throw new Error('Invalid fee params. EIP1559 not supported');
687
+ }
688
+ if (params.replayProtectionOptions?.hardfork === 'london') {
689
+ throw new Error('Invalid replayProtection options. Cannot use the hardfork "london" for this chain');
690
+ }
691
+ }
692
+ }
693
+ /**
694
+ * Helper which Adds signatures to tx object and re-serializes tx
695
+ * @param {EthLikeCommon.default} ethCommon
696
+ * @param {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction} tx
697
+ * @param {ECDSAMethodTypes.Signature} signature
698
+ * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}
699
+ */
700
+ getSignedTxFromSignature(ethCommon, tx, signature) {
701
+ // get signed Tx from signature
702
+ const txData = tx.toJSON();
703
+ const yParity = signature.recid;
704
+ const baseParams = {
705
+ to: txData.to,
706
+ nonce: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.nonce), 'hex'),
707
+ value: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.value), 'hex'),
708
+ gasLimit: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.gasLimit), 'hex'),
709
+ data: txData.data,
710
+ r: (0, ethereumjs_util_1.addHexPrefix)(signature.r),
711
+ s: (0, ethereumjs_util_1.addHexPrefix)(signature.s),
712
+ };
713
+ let finalTx;
714
+ if (txData.maxFeePerGas && txData.maxPriorityFeePerGas) {
715
+ finalTx = tx_1.FeeMarketEIP1559Transaction.fromTxData({
716
+ ...baseParams,
717
+ maxPriorityFeePerGas: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.maxPriorityFeePerGas), 'hex'),
718
+ maxFeePerGas: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.maxFeePerGas), 'hex'),
719
+ v: new bn_js_1.default(yParity.toString()),
720
+ }, { common: ethCommon });
721
+ }
722
+ else if (txData.gasPrice) {
723
+ const v = BigInt(35) + BigInt(yParity) + BigInt(ethCommon.chainIdBN().toNumber()) * BigInt(2);
724
+ finalTx = tx_1.Transaction.fromTxData({
725
+ ...baseParams,
726
+ v: new bn_js_1.default(v.toString()),
727
+ gasPrice: new bn_js_1.default((0, ethereumjs_util_1.stripHexPrefix)(txData.gasPrice.toString()), 'hex'),
728
+ }, { common: ethCommon });
729
+ }
730
+ return finalTx;
731
+ }
732
+ /**
733
+ * Builds a funds recovery transaction without BitGo
734
+ * @param params
735
+ * @param {string} params.userKey - [encrypted] xprv
736
+ * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
737
+ * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
738
+ * @param {string} params.walletContractAddress - the ETH address of the wallet contract
739
+ * @param {string} params.krsProvider - necessary if backup key is held by KRS
740
+ * @param {string} params.recoveryDestination - target address to send recovered funds to
741
+ * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn
742
+ * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn
743
+ */
744
+ async recover(params) {
745
+ if (params.isTss === true) {
746
+ return this.recoverTSS(params);
747
+ }
748
+ return this.recoverEthLike(params);
749
+ }
750
+ generateForwarderAddress(baseAddress, feeAddress, forwarderFactoryAddress, forwarderImplementationAddress, index) {
751
+ const salt = (0, ethereumjs_util_1.addHexPrefix)(index.toString(16));
752
+ const saltBuffer = (0, ethereumjs_util_1.setLengthLeft)((0, ethereumjs_util_1.toBuffer)(salt), 32);
753
+ const { createForwarderParams, createForwarderTypes } = (0, lib_1.getCreateForwarderParamsAndTypes)(baseAddress, saltBuffer, feeAddress);
754
+ const calculationSalt = (0, ethereumjs_util_1.bufferToHex)(exports.optionalDeps.ethAbi.soliditySHA3(createForwarderTypes, createForwarderParams));
755
+ const initCode = (0, lib_1.getProxyInitcode)(forwarderImplementationAddress);
756
+ return (0, lib_1.calculateForwarderV1Address)(forwarderFactoryAddress, calculationSalt, initCode);
757
+ }
758
+ deriveAddressFromPublicKey(commonKeychain, index) {
759
+ const derivationPath = `m/${index}`;
760
+ const pubkeySize = 33;
761
+ const ecdsaMpc = new sdk_core_1.Ecdsa();
762
+ const derivedPublicKey = Buffer.from(ecdsaMpc.deriveUnhardened(commonKeychain, derivationPath), 'hex')
763
+ .subarray(0, pubkeySize)
764
+ .toString('hex');
765
+ const publicKey = Buffer.from(derivedPublicKey, 'hex').slice(0, 66).toString('hex');
766
+ const keyPair = new lib_1.KeyPair({ pub: publicKey });
767
+ const address = keyPair.getAddress();
768
+ return address;
769
+ }
770
+ getConsolidationAddress(params, index) {
771
+ const possibleConsolidationAddresses = [];
772
+ if (params.walletContractAddress && params.bitgoFeeAddress) {
773
+ const ethNetwork = this.getNetwork();
774
+ const forwarderFactoryAddress = ethNetwork?.walletV4ForwarderFactoryAddress;
775
+ const forwarderImplementationAddress = ethNetwork?.walletV4ForwarderImplementationAddress;
776
+ try {
777
+ const forwarderAddress = this.generateForwarderAddress(params.walletContractAddress, params.bitgoFeeAddress, forwarderFactoryAddress, forwarderImplementationAddress, index);
778
+ possibleConsolidationAddresses.push(forwarderAddress);
779
+ }
780
+ catch (e) {
781
+ console.log(`Failed to generate forwarder address: ${e.message}`);
782
+ }
783
+ }
784
+ if (params.userKey) {
785
+ try {
786
+ const derivedAddress = this.deriveAddressFromPublicKey(params.userKey, index);
787
+ possibleConsolidationAddresses.push(derivedAddress);
788
+ }
789
+ catch (e) {
790
+ console.log(`Failed to generate derived address: ${e}`);
791
+ }
792
+ }
793
+ if (possibleConsolidationAddresses.length === 0) {
794
+ throw new Error('Unable to generate consolidation address. Check that wallet contract address, fee address, or user key is valid.');
795
+ }
796
+ return possibleConsolidationAddresses;
797
+ }
798
+ async recoverConsolidations(params) {
799
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
800
+ const startIdx = params.startingScanIndex || 1;
801
+ const endIdx = params.endingScanIndex || startIdx + exports.DEFAULT_SCAN_FACTOR;
802
+ if (!params.walletContractAddress || params.walletContractAddress === '') {
803
+ throw new Error(`Invalid wallet contract address ${params.walletContractAddress}`);
804
+ }
805
+ if (!params.bitgoFeeAddress || params.bitgoFeeAddress === '') {
806
+ throw new Error(`Invalid fee address ${params.bitgoFeeAddress}`);
807
+ }
808
+ if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * exports.DEFAULT_SCAN_FACTOR) {
809
+ throw new Error(`Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`);
810
+ }
811
+ const consolidatedTransactions = [];
812
+ let lastScanIndex = startIdx;
813
+ for (let i = startIdx; i < endIdx; i++) {
814
+ const consolidationAddress = this.getConsolidationAddress(params, i);
815
+ for (const address of consolidationAddress) {
816
+ const recoverParams = {
817
+ apiKey: params.apiKey,
818
+ backupKey: params.backupKey || '',
819
+ gasLimit: params.gasLimit,
820
+ recoveryDestination: params.recoveryDestination || '',
821
+ userKey: params.userKey || '',
822
+ walletContractAddress: address,
823
+ derivationSeed: '',
824
+ isTss: params.isTss,
825
+ eip1559: {
826
+ maxFeePerGas: params.eip1559?.maxFeePerGas || 20,
827
+ maxPriorityFeePerGas: params.eip1559?.maxPriorityFeePerGas || 200000,
828
+ },
829
+ replayProtectionOptions: {
830
+ chain: params.replayProtectionOptions?.chain || 0,
831
+ hardfork: params.replayProtectionOptions?.hardfork || 'london',
832
+ },
833
+ bitgoKey: '',
834
+ ignoreAddressTypes: [],
835
+ };
836
+ let recoveryTransaction;
837
+ try {
838
+ recoveryTransaction = await this.recover(recoverParams);
839
+ }
840
+ catch (e) {
841
+ if (e.message === 'Did not find address with funds to recover' ||
842
+ e.message === 'Did not find token account to recover tokens, please check token account' ||
843
+ e.message === 'Not enough token funds to recover') {
844
+ lastScanIndex = i;
845
+ continue;
846
+ }
847
+ throw e;
848
+ }
849
+ if (isUnsignedSweep) {
850
+ consolidatedTransactions.push(recoveryTransaction.txRequests[0]);
851
+ }
852
+ else {
853
+ consolidatedTransactions.push(recoveryTransaction);
854
+ }
855
+ }
856
+ // To avoid rate limit for etherscan
857
+ await new Promise((resolve) => setTimeout(resolve, 1000));
858
+ // lastScanIndex = i;
859
+ }
860
+ if (consolidatedTransactions.length === 0) {
861
+ throw new Error(`Did not find an address with sufficient funds to recover. Please start the next scan at address index ${lastScanIndex + 1}.`);
862
+ }
863
+ return { transactions: consolidatedTransactions, lastScanIndex };
864
+ }
865
+ /**
866
+ * Builds a funds recovery transaction without BitGo for non-TSS transaction
867
+ * @param params
868
+ * @param {string} params.userKey [encrypted] xprv or xpub
869
+ * @param {string} params.backupKey [encrypted] xprv or xpub if the xprv is held by a KRS provider
870
+ * @param {string} params.walletPassphrase used to decrypt userKey and backupKey
871
+ * @param {string} params.walletContractAddress the EthLike address of the wallet contract
872
+ * @param {string} params.krsProvider necessary if backup key is held by KRS
873
+ * @param {string} params.recoveryDestination target address to send recovered funds to
874
+ * @param {string} params.bitgoFeeAddress wrong chain wallet fee address for evm based cross chain recovery txn
875
+ * @param {string} params.bitgoDestinationAddress target bitgo address where fee will be sent for evm based cross chain recovery txn
876
+ * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}
877
+ */
878
+ async recoverEthLike(params) {
879
+ // bitgoFeeAddress is only defined when it is a evm cross chain recovery
880
+ // as we use fee from this wrong chain address for the recovery txn on the correct chain.
881
+ if (params.bitgoFeeAddress) {
882
+ return this.recoverEthLikeforEvmBasedRecovery(params);
883
+ }
884
+ this.validateRecoveryParams(params);
885
+ const isUnsignedSweep = params.isUnsignedSweep ?? (0, sdk_core_1.getIsUnsignedSweep)(params);
886
+ // Clean up whitespace from entered values
887
+ let userKey = params.userKey.replace(/\s/g, '');
888
+ const backupKey = params.backupKey.replace(/\s/g, '');
889
+ const gasLimit = new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
890
+ const gasPrice = params.eip1559
891
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
892
+ : new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
893
+ if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
894
+ try {
895
+ userKey = this.bitgo.decrypt({
896
+ input: userKey,
897
+ password: params.walletPassphrase,
898
+ });
899
+ }
900
+ catch (e) {
901
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
902
+ }
903
+ }
904
+ let backupKeyAddress;
905
+ let backupSigningKey;
906
+ if (isUnsignedSweep) {
907
+ const backupHDNode = secp256k1_1.bip32.fromBase58(backupKey);
908
+ backupSigningKey = backupHDNode.publicKey;
909
+ backupKeyAddress = `0x${exports.optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`;
910
+ }
911
+ else {
912
+ // Decrypt backup private key and get address
913
+ let backupPrv;
914
+ try {
915
+ backupPrv = this.bitgo.decrypt({
916
+ input: backupKey,
917
+ password: params.walletPassphrase,
918
+ });
919
+ }
920
+ catch (e) {
921
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
922
+ }
923
+ const keyPair = new lib_1.KeyPair({ prv: backupPrv });
924
+ backupSigningKey = keyPair.getKeys().prv;
925
+ if (!backupSigningKey) {
926
+ throw new Error('no private key');
927
+ }
928
+ backupKeyAddress = keyPair.getAddress();
929
+ }
930
+ const backupKeyNonce = await this.getAddressNonce(backupKeyAddress, params.apiKey);
931
+ // get balance of backupKey to ensure funds are available to pay fees
932
+ const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress, params.apiKey);
933
+ let totalGasNeeded = gasPrice.mul(gasLimit);
934
+ // On L2 chains with L1 data fees, add buffer for L1 fees
935
+ if (this.staticsCoin?.family !== undefined && lib_1.coinFamiliesWithL1Fees.includes(this.staticsCoin.family)) {
936
+ totalGasNeeded = totalGasNeeded.add(new exports.optionalDeps.ethUtil.BN(statics_1.ethGasConfigs.l1GasFeeBuffer));
937
+ }
938
+ const weiToGwei = 10 ** 9;
939
+ if (backupKeyBalance.lt(totalGasNeeded)) {
940
+ throw new Error(`Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +
941
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
942
+ ` Gwei to perform recoveries. Try sending some funds to this address then retry.`);
943
+ }
944
+ // get balance of wallet
945
+ const txAmount = await this.queryAddressBalance(params.walletContractAddress, params.apiKey);
946
+ if (new bignumber_js_1.BigNumber(txAmount).isLessThanOrEqualTo(0)) {
947
+ throw new Error('Wallet does not have enough funds to recover');
948
+ }
949
+ // build recipients object
950
+ const recipients = [
951
+ {
952
+ address: params.recoveryDestination,
953
+ amount: txAmount.toString(10),
954
+ },
955
+ ];
956
+ // Get sequence ID using contract call
957
+ // we need to wait between making two explorer api calls to avoid getting banned
958
+ await new Promise((resolve) => setTimeout(resolve, 1000));
959
+ const sequenceId = await this.querySequenceId(params.walletContractAddress, params.apiKey);
960
+ let operationHash, signature;
961
+ // Get operation hash and sign it
962
+ if (!isUnsignedSweep) {
963
+ operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);
964
+ signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userKey));
965
+ try {
966
+ sdk_core_1.Util.ecRecoverEthAddress(operationHash, signature);
967
+ }
968
+ catch (e) {
969
+ throw new Error('Invalid signature');
970
+ }
971
+ }
972
+ const txInfo = {
973
+ recipient: recipients[0],
974
+ expireTime: this.getDefaultExpireTime(),
975
+ contractSequenceId: sequenceId,
976
+ operationHash: operationHash,
977
+ signature: signature,
978
+ gasLimit: gasLimit.toString(10),
979
+ };
980
+ const txBuilder = this.getTransactionBuilder(params.common);
981
+ txBuilder.counter(backupKeyNonce);
982
+ txBuilder.contract(params.walletContractAddress);
983
+ let txFee;
984
+ if (params.eip1559) {
985
+ txFee = {
986
+ eip1559: {
987
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
988
+ maxFeePerGas: params.eip1559.maxFeePerGas,
989
+ },
990
+ };
991
+ }
992
+ else {
993
+ txFee = { fee: gasPrice.toString() };
994
+ }
995
+ txBuilder.fee({
996
+ ...txFee,
997
+ gasLimit: gasLimit.toString(),
998
+ });
999
+ const transferBuilder = txBuilder.transfer();
1000
+ transferBuilder
1001
+ .coin(this.staticsCoin?.name)
1002
+ .amount(recipients[0].amount)
1003
+ .contractSequenceId(sequenceId)
1004
+ .expirationTime(this.getDefaultExpireTime())
1005
+ .to(params.recoveryDestination);
1006
+ const tx = await txBuilder.build();
1007
+ if (isUnsignedSweep) {
1008
+ const response = {
1009
+ txHex: tx.toBroadcastFormat(),
1010
+ userKey,
1011
+ backupKey,
1012
+ coin: this.getChain(),
1013
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1014
+ gasLimit,
1015
+ recipients: [txInfo.recipient],
1016
+ walletContractAddress: tx.toJson().to,
1017
+ amount: txInfo.recipient.amount,
1018
+ backupKeyNonce,
1019
+ eip1559: params.eip1559,
1020
+ };
1021
+ lodash_1.default.extend(response, txInfo);
1022
+ response.nextContractSequenceId = response.contractSequenceId;
1023
+ return response;
1024
+ }
1025
+ txBuilder
1026
+ .transfer()
1027
+ .coin(this.staticsCoin?.name)
1028
+ .key(new lib_1.KeyPair({ prv: userKey }).getKeys().prv);
1029
+ txBuilder.sign({ key: backupSigningKey });
1030
+ const signedTx = await txBuilder.build();
1031
+ return {
1032
+ id: signedTx.toJson().id,
1033
+ tx: signedTx.toBroadcastFormat(),
1034
+ };
1035
+ }
1036
+ /**
1037
+ * Extract recipients from transaction hex
1038
+ * @param txHex - The transaction hex string
1039
+ * @returns Array of recipients with address and amount
1040
+ */
1041
+ async extractRecipientsFromTxHex(txHex) {
1042
+ const txBuffer = exports.optionalDeps.ethUtil.toBuffer(txHex);
1043
+ const decodedTx = exports.optionalDeps.EthTx.TransactionFactory.fromSerializedData(txBuffer);
1044
+ const recipients = [];
1045
+ if (decodedTx.data && decodedTx.data.length > 0) {
1046
+ const dataHex = exports.optionalDeps.ethUtil.bufferToHex(decodedTx.data);
1047
+ const transferData = (0, lib_1.decodeTransferData)(dataHex);
1048
+ if (transferData.to) {
1049
+ recipients.push({
1050
+ address: transferData.to,
1051
+ amount: transferData.amount,
1052
+ });
1053
+ }
1054
+ }
1055
+ return recipients;
1056
+ }
1057
+ async sendCrossChainRecoveryTransaction(params) {
1058
+ const buildResponse = await this.buildCrossChainRecoveryTransaction(params.recoveryId);
1059
+ if (params.walletType === 'cold') {
1060
+ return buildResponse;
1061
+ }
1062
+ if (!params.encryptedPrv) {
1063
+ throw new Error('missing encryptedPrv');
1064
+ }
1065
+ let userKeyPrv;
1066
+ try {
1067
+ userKeyPrv = this.bitgo.decrypt({
1068
+ input: params.encryptedPrv,
1069
+ password: params.walletPassphrase,
1070
+ });
1071
+ }
1072
+ catch (e) {
1073
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
1074
+ }
1075
+ const keyPair = new lib_1.KeyPair({ prv: userKeyPrv });
1076
+ const userSigningKey = keyPair.getKeys().prv;
1077
+ if (!userSigningKey) {
1078
+ throw new Error('no private key');
1079
+ }
1080
+ const txBuilder = this.getTransactionBuilder(params.common);
1081
+ const txHex = buildResponse.txHex;
1082
+ txBuilder.from(txHex);
1083
+ if (buildResponse.walletVersion) {
1084
+ // If walletVersion is provided, set it in the txBuilder
1085
+ txBuilder.walletVersion(buildResponse.walletVersion);
1086
+ }
1087
+ txBuilder
1088
+ .transfer()
1089
+ .coin(this.staticsCoin?.name)
1090
+ .key(userSigningKey);
1091
+ const tx = await txBuilder.build();
1092
+ const res = await this.bitgo
1093
+ .post(this.bitgo.microservicesUrl(`/api/recovery/v1/crosschain/${params.recoveryId}/sign`))
1094
+ .send({ txHex: tx.toBroadcastFormat() });
1095
+ return {
1096
+ coin: this.staticsCoin?.name,
1097
+ txid: res.body.txid,
1098
+ };
1099
+ }
1100
+ async buildCrossChainRecoveryTransaction(recoveryId) {
1101
+ const res = await this.bitgo.get(this.bitgo.microservicesUrl(`/api/recovery/v1/crosschain/${recoveryId}/buildtx`));
1102
+ // Extract recipients from the transaction hex
1103
+ const recipients = await this.extractRecipientsFromTxHex(res.body.txHex);
1104
+ return {
1105
+ coin: res.body.coin,
1106
+ txHex: res.body.txHex,
1107
+ txid: res.body.txid,
1108
+ walletVersion: res.body.walletVersion,
1109
+ recipients,
1110
+ };
1111
+ }
1112
+ /**
1113
+ * Builds a unsigned (for cold, custody wallet) or
1114
+ * half-signed (for hot wallet) evm cross chain recovery transaction with
1115
+ * same expected arguments as recover method.
1116
+ * This helps recover funds from evm based wrong chain.
1117
+ * @param {RecoverOptions} params
1118
+ * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}
1119
+ */
1120
+ async recoverEthLikeforEvmBasedRecovery(params) {
1121
+ this.validateEvmBasedRecoveryParams(params);
1122
+ // Clean up whitespace from entered values
1123
+ const userKey = params.userKey.replace(/\s/g, '');
1124
+ const bitgoFeeAddress = params.bitgoFeeAddress?.replace(/\s/g, '').toLowerCase();
1125
+ const bitgoDestinationAddress = params.bitgoDestinationAddress?.replace(/\s/g, '').toLowerCase();
1126
+ const recoveryDestination = params.recoveryDestination?.replace(/\s/g, '').toLowerCase();
1127
+ const walletContractAddress = params.walletContractAddress?.replace(/\s/g, '').toLowerCase();
1128
+ const tokenContractAddress = params.tokenContractAddress?.replace(/\s/g, '').toLowerCase();
1129
+ let userSigningKey;
1130
+ let userKeyPrv;
1131
+ if (params.walletPassphrase) {
1132
+ if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
1133
+ try {
1134
+ userKeyPrv = this.bitgo.decrypt({
1135
+ input: userKey,
1136
+ password: params.walletPassphrase,
1137
+ });
1138
+ }
1139
+ catch (e) {
1140
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
1141
+ }
1142
+ }
1143
+ const keyPair = new lib_1.KeyPair({ prv: userKeyPrv });
1144
+ userSigningKey = keyPair.getKeys().prv;
1145
+ if (!userSigningKey) {
1146
+ throw new Error('no private key');
1147
+ }
1148
+ }
1149
+ // Use default gasLimit for cold and custody wallets
1150
+ let gasLimit = params.gasLimit || userKey.startsWith('xpub') || !userKey
1151
+ ? new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit))
1152
+ : new exports.optionalDeps.ethUtil.BN(0);
1153
+ const gasPrice = params.eip1559
1154
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
1155
+ : params.gasPrice
1156
+ ? new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice))
1157
+ : await this.getGasPriceFromExternalAPI(this.staticsCoin?.name, params.apiKey);
1158
+ const bitgoFeeAddressNonce = await this.getAddressNonce(bitgoFeeAddress, params.apiKey);
1159
+ if (tokenContractAddress) {
1160
+ return this.recoverEthLikeTokenforEvmBasedRecovery(params, bitgoFeeAddressNonce, gasLimit, gasPrice, userKey, userSigningKey, params.apiKey);
1161
+ }
1162
+ // get balance of wallet
1163
+ const txAmount = await this.queryAddressBalance(walletContractAddress, params.apiKey);
1164
+ const bitgoFeePercentage = 0; // TODO: BG-71912 can change the fee% here.
1165
+ const bitgoFeeAmount = txAmount * (bitgoFeePercentage / 100);
1166
+ // build recipients object
1167
+ const recipients = [
1168
+ {
1169
+ address: recoveryDestination,
1170
+ amount: new bignumber_js_1.BigNumber(txAmount).minus(bitgoFeeAmount).toFixed(),
1171
+ },
1172
+ ];
1173
+ if (bitgoFeePercentage > 0) {
1174
+ if (lodash_1.default.isUndefined(bitgoDestinationAddress) || !this.isValidAddress(bitgoDestinationAddress)) {
1175
+ throw new Error('invalid bitgoDestinationAddress');
1176
+ }
1177
+ recipients.push({
1178
+ address: bitgoDestinationAddress,
1179
+ amount: bitgoFeeAmount.toString(10),
1180
+ });
1181
+ }
1182
+ // calculate batch data
1183
+ const BATCH_METHOD_NAME = 'batch';
1184
+ const BATCH_METHOD_TYPES = ['address[]', 'uint256[]'];
1185
+ const batchExecutionInfo = this.getBatchExecutionInfo(recipients);
1186
+ const batchData = exports.optionalDeps.ethUtil.addHexPrefix(this.getMethodCallData(BATCH_METHOD_NAME, BATCH_METHOD_TYPES, batchExecutionInfo.values).toString('hex'));
1187
+ // Get sequence ID using contract call
1188
+ // we need to wait between making two explorer api calls to avoid getting banned
1189
+ await new Promise((resolve) => setTimeout(resolve, 1000));
1190
+ const sequenceId = await this.querySequenceId(walletContractAddress, params.apiKey);
1191
+ const network = this.getNetwork();
1192
+ const batcherContractAddress = network?.batcherContractAddress;
1193
+ const txBuilder = this.getTransactionBuilder(params.common);
1194
+ txBuilder.counter(bitgoFeeAddressNonce);
1195
+ txBuilder.contract(walletContractAddress);
1196
+ let txFee;
1197
+ if (params.eip1559) {
1198
+ txFee = {
1199
+ eip1559: {
1200
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
1201
+ maxFeePerGas: params.eip1559.maxFeePerGas,
1202
+ },
1203
+ };
1204
+ }
1205
+ else {
1206
+ txFee = { fee: gasPrice.toString() };
1207
+ }
1208
+ txBuilder.fee({
1209
+ ...txFee,
1210
+ gasLimit: gasLimit.toString(),
1211
+ });
1212
+ const transferBuilder = txBuilder.transfer();
1213
+ if (!batcherContractAddress) {
1214
+ transferBuilder
1215
+ .coin(this.staticsCoin?.name)
1216
+ .amount(batchExecutionInfo.totalAmount)
1217
+ .contractSequenceId(sequenceId)
1218
+ .expirationTime(this.getDefaultExpireTime())
1219
+ .to(recoveryDestination);
1220
+ }
1221
+ else {
1222
+ transferBuilder
1223
+ .coin(this.staticsCoin?.name)
1224
+ .amount(batchExecutionInfo.totalAmount)
1225
+ .contractSequenceId(sequenceId)
1226
+ .expirationTime(this.getDefaultExpireTime())
1227
+ .to(batcherContractAddress)
1228
+ .data(batchData);
1229
+ }
1230
+ if (params.walletPassphrase) {
1231
+ transferBuilder.key(userSigningKey);
1232
+ }
1233
+ // If the intended chain is arbitrum or optimism, we need to use wallet version 4
1234
+ // since these contracts construct operationHash differently
1235
+ if (params.intendedChain && ['arbeth', 'opeth'].includes(statics_1.coins.get(params.intendedChain).family)) {
1236
+ txBuilder.walletVersion(4);
1237
+ }
1238
+ // If gasLimit was not passed as a param or if it is not cold/custody wallet, then fetch the gasLimit from Explorer
1239
+ if (!params.gasLimit && userKey && !userKey.startsWith('xpub')) {
1240
+ const sendData = txBuilder.getSendData();
1241
+ gasLimit = await this.getGasLimitFromExternalAPI(params.intendedChain, params.bitgoFeeAddress, params.walletContractAddress, sendData, params.apiKey);
1242
+ txBuilder.fee({
1243
+ ...txFee,
1244
+ gasLimit: gasLimit.toString(),
1245
+ });
1246
+ }
1247
+ // Get the balance of bitgoFeeAddress to ensure funds are available to pay fees
1248
+ await this.ensureSufficientBalance(bitgoFeeAddress, gasPrice, gasLimit, params.apiKey);
1249
+ const tx = await txBuilder.build();
1250
+ const txInfo = {
1251
+ recipients: recipients,
1252
+ expireTime: this.getDefaultExpireTime(),
1253
+ contractSequenceId: sequenceId,
1254
+ gasLimit: gasLimit.toString(10),
1255
+ isEvmBasedCrossChainRecovery: true,
1256
+ };
1257
+ const response = {
1258
+ txHex: tx.toBroadcastFormat(),
1259
+ userKey,
1260
+ coin: this.getChain(),
1261
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1262
+ gasLimit,
1263
+ recipients: txInfo.recipients,
1264
+ walletContractAddress: tx.toJson().to,
1265
+ amount: batchExecutionInfo.totalAmount,
1266
+ backupKeyNonce: bitgoFeeAddressNonce,
1267
+ eip1559: params.eip1559,
1268
+ ...(txBuilder.getWalletVersion() === 4 ? { walletVersion: txBuilder.getWalletVersion() } : {}),
1269
+ };
1270
+ lodash_1.default.extend(response, txInfo);
1271
+ response.nextContractSequenceId = response.contractSequenceId;
1272
+ if (params.walletPassphrase) {
1273
+ const halfSignedTxn = {
1274
+ halfSigned: {
1275
+ txHex: tx.toBroadcastFormat(),
1276
+ recipients: txInfo.recipients,
1277
+ expireTime: txInfo.expireTime,
1278
+ },
1279
+ };
1280
+ lodash_1.default.extend(response, halfSignedTxn);
1281
+ const feesUsed = {
1282
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1283
+ gasLimit: exports.optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),
1284
+ };
1285
+ response['feesUsed'] = feesUsed;
1286
+ }
1287
+ return response;
1288
+ }
1289
+ /**
1290
+ * Query explorer for the balance of an address for a token
1291
+ * @param {string} tokenContractAddress - address where the token smart contract is hosted
1292
+ * @param {string} walletContractAddress - address of the wallet
1293
+ * @param {string} apiKey - optional API key to use instead of the one from the environment
1294
+ * @returns {BigNumber} token balaance in base units
1295
+ */
1296
+ async queryAddressTokenBalance(tokenContractAddress, walletContractAddress, apiKey) {
1297
+ if (!exports.optionalDeps.ethUtil.isValidAddress(tokenContractAddress)) {
1298
+ throw new Error('cannot get balance for invalid token address');
1299
+ }
1300
+ if (!exports.optionalDeps.ethUtil.isValidAddress(walletContractAddress)) {
1301
+ throw new Error('cannot get token balance for invalid wallet address');
1302
+ }
1303
+ const result = await this.recoveryBlockchainExplorerQuery({
1304
+ chainid: this.getChainId().toString(),
1305
+ module: 'account',
1306
+ action: 'tokenbalance',
1307
+ contractaddress: tokenContractAddress,
1308
+ address: walletContractAddress,
1309
+ tag: 'latest',
1310
+ }, apiKey);
1311
+ // throw if the result does not exist or the result is not a valid number
1312
+ if (!result || !result.result || isNaN(result.result)) {
1313
+ throw new Error(`Could not obtain token address balance for ${tokenContractAddress} from Etherscan, got: ${result.result}`);
1314
+ }
1315
+ return new exports.optionalDeps.ethUtil.BN(result.result, 10);
1316
+ }
1317
+ async recoverEthLikeTokenforEvmBasedRecovery(params, bitgoFeeAddressNonce, gasLimit, gasPrice, userKey, userSigningKey, apiKey) {
1318
+ // get token balance of wallet
1319
+ const txAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, params.walletContractAddress, apiKey);
1320
+ // build recipients object
1321
+ const recipients = [
1322
+ {
1323
+ address: params.recoveryDestination,
1324
+ amount: new bignumber_js_1.BigNumber(txAmount).toFixed(),
1325
+ },
1326
+ ];
1327
+ // Get sequence ID using contract call
1328
+ // we need to wait between making two explorer api calls to avoid getting banned
1329
+ await new Promise((resolve) => setTimeout(resolve, 1000));
1330
+ const sequenceId = await this.querySequenceId(params.walletContractAddress, params.apiKey);
1331
+ const txBuilder = this.getTransactionBuilder(params.common);
1332
+ txBuilder.counter(bitgoFeeAddressNonce);
1333
+ txBuilder.contract(params.walletContractAddress);
1334
+ let txFee;
1335
+ if (params.eip1559) {
1336
+ txFee = {
1337
+ eip1559: {
1338
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
1339
+ maxFeePerGas: params.eip1559.maxFeePerGas,
1340
+ },
1341
+ };
1342
+ }
1343
+ else {
1344
+ txFee = { fee: gasPrice.toString() };
1345
+ }
1346
+ txBuilder.fee({
1347
+ ...txFee,
1348
+ gasLimit: gasLimit.toString(),
1349
+ });
1350
+ const transferBuilder = txBuilder.transfer();
1351
+ const network = this.getNetwork();
1352
+ const token = (0, lib_1.getToken)(params.tokenContractAddress, network, this.staticsCoin?.family)?.name;
1353
+ transferBuilder
1354
+ .amount(txAmount)
1355
+ .contractSequenceId(sequenceId)
1356
+ .expirationTime(this.getDefaultExpireTime())
1357
+ .to(params.recoveryDestination);
1358
+ if (token) {
1359
+ transferBuilder.coin(token);
1360
+ }
1361
+ else {
1362
+ transferBuilder
1363
+ .coin(this.staticsCoin?.name)
1364
+ .tokenContractAddress(params.tokenContractAddress);
1365
+ }
1366
+ if (params.walletPassphrase) {
1367
+ txBuilder.transfer().key(userSigningKey);
1368
+ }
1369
+ // If the intended chain is arbitrum or optimism, we need to use wallet version 4
1370
+ // since these contracts construct operationHash differently
1371
+ if (params.intendedChain && ['arbeth', 'opeth'].includes(statics_1.coins.get(params.intendedChain).family)) {
1372
+ txBuilder.walletVersion(4);
1373
+ }
1374
+ if (!params.gasLimit && userKey && !userKey.startsWith('xpub')) {
1375
+ const sendData = txBuilder.getSendData();
1376
+ gasLimit = await this.getGasLimitFromExternalAPI(params.intendedChain, params.bitgoFeeAddress, params.walletContractAddress, sendData, apiKey);
1377
+ txBuilder.fee({
1378
+ ...txFee,
1379
+ gasLimit: gasLimit.toString(),
1380
+ });
1381
+ }
1382
+ // Get the balance of bitgoFeeAddress to ensure funds are available to pay fees
1383
+ await this.ensureSufficientBalance(params.bitgoFeeAddress, gasPrice, gasLimit, params.apiKey);
1384
+ const tx = await txBuilder.build();
1385
+ const txInfo = {
1386
+ recipients: recipients,
1387
+ expireTime: this.getDefaultExpireTime(),
1388
+ contractSequenceId: sequenceId,
1389
+ gasLimit: gasLimit.toString(10),
1390
+ isEvmBasedCrossChainRecovery: true,
1391
+ };
1392
+ const response = {
1393
+ txHex: tx.toBroadcastFormat(),
1394
+ userKey,
1395
+ coin: token ? token : this.getChain(),
1396
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1397
+ gasLimit,
1398
+ recipients: txInfo.recipients,
1399
+ walletContractAddress: tx.toJson().to,
1400
+ amount: txAmount.toString(),
1401
+ backupKeyNonce: bitgoFeeAddressNonce,
1402
+ eip1559: params.eip1559,
1403
+ ...(txBuilder.getWalletVersion() === 4 ? { walletVersion: txBuilder.getWalletVersion() } : {}),
1404
+ };
1405
+ lodash_1.default.extend(response, txInfo);
1406
+ response.nextContractSequenceId = response.contractSequenceId;
1407
+ if (params.walletPassphrase) {
1408
+ const halfSignedTxn = {
1409
+ halfSigned: {
1410
+ txHex: tx.toBroadcastFormat(),
1411
+ recipients: txInfo.recipients,
1412
+ expireTime: txInfo.expireTime,
1413
+ },
1414
+ };
1415
+ lodash_1.default.extend(response, halfSignedTxn);
1416
+ const feesUsed = {
1417
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1418
+ gasLimit: exports.optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),
1419
+ };
1420
+ response['feesUsed'] = feesUsed;
1421
+ }
1422
+ return response;
1423
+ }
1424
+ /**
1425
+ * Validate evm based cross chain recovery params
1426
+ * @param params {RecoverOptions}
1427
+ * @returns {void}
1428
+ */
1429
+ validateEvmBasedRecoveryParams(params) {
1430
+ if (lodash_1.default.isUndefined(params.bitgoFeeAddress) || !this.isValidAddress(params.bitgoFeeAddress)) {
1431
+ throw new Error('invalid bitgoFeeAddress');
1432
+ }
1433
+ if (lodash_1.default.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
1434
+ throw new Error('invalid walletContractAddress');
1435
+ }
1436
+ if (lodash_1.default.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
1437
+ throw new Error('invalid recoveryDestination');
1438
+ }
1439
+ }
1440
+ /**
1441
+ * Return types, values, and total amount in wei to send in a batch transaction, using the method signature
1442
+ * `distributeBatch(address[], uint256[])`
1443
+ * @param {Recipient[]} recipients - transaction recipients
1444
+ * @returns {GetBatchExecutionInfoRT} information needed to execute the batch transaction
1445
+ */
1446
+ getBatchExecutionInfo(recipients) {
1447
+ const addresses = [];
1448
+ const amounts = [];
1449
+ let sum = new bignumber_js_1.BigNumber('0');
1450
+ lodash_1.default.forEach(recipients, ({ address, amount }) => {
1451
+ addresses.push(address);
1452
+ amounts.push(amount);
1453
+ sum = sum.plus(amount);
1454
+ });
1455
+ return {
1456
+ values: [addresses, amounts],
1457
+ totalAmount: sum.toFixed(),
1458
+ };
1459
+ }
1460
+ /**
1461
+ * Build arguments to call the send method on the wallet contract
1462
+ * @param txInfo
1463
+ */
1464
+ getSendMethodArgs(txInfo) {
1465
+ // Method signature is
1466
+ // sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)
1467
+ return [
1468
+ {
1469
+ name: 'toAddress',
1470
+ type: 'address',
1471
+ value: txInfo.recipient.address,
1472
+ },
1473
+ {
1474
+ name: 'value',
1475
+ type: 'uint',
1476
+ value: txInfo.recipient.amount,
1477
+ },
1478
+ {
1479
+ name: 'data',
1480
+ type: 'bytes',
1481
+ value: exports.optionalDeps.ethUtil.toBuffer(exports.optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),
1482
+ },
1483
+ {
1484
+ name: 'expireTime',
1485
+ type: 'uint',
1486
+ value: txInfo.expireTime,
1487
+ },
1488
+ {
1489
+ name: 'sequenceId',
1490
+ type: 'uint',
1491
+ value: txInfo.contractSequenceId,
1492
+ },
1493
+ {
1494
+ name: 'signature',
1495
+ type: 'bytes',
1496
+ value: exports.optionalDeps.ethUtil.toBuffer(exports.optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),
1497
+ },
1498
+ ];
1499
+ }
1500
+ /**
1501
+ * Recovers a tx with TSS key shares
1502
+ * same expected arguments as recover method, but with TSS key shares
1503
+ */
1504
+ async recoverTSS(params) {
1505
+ this.validateRecoveryParams(params);
1506
+ // Clean up whitespace from entered values
1507
+ const userPublicOrPrivateKeyShare = params.userKey.replace(/\s/g, '');
1508
+ const backupPrivateOrPublicKeyShare = params.backupKey.replace(/\s/g, '');
1509
+ if ((0, sdk_core_1.getIsUnsignedSweep)({
1510
+ userKey: userPublicOrPrivateKeyShare,
1511
+ backupKey: backupPrivateOrPublicKeyShare,
1512
+ isTss: params.isTss,
1513
+ })) {
1514
+ return this.buildUnsignedSweepTxnTSS(params);
1515
+ }
1516
+ else {
1517
+ const { userKeyShare, backupKeyShare, commonKeyChain } = await sdk_core_1.ECDSAUtils.getMpcV2RecoveryKeyShares(userPublicOrPrivateKeyShare, backupPrivateOrPublicKeyShare, params.walletPassphrase);
1518
+ const { gasLimit, gasPrice } = await this.getGasValues(params);
1519
+ const MPC = new sdk_core_1.Ecdsa();
1520
+ const derivedCommonKeyChain = MPC.deriveUnhardened(commonKeyChain, 'm/0');
1521
+ const backupKeyPair = new lib_1.KeyPair({ pub: derivedCommonKeyChain.slice(0, 66) });
1522
+ const baseAddress = backupKeyPair.getAddress();
1523
+ const unsignedTx = (await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params)).tx;
1524
+ const messageHash = unsignedTx.getMessageToSign(true);
1525
+ const signature = await sdk_core_1.ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);
1526
+ const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);
1527
+ const signedTx = this.getSignedTxFromSignature(ethCommmon, unsignedTx, signature);
1528
+ return {
1529
+ id: (0, ethereumjs_util_1.addHexPrefix)(signedTx.hash().toString('hex')),
1530
+ tx: (0, ethereumjs_util_1.addHexPrefix)(signedTx.serialize().toString('hex')),
1531
+ };
1532
+ }
1533
+ }
1534
+ async getGasValues(params) {
1535
+ const gasLimit = new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
1536
+ const gasPrice = params.eip1559
1537
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
1538
+ : new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
1539
+ return { gasLimit, gasPrice };
1540
+ }
1541
+ async buildUnsignedSweepTxnTSS(params) {
1542
+ const userPublicOrPrivateKeyShare = params.userKey.replace(/\s/g, '');
1543
+ const backupPrivateOrPublicKeyShare = params.backupKey.replace(/\s/g, '');
1544
+ const { gasLimit, gasPrice } = await this.getGasValues(params);
1545
+ const backupKeyPair = new lib_1.KeyPair({ pub: backupPrivateOrPublicKeyShare });
1546
+ const baseAddress = backupKeyPair.getAddress();
1547
+ const { txInfo, tx, nonce } = await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params);
1548
+ return this.formatForOfflineVaultTSS(txInfo, tx, userPublicOrPrivateKeyShare, backupPrivateOrPublicKeyShare, gasPrice, gasLimit, nonce, params.eip1559, params.replayProtectionOptions);
1549
+ }
1550
+ async buildUnsignedSweepTxnMPCv2(params) {
1551
+ const { gasLimit, gasPrice } = await this.getGasValues(params);
1552
+ const recoverParams = params;
1553
+ this.validateUnsignedSweepTSSParams(recoverParams);
1554
+ const derivationPath = recoverParams.derivationSeed ? (0, sdk_lib_mpc_1.getDerivationPath)(recoverParams.derivationSeed) : 'm/0';
1555
+ const MPC = new sdk_core_1.Ecdsa();
1556
+ const derivedCommonKeyChain = MPC.deriveUnhardened(recoverParams.backupKey, derivationPath);
1557
+ const backupKeyPair = new lib_1.KeyPair({ pub: derivedCommonKeyChain.slice(0, 66) });
1558
+ const baseAddress = backupKeyPair.getAddress();
1559
+ const { txInfo, tx, nonce } = await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params);
1560
+ return this.buildTxRequestForOfflineVaultMPCv2(txInfo, tx, derivationPath, nonce, gasPrice, gasLimit, params.eip1559, params.replayProtectionOptions, recoverParams.backupKey);
1561
+ }
1562
+ async createBroadcastableSweepTransaction(params) {
1563
+ const req = params.signatureShares;
1564
+ const broadcastableTransactions = [];
1565
+ let lastScanIndex = 0;
1566
+ for (let i = 0; i < req.length; i++) {
1567
+ const transaction = req[i]?.txRequest?.transactions?.[0]?.unsignedTx;
1568
+ if (!req[i].ovc || !req[i].ovc[0].ecdsaSignature) {
1569
+ throw new Error('Missing signature(s)');
1570
+ }
1571
+ if (!transaction.signableHex) {
1572
+ throw new Error('Missing signable hex');
1573
+ }
1574
+ const signature = req[i].ovc[0].ecdsaSignature;
1575
+ if (!signature) {
1576
+ throw new Error('Signature is undefined');
1577
+ }
1578
+ const shares = signature.toString().split(':');
1579
+ if (shares.length !== 4) {
1580
+ throw new Error('Invalid signature');
1581
+ }
1582
+ const finalSignature = {
1583
+ recid: Number(shares[0]),
1584
+ r: shares[1],
1585
+ s: shares[2],
1586
+ y: shares[3],
1587
+ };
1588
+ if (!transaction.coinSpecific?.commonKeyChain) {
1589
+ throw new Error(`Missing common keychain for transaction at index ${i}`);
1590
+ }
1591
+ const commonKeyChain = transaction.coinSpecific.commonKeyChain;
1592
+ if (!transaction.derivationPath) {
1593
+ throw new Error(`Missing derivation path for transaction at index ${i}`);
1594
+ }
1595
+ if (!commonKeyChain) {
1596
+ throw new Error(`Missing common key chain for transaction at index ${i}`);
1597
+ }
1598
+ const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(transaction.eip1559, transaction.replayProtectionOptions);
1599
+ let unsignedTx;
1600
+ if (transaction.eip1559) {
1601
+ unsignedTx = tx_1.FeeMarketEIP1559Transaction.fromSerializedTx(Buffer.from(transaction.serializedTxHex, 'hex'));
1602
+ }
1603
+ else {
1604
+ unsignedTx = tx_1.Transaction.fromSerializedTx(Buffer.from(transaction.serializedTxHex, 'hex'));
1605
+ }
1606
+ const signedTx = this.getSignedTxFromSignature(ethCommmon, unsignedTx, finalSignature);
1607
+ broadcastableTransactions.push({
1608
+ serializedTx: (0, ethereumjs_util_1.addHexPrefix)(signedTx.serialize().toString('hex')),
1609
+ });
1610
+ if (i === req.length - 1 && transaction.coinSpecific?.lastScanIndex) {
1611
+ lastScanIndex = transaction.coinSpecific?.lastScanIndex;
1612
+ }
1613
+ }
1614
+ return { transactions: broadcastableTransactions, lastScanIndex };
1615
+ }
1616
+ /**
1617
+ * Method to validate recovery params
1618
+ * @param {RecoverOptions} params
1619
+ * @returns {void}
1620
+ */
1621
+ async validateUnsignedSweepTSSParams(params) {
1622
+ if (lodash_1.default.isUndefined(params.backupKey) && params.backupKey === '') {
1623
+ throw new Error('missing commonKeyChain');
1624
+ }
1625
+ if (!lodash_1.default.isUndefined(params.derivationSeed) && typeof params.derivationSeed !== 'string') {
1626
+ throw new Error('invalid derivationSeed');
1627
+ }
1628
+ if (lodash_1.default.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
1629
+ throw new Error('missing or invalid destinationAddress');
1630
+ }
1631
+ }
1632
+ /**
1633
+ * Helper function for recover()
1634
+ * This transforms the unsigned transaction information into a format the BitGo offline vault expects
1635
+ * @param {UnformattedTxInfo} txInfo - tx info
1636
+ * @param {LegacyTransaction | FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object
1637
+ * @param {string} derivationPath - the derivationPath
1638
+ * @param {number} nonce - the nonce of the backup key address
1639
+ * @param {Buffer} gasPrice - gas price for the tx
1640
+ * @param {number} gasLimit - gas limit for the tx
1641
+ * @param {EIP1559} eip1559 - eip1559 params
1642
+ * @param replayProtectionOptions
1643
+ * @param commonKeyChain
1644
+ * @returns {Promise<OfflineVaultTxInfo>}
1645
+ */
1646
+ buildTxRequestForOfflineVaultMPCv2(txInfo, ethTx, derivationPath, nonce, gasPrice, gasLimit, eip1559, replayProtectionOptions, commonKeyChain) {
1647
+ if (!ethTx.to) {
1648
+ throw new Error('Eth tx must have a `to` address');
1649
+ }
1650
+ const fee = eip1559
1651
+ ? gasLimit * eip1559.maxFeePerGas
1652
+ : gasLimit * exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed();
1653
+ const unsignedTx = {
1654
+ serializedTxHex: ethTx.serialize().toString('hex'),
1655
+ signableHex: ethTx instanceof tx_1.FeeMarketEIP1559Transaction
1656
+ ? ethTx.getMessageToSign(false).toString('hex')
1657
+ : Buffer.from(rlp_1.RLP.encode((0, ethereumjs_util_1.bufArrToArr)(ethTx.getMessageToSign(false)))).toString('hex'),
1658
+ derivationPath: derivationPath,
1659
+ feeInfo: {
1660
+ fee: fee,
1661
+ feeString: fee.toString(),
1662
+ },
1663
+ parsedTx: {
1664
+ spendAmount: txInfo.recipient.amount,
1665
+ outputs: [
1666
+ {
1667
+ coinName: this.getChain(),
1668
+ address: txInfo.recipient.address,
1669
+ valueString: txInfo.recipient.amount,
1670
+ },
1671
+ ],
1672
+ },
1673
+ coinSpecific: {
1674
+ commonKeyChain: commonKeyChain,
1675
+ },
1676
+ eip1559: eip1559,
1677
+ replayProtectionOptions: replayProtectionOptions,
1678
+ };
1679
+ return {
1680
+ txRequests: [
1681
+ {
1682
+ walletCoin: this.getChain(),
1683
+ transactions: [
1684
+ {
1685
+ unsignedTx: unsignedTx,
1686
+ nonce: nonce,
1687
+ signatureShares: [],
1688
+ },
1689
+ ],
1690
+ },
1691
+ ],
1692
+ };
1693
+ }
1694
+ async buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params) {
1695
+ const txAmount = await this.validateBalanceAndGetTxAmount(baseAddress, gasPrice, gasLimit, params.apiKey);
1696
+ const nonce = await this.getAddressNonce(baseAddress, params.apiKey);
1697
+ const recipients = [
1698
+ {
1699
+ address: params.recoveryDestination,
1700
+ amount: txAmount.toString(10),
1701
+ },
1702
+ ];
1703
+ const txInfo = {
1704
+ recipient: recipients[0],
1705
+ expireTime: this.getDefaultExpireTime(),
1706
+ gasLimit: gasLimit.toString(10),
1707
+ };
1708
+ const txParams = {
1709
+ to: params.recoveryDestination,
1710
+ nonce: nonce,
1711
+ value: txAmount,
1712
+ gasPrice: gasPrice,
1713
+ gasLimit: gasLimit,
1714
+ data: Buffer.from('0x'),
1715
+ eip1559: params.eip1559,
1716
+ replayProtectionOptions: params.replayProtectionOptions,
1717
+ };
1718
+ const tx = AbstractEthLikeNewCoins.buildTransaction(txParams);
1719
+ return { txInfo, tx, nonce };
1720
+ }
1721
+ async validateBalanceAndGetTxAmount(baseAddress, gasPrice, gasLimit, apiKey) {
1722
+ const baseAddressBalance = await this.queryAddressBalance(baseAddress, apiKey);
1723
+ let totalGasNeeded = gasPrice.mul(gasLimit);
1724
+ // On L2 chains with L1 data fees, add buffer for L1 fees
1725
+ if (this.staticsCoin?.family !== undefined && lib_1.coinFamiliesWithL1Fees.includes(this.staticsCoin.family)) {
1726
+ totalGasNeeded = totalGasNeeded.add(new exports.optionalDeps.ethUtil.BN(statics_1.ethGasConfigs.l1GasFeeBuffer));
1727
+ }
1728
+ const weiToGwei = new bn_js_1.default(10 ** 9);
1729
+ if (baseAddressBalance.lt(totalGasNeeded)) {
1730
+ throw new Error(`Backup key address ${baseAddress} has balance ${baseAddressBalance.div(weiToGwei).toString()} Gwei.` +
1731
+ `This address must have a balance of at least ${totalGasNeeded.div(weiToGwei).toString()}` +
1732
+ ` Gwei to perform recoveries. Try sending some ETH to this address then retry.`);
1733
+ }
1734
+ const txAmount = baseAddressBalance.sub(totalGasNeeded);
1735
+ return txAmount;
1736
+ }
1737
+ /**
1738
+ * Make a query to blockchain explorer for information such as balance, token balance, solidity calls
1739
+ * @param query {Object} key-value pairs of parameters to append after /api
1740
+ * @param apiKey {string} optional API key to use instead of the one from the environment
1741
+ * @returns {Object} response from the blockchain explorer
1742
+ */
1743
+ async recoveryBlockchainExplorerQuery(query, apiKey) {
1744
+ throw new Error('method not implemented');
1745
+ }
1746
+ /**
1747
+ * Creates the extra parameters needed to build a hop transaction
1748
+ * @param buildParams The original build parameters
1749
+ * @returns extra parameters object to merge with the original build parameters object and send to the platform
1750
+ */
1751
+ async createHopTransactionParams(buildParams) {
1752
+ const wallet = buildParams.wallet;
1753
+ const recipients = buildParams.recipients;
1754
+ const walletPassphrase = buildParams.walletPassphrase;
1755
+ const userKeychain = await this.keychains().get({ id: wallet.keyIds()[0] });
1756
+ const userPrv = wallet.getUserPrv({ keychain: userKeychain, walletPassphrase });
1757
+ const userPrvBuffer = secp256k1_1.bip32.fromBase58(userPrv).privateKey;
1758
+ if (!userPrvBuffer) {
1759
+ throw new Error('invalid userPrv');
1760
+ }
1761
+ if (!recipients || !Array.isArray(recipients)) {
1762
+ throw new Error('expecting array of recipients');
1763
+ }
1764
+ // Right now we only support 1 recipient
1765
+ if (recipients.length !== 1) {
1766
+ throw new Error('must send to exactly 1 recipient');
1767
+ }
1768
+ const recipientAddress = recipients[0].address;
1769
+ const recipientAmount = recipients[0].amount;
1770
+ const feeEstimateParams = {
1771
+ recipient: recipientAddress,
1772
+ amount: recipientAmount,
1773
+ hop: true,
1774
+ };
1775
+ const feeEstimate = await this.feeEstimate(feeEstimateParams);
1776
+ const gasLimit = feeEstimate.gasLimitEstimate;
1777
+ const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);
1778
+ const gasPriceMax = gasPrice * 5;
1779
+ // Payment id a random number so its different for every tx
1780
+ const paymentId = Math.floor(Math.random() * 10000000000).toString();
1781
+ const hopDigest = AbstractEthLikeNewCoins.getHopDigest([
1782
+ recipientAddress,
1783
+ recipientAmount,
1784
+ gasPriceMax.toString(),
1785
+ gasLimit.toString(),
1786
+ paymentId,
1787
+ ]);
1788
+ const userReqSig = exports.optionalDeps.ethUtil.addHexPrefix(Buffer.from(secp256k1_2.default.ecdsaSign(hopDigest, userPrvBuffer).signature).toString('hex'));
1789
+ return {
1790
+ hopParams: {
1791
+ gasPriceMax,
1792
+ userReqSig,
1793
+ paymentId,
1794
+ gasLimit,
1795
+ },
1796
+ };
1797
+ }
1798
+ /**
1799
+ * Validates that the hop prebuild from the HSM is valid and correct
1800
+ * @param {IWallet} wallet - The wallet that the prebuild is for
1801
+ * @param {HopPrebuild} hopPrebuild - The prebuild to validate
1802
+ * @param {Object} originalParams - The original parameters passed to prebuildTransaction
1803
+ * @param {Recipient[]} originalParams.recipients - The original recipients array
1804
+ * @returns {void}
1805
+ * @throws Error if The prebuild is invalid
1806
+ */
1807
+ async validateHopPrebuild(wallet, hopPrebuild, originalParams) {
1808
+ const { tx, id, signature } = hopPrebuild;
1809
+ // first, validate the HSM signature
1810
+ const serverXpub = sdk_core_1.common.Environments[this.bitgo.getEnv()].hsmXpub;
1811
+ const serverPubkeyBuffer = secp256k1_1.bip32.fromBase58(serverXpub).publicKey;
1812
+ const signatureBuffer = Buffer.from(exports.optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');
1813
+ const messageBuffer = Buffer.from(exports.optionalDeps.ethUtil.padToEven(exports.optionalDeps.ethUtil.stripHexPrefix(id)), 'hex');
1814
+ const sig = new Uint8Array(signatureBuffer.slice(1));
1815
+ const isValidSignature = secp256k1_2.default.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);
1816
+ if (!isValidSignature) {
1817
+ throw new Error(`Hop txid signature invalid - pub: ${serverXpub}, msg: ${messageBuffer?.toString()}, sig: ${signatureBuffer?.toString()}`);
1818
+ }
1819
+ const builtHopTx = exports.optionalDeps.EthTx.TransactionFactory.fromSerializedData(exports.optionalDeps.ethUtil.toBuffer(tx));
1820
+ // If original params are given, we can check them against the transaction prebuild params
1821
+ if (!lodash_1.default.isNil(originalParams)) {
1822
+ const { recipients } = originalParams;
1823
+ // Then validate that the tx params actually equal the requested params
1824
+ const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount);
1825
+ const originalDestination = recipients[0].address;
1826
+ const hopAmount = new bignumber_js_1.BigNumber(exports.optionalDeps.ethUtil.bufferToHex(builtHopTx.value));
1827
+ if (!builtHopTx.to) {
1828
+ throw new Error(`Transaction does not have a destination address`);
1829
+ }
1830
+ const hopDestination = builtHopTx.to.toString();
1831
+ if (!hopAmount.eq(originalAmount)) {
1832
+ throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
1833
+ }
1834
+ if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
1835
+ throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);
1836
+ }
1837
+ }
1838
+ if (!builtHopTx.verifySignature()) {
1839
+ // We dont want to continue at all in this case, at risk of ETH being stuck on the hop address
1840
+ throw new Error(`Invalid hop transaction signature, txid: ${id}`);
1841
+ }
1842
+ if (exports.optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {
1843
+ throw new Error(`Signed hop txid does not equal actual txid`);
1844
+ }
1845
+ }
1846
+ /**
1847
+ * Gets the hop digest for the user to sign. This is validated in the HSM to prove that the user requested this tx
1848
+ * @param {string[]} paramsArr - The parameters to hash together for the digest
1849
+ * @returns {Buffer}
1850
+ */
1851
+ static getHopDigest(paramsArr) {
1852
+ const hash = (0, keccak_1.default)('keccak256');
1853
+ hash.update([AbstractEthLikeNewCoins.hopTransactionSalt, ...paramsArr].join('$'));
1854
+ return hash.digest();
1855
+ }
1856
+ /**
1857
+ * Modify prebuild before sending it to the server. Add things like hop transaction params
1858
+ * @param {BuildOptions} buildParams - The whitelisted parameters for this prebuild
1859
+ * @param {boolean} buildParams.hop - True if this should prebuild a hop tx, else false
1860
+ * @param {Recipient[]} buildParams.recipients - The recipients array of this transaction
1861
+ * @param {Wallet} buildParams.wallet - The wallet sending this tx
1862
+ * @param {string} buildParams.walletPassphrase - the passphrase for this wallet
1863
+ * @returns {Promise<BuildOptions>}
1864
+ */
1865
+ async getExtraPrebuildParams(buildParams) {
1866
+ if (!lodash_1.default.isUndefined(buildParams.hop) &&
1867
+ buildParams.hop &&
1868
+ !lodash_1.default.isUndefined(buildParams.wallet) &&
1869
+ !lodash_1.default.isUndefined(buildParams.recipients) &&
1870
+ !lodash_1.default.isUndefined(buildParams.walletPassphrase)) {
1871
+ if (this instanceof ethLikeToken_1.EthLikeToken) {
1872
+ throw new Error(`Hop transactions are not enabled for ERC-20 tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`);
1873
+ }
1874
+ return (await this.createHopTransactionParams({
1875
+ wallet: buildParams.wallet,
1876
+ recipients: buildParams.recipients,
1877
+ walletPassphrase: buildParams.walletPassphrase,
1878
+ }));
1879
+ }
1880
+ return {};
1881
+ }
1882
+ /**
1883
+ * Modify prebuild after receiving it from the server. Add things like nlocktime
1884
+ * @param {TransactionPrebuild} params - The prebuild to modify
1885
+ * @returns {TransactionPrebuild} The modified prebuild
1886
+ */
1887
+ async postProcessPrebuild(params) {
1888
+ if (!lodash_1.default.isUndefined(params.hopTransaction) && !lodash_1.default.isUndefined(params.wallet) && !lodash_1.default.isUndefined(params.buildParams)) {
1889
+ await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);
1890
+ }
1891
+ return params;
1892
+ }
1893
+ /**
1894
+ * Coin-specific things done before signing a transaction, i.e. verification
1895
+ * @param {PresignTransactionOptions} params
1896
+ * @returns {Promise<PresignTransactionOptions>}
1897
+ */
1898
+ async presignTransaction(params) {
1899
+ if (!lodash_1.default.isUndefined(params.hopTransaction) && !lodash_1.default.isUndefined(params.wallet) && !lodash_1.default.isUndefined(params.buildParams)) {
1900
+ await this.validateHopPrebuild(params.wallet, params.hopTransaction);
1901
+ }
1902
+ return params;
1903
+ }
1904
+ /**
1905
+ * Fetch fee estimate information from the server
1906
+ * @param {Object} params - The params passed into the function
1907
+ * @param {boolean} [params.hop] - True if we should estimate fee for a hop transaction
1908
+ * @param {string} [params.recipient] - The recipient of the transaction to estimate a send to
1909
+ * @param {string} [params.data] - The ETH tx data to estimate a send for
1910
+ * @returns {Object} The fee info returned from the server
1911
+ */
1912
+ async feeEstimate(params) {
1913
+ const query = {};
1914
+ if (params && params.hop) {
1915
+ query.hop = params.hop;
1916
+ }
1917
+ if (params && params.recipient) {
1918
+ query.recipient = params.recipient;
1919
+ }
1920
+ if (params && params.data) {
1921
+ query.data = params.data;
1922
+ }
1923
+ if (params && params.amount) {
1924
+ query.amount = params.amount;
1925
+ }
1926
+ return await this.bitgo.get(this.url('/tx/fee')).query(query).result();
1927
+ }
1928
+ /**
1929
+ * Generate secp256k1 key pair
1930
+ *
1931
+ * @param {Buffer} seed
1932
+ * @returns {KeyPair} object with generated pub and prv
1933
+ */
1934
+ generateKeyPair(seed) {
1935
+ if (!seed) {
1936
+ // An extended private key has both a normal 256 bit private key and a 256
1937
+ // bit chain code, both of which must be random. 512 bits is therefore the
1938
+ // maximum entropy and gives us maximum security against cracking.
1939
+ seed = (0, crypto_1.randomBytes)(512 / 8);
1940
+ }
1941
+ const extendedKey = secp256k1_1.bip32.fromSeed(seed);
1942
+ const xpub = extendedKey.neutered().toBase58();
1943
+ return {
1944
+ pub: xpub,
1945
+ prv: extendedKey.toBase58(),
1946
+ };
1947
+ }
1948
+ async parseTransaction(params) {
1949
+ return {};
1950
+ }
1951
+ /**
1952
+ * Get forwarder factory and implementation addresses for deposit address verification.
1953
+ * Forwarders are smart contracts that forward funds to the base wallet address.
1954
+ *
1955
+ * @param {number | undefined} forwarderVersion - The wallet version
1956
+ * @returns {object} Factory and implementation addresses for forwarders
1957
+ */
1958
+ getForwarderFactoryAddressesAndForwarderImplementationAddress(forwarderVersion) {
1959
+ const ethNetwork = this.getNetwork();
1960
+ switch (forwarderVersion) {
1961
+ case 1:
1962
+ if (!ethNetwork?.forwarderFactoryAddress || !ethNetwork?.forwarderImplementationAddress) {
1963
+ throw new Error('Forwarder factory addresses not configured for this network');
1964
+ }
1965
+ return {
1966
+ forwarderFactoryAddress: ethNetwork.forwarderFactoryAddress,
1967
+ forwarderImplementationAddress: ethNetwork.forwarderImplementationAddress,
1968
+ };
1969
+ case 2:
1970
+ if (!ethNetwork?.walletV2ForwarderFactoryAddress || !ethNetwork?.walletV2ForwarderImplementationAddress) {
1971
+ throw new Error('Wallet v2 factory addresses not configured for this network');
1972
+ }
1973
+ return {
1974
+ forwarderFactoryAddress: ethNetwork.walletV2ForwarderFactoryAddress,
1975
+ forwarderImplementationAddress: ethNetwork.walletV2ForwarderImplementationAddress,
1976
+ };
1977
+ case 4:
1978
+ case 5:
1979
+ if (!ethNetwork?.walletV4ForwarderFactoryAddress || !ethNetwork?.walletV4ForwarderImplementationAddress) {
1980
+ throw new Error(`Forwarder v${forwarderVersion} factory addresses not configured for this network`);
1981
+ }
1982
+ return {
1983
+ forwarderFactoryAddress: ethNetwork.walletV4ForwarderFactoryAddress,
1984
+ forwarderImplementationAddress: ethNetwork.walletV4ForwarderImplementationAddress,
1985
+ };
1986
+ default:
1987
+ throw new Error(`Forwarder version ${forwarderVersion} not supported`);
1988
+ }
1989
+ }
1990
+ /**
1991
+ * Get wallet base address factory and implementation addresses.
1992
+ * This is used for base address verification for V1, V2, V4, and V5 wallets.
1993
+ * The base address is the main wallet contract deployed via CREATE2.
1994
+ *
1995
+ * @param {number} walletVersion - The wallet version (1, 2, 4, or 5)
1996
+ * @returns {object} Factory and implementation addresses for the wallet base address
1997
+ * @throws {Error} if wallet version addresses are not configured
1998
+ */
1999
+ getWalletAddressFactoryAddressesAndImplementationAddress(walletVersion) {
2000
+ const ethNetwork = this.getNetwork();
2001
+ switch (walletVersion) {
2002
+ case 1:
2003
+ if (!ethNetwork?.walletFactoryAddress || !ethNetwork?.walletImplementationAddress) {
2004
+ throw new Error('Wallet v1 factory addresses not configured for this network');
2005
+ }
2006
+ return {
2007
+ walletFactoryAddress: ethNetwork.walletFactoryAddress,
2008
+ walletImplementationAddress: ethNetwork.walletImplementationAddress,
2009
+ };
2010
+ case 2:
2011
+ if (!ethNetwork?.walletV2FactoryAddress || !ethNetwork?.walletV2ImplementationAddress) {
2012
+ throw new Error('Wallet v2 factory addresses not configured for this network');
2013
+ }
2014
+ return {
2015
+ walletFactoryAddress: ethNetwork.walletV2FactoryAddress,
2016
+ walletImplementationAddress: ethNetwork.walletV2ImplementationAddress,
2017
+ };
2018
+ case 4:
2019
+ case 5:
2020
+ if (!ethNetwork?.walletV4ForwarderFactoryAddress || !ethNetwork?.walletV4ForwarderImplementationAddress) {
2021
+ throw new Error(`Wallet v${walletVersion} factory addresses not configured for this network`);
2022
+ }
2023
+ return {
2024
+ walletFactoryAddress: ethNetwork.walletV4ForwarderFactoryAddress,
2025
+ walletImplementationAddress: ethNetwork.walletV4ForwarderImplementationAddress,
2026
+ };
2027
+ default:
2028
+ throw new Error(`Wallet version ${walletVersion} not supported`);
2029
+ }
2030
+ }
2031
+ /**
2032
+ * Helper method to create a salt buffer from hex string.
2033
+ * Converts a hex salt string to a 32-byte buffer.
2034
+ *
2035
+ * @param {string} salt - The hex salt string
2036
+ * @returns {Buffer} 32-byte salt buffer
2037
+ */
2038
+ createSaltBuffer(salt) {
2039
+ const ethUtil = exports.optionalDeps.ethUtil;
2040
+ return ethUtil.setLengthLeft(Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(salt || '')), 'hex'), 32);
2041
+ }
2042
+ /**
2043
+ * Verify BIP32 wallet base address (V1, V2, V4).
2044
+ * These wallets use a wallet factory to deploy base addresses with CREATE2.
2045
+ * The address is derived from the keychains' ethAddresses and a salt.
2046
+ *
2047
+ * @param {VerifyBip32BaseAddressOptions} params - Verification parameters
2048
+ * @returns {object} Expected and actual addresses for comparison
2049
+ */
2050
+ verifyCreate2BaseAddress(params) {
2051
+ const { address, coinSpecific, keychains, walletVersion } = params;
2052
+ if (!coinSpecific.salt) {
2053
+ throw new Error(`missing salt for v${walletVersion} base address verification`);
2054
+ }
2055
+ // Get wallet factory and implementation addresses for the wallet version
2056
+ const { walletFactoryAddress, walletImplementationAddress } = this.getWalletAddressFactoryAddressesAndImplementationAddress(walletVersion);
2057
+ const initcode = (0, lib_1.getProxyInitcode)(walletImplementationAddress);
2058
+ // Convert the wallet salt to a buffer, pad to 32 bytes
2059
+ const saltBuffer = this.createSaltBuffer(coinSpecific.salt);
2060
+ // Reconstruct calculationSalt using keychains' ethAddresses and wallet salt
2061
+ const ethAddresses = keychains.map((kc) => {
2062
+ if (!kc.ethAddress) {
2063
+ throw new Error(`keychain missing ethAddress for v${walletVersion} base address verification`);
2064
+ }
2065
+ return kc.ethAddress;
2066
+ });
2067
+ const calculationSalt = exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(['address[]', 'bytes32'], [ethAddresses, saltBuffer]));
2068
+ const expectedAddress = (0, lib_1.calculateForwarderV1Address)(walletFactoryAddress, calculationSalt, initcode);
2069
+ if (expectedAddress !== address) {
2070
+ throw new sdk_core_1.UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);
2071
+ }
2072
+ return true;
2073
+ }
2074
+ /**
2075
+ * Verify forwarder receive address (deposit address).
2076
+ * Forwarder addresses are derived using CREATE2 from the base address and salt.
2077
+ *
2078
+ * @param {VerifyEthAddressOptions} params - Verification parameters
2079
+ * @param {number} forwarderVersion - The forwarder version
2080
+ * @returns {object} Expected and actual addresses for comparison
2081
+ */
2082
+ verifyForwarderAddress(params, forwarderVersion) {
2083
+ const { address, coinSpecific, baseAddress } = params;
2084
+ const { forwarderFactoryAddress, forwarderImplementationAddress } = this.getForwarderFactoryAddressesAndForwarderImplementationAddress(forwarderVersion);
2085
+ const initcode = (0, lib_1.getProxyInitcode)(forwarderImplementationAddress);
2086
+ const saltBuffer = this.createSaltBuffer(coinSpecific.salt || '');
2087
+ const { createForwarderParams, createForwarderTypes } = forwarderVersion === 4
2088
+ ? (0, lib_1.getCreateForwarderParamsAndTypes)(baseAddress, saltBuffer, coinSpecific.feeAddress)
2089
+ : (0, lib_1.getCreateForwarderParamsAndTypes)(baseAddress, saltBuffer);
2090
+ const calculationSalt = exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(createForwarderTypes, createForwarderParams));
2091
+ const expectedAddress = (0, lib_1.calculateForwarderV1Address)(forwarderFactoryAddress, calculationSalt, initcode);
2092
+ if (expectedAddress !== address) {
2093
+ throw new sdk_core_1.UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);
2094
+ }
2095
+ return true;
2096
+ }
2097
+ /**
2098
+ * Make sure an address is a wallet address and throw an error if it's not.
2099
+ * @param {Object} params
2100
+ * @param {string} params.address - The derived address string on the network
2101
+ * @param {Object} params.coinSpecific - Coin-specific details for the address such as a forwarderVersion
2102
+ * @param {string} params.baseAddress - The base address of the wallet on the network
2103
+ * @throws {InvalidAddressError}
2104
+ * @throws {InvalidAddressVerificationObjectPropertyError}
2105
+ * @throws {UnexpectedAddressError}
2106
+ * @returns {boolean} True iff address is a wallet address
2107
+ */
2108
+ async isWalletAddress(params) {
2109
+ const { address, impliedForwarderVersion, coinSpecific, baseAddress } = params;
2110
+ const forwarderVersion = impliedForwarderVersion ?? coinSpecific?.forwarderVersion;
2111
+ // Validate address format
2112
+ if (address && !this.isValidAddress(address)) {
2113
+ throw new sdk_core_1.InvalidAddressError(`invalid address: ${address}`);
2114
+ }
2115
+ // Forwarder version 0 addresses cannot be verified because we do not store the nonce value required for address derivation.
2116
+ if (forwarderVersion === 0) {
2117
+ return true;
2118
+ }
2119
+ // Determine if we are verifying a base address
2120
+ const isVerifyingBaseAddress = baseAddress && address === baseAddress;
2121
+ // TSS/MPC wallet address verification (V3, V5, V6)
2122
+ // V5 base addresses use TSS, but V5 forwarders use the regular forwarder verification
2123
+ const isTssWalletVersion = params.walletVersion === 3 || params.walletVersion === 5 || params.walletVersion === 6;
2124
+ const shouldUseTssVerification = (0, sdk_core_1.isTssVerifyAddressOptions)(params) && isTssWalletVersion && (params.walletVersion !== 5 || isVerifyingBaseAddress);
2125
+ if (shouldUseTssVerification) {
2126
+ if (isVerifyingBaseAddress) {
2127
+ const index = typeof params.index === 'string' ? parseInt(params.index, 10) : params.index;
2128
+ if (index !== 0) {
2129
+ throw new Error(`Base address verification requires index 0, but got index ${params.index}. ` +
2130
+ `The base address is always derived at index 0.`);
2131
+ }
2132
+ }
2133
+ return (0, sdk_core_1.verifyMPCWalletAddress)({ ...params, keyCurve: 'secp256k1' }, this.isValidAddress, (pubKey) => {
2134
+ return new lib_1.KeyPair({ pub: pubKey }).getAddress();
2135
+ });
2136
+ }
2137
+ // From here on, we need baseAddress and coinSpecific for non-TSS verifications
2138
+ if (lodash_1.default.isUndefined(baseAddress) || !this.isValidAddress(baseAddress)) {
2139
+ throw new sdk_core_1.InvalidAddressError('invalid base address');
2140
+ }
2141
+ if (!lodash_1.default.isObject(coinSpecific)) {
2142
+ throw new sdk_core_1.InvalidAddressVerificationObjectPropertyError('address validation failure: coinSpecific field must be an object');
2143
+ }
2144
+ // BIP32 wallet base address verification (V1, V2, V4)
2145
+ if (isVerifyingBaseAddress && isVerifyContractBaseAddressOptions(params)) {
2146
+ return this.verifyCreate2BaseAddress(params);
2147
+ }
2148
+ // Forwarder receive address verification (deposit addresses)
2149
+ if (!isVerifyingBaseAddress) {
2150
+ return this.verifyForwarderAddress(params, forwarderVersion);
2151
+ }
2152
+ // If we reach here, it's a base address verification for an unsupported wallet version
2153
+ throw new Error(`Base address verification not supported for wallet version ${params.walletVersion}`);
2154
+ }
2155
+ /**
2156
+ *
2157
+ * @param {TransactionPrebuild} txPrebuild
2158
+ * @returns {boolean}
2159
+ */
2160
+ verifyCoin(txPrebuild) {
2161
+ const nativeCoin = this.getChain().split(':')[0];
2162
+ return txPrebuild.coin === nativeCoin;
2163
+ }
2164
+ /**
2165
+ * Generate transaction explanation for error reporting
2166
+ * @param txPrebuild - Transaction prebuild containing txHex and fee info
2167
+ * @returns Stringified JSON explanation
2168
+ */
2169
+ async getTxExplanation(txPrebuild) {
2170
+ if (!txPrebuild?.txHex || !txPrebuild?.gasPrice) {
2171
+ return undefined;
2172
+ }
2173
+ try {
2174
+ const explanation = await this.explainTransaction({
2175
+ txHex: txPrebuild.txHex,
2176
+ feeInfo: {
2177
+ fee: txPrebuild.gasPrice.toString(),
2178
+ },
2179
+ });
2180
+ return JSON.stringify(explanation, null, 2);
2181
+ }
2182
+ catch (e) {
2183
+ const errorDetails = {
2184
+ error: 'Failed to parse transaction explanation',
2185
+ txHex: txPrebuild.txHex,
2186
+ details: e instanceof Error ? e.message : String(e),
2187
+ };
2188
+ return JSON.stringify(errorDetails, null, 2);
2189
+ }
2190
+ }
2191
+ /**
2192
+ * Verify if a tss transaction is valid
2193
+ *
2194
+ * @param {VerifyEthTransactionOptions} params
2195
+ * @param {TransactionParams} params.txParams - params object passed to send
2196
+ * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server
2197
+ * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against
2198
+ * @returns {boolean}
2199
+ * @throws {TxIntentMismatchRecipientError} if transaction recipients don't match user intent
2200
+ */
2201
+ async verifyTssTransaction(params) {
2202
+ const { txParams, txPrebuild, wallet } = params;
2203
+ // Helper to throw TxIntentMismatchRecipientError with recipient details
2204
+ const throwRecipientMismatch = async (message, mismatchedRecipients) => {
2205
+ const txExplanation = await this.getTxExplanation(txPrebuild);
2206
+ throw new sdk_core_1.TxIntentMismatchRecipientError(message, undefined, [txParams], txPrebuild?.txHex, mismatchedRecipients, txExplanation);
2207
+ };
2208
+ if (!txParams?.recipients &&
2209
+ !(txParams.prebuildTx?.consolidateId ||
2210
+ txPrebuild?.consolidateId ||
2211
+ (txParams.type &&
2212
+ ['acceleration', 'fillNonce', 'transferToken', 'tokenApproval', 'consolidate', 'bridgeFunds'].includes(txParams.type)))) {
2213
+ throw new Error('missing txParams');
2214
+ }
2215
+ if (!wallet || !txPrebuild) {
2216
+ throw new Error('missing params');
2217
+ }
2218
+ if (txParams.hop && txParams.recipients && txParams.recipients.length > 1) {
2219
+ throw new Error('tx cannot be both a batch and hop transaction');
2220
+ }
2221
+ if (txParams.type && ['transfer'].includes(txParams.type)) {
2222
+ if (txParams.recipients && txParams.recipients.length === 1) {
2223
+ const recipients = txParams.recipients;
2224
+ const expectedAmount = recipients[0].amount.toString();
2225
+ const expectedDestination = recipients[0].address;
2226
+ const txBuilder = this.getTransactionBuilder();
2227
+ txBuilder.from(txPrebuild.txHex);
2228
+ const tx = await txBuilder.build();
2229
+ const txJson = tx.toJson();
2230
+ if (txJson.data === '0x') {
2231
+ if (expectedAmount !== txJson.value) {
2232
+ await throwRecipientMismatch('the transaction amount in txPrebuild does not match the value given by client', [{ address: txJson.to, amount: txJson.value }]);
2233
+ }
2234
+ if (expectedDestination.toLowerCase() !== txJson.to.toLowerCase()) {
2235
+ await throwRecipientMismatch('destination address does not match with the recipient address', [
2236
+ { address: txJson.to, amount: txJson.value },
2237
+ ]);
2238
+ }
2239
+ }
2240
+ else if (txJson.data.startsWith('0xa9059cbb')) {
2241
+ const [recipientAddress, amount] = (0, lib_1.getRawDecoded)(['address', 'uint256'], (0, lib_1.getBufferedByteCode)('0xa9059cbb', txJson.data));
2242
+ // Check if recipients[0].data exists (WalletConnect flow)
2243
+ let expectedRecipientAddress;
2244
+ let expectedTokenAmount;
2245
+ const recipientData = recipients[0].data;
2246
+ if (recipientData && recipientData.startsWith('0xa9059cbb')) {
2247
+ // WalletConnect: decode expected recipient and amount from recipients[0].data
2248
+ const [expectedRecipient, expectedAmount] = (0, lib_1.getRawDecoded)(['address', 'uint256'], (0, lib_1.getBufferedByteCode)('0xa9059cbb', recipientData));
2249
+ expectedRecipientAddress = (0, ethereumjs_util_1.addHexPrefix)(expectedRecipient.toString()).toLowerCase();
2250
+ expectedTokenAmount = expectedAmount.toString();
2251
+ }
2252
+ else {
2253
+ // Normal flow: use recipients[0].address and recipients[0].amount
2254
+ expectedRecipientAddress = expectedDestination.toLowerCase();
2255
+ expectedTokenAmount = expectedAmount;
2256
+ }
2257
+ if (expectedTokenAmount !== amount.toString()) {
2258
+ await throwRecipientMismatch('the transaction amount in txPrebuild does not match the value given by client', [{ address: (0, ethereumjs_util_1.addHexPrefix)(recipientAddress.toString()), amount: amount.toString() }]);
2259
+ }
2260
+ if (expectedRecipientAddress !== (0, ethereumjs_util_1.addHexPrefix)(recipientAddress.toString()).toLowerCase()) {
2261
+ await throwRecipientMismatch('destination address does not match with the recipient address', [
2262
+ { address: (0, ethereumjs_util_1.addHexPrefix)(recipientAddress.toString()), amount: amount.toString() },
2263
+ ]);
2264
+ }
2265
+ }
2266
+ }
2267
+ }
2268
+ // Verify consolidation transactions send to base address
2269
+ if (params.verification?.consolidationToBaseAddress) {
2270
+ const coinSpecific = wallet.coinSpecific();
2271
+ if (!coinSpecific || !coinSpecific.baseAddress) {
2272
+ throw new Error('Unable to determine base address for consolidation');
2273
+ }
2274
+ const baseAddress = coinSpecific.baseAddress;
2275
+ if (!txPrebuild.txHex) {
2276
+ throw new Error('missing txHex in txPrebuild');
2277
+ }
2278
+ const txBuilder = this.getTransactionBuilder();
2279
+ txBuilder.from(txPrebuild.txHex);
2280
+ const tx = await txBuilder.build();
2281
+ const txJson = tx.toJson();
2282
+ // Verify the transaction recipient matches the base address
2283
+ if (!txJson.to) {
2284
+ throw new Error('Consolidation transaction is missing recipient address');
2285
+ }
2286
+ if (txJson.to.toLowerCase() !== baseAddress.toLowerCase()) {
2287
+ await throwRecipientMismatch('Consolidation transaction recipient does not match wallet base address', [
2288
+ { address: txJson.to, amount: txJson.value },
2289
+ ]);
2290
+ }
2291
+ }
2292
+ return true;
2293
+ }
2294
+ /**
2295
+ * Verify that a transaction prebuild complies with the original intention
2296
+ *
2297
+ * @param {VerifyEthTransactionOptions} params
2298
+ * @param {TransactionParams} params.txParams - params object passed to send
2299
+ * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server
2300
+ * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against
2301
+ * @returns {boolean}
2302
+ * @throws {TxIntentMismatchError} if transaction validation fails
2303
+ * @throws {TxIntentMismatchRecipientError} if transaction recipients don't match user intent
2304
+ */
2305
+ async verifyTransaction(params) {
2306
+ const ethNetwork = this.getNetwork();
2307
+ const { txParams, txPrebuild, wallet, walletType } = params;
2308
+ if (walletType === 'tss') {
2309
+ return this.verifyTssTransaction(params);
2310
+ }
2311
+ // Helper to throw TxIntentMismatchRecipientError with recipient details
2312
+ const throwRecipientMismatch = async (message, mismatchedRecipients) => {
2313
+ const txExplanation = await this.getTxExplanation(txPrebuild);
2314
+ throw new sdk_core_1.TxIntentMismatchRecipientError(message, undefined, [txParams], txPrebuild?.txHex, mismatchedRecipients, txExplanation);
2315
+ };
2316
+ if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {
2317
+ throw new Error('missing params');
2318
+ }
2319
+ const recipients = txParams.recipients;
2320
+ if (txParams.hop && recipients.length > 1) {
2321
+ throw new Error('tx cannot be both a batch and hop transaction');
2322
+ }
2323
+ if (txPrebuild.recipients.length > 1) {
2324
+ throw new Error(`${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
2325
+ }
2326
+ if (txParams.hop && txPrebuild.hopTransaction) {
2327
+ // Check recipient amount for hop transaction
2328
+ if (recipients.length !== 1) {
2329
+ throw new Error(`hop transaction only supports 1 recipient but ${recipients.length} found`);
2330
+ }
2331
+ // Check tx sends to hop address
2332
+ const decodedHopTx = exports.optionalDeps.EthTx.TransactionFactory.fromSerializedData(exports.optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx));
2333
+ const expectedHopAddress = exports.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());
2334
+ const actualHopAddress = exports.optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);
2335
+ if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {
2336
+ await throwRecipientMismatch('recipient address of txPrebuild does not match hop address', [
2337
+ { address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() },
2338
+ ]);
2339
+ }
2340
+ // Convert TransactionRecipient array to Recipient array
2341
+ const hopRecipients = recipients.map((r) => {
2342
+ return {
2343
+ address: r.address,
2344
+ amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,
2345
+ };
2346
+ });
2347
+ // Check destination address and amount
2348
+ await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients: hopRecipients });
2349
+ }
2350
+ else if (recipients.length > 1) {
2351
+ // Check total amount for batch transaction
2352
+ if (txParams.tokenName) {
2353
+ const expectedTotalAmount = new bignumber_js_1.BigNumber(0);
2354
+ if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
2355
+ await throwRecipientMismatch('batch token transaction amount in txPrebuild should be zero for token transfers', [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]);
2356
+ }
2357
+ }
2358
+ else {
2359
+ let expectedTotalAmount = new bignumber_js_1.BigNumber(0);
2360
+ for (let i = 0; i < recipients.length; i++) {
2361
+ expectedTotalAmount = expectedTotalAmount.plus(recipients[i].amount);
2362
+ }
2363
+ if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
2364
+ await throwRecipientMismatch('batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client', [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]);
2365
+ }
2366
+ }
2367
+ // Check batch transaction is sent to the batcher contract address for the chain
2368
+ const batcherContractAddress = ethNetwork?.batcherContractAddress;
2369
+ if (!batcherContractAddress ||
2370
+ batcherContractAddress.toLowerCase() !== txPrebuild.recipients[0].address.toLowerCase()) {
2371
+ await throwRecipientMismatch('recipient address of txPrebuild does not match batcher address', [
2372
+ { address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() },
2373
+ ]);
2374
+ }
2375
+ }
2376
+ else {
2377
+ // Check recipient address and amount for normal transaction
2378
+ if (recipients.length !== 1) {
2379
+ throw new Error(`normal transaction only supports 1 recipient but ${recipients.length} found`);
2380
+ }
2381
+ const expectedAmount = new bignumber_js_1.BigNumber(recipients[0].amount);
2382
+ if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
2383
+ await throwRecipientMismatch('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client', [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]);
2384
+ }
2385
+ if (this.isETHAddress(recipients[0].address) && recipients[0].address !== txPrebuild.recipients[0].address) {
2386
+ await throwRecipientMismatch('destination address in normal txPrebuild does not match that in txParams supplied by client', [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]);
2387
+ }
2388
+ }
2389
+ // Check coin is correct for all transaction types
2390
+ if (!this.verifyCoin(txPrebuild)) {
2391
+ const txExplanation = await this.getTxExplanation(txPrebuild);
2392
+ throw new sdk_core_1.TxIntentMismatchError('coin in txPrebuild did not match that in txParams supplied by client', undefined, [txParams], txPrebuild?.txHex, txExplanation);
2393
+ }
2394
+ return true;
2395
+ }
2396
+ /**
2397
+ * Check if address is valid eth address
2398
+ * @param address
2399
+ * @returns {boolean}
2400
+ */
2401
+ isETHAddress(address) {
2402
+ return !!address.match(/0x[a-fA-F0-9]{40}/);
2403
+ }
2404
+ /**
2405
+ * Transform message to accommodate specific blockchain requirements.
2406
+ * @param {string} message - the message to prepare
2407
+ * @return {string} the prepared message as a hex encoded string.
2408
+ */
2409
+ encodeMessage(message) {
2410
+ const prefix = `\u0019Ethereum Signed Message:\n${message.length}`;
2411
+ return Buffer.from(prefix.concat(message)).toString('hex');
2412
+ }
2413
+ /**
2414
+ * Transform the Typed data to accomodate the blockchain requirements (EIP-712)
2415
+ * @param {TypedData} typedData - the typed data to prepare
2416
+ * @return {Buffer} a buffer of the result
2417
+ */
2418
+ encodeTypedData(typedData) {
2419
+ const version = typedData.version;
2420
+ if (version === eth_sig_util_1.SignTypedDataVersion.V1) {
2421
+ throw new Error('SignTypedData v1 is not supported due to security concerns');
2422
+ }
2423
+ const typedDataRaw = JSON.parse(typedData.typedDataRaw);
2424
+ const sanitizedData = eth_sig_util_1.TypedDataUtils.sanitizeData(typedDataRaw);
2425
+ const parts = [Buffer.from('1901', 'hex')];
2426
+ const eip712Domain = 'EIP712Domain';
2427
+ parts.push(eth_sig_util_1.TypedDataUtils.hashStruct(eip712Domain, sanitizedData.domain, sanitizedData.types, version));
2428
+ if (sanitizedData.primaryType !== eip712Domain) {
2429
+ parts.push(eth_sig_util_1.TypedDataUtils.hashStruct(sanitizedData.primaryType, sanitizedData.message, sanitizedData.types, version));
2430
+ }
2431
+ return Buffer.concat(parts);
2432
+ }
2433
+ /**
2434
+ * Build the data to transfer an ERC-721 or ERC-1155 token to another address
2435
+ * @param params
2436
+ */
2437
+ buildNftTransferData(params) {
2438
+ const { tokenContractAddress, recipientAddress, fromAddress } = params;
2439
+ switch (params.type) {
2440
+ case 'ERC721': {
2441
+ const tokenId = params.tokenId;
2442
+ const contractData = new lib_1.ERC721TransferBuilder()
2443
+ .tokenContractAddress(tokenContractAddress)
2444
+ .to(recipientAddress)
2445
+ .from(fromAddress)
2446
+ .tokenId(tokenId)
2447
+ .build();
2448
+ return contractData;
2449
+ }
2450
+ case 'ERC1155': {
2451
+ const entries = params.entries;
2452
+ const transferBuilder = new lib_1.ERC1155TransferBuilder()
2453
+ .tokenContractAddress(tokenContractAddress)
2454
+ .to(recipientAddress)
2455
+ .from(fromAddress);
2456
+ for (const entry of entries) {
2457
+ transferBuilder.entry(parseInt(entry.tokenId, 10), entry.amount);
2458
+ }
2459
+ return transferBuilder.build();
2460
+ }
2461
+ default:
2462
+ throw new Error(`Unsupported NFT type: ${params.type}`);
2463
+ }
2464
+ }
2465
+ /**
2466
+ * Fetch the gas price from the explorer
2467
+ * @param {string} wrongChainCoin - the coin that we're getting gas price for
2468
+ * @param {string} apiKey - optional API key to use instead of the one from the environment
2469
+ */
2470
+ async getGasPriceFromExternalAPI(wrongChainCoin, apiKey) {
2471
+ try {
2472
+ const res = await this.recoveryBlockchainExplorerQuery({
2473
+ chainid: this.getChainId().toString(),
2474
+ module: 'proxy',
2475
+ action: 'eth_gasPrice',
2476
+ }, apiKey);
2477
+ const gasPrice = new bn_js_1.default(res.result.slice(2), 16);
2478
+ console.log(` Got gas price: ${gasPrice}`);
2479
+ return gasPrice;
2480
+ }
2481
+ catch (e) {
2482
+ throw new Error(`Failed to get gas price. Please make sure to use the api key of ${wrongChainCoin}`);
2483
+ }
2484
+ }
2485
+ /**
2486
+ * Fetch the gas limit from the explorer
2487
+ * @param intendedChain
2488
+ * @param from
2489
+ * @param to
2490
+ * @param data
2491
+ * @param {string} apiKey - optional API key to use instead of the one from the environment
2492
+ */
2493
+ async getGasLimitFromExternalAPI(intendedChain, from, to, data, apiKey) {
2494
+ try {
2495
+ const res = await this.recoveryBlockchainExplorerQuery({
2496
+ chainid: this.getChainId().toString(),
2497
+ module: 'proxy',
2498
+ action: 'eth_estimateGas',
2499
+ from,
2500
+ to,
2501
+ data,
2502
+ }, apiKey);
2503
+ const gasLimit = new bn_js_1.default(res.result.slice(2), 16);
2504
+ console.log(`Got gas limit: ${gasLimit}`);
2505
+ return gasLimit;
2506
+ }
2507
+ catch (e) {
2508
+ throw new Error(`Failed to get gas limit. Please make sure to use the privateKey aka userKey of ${intendedChain} wallet ${to}`);
2509
+ }
2510
+ }
2511
+ /**
2512
+ * Get the balance of bitgoFeeAddress to ensure funds are available to pay fees
2513
+ * @param bitgoFeeAddress
2514
+ * @param gasPrice
2515
+ * @param gasLimit
2516
+ * @param apiKey - optional API key to use instead of the one from the environment
2517
+ */
2518
+ async ensureSufficientBalance(bitgoFeeAddress, gasPrice, gasLimit, apiKey) {
2519
+ const bitgoFeeAddressBalance = await this.queryAddressBalance(bitgoFeeAddress, apiKey);
2520
+ const totalGasNeeded = Number(gasPrice.mul(gasLimit));
2521
+ const weiToGwei = 10 ** 9;
2522
+ if (bitgoFeeAddressBalance.lt(totalGasNeeded)) {
2523
+ throw new Error(`Fee address ${bitgoFeeAddress} has balance ${(bitgoFeeAddressBalance / weiToGwei).toString()} Gwei.` +
2524
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
2525
+ ` Gwei to perform recoveries. Try sending some ${this.getChain()} to this address then retry.`);
2526
+ }
2527
+ }
2528
+ }
2529
+ exports.AbstractEthLikeNewCoins = AbstractEthLikeNewCoins;
2530
+ AbstractEthLikeNewCoins.hopTransactionSalt = 'bitgoHopAddressRequestSalt';
2531
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractEthLikeNewCoins.js","sourceRoot":"","sources":["../../src/abstractEthLikeNewCoins.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport {\n  AddressCoinSpecific,\n  BitGoBase,\n  BuildNftTransferDataOptions,\n  common,\n  Ecdsa,\n  ECDSAMethodTypes,\n  ECDSAUtils,\n  EthereumLibraryUnavailableError,\n  FeeEstimateOptions,\n  FullySignedTransaction,\n  getIsUnsignedSweep,\n  HalfSignedTransaction,\n  InvalidAddressError,\n  InvalidAddressVerificationObjectPropertyError,\n  IWallet,\n  KeyPair,\n  MPCSweepRecoveryOptions,\n  MPCSweepTxs,\n  MPCTx,\n  MPCTxs,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  PrebuildTransactionResult,\n  PresignTransactionOptions as BasePresignTransactionOptions,\n  Recipient,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  TxIntentMismatchError,\n  TxIntentMismatchRecipientError,\n  TransactionParams,\n  TransactionPrebuild as BaseTransactionPrebuild,\n  TransactionRecipient,\n  TypedData,\n  UnexpectedAddressError,\n  UnsignedTransactionTss,\n  Util,\n  VerifyAddressOptions as BaseVerifyAddressOptions,\n  VerifyTransactionOptions,\n  Wallet,\n  verifyMPCWalletAddress,\n  TssVerifyAddressOptions,\n  isTssVerifyAddressOptions,\n} from '@bitgo-beta/sdk-core';\nimport { getDerivationPath } from '@bitgo-beta/sdk-lib-mpc';\nimport { bip32 } from '@bitgo-beta/secp256k1';\nimport {\n  BaseCoin as StaticsBaseCoin,\n  ChainIdNotFoundError,\n  CoinFeature,\n  coins,\n  EthereumNetwork as EthLikeNetwork,\n  ethGasConfigs,\n} from '@bitgo-beta/statics';\nimport type * as EthLikeCommon from '@ethereumjs/common';\nimport type * as EthLikeTxLib from '@ethereumjs/tx';\nimport { FeeMarketEIP1559Transaction, Transaction as LegacyTransaction } from '@ethereumjs/tx';\nimport { RLP } from '@ethereumjs/rlp';\nimport { SignTypedDataVersion, TypedDataUtils, TypedMessage } from '@metamask/eth-sig-util';\nimport { BigNumber } from 'bignumber.js';\nimport BN from 'bn.js';\nimport { randomBytes } from 'crypto';\nimport debugLib from 'debug';\nimport { addHexPrefix, bufArrToArr, stripHexPrefix, bufferToHex, setLengthLeft, toBuffer } from 'ethereumjs-util';\nimport Keccak from 'keccak';\nimport _ from 'lodash';\nimport secp256k1 from 'secp256k1';\n\nimport { AbstractEthLikeCoin } from './abstractEthLikeCoin';\nimport { EthLikeToken } from './ethLikeToken';\nimport {\n  calculateForwarderV1Address,\n  coinFamiliesWithL1Fees,\n  decodeTransferData,\n  ERC1155TransferBuilder,\n  ERC721TransferBuilder,\n  getBufferedByteCode,\n  getCommon,\n  getCreateForwarderParamsAndTypes,\n  getProxyInitcode,\n  getRawDecoded,\n  getToken,\n  KeyPair as KeyPairLib,\n  TransactionBuilder,\n  TransferBuilder,\n} from './lib';\nimport { SendCrossChainRecoveryOptions } from './types';\n\n/**\n * The prebuilt hop transaction returned from the HSM\n */\ninterface HopPrebuild {\n  tx: string;\n  id: string;\n  signature: string;\n  paymentId: string;\n  gasPrice: number;\n  gasLimit: number;\n  amount: number;\n  recipient: string;\n  nonce: number;\n  userReqSig: string;\n  gasPriceMax: number;\n}\n\n/**\n * The extra parameters to send to platform build route for hop transactions\n */\ninterface HopParams {\n  hopParams: {\n    gasPriceMax: number;\n    userReqSig: string;\n    paymentId: string;\n    gasLimit: number;\n  };\n}\n\nexport interface EIP1559 {\n  maxPriorityFeePerGas: number;\n  maxFeePerGas: number;\n}\n\nexport interface ReplayProtectionOptions {\n  chain: string | number;\n  hardfork: string;\n}\n\nexport interface TransactionPrebuild extends BaseTransactionPrebuild {\n  hopTransaction?: HopPrebuild;\n  buildParams: {\n    recipients: Recipient[];\n  };\n  recipients: TransactionRecipient[];\n  nextContractSequenceId: number;\n  gasPrice: number;\n  gasLimit: number;\n  isBatch: boolean;\n  coin: string;\n  token?: string;\n}\n\nexport interface SignFinalOptions {\n  txPrebuild: {\n    eip1559?: EIP1559;\n    replayProtectionOptions?: ReplayProtectionOptions;\n    gasPrice?: string;\n    gasLimit?: string;\n    recipients?: Recipient[];\n    halfSigned?: {\n      expireTime: number;\n      contractSequenceId: number;\n      backupKeyNonce?: number;\n      signature: string;\n      txHex?: string;\n    };\n    nextContractSequenceId?: number;\n    hopTransaction?: string;\n    backupKeyNonce?: number;\n    isBatch?: boolean;\n    txHex?: string;\n    expireTime?: number;\n  };\n  signingKeyNonce?: number;\n  walletContractAddress?: string;\n  prv: string;\n  recipients?: Recipient[];\n  common?: EthLikeCommon.default;\n}\n\nexport interface SignTransactionOptions extends BaseSignTransactionOptions, SignFinalOptions {\n  isLastSignature?: boolean;\n  expireTime?: number;\n  sequenceId?: number;\n  gasLimit?: number;\n  gasPrice?: number;\n  custodianTransactionId?: string;\n  common?: EthLikeCommon.default;\n  walletVersion?: number;\n}\n\nexport type SignedTransaction = HalfSignedTransaction | FullySignedTransaction;\n\nexport interface FeesUsed {\n  gasPrice: number;\n  gasLimit: number;\n}\n\ninterface PrecreateBitGoOptions {\n  enterprise?: string;\n  newFeeAddress?: string;\n}\n\nexport interface OfflineVaultTxInfo {\n  nextContractSequenceId?: string;\n  contractSequenceId?: string;\n  tx?: string;\n  txHex?: string;\n  userKey?: string;\n  backupKey?: string;\n  coin: string;\n  gasPrice: number;\n  gasLimit: number;\n  recipients: Recipient[];\n  walletContractAddress: string;\n  amount: string;\n  backupKeyNonce: number;\n  // For Eth Specific Coins\n  eip1559?: EIP1559;\n  replayProtectionOptions?: ReplayProtectionOptions;\n  // For Hot Wallet EvmBasedCrossChainRecovery Specific\n  halfSigned?: HalfSignedTransaction;\n  feesUsed?: FeesUsed;\n  isEvmBasedCrossChainRecovery?: boolean;\n  walletVersion?: number;\n}\n\ninterface UnformattedTxInfo {\n  recipient: Recipient;\n}\n\nexport type UnsignedSweepTxMPCv2 = {\n  txRequests: {\n    transactions: [\n      {\n        unsignedTx: UnsignedTransactionTss;\n        nonce: number;\n        signatureShares: [];\n      }\n    ];\n    walletCoin: string;\n  }[];\n};\n\nexport type UnsignedBuilConsolidation = {\n  transactions: MPCSweepTxs[] | UnsignedSweepTxMPCv2[] | RecoveryInfo[] | OfflineVaultTxInfo[];\n  lastScanIndex: number;\n};\n\nexport type RecoverOptionsWithBytes = {\n  isTss: true;\n  /**\n   * @deprecated this is no longer used\n   */\n  openSSLBytes?: Uint8Array;\n};\n\nexport type NonTSSRecoverOptions = {\n  isTss?: false | undefined;\n};\n\nexport type TSSRecoverOptions = RecoverOptionsWithBytes | NonTSSRecoverOptions;\n\nexport type RecoverOptions = {\n  userKey: string;\n  backupKey: string;\n  walletPassphrase?: string;\n  walletContractAddress: string; // use this as walletBaseAddress for TSS\n  recoveryDestination: string;\n  krsProvider?: string;\n  gasPrice?: number;\n  gasLimit?: number;\n  eip1559?: EIP1559;\n  replayProtectionOptions?: ReplayProtectionOptions;\n  bitgoFeeAddress?: string;\n  bitgoDestinationAddress?: string;\n  tokenContractAddress?: string;\n  intendedChain?: string;\n  common?: EthLikeCommon.default;\n  derivationSeed?: string;\n  apiKey?: string; // optional API key to use instead of the one from the environment\n  isUnsignedSweep?: boolean; // specify if this is an unsigned recovery\n} & TSSRecoverOptions;\n\nexport type GetBatchExecutionInfoRT = {\n  values: [string[], string[]];\n  totalAmount: string;\n};\n\nexport interface BuildTransactionParams {\n  to: string;\n  nonce?: number;\n  value: number;\n  data?: Buffer;\n  gasPrice?: number;\n  gasLimit?: number;\n  eip1559?: EIP1559;\n  replayProtectionOptions?: ReplayProtectionOptions;\n}\n\nexport interface RecoveryInfo {\n  id: string;\n  tx: string;\n  backupKey?: string;\n  coin?: string;\n}\n\nexport interface RecoverTokenTransaction {\n  halfSigned: {\n    recipient: Recipient;\n    expireTime: number;\n    contractSequenceId: number;\n    operationHash: string;\n    signature: string;\n    gasLimit: number;\n    gasPrice: number;\n    tokenContractAddress: string;\n    walletId: string;\n  };\n}\n\nexport interface RecoverTokenOptions {\n  tokenContractAddress: string;\n  wallet: Wallet;\n  recipient: string;\n  broadcast?: boolean;\n  walletPassphrase?: string;\n  prv?: string;\n}\n\nexport interface GetSendMethodArgsOptions {\n  recipient: Recipient;\n  expireTime: number;\n  contractSequenceId: number;\n  signature: string;\n}\n\nexport interface SendMethodArgs {\n  name: string;\n  type: string;\n  value: any;\n}\n\ninterface HopTransactionBuildOptions {\n  wallet: Wallet;\n  recipients: Recipient[];\n  walletPassphrase: string;\n}\n\nexport interface BuildOptions {\n  hop?: boolean;\n  wallet?: Wallet;\n  recipients?: Recipient[];\n  walletPassphrase?: string;\n  [index: string]: unknown;\n}\n\ninterface FeeEstimate {\n  gasLimitEstimate: number;\n  feeEstimate: number;\n}\n\n// TODO: This interface will need to be updated for the new fee model introduced in the London Hard Fork\ninterface EthTransactionParams extends TransactionParams {\n  gasPrice?: number;\n  gasLimit?: number;\n  hopParams?: HopParams;\n  hop?: boolean;\n  prebuildTx?: PrebuildTransactionResult;\n  tokenName?: string;\n  feeToken?: string;\n}\n\nexport interface VerifyEthTransactionOptions extends VerifyTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  txParams: EthTransactionParams;\n}\n\ninterface PresignTransactionOptions extends TransactionPrebuild, BasePresignTransactionOptions {\n  wallet: Wallet;\n}\n\ninterface EthAddressCoinSpecifics extends AddressCoinSpecific {\n  forwarderVersion: number;\n  salt?: string;\n  feeAddress?: string;\n}\n\nexport const DEFAULT_SCAN_FACTOR = 20;\nexport interface EthConsolidationRecoveryOptions {\n  coinName?: string;\n  walletContractAddress?: string;\n  apiKey?: string;\n  isTss?: boolean;\n  userKey?: string;\n  backupKey?: string;\n  walletPassphrase?: string;\n  recoveryDestination?: string;\n  krsProvider?: string;\n  gasPrice?: number;\n  gasLimit?: number;\n  eip1559?: EIP1559;\n  replayProtectionOptions?: ReplayProtectionOptions;\n  bitgoFeeAddress?: string;\n  bitgoDestinationAddress?: string;\n  tokenContractAddress?: string;\n  intendedChain?: string;\n  common?: EthLikeCommon.default;\n  derivationSeed?: string;\n  bitgoKey?: string;\n  startingScanIndex?: number;\n  endingScanIndex?: number;\n  ignoreAddressTypes?: unknown;\n}\n\nexport interface VerifyEthAddressOptions extends BaseVerifyAddressOptions {\n  baseAddress: string;\n  coinSpecific: EthAddressCoinSpecifics;\n  forwarderVersion?: number;\n  walletVersion?: number;\n}\n\nexport type TssVerifyEthAddressOptions = TssVerifyAddressOptions & VerifyEthAddressOptions;\n\n/**\n * Keychain with ethAddress for BIP32 wallet verification (V1, V2, V4)\n * Used for wallets that derive addresses using Ethereum addresses from keychains\n */\nexport interface KeychainWithEthAddress {\n  ethAddress: string;\n  pub: string;\n}\n\n/**\n * BIP32 wallet base address verification options\n * Supports V1, V2, and V4 wallets that use ethAddress-based derivation\n */\nexport interface VerifyContractBaseAddressOptions extends VerifyEthAddressOptions {\n  walletVersion: number;\n  keychains: KeychainWithEthAddress[];\n}\n\n/**\n * Type guard to check if params are for BIP32 base address verification (V1, V2, V4)\n * These wallet versions use ethAddress for address derivation\n */\nexport function isVerifyContractBaseAddressOptions(\n  params: VerifyEthAddressOptions | TssVerifyEthAddressOptions\n): params is VerifyContractBaseAddressOptions {\n  return (\n    (params.walletVersion === 1 || params.walletVersion === 2 || params.walletVersion === 4) &&\n    'keychains' in params &&\n    Array.isArray(params.keychains) &&\n    params.keychains.length === 3 &&\n    params.keychains.every((kc: any) => 'ethAddress' in kc && typeof kc.ethAddress === 'string')\n  );\n}\n\nconst debug = debugLib('bitgo:v2:ethlike');\n\nexport const optionalDeps = {\n  get ethAbi() {\n    try {\n      return require('ethereumjs-abi');\n    } catch (e) {\n      debug('unable to load ethereumjs-abi:');\n      debug(e.stack);\n      throw new EthereumLibraryUnavailableError(`ethereumjs-abi`);\n    }\n  },\n\n  get ethUtil() {\n    try {\n      return require('ethereumjs-util');\n    } catch (e) {\n      debug('unable to load ethereumjs-util:');\n      debug(e.stack);\n      throw new EthereumLibraryUnavailableError(`ethereumjs-util`);\n    }\n  },\n\n  get EthTx(): typeof EthLikeTxLib {\n    try {\n      return require('@ethereumjs/tx');\n    } catch (e) {\n      debug('unable to load @ethereumjs/tx');\n      debug(e.stack);\n      throw new EthereumLibraryUnavailableError(`@ethereumjs/tx`);\n    }\n  },\n\n  get EthCommon(): typeof EthLikeCommon {\n    try {\n      return require('@ethereumjs/common');\n    } catch (e) {\n      debug('unable to load @ethereumjs/common:');\n      debug(e.stack);\n      throw new EthereumLibraryUnavailableError(`@ethereumjs/common`);\n    }\n  },\n};\n\nexport abstract class AbstractEthLikeNewCoins extends AbstractEthLikeCoin {\n  static hopTransactionSalt = 'bitgoHopAddressRequestSalt';\n  protected readonly sendMethodName: 'sendMultiSig' | 'sendMultiSigToken';\n\n  readonly staticsCoin?: Readonly<StaticsBaseCoin>;\n\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this.staticsCoin = staticsCoin;\n    this.sendMethodName = 'sendMultiSig';\n  }\n\n  /**\n   * Method to return the coin's network object\n   * @returns {EthLikeNetwork | undefined}\n   */\n  getNetwork(): EthLikeNetwork | undefined {\n    return this.staticsCoin?.network as EthLikeNetwork;\n  }\n\n  /**\n   * Evaluates whether an address string is valid for this coin\n   * @param {string} address\n   * @returns {boolean} True if address is the valid ethlike adderss\n   */\n  isValidAddress(address: string): boolean {\n    return optionalDeps.ethUtil.isValidAddress(optionalDeps.ethUtil.addHexPrefix(address));\n  }\n\n  /**\n   * Flag for sending data along with transactions\n   * @returns {boolean} True if okay to send tx data (ETH), false otherwise\n   */\n  transactionDataAllowed() {\n    return true;\n  }\n\n  /** @inheritDoc */\n  supportsMessageSigning(): boolean {\n    return true;\n  }\n\n  /** @inheritDoc */\n  supportsSigningTypedData(): boolean {\n    return true;\n  }\n\n  /**\n   * Default expire time for a contract call (1 week)\n   * @returns {number} Time in seconds\n   */\n  getDefaultExpireTime(): number {\n    return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;\n  }\n\n  /**\n   * Method to get the custom chain common object based on params from recovery\n   * @param {number} chainId - the chain id of the custom chain\n   * @returns {EthLikeCommon.default}\n   */\n  static getCustomChainCommon(chainId: number): EthLikeCommon.default {\n    const coinName = coins.coinNameFromChainId(chainId);\n    if (!coinName) {\n      throw new ChainIdNotFoundError(chainId);\n    }\n    const coin = coins.get(coinName);\n    const ethLikeCommon = getCommon(coin.network as EthLikeNetwork);\n    return ethLikeCommon;\n  }\n\n  /**\n   * Gets correct Eth Common object based on params from either recovery or tx building\n   * @param {EIP1559} eip1559 - configs that specify whether we should construct an eip1559 tx\n   * @param {ReplayProtectionOptions} replayProtectionOptions - check if chain id supports replay protection\n   * @returns {EthLikeCommon.default}\n   */\n  private static getEthLikeCommon(\n    eip1559?: EIP1559,\n    replayProtectionOptions?: ReplayProtectionOptions\n  ): EthLikeCommon.default {\n    // if eip1559 params are specified, default to london hardfork, otherwise,\n    // default to petersburg to avoid replay protection issues\n    const defaultHardfork = !!eip1559 ? 'london' : optionalDeps.EthCommon.Hardfork.Petersburg;\n    const ethLikeCommon = AbstractEthLikeNewCoins.getCustomChainCommon(replayProtectionOptions?.chain as number);\n    ethLikeCommon.setHardfork(replayProtectionOptions?.hardfork ?? defaultHardfork);\n    return ethLikeCommon;\n  }\n\n  /**\n   * Method to build the tx object\n   * @param {BuildTransactionParams} params - params to build transaction\n   * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}\n   */\n  static buildTransaction(\n    params: BuildTransactionParams\n  ): EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction {\n    // if eip1559 params are specified, default to london hardfork, otherwise,\n    // default to tangerine whistle to avoid replay protection issues\n    const ethLikeCommon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);\n    const baseParams = {\n      to: params.to,\n      nonce: params.nonce,\n      value: params.value,\n      data: params.data,\n      gasLimit: new optionalDeps.ethUtil.BN(params.gasLimit),\n    };\n\n    const unsignedEthTx = !!params.eip1559\n      ? optionalDeps.EthTx.FeeMarketEIP1559Transaction.fromTxData(\n          {\n            ...baseParams,\n            maxFeePerGas: new optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas),\n            maxPriorityFeePerGas: new optionalDeps.ethUtil.BN(params.eip1559.maxPriorityFeePerGas),\n          },\n          { common: ethLikeCommon }\n        )\n      : optionalDeps.EthTx.Transaction.fromTxData(\n          {\n            ...baseParams,\n            gasPrice: new optionalDeps.ethUtil.BN(params.gasPrice),\n          },\n          { common: ethLikeCommon }\n        );\n\n    return unsignedEthTx;\n  }\n\n  /**\n   * Query explorer for the balance of an address\n   * @param {String} address - the ETHLike address\n   * @param {String} apiKey - optional API key to use instead of the one from the environment\n   * @returns {BigNumber} address balance\n   */\n  async queryAddressBalance(address: string, apiKey?: string): Promise<any> {\n    const result = await this.recoveryBlockchainExplorerQuery(\n      {\n        chainid: this.getChainId().toString(),\n        module: 'account',\n        action: 'balance',\n        address: address,\n      },\n      apiKey\n    );\n    // throw if the result does not exist or the result is not a valid number\n    if (!result || !result.result || isNaN(result.result)) {\n      throw new Error(`Could not obtain address balance for ${address} from the explorer, got: ${result.result}`);\n    }\n    return new optionalDeps.ethUtil.BN(result.result, 10);\n  }\n\n  /**\n   * @param {Recipient[]} recipients - the recipients of the transaction\n   * @param {number} expireTime - the expire time of the transaction\n   * @param {number} contractSequenceId - the contract sequence id of the transaction\n   * @returns {string}\n   */\n  getOperationSha3ForExecuteAndConfirm(\n    recipients: Recipient[],\n    expireTime: number,\n    contractSequenceId: number\n  ): string {\n    if (!recipients || !Array.isArray(recipients)) {\n      throw new Error('expecting array of recipients');\n    }\n\n    // Right now we only support 1 recipient\n    if (recipients.length !== 1) {\n      throw new Error('must send to exactly 1 recipient');\n    }\n\n    if (!_.isNumber(expireTime)) {\n      throw new Error('expireTime must be number of seconds since epoch');\n    }\n\n    if (!_.isNumber(contractSequenceId)) {\n      throw new Error('contractSequenceId must be number');\n    }\n\n    // Check inputs\n    recipients.forEach(function (recipient) {\n      if (\n        !_.isString(recipient.address) ||\n        !optionalDeps.ethUtil.isValidAddress(optionalDeps.ethUtil.addHexPrefix(recipient.address))\n      ) {\n        throw new Error('Invalid address: ' + recipient.address);\n      }\n\n      let amount: BigNumber;\n      try {\n        amount = new BigNumber(recipient.amount);\n      } catch (e) {\n        throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');\n      }\n\n      recipient.amount = amount.toFixed(0);\n\n      if (recipient.data && !_.isString(recipient.data)) {\n        throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');\n      }\n    });\n\n    const recipient = recipients[0];\n    return optionalDeps.ethUtil.bufferToHex(\n      optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId))\n    );\n  }\n\n  /**\n   * Get transfer operation for coin\n   * @param {Recipient} recipient - recipient info\n   * @param {number} expireTime - expiry time\n   * @param {number} contractSequenceId - sequence id\n   * @returns {Array} operation array\n   */\n  getOperation(recipient: Recipient, expireTime: number, contractSequenceId: number): (string | Buffer)[][] {\n    const network = this.getNetwork() as EthLikeNetwork;\n    return [\n      ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],\n      [\n        network.nativeCoinOperationHashPrefix,\n        new optionalDeps.ethUtil.BN(optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),\n        recipient.amount,\n        Buffer.from(optionalDeps.ethUtil.stripHexPrefix(optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),\n        expireTime,\n        contractSequenceId,\n      ],\n    ];\n  }\n\n  /**\n   * Queries the contract (via explorer API) for the next sequence ID\n   * @param {String} address - address of the contract\n   * @param {String} apiKey - optional API key to use instead of the one from the environment\n   * @returns {Promise<Number>} sequence ID\n   */\n  async querySequenceId(address: string, apiKey?: string): Promise<number> {\n    // Get sequence ID using contract call\n    const sequenceIdMethodSignature = optionalDeps.ethAbi.methodID('getNextSequenceId', []);\n    const sequenceIdArgs = optionalDeps.ethAbi.rawEncode([], []);\n    const sequenceIdData = Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');\n    const result = await this.recoveryBlockchainExplorerQuery(\n      {\n        chainid: this.getChainId().toString(),\n        module: 'proxy',\n        action: 'eth_call',\n        to: address,\n        data: sequenceIdData,\n        tag: 'latest',\n      },\n      apiKey\n    );\n    if (!result || !result.result) {\n      throw new Error('Could not obtain sequence ID from explorer, got: ' + result.result);\n    }\n    const sequenceIdHex = result.result;\n    return new optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();\n  }\n\n  /**\n   * Recover an unsupported token from a BitGo multisig wallet\n   * This builds a half-signed transaction, for which there will be an admin route to co-sign and broadcast. Optionally\n   * the user can set params.broadcast = true and the half-signed tx will be sent to BitGo for cosigning and broadcasting\n   * @param {RecoverTokenOptions} params\n   * @param {Wallet} params.wallet - the wallet to recover the token from\n   * @param {string} params.tokenContractAddress - the contract address of the unsupported token\n   * @param {string} params.recipient - the destination address recovered tokens should be sent to\n   * @param {string} params.walletPassphrase - the wallet passphrase\n   * @param {string} params.prv - the xprv\n   * @param {boolean} params.broadcast - if true, we will automatically submit the half-signed tx to BitGo for cosigning and broadcasting\n   * @returns {Promise<RecoverTokenTransaction>}\n   */\n  async recoverToken(params: RecoverTokenOptions): Promise<RecoverTokenTransaction> {\n    const network = this.getNetwork() as EthLikeNetwork;\n    if (!_.isObject(params)) {\n      throw new Error(`recoverToken must be passed a params object. Got ${params} (type ${typeof params})`);\n    }\n\n    if (_.isUndefined(params.tokenContractAddress) || !_.isString(params.tokenContractAddress)) {\n      throw new Error(\n        `tokenContractAddress must be a string, got ${\n          params.tokenContractAddress\n        } (type ${typeof params.tokenContractAddress})`\n      );\n    }\n\n    if (!this.isValidAddress(params.tokenContractAddress)) {\n      throw new Error('tokenContractAddress not a valid address');\n    }\n\n    if (_.isUndefined(params.wallet) || !(params.wallet instanceof Wallet)) {\n      throw new Error(`wallet must be a wallet instance, got ${params.wallet} (type ${typeof params.wallet})`);\n    }\n\n    if (_.isUndefined(params.recipient) || !_.isString(params.recipient)) {\n      throw new Error(`recipient must be a string, got ${params.recipient} (type ${typeof params.recipient})`);\n    }\n\n    if (!this.isValidAddress(params.recipient)) {\n      throw new Error('recipient not a valid address');\n    }\n\n    if (!optionalDeps.ethUtil.bufferToHex || !optionalDeps.ethAbi.soliditySHA3) {\n      throw new Error('ethereum not fully supported in this environment');\n    }\n\n    // Get token balance from external API\n    const coinSpecific = params.wallet.coinSpecific();\n    if (!coinSpecific || !_.isString(coinSpecific.baseAddress)) {\n      throw new Error('missing required coin specific property baseAddress');\n    }\n    const recoveryAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, coinSpecific.baseAddress);\n\n    if (params.broadcast) {\n      // We're going to create a normal ETH transaction that sends an amount of 0 ETH to the\n      // tokenContractAddress and encode the unsupported-token-send data in the data field\n      // #tricksy\n      const sendMethodArgs = [\n        {\n          name: '_to',\n          type: 'address',\n          value: params.recipient,\n        },\n        {\n          name: '_value',\n          type: 'uint256',\n          value: recoveryAmount.toString(10),\n        },\n      ];\n      const methodSignature = optionalDeps.ethAbi.methodID('transfer', _.map(sendMethodArgs, 'type'));\n      const encodedArgs = optionalDeps.ethAbi.rawEncode(_.map(sendMethodArgs, 'type'), _.map(sendMethodArgs, 'value'));\n      const sendData = Buffer.concat([methodSignature, encodedArgs]);\n\n      const broadcastParams: any = {\n        address: params.tokenContractAddress,\n        amount: '0',\n        data: sendData.toString('hex'),\n      };\n\n      if (params.walletPassphrase) {\n        broadcastParams.walletPassphrase = params.walletPassphrase;\n      } else if (params.prv) {\n        broadcastParams.prv = params.prv;\n      }\n\n      return await params.wallet.send(broadcastParams);\n    }\n\n    const recipient = {\n      address: params.recipient,\n      amount: recoveryAmount.toString(10),\n    };\n\n    // This signature will be valid for one week\n    const expireTime = Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;\n\n    // Get sequence ID. We do this by building a 'fake' eth transaction, so the platform will increment and return us the new sequence id\n    // This _does_ require the user to have a non-zero wallet balance\n    const { nextContractSequenceId, gasPrice, gasLimit } = (await params.wallet.prebuildTransaction({\n      recipients: [\n        {\n          address: params.recipient,\n          amount: '1',\n        },\n      ],\n    })) as any;\n\n    // these recoveries need to be processed by support, but if the customer sends any transactions before recovery is\n    // complete the sequence ID will be invalid. artificially inflate the sequence ID to allow more time for processing\n    const safeSequenceId = nextContractSequenceId + 1000;\n\n    // Build sendData for ethereum tx\n    const operationTypes = ['string', 'address', 'uint', 'address', 'uint', 'uint'];\n    const operationArgs = [\n      // Token operation has prefix has been added here so that ether operation hashes, signatures cannot be re-used for tokenSending\n      network.tokenOperationHashPrefix,\n      new optionalDeps.ethUtil.BN(optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),\n      recipient.amount,\n      new optionalDeps.ethUtil.BN(optionalDeps.ethUtil.stripHexPrefix(params.tokenContractAddress), 16),\n      expireTime,\n      safeSequenceId,\n    ];\n\n    const operationHash = optionalDeps.ethUtil.bufferToHex(\n      optionalDeps.ethAbi.soliditySHA3(operationTypes, operationArgs)\n    );\n\n    const userPrv = await params.wallet.getPrv({\n      prv: params.prv,\n      walletPassphrase: params.walletPassphrase,\n    });\n\n    const signature = Util.ethSignMsgHash(operationHash, Util.xprvToEthPrivateKey(userPrv));\n\n    return {\n      halfSigned: {\n        recipient: recipient,\n        expireTime: expireTime,\n        contractSequenceId: safeSequenceId,\n        operationHash: operationHash,\n        signature: signature,\n        gasLimit: gasLimit,\n        gasPrice: gasPrice,\n        tokenContractAddress: params.tokenContractAddress,\n        walletId: params.wallet.id(),\n      },\n    };\n  }\n\n  /**\n   * Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key\n   * @param {PrecreateBitGoOptions} params\n   * @param {string} params.enterprise {String} the enterprise id to associate with this key\n   * @param {string} params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)\n   * @returns {void}\n   */\n  preCreateBitGo(params: PrecreateBitGoOptions): void {\n    // We always need params object, since either enterprise or newFeeAddress is required\n    if (!_.isObject(params)) {\n      throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);\n    }\n\n    if (_.isUndefined(params.enterprise) && _.isUndefined(params.newFeeAddress)) {\n      throw new Error(\n        'expecting enterprise when adding BitGo key. If you want to create a new ETH bitgo key, set the newFeeAddress parameter to true.'\n      );\n    }\n\n    // Check whether key should be an enterprise key or a BitGo key for a new fee address\n    if (!_.isUndefined(params.enterprise) && !_.isUndefined(params.newFeeAddress)) {\n      throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);\n    }\n\n    if (!_.isUndefined(params.enterprise) && !_.isString(params.enterprise)) {\n      throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);\n    }\n\n    if (!_.isUndefined(params.newFeeAddress) && !_.isBoolean(params.newFeeAddress)) {\n      throw new Error(\n        `newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`\n      );\n    }\n  }\n\n  /**\n   * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address\n   * @param {string} address\n   * @param {string} apiKey - optional API key to use instead of the one from the environment\n   * @returns {Promise<number>}\n   */\n  async getAddressNonce(address: string, apiKey?: string): Promise<number> {\n    // Get nonce for backup key (should be 0)\n    let nonce = 0;\n\n    const result = await this.recoveryBlockchainExplorerQuery(\n      {\n        chainid: this.getChainId().toString(),\n        module: 'account',\n        action: 'txlist',\n        address,\n      },\n      apiKey\n    );\n\n    if (result && typeof result?.nonce === 'number') {\n      return Number(result.nonce);\n    }\n\n    if (!result || !Array.isArray(result.result)) {\n      throw new Error('Unable to find next nonce from Etherscan, got: ' + JSON.stringify(result));\n    }\n    const backupKeyTxList = result.result;\n    if (backupKeyTxList.length > 0) {\n      // Calculate last nonce used\n      const outgoingTxs = backupKeyTxList.filter((tx) => tx.from === address);\n      nonce = outgoingTxs.length;\n    }\n    return nonce;\n  }\n\n  /**\n   * Helper function for recover()\n   * This transforms the unsigned transaction information into a format the BitGo offline vault expects\n   * @param {UnformattedTxInfo} txInfo - tx info\n   * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object\n   * @param {string} userKey - the user's key\n   * @param {string} backupKey - the backup key\n   * @param {Buffer} gasPrice - gas price for the tx\n   * @param {number} gasLimit - gas limit for the tx\n   * @param {EIP1559} eip1559 - eip1559 params\n   * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options\n   * @param {apiKey} apiKey - optional apiKey to use when retrieving block chain data\n   * @returns {Promise<OfflineVaultTxInfo>}\n   */\n  async formatForOfflineVault(\n    txInfo: UnformattedTxInfo,\n    ethTx: EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction,\n    userKey: string,\n    backupKey: string,\n    gasPrice: Buffer,\n    gasLimit: number,\n    eip1559?: EIP1559,\n    replayProtectionOptions?: ReplayProtectionOptions,\n    apiKey?: string\n  ): Promise<OfflineVaultTxInfo> {\n    if (!ethTx.to) {\n      throw new Error('Eth tx must have a `to` address');\n    }\n    const backupHDNode = bip32.fromBase58(backupKey);\n    const backupSigningKey = backupHDNode.publicKey;\n    const response: OfflineVaultTxInfo = {\n      tx: ethTx.serialize().toString('hex'),\n      userKey,\n      backupKey,\n      coin: this.getChain(),\n      gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n      gasLimit,\n      recipients: [txInfo.recipient],\n      walletContractAddress: ethTx.to.toString(),\n      amount: txInfo.recipient.amount as string,\n      backupKeyNonce: await this.getAddressNonce(\n        `0x${optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`,\n        apiKey\n      ),\n      eip1559,\n      replayProtectionOptions,\n    };\n    _.extend(response, txInfo);\n    response.nextContractSequenceId = response.contractSequenceId;\n    return response;\n  }\n\n  /**\n   * Helper function for recover()\n   * This transforms the unsigned transaction information into a format the BitGo offline vault expects\n   * @param {UnformattedTxInfo} txInfo - tx info\n   * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object\n   * @param {string} userKey - the user's key\n   * @param {string} backupKey - the backup key\n   * @param {Buffer} gasPrice - gas price for the tx\n   * @param {number} gasLimit - gas limit for the tx\n   * @param {number} backupKeyNonce - the nonce of the backup key address\n   * @param {EIP1559} eip1559 - eip1559 params\n   * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options\n   * @returns {Promise<OfflineVaultTxInfo>}\n   */\n  formatForOfflineVaultTSS(\n    txInfo: UnformattedTxInfo,\n    ethTx: EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction,\n    userKey: string,\n    backupKey: string,\n    gasPrice: Buffer,\n    gasLimit: number,\n    backupKeyNonce: number,\n    eip1559?: EIP1559,\n    replayProtectionOptions?: ReplayProtectionOptions\n  ): OfflineVaultTxInfo {\n    if (!ethTx.to) {\n      throw new Error('Eth tx must have a `to` address');\n    }\n    const response: OfflineVaultTxInfo = {\n      tx: ethTx.serialize().toString('hex'),\n      txHex: ethTx.getMessageToSign(false).toString(),\n      userKey,\n      backupKey,\n      coin: this.getChain(),\n      gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n      gasLimit,\n      recipients: [txInfo.recipient],\n      walletContractAddress: ethTx.to.toString(),\n      amount: txInfo.recipient.amount as string,\n      backupKeyNonce: backupKeyNonce,\n      eip1559,\n      replayProtectionOptions,\n    };\n    _.extend(response, txInfo);\n    return response;\n  }\n\n  /**\n   * Check whether the gas price passed in by user are within our max and min bounds\n   * If they are not set, set them to the defaults\n   * @param {number} userGasPrice - user defined gas price\n   * @returns {number} the gas price to use for this transaction\n   */\n  setGasPrice(userGasPrice?: number): number {\n    if (!userGasPrice) {\n      return ethGasConfigs.defaultGasPrice;\n    }\n\n    const gasPriceMax = ethGasConfigs.maximumGasPrice;\n    const gasPriceMin = ethGasConfigs.minimumGasPrice;\n    if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {\n      throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);\n    }\n    return userGasPrice;\n  }\n  /**\n   * Check whether gas limit passed in by user are within our max and min bounds\n   * If they are not set, set them to the defaults\n   * @param {number} userGasLimit user defined gas limit\n   * @returns {number} the gas limit to use for this transaction\n   */\n  setGasLimit(userGasLimit?: number): number {\n    if (!userGasLimit) {\n      return ethGasConfigs.defaultGasLimit;\n    }\n    const gasLimitMax = ethGasConfigs.maximumGasLimit;\n    const gasLimitMin = ethGasConfigs.minimumGasLimit;\n    if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {\n      throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);\n    }\n    return userGasLimit;\n  }\n\n  /**\n   * Helper function for signTransaction for the rare case that SDK is doing the second signature\n   * Note: we are expecting this to be called from the offline vault\n   * @param {SignFinalOptions.txPrebuild} params.txPrebuild\n   * @param {string} params.prv\n   * @returns {{txHex: string}}\n   */\n  async signFinalEthLike(params: SignFinalOptions): Promise<FullySignedTransaction> {\n    const signingKey = new KeyPairLib({ prv: params.prv }).getKeys().prv;\n    if (_.isUndefined(signingKey)) {\n      throw new Error('missing private key');\n    }\n    const txBuilder = this.getTransactionBuilder(params.common);\n    try {\n      txBuilder.from(params.txPrebuild.halfSigned?.txHex);\n    } catch (e) {\n      throw new Error('invalid half-signed transaction');\n    }\n    txBuilder.sign({ key: signingKey });\n    const tx = await txBuilder.build();\n    return {\n      txHex: tx.toBroadcastFormat(),\n    };\n  }\n\n  /**\n   * Assemble half-sign prebuilt transaction\n   * @param {SignTransactionOptions} params\n   */\n  async signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    // Normally the SDK provides the first signature for an EthLike tx, but occasionally it provides the second and final one.\n    if (params.isLastSignature) {\n      // In this case when we're doing the second (final) signature, the logic is different.\n      return await this.signFinalEthLike(params);\n    }\n    const txBuilder = this.getTransactionBuilder(params.common);\n    txBuilder.from(params.txPrebuild.txHex);\n    txBuilder\n      .transfer()\n      .coin(this.staticsCoin?.name as string)\n      .key(new KeyPairLib({ prv: params.prv }).getKeys().prv!);\n    if (params.walletVersion) {\n      txBuilder.walletVersion(params.walletVersion);\n    }\n    const transaction = await txBuilder.build();\n\n    // In case of tx with contract data from a custodial wallet, we are running into an issue\n    // as halfSigned is not having the data field. So, we are adding the data field to the halfSigned tx\n    let recipients = params.txPrebuild.recipients || params.recipients;\n    if (recipients === undefined) {\n      recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));\n    }\n\n    const txParams = {\n      eip1559: params.txPrebuild.eip1559,\n      txHex: transaction.toBroadcastFormat(),\n      recipients: recipients,\n      expiration: params.txPrebuild.expireTime,\n      hopTransaction: params.txPrebuild.hopTransaction,\n      custodianTransactionId: params.custodianTransactionId,\n      expireTime: params.expireTime,\n      contractSequenceId: params.txPrebuild.nextContractSequenceId as number,\n      sequenceId: params.sequenceId,\n      ...(params.txPrebuild.isBatch ? { isBatch: params.txPrebuild.isBatch } : {}),\n    };\n\n    return { halfSigned: txParams };\n  }\n\n  /**\n   * Method to validate recovery params\n   * @param {RecoverOptions} params\n   * @returns {void}\n   */\n  validateRecoveryParams(params: RecoverOptions): void {\n    if (params.userKey === undefined) {\n      throw new Error('missing userKey');\n    }\n\n    if (params.backupKey === undefined) {\n      throw new Error('missing backupKey');\n    }\n\n    if (\n      !params.isUnsignedSweep &&\n      params.walletPassphrase === undefined &&\n      !params.userKey.startsWith('xpub') &&\n      !params.isTss\n    ) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    if (params.walletContractAddress === undefined || !this.isValidAddress(params.walletContractAddress)) {\n      throw new Error('invalid walletContractAddress');\n    }\n\n    if (params.recoveryDestination === undefined || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    if (!this.staticsCoin?.features.includes(CoinFeature.EIP1559)) {\n      if (params.eip1559) {\n        throw new Error('Invalid fee params. EIP1559 not supported');\n      }\n      if (params.replayProtectionOptions?.hardfork === 'london') {\n        throw new Error('Invalid replayProtection options. Cannot use the hardfork \"london\" for this chain');\n      }\n    }\n  }\n\n  /**\n   * Helper which Adds signatures to tx object and re-serializes tx\n   * @param {EthLikeCommon.default} ethCommon\n   * @param {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction} tx\n   * @param {ECDSAMethodTypes.Signature} signature\n   * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}\n   */\n  private getSignedTxFromSignature(\n    ethCommon: EthLikeCommon.default,\n    tx: EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction,\n    signature: ECDSAMethodTypes.Signature\n  ) {\n    // get signed Tx from signature\n    const txData = tx.toJSON();\n    const yParity = signature.recid;\n    const baseParams = {\n      to: txData.to,\n      nonce: new BN(stripHexPrefix(txData.nonce!), 'hex'),\n      value: new BN(stripHexPrefix(txData.value!), 'hex'),\n      gasLimit: new BN(stripHexPrefix(txData.gasLimit!), 'hex'),\n      data: txData.data,\n      r: addHexPrefix(signature.r),\n      s: addHexPrefix(signature.s),\n    };\n\n    let finalTx;\n    if (txData.maxFeePerGas && txData.maxPriorityFeePerGas) {\n      finalTx = FeeMarketEIP1559Transaction.fromTxData(\n        {\n          ...baseParams,\n          maxPriorityFeePerGas: new BN(stripHexPrefix(txData.maxPriorityFeePerGas!), 'hex'),\n          maxFeePerGas: new BN(stripHexPrefix(txData.maxFeePerGas!), 'hex'),\n          v: new BN(yParity.toString()),\n        },\n        { common: ethCommon }\n      );\n    } else if (txData.gasPrice) {\n      const v = BigInt(35) + BigInt(yParity) + BigInt(ethCommon.chainIdBN().toNumber()) * BigInt(2);\n      finalTx = LegacyTransaction.fromTxData(\n        {\n          ...baseParams,\n          v: new BN(v.toString()),\n          gasPrice: new BN(stripHexPrefix(txData.gasPrice!.toString()), 'hex'),\n        },\n        { common: ethCommon }\n      );\n    }\n\n    return finalTx;\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo\n   * @param params\n   * @param {string} params.userKey - [encrypted] xprv\n   * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider\n   * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey\n   * @param {string} params.walletContractAddress - the ETH address of the wallet contract\n   * @param {string} params.krsProvider - necessary if backup key is held by KRS\n   * @param {string} params.recoveryDestination - target address to send recovered funds to\n   * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn\n   * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn\n   */\n  async recover(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo | UnsignedSweepTxMPCv2> {\n    if (params.isTss === true) {\n      return this.recoverTSS(params);\n    }\n    return this.recoverEthLike(params);\n  }\n\n  generateForwarderAddress(\n    baseAddress: string,\n    feeAddress: string,\n    forwarderFactoryAddress: string,\n    forwarderImplementationAddress: string,\n    index: number\n  ): string {\n    const salt = addHexPrefix(index.toString(16));\n    const saltBuffer = setLengthLeft(toBuffer(salt), 32);\n\n    const { createForwarderParams, createForwarderTypes } = getCreateForwarderParamsAndTypes(\n      baseAddress,\n      saltBuffer,\n      feeAddress\n    );\n\n    const calculationSalt = bufferToHex(optionalDeps.ethAbi.soliditySHA3(createForwarderTypes, createForwarderParams));\n\n    const initCode = getProxyInitcode(forwarderImplementationAddress);\n    return calculateForwarderV1Address(forwarderFactoryAddress, calculationSalt, initCode);\n  }\n\n  deriveAddressFromPublicKey(commonKeychain: string, index: number): string {\n    const derivationPath = `m/${index}`;\n    const pubkeySize = 33;\n\n    const ecdsaMpc = new Ecdsa();\n    const derivedPublicKey = Buffer.from(ecdsaMpc.deriveUnhardened(commonKeychain, derivationPath), 'hex')\n      .subarray(0, pubkeySize)\n      .toString('hex');\n\n    const publicKey = Buffer.from(derivedPublicKey, 'hex').slice(0, 66).toString('hex');\n\n    const keyPair = new KeyPairLib({ pub: publicKey });\n    const address = keyPair.getAddress();\n    return address;\n  }\n\n  getConsolidationAddress(params: EthConsolidationRecoveryOptions, index: number): string[] {\n    const possibleConsolidationAddresses: string[] = [];\n    if (params.walletContractAddress && params.bitgoFeeAddress) {\n      const ethNetwork = this.getNetwork();\n      const forwarderFactoryAddress = ethNetwork?.walletV4ForwarderFactoryAddress as string;\n      const forwarderImplementationAddress = ethNetwork?.walletV4ForwarderImplementationAddress as string;\n      try {\n        const forwarderAddress = this.generateForwarderAddress(\n          params.walletContractAddress,\n          params.bitgoFeeAddress,\n          forwarderFactoryAddress,\n          forwarderImplementationAddress,\n          index\n        );\n        possibleConsolidationAddresses.push(forwarderAddress);\n      } catch (e) {\n        console.log(`Failed to generate forwarder address: ${e.message}`);\n      }\n    }\n\n    if (params.userKey) {\n      try {\n        const derivedAddress = this.deriveAddressFromPublicKey(params.userKey, index);\n        possibleConsolidationAddresses.push(derivedAddress);\n      } catch (e) {\n        console.log(`Failed to generate derived address: ${e}`);\n      }\n    }\n\n    if (possibleConsolidationAddresses.length === 0) {\n      throw new Error(\n        'Unable to generate consolidation address. Check that wallet contract address, fee address, or user key is valid.'\n      );\n    }\n    return possibleConsolidationAddresses;\n  }\n\n  async recoverConsolidations(params: EthConsolidationRecoveryOptions): Promise<UnsignedBuilConsolidation> {\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n    const startIdx = params.startingScanIndex || 1;\n    const endIdx = params.endingScanIndex || startIdx + DEFAULT_SCAN_FACTOR;\n\n    if (!params.walletContractAddress || params.walletContractAddress === '') {\n      throw new Error(`Invalid wallet contract address ${params.walletContractAddress}`);\n    }\n\n    if (!params.bitgoFeeAddress || params.bitgoFeeAddress === '') {\n      throw new Error(`Invalid fee address ${params.bitgoFeeAddress}`);\n    }\n\n    if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * DEFAULT_SCAN_FACTOR) {\n      throw new Error(\n        `Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`\n      );\n    }\n\n    const consolidatedTransactions: any[] = [];\n    let lastScanIndex = startIdx;\n\n    for (let i = startIdx; i < endIdx; i++) {\n      const consolidationAddress = this.getConsolidationAddress(params, i);\n      for (const address of consolidationAddress) {\n        const recoverParams = {\n          apiKey: params.apiKey,\n          backupKey: params.backupKey || '',\n          gasLimit: params.gasLimit,\n          recoveryDestination: params.recoveryDestination || '',\n          userKey: params.userKey || '',\n          walletContractAddress: address,\n          derivationSeed: '',\n          isTss: params.isTss,\n          eip1559: {\n            maxFeePerGas: params.eip1559?.maxFeePerGas || 20,\n            maxPriorityFeePerGas: params.eip1559?.maxPriorityFeePerGas || 200000,\n          },\n          replayProtectionOptions: {\n            chain: params.replayProtectionOptions?.chain || 0,\n            hardfork: params.replayProtectionOptions?.hardfork || 'london',\n          },\n          bitgoKey: '',\n          ignoreAddressTypes: [],\n        };\n        let recoveryTransaction;\n        try {\n          recoveryTransaction = await this.recover(recoverParams);\n        } catch (e) {\n          if (\n            e.message === 'Did not find address with funds to recover' ||\n            e.message === 'Did not find token account to recover tokens, please check token account' ||\n            e.message === 'Not enough token funds to recover'\n          ) {\n            lastScanIndex = i;\n            continue;\n          }\n          throw e;\n        }\n        if (isUnsignedSweep) {\n          consolidatedTransactions.push((recoveryTransaction as MPCSweepTxs).txRequests[0]);\n        } else {\n          consolidatedTransactions.push(recoveryTransaction);\n        }\n      }\n      // To avoid rate limit for etherscan\n      await new Promise((resolve) => setTimeout(resolve, 1000));\n      // lastScanIndex = i;\n    }\n\n    if (consolidatedTransactions.length === 0) {\n      throw new Error(\n        `Did not find an address with sufficient funds to recover. Please start the next scan at address index ${\n          lastScanIndex + 1\n        }.`\n      );\n    }\n    return { transactions: consolidatedTransactions, lastScanIndex };\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo for non-TSS transaction\n   * @param params\n   * @param {string} params.userKey [encrypted] xprv or xpub\n   * @param {string} params.backupKey [encrypted] xprv or xpub if the xprv is held by a KRS provider\n   * @param {string} params.walletPassphrase used to decrypt userKey and backupKey\n   * @param {string} params.walletContractAddress the EthLike address of the wallet contract\n   * @param {string} params.krsProvider necessary if backup key is held by KRS\n   * @param {string} params.recoveryDestination target address to send recovered funds to\n   * @param {string} params.bitgoFeeAddress wrong chain wallet fee address for evm based cross chain recovery txn\n   * @param {string} params.bitgoDestinationAddress target bitgo address where fee will be sent for evm based cross chain recovery txn\n   * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}\n   */\n  protected async recoverEthLike(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    // bitgoFeeAddress is only defined when it is a evm cross chain recovery\n    // as we use fee from this wrong chain address for the recovery txn on the correct chain.\n    if (params.bitgoFeeAddress) {\n      return this.recoverEthLikeforEvmBasedRecovery(params);\n    }\n\n    this.validateRecoveryParams(params);\n    const isUnsignedSweep = params.isUnsignedSweep ?? getIsUnsignedSweep(params);\n\n    // Clean up whitespace from entered values\n    let userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n    const gasLimit = new optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));\n    const gasPrice = params.eip1559\n      ? new optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)\n      : new optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));\n\n    if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {\n      try {\n        userKey = this.bitgo.decrypt({\n          input: userKey,\n          password: params.walletPassphrase,\n        });\n      } catch (e) {\n        throw new Error(`Error decrypting user keychain: ${e.message}`);\n      }\n    }\n    let backupKeyAddress;\n    let backupSigningKey;\n    if (isUnsignedSweep) {\n      const backupHDNode = bip32.fromBase58(backupKey);\n      backupSigningKey = backupHDNode.publicKey;\n      backupKeyAddress = `0x${optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`;\n    } else {\n      // Decrypt backup private key and get address\n      let backupPrv;\n\n      try {\n        backupPrv = this.bitgo.decrypt({\n          input: backupKey,\n          password: params.walletPassphrase,\n        });\n      } catch (e) {\n        throw new Error(`Error decrypting backup keychain: ${e.message}`);\n      }\n\n      const keyPair = new KeyPairLib({ prv: backupPrv });\n      backupSigningKey = keyPair.getKeys().prv;\n      if (!backupSigningKey) {\n        throw new Error('no private key');\n      }\n      backupKeyAddress = keyPair.getAddress();\n    }\n\n    const backupKeyNonce = await this.getAddressNonce(backupKeyAddress, params.apiKey);\n    // get balance of backupKey to ensure funds are available to pay fees\n    const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress, params.apiKey);\n    let totalGasNeeded = gasPrice.mul(gasLimit);\n\n    // On L2 chains with L1 data fees, add buffer for L1 fees\n    if (this.staticsCoin?.family !== undefined && coinFamiliesWithL1Fees.includes(this.staticsCoin.family)) {\n      totalGasNeeded = totalGasNeeded.add(new optionalDeps.ethUtil.BN(ethGasConfigs.l1GasFeeBuffer));\n    }\n\n    const weiToGwei = 10 ** 9;\n    if (backupKeyBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +\n          `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +\n          ` Gwei to perform recoveries. Try sending some funds to this address then retry.`\n      );\n    }\n\n    // get balance of wallet\n    const txAmount = await this.queryAddressBalance(params.walletContractAddress, params.apiKey);\n    if (new BigNumber(txAmount).isLessThanOrEqualTo(0)) {\n      throw new Error('Wallet does not have enough funds to recover');\n    }\n\n    // build recipients object\n    const recipients = [\n      {\n        address: params.recoveryDestination,\n        amount: txAmount.toString(10),\n      },\n    ];\n\n    // Get sequence ID using contract call\n    // we need to wait between making two explorer api calls to avoid getting banned\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const sequenceId = await this.querySequenceId(params.walletContractAddress, params.apiKey);\n\n    let operationHash, signature;\n    // Get operation hash and sign it\n    if (!isUnsignedSweep) {\n      operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);\n      signature = Util.ethSignMsgHash(operationHash, Util.xprvToEthPrivateKey(userKey));\n\n      try {\n        Util.ecRecoverEthAddress(operationHash, signature);\n      } catch (e) {\n        throw new Error('Invalid signature');\n      }\n    }\n\n    const txInfo = {\n      recipient: recipients[0],\n      expireTime: this.getDefaultExpireTime(),\n      contractSequenceId: sequenceId,\n      operationHash: operationHash,\n      signature: signature,\n      gasLimit: gasLimit.toString(10),\n    };\n\n    const txBuilder = this.getTransactionBuilder(params.common) as TransactionBuilder;\n    txBuilder.counter(backupKeyNonce);\n    txBuilder.contract(params.walletContractAddress);\n    let txFee;\n    if (params.eip1559) {\n      txFee = {\n        eip1559: {\n          maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,\n          maxFeePerGas: params.eip1559.maxFeePerGas,\n        },\n      };\n    } else {\n      txFee = { fee: gasPrice.toString() };\n    }\n    txBuilder.fee({\n      ...txFee,\n      gasLimit: gasLimit.toString(),\n    });\n    const transferBuilder = txBuilder.transfer() as TransferBuilder;\n    transferBuilder\n      .coin(this.staticsCoin?.name as string)\n      .amount(recipients[0].amount)\n      .contractSequenceId(sequenceId)\n      .expirationTime(this.getDefaultExpireTime())\n      .to(params.recoveryDestination);\n\n    const tx = await txBuilder.build();\n    if (isUnsignedSweep) {\n      const response: OfflineVaultTxInfo = {\n        txHex: tx.toBroadcastFormat(),\n        userKey,\n        backupKey,\n        coin: this.getChain(),\n        gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n        gasLimit,\n        recipients: [txInfo.recipient],\n        walletContractAddress: tx.toJson().to,\n        amount: txInfo.recipient.amount,\n        backupKeyNonce,\n        eip1559: params.eip1559,\n      };\n      _.extend(response, txInfo);\n      response.nextContractSequenceId = response.contractSequenceId;\n      return response;\n    }\n\n    txBuilder\n      .transfer()\n      .coin(this.staticsCoin?.name as string)\n      .key(new KeyPairLib({ prv: userKey }).getKeys().prv as string);\n    txBuilder.sign({ key: backupSigningKey });\n\n    const signedTx = await txBuilder.build();\n\n    return {\n      id: signedTx.toJson().id,\n      tx: signedTx.toBroadcastFormat(),\n    };\n  }\n\n  /**\n   * Extract recipients from transaction hex\n   * @param txHex - The transaction hex string\n   * @returns Array of recipients with address and amount\n   */\n  private async extractRecipientsFromTxHex(txHex: string): Promise<Array<{ address: string; amount: string }>> {\n    const txBuffer = optionalDeps.ethUtil.toBuffer(txHex);\n    const decodedTx = optionalDeps.EthTx.TransactionFactory.fromSerializedData(txBuffer);\n    const recipients: Array<{ address: string; amount: string }> = [];\n\n    if (decodedTx.data && decodedTx.data.length > 0) {\n      const dataHex = optionalDeps.ethUtil.bufferToHex(decodedTx.data);\n      const transferData = decodeTransferData(dataHex);\n      if (transferData.to) {\n        recipients.push({\n          address: transferData.to,\n          amount: transferData.amount,\n        });\n      }\n    }\n\n    return recipients;\n  }\n\n  async sendCrossChainRecoveryTransaction(\n    params: SendCrossChainRecoveryOptions\n  ): Promise<{ coin: string; txHex?: string; txid: string }> {\n    const buildResponse = await this.buildCrossChainRecoveryTransaction(params.recoveryId);\n    if (params.walletType === 'cold') {\n      return buildResponse;\n    }\n    if (!params.encryptedPrv) {\n      throw new Error('missing encryptedPrv');\n    }\n\n    let userKeyPrv;\n    try {\n      userKeyPrv = this.bitgo.decrypt({\n        input: params.encryptedPrv,\n        password: params.walletPassphrase,\n      });\n    } catch (e) {\n      throw new Error(`Error decrypting user keychain: ${e.message}`);\n    }\n    const keyPair = new KeyPairLib({ prv: userKeyPrv });\n    const userSigningKey = keyPair.getKeys().prv;\n    if (!userSigningKey) {\n      throw new Error('no private key');\n    }\n\n    const txBuilder = this.getTransactionBuilder(params.common) as TransactionBuilder;\n    const txHex = buildResponse.txHex;\n    txBuilder.from(txHex);\n    if (buildResponse.walletVersion) {\n      // If walletVersion is provided, set it in the txBuilder\n      txBuilder.walletVersion(buildResponse.walletVersion);\n    }\n    txBuilder\n      .transfer()\n      .coin(this.staticsCoin?.name as string)\n      .key(userSigningKey);\n    const tx = await txBuilder.build();\n    const res = await this.bitgo\n      .post(this.bitgo.microservicesUrl(`/api/recovery/v1/crosschain/${params.recoveryId}/sign`))\n      .send({ txHex: tx.toBroadcastFormat() });\n    return {\n      coin: this.staticsCoin?.name as string,\n      txid: res.body.txid,\n    };\n  }\n\n  async buildCrossChainRecoveryTransaction(recoveryId: string): Promise<{\n    coin: string;\n    txHex: string;\n    txid: string;\n    walletVersion?: number;\n    recipients: Array<{ address: string; amount: string }>;\n  }> {\n    const res = await this.bitgo.get(this.bitgo.microservicesUrl(`/api/recovery/v1/crosschain/${recoveryId}/buildtx`));\n    // Extract recipients from the transaction hex\n    const recipients = await this.extractRecipientsFromTxHex(res.body.txHex);\n    return {\n      coin: res.body.coin,\n      txHex: res.body.txHex,\n      txid: res.body.txid,\n      walletVersion: res.body.walletVersion,\n      recipients,\n    };\n  }\n\n  /**\n   * Builds a unsigned (for cold, custody wallet) or\n   * half-signed (for hot wallet) evm cross chain recovery transaction with\n   * same expected arguments as recover method.\n   * This helps recover funds from evm based wrong chain.\n   * @param {RecoverOptions} params\n   * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}\n   */\n  protected async recoverEthLikeforEvmBasedRecovery(\n    params: RecoverOptions\n  ): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    this.validateEvmBasedRecoveryParams(params);\n\n    // Clean up whitespace from entered values\n    const userKey = params.userKey.replace(/\\s/g, '');\n    const bitgoFeeAddress = params.bitgoFeeAddress?.replace(/\\s/g, '').toLowerCase() as string;\n    const bitgoDestinationAddress = params.bitgoDestinationAddress?.replace(/\\s/g, '').toLowerCase() as string;\n    const recoveryDestination = params.recoveryDestination?.replace(/\\s/g, '').toLowerCase() as string;\n    const walletContractAddress = params.walletContractAddress?.replace(/\\s/g, '').toLowerCase() as string;\n    const tokenContractAddress = params.tokenContractAddress?.replace(/\\s/g, '').toLowerCase() as string;\n\n    let userSigningKey;\n    let userKeyPrv;\n    if (params.walletPassphrase) {\n      if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {\n        try {\n          userKeyPrv = this.bitgo.decrypt({\n            input: userKey,\n            password: params.walletPassphrase,\n          });\n        } catch (e) {\n          throw new Error(`Error decrypting user keychain: ${e.message}`);\n        }\n      }\n\n      const keyPair = new KeyPairLib({ prv: userKeyPrv });\n      userSigningKey = keyPair.getKeys().prv;\n      if (!userSigningKey) {\n        throw new Error('no private key');\n      }\n    }\n\n    // Use default gasLimit for cold and custody wallets\n    let gasLimit =\n      params.gasLimit || userKey.startsWith('xpub') || !userKey\n        ? new optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit))\n        : new optionalDeps.ethUtil.BN(0);\n\n    const gasPrice = params.eip1559\n      ? new optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)\n      : params.gasPrice\n      ? new optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice))\n      : await this.getGasPriceFromExternalAPI(this.staticsCoin?.name as string, params.apiKey);\n\n    const bitgoFeeAddressNonce = await this.getAddressNonce(bitgoFeeAddress, params.apiKey);\n\n    if (tokenContractAddress) {\n      return this.recoverEthLikeTokenforEvmBasedRecovery(\n        params,\n        bitgoFeeAddressNonce,\n        gasLimit,\n        gasPrice,\n        userKey,\n        userSigningKey,\n        params.apiKey\n      );\n    }\n\n    // get balance of wallet\n    const txAmount = await this.queryAddressBalance(walletContractAddress, params.apiKey);\n\n    const bitgoFeePercentage = 0; // TODO: BG-71912 can change the fee% here.\n    const bitgoFeeAmount = txAmount * (bitgoFeePercentage / 100);\n\n    // build recipients object\n    const recipients: Recipient[] = [\n      {\n        address: recoveryDestination,\n        amount: new BigNumber(txAmount).minus(bitgoFeeAmount).toFixed(),\n      },\n    ];\n\n    if (bitgoFeePercentage > 0) {\n      if (_.isUndefined(bitgoDestinationAddress) || !this.isValidAddress(bitgoDestinationAddress)) {\n        throw new Error('invalid bitgoDestinationAddress');\n      }\n\n      recipients.push({\n        address: bitgoDestinationAddress,\n        amount: bitgoFeeAmount.toString(10),\n      });\n    }\n\n    // calculate batch data\n    const BATCH_METHOD_NAME = 'batch';\n    const BATCH_METHOD_TYPES = ['address[]', 'uint256[]'];\n    const batchExecutionInfo = this.getBatchExecutionInfo(recipients);\n    const batchData = optionalDeps.ethUtil.addHexPrefix(\n      this.getMethodCallData(BATCH_METHOD_NAME, BATCH_METHOD_TYPES, batchExecutionInfo.values).toString('hex')\n    );\n\n    // Get sequence ID using contract call\n    // we need to wait between making two explorer api calls to avoid getting banned\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const sequenceId = await this.querySequenceId(walletContractAddress, params.apiKey);\n\n    const network = this.getNetwork();\n    const batcherContractAddress = network?.batcherContractAddress as string;\n\n    const txBuilder = this.getTransactionBuilder(params.common) as TransactionBuilder;\n    txBuilder.counter(bitgoFeeAddressNonce);\n    txBuilder.contract(walletContractAddress);\n    let txFee;\n    if (params.eip1559) {\n      txFee = {\n        eip1559: {\n          maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,\n          maxFeePerGas: params.eip1559.maxFeePerGas,\n        },\n      };\n    } else {\n      txFee = { fee: gasPrice.toString() };\n    }\n    txBuilder.fee({\n      ...txFee,\n      gasLimit: gasLimit.toString(),\n    });\n\n    const transferBuilder = txBuilder.transfer() as TransferBuilder;\n    if (!batcherContractAddress) {\n      transferBuilder\n        .coin(this.staticsCoin?.name as string)\n        .amount(batchExecutionInfo.totalAmount)\n        .contractSequenceId(sequenceId)\n        .expirationTime(this.getDefaultExpireTime())\n        .to(recoveryDestination);\n    } else {\n      transferBuilder\n        .coin(this.staticsCoin?.name as string)\n        .amount(batchExecutionInfo.totalAmount)\n        .contractSequenceId(sequenceId)\n        .expirationTime(this.getDefaultExpireTime())\n        .to(batcherContractAddress)\n        .data(batchData);\n    }\n\n    if (params.walletPassphrase) {\n      transferBuilder.key(userSigningKey);\n    }\n\n    // If the intended chain is arbitrum or optimism, we need to use wallet version 4\n    // since these contracts construct operationHash differently\n    if (params.intendedChain && ['arbeth', 'opeth'].includes(coins.get(params.intendedChain).family)) {\n      txBuilder.walletVersion(4);\n    }\n\n    // If gasLimit was not passed as a param or if it is not cold/custody wallet, then fetch the gasLimit from Explorer\n    if (!params.gasLimit && userKey && !userKey.startsWith('xpub')) {\n      const sendData = txBuilder.getSendData();\n      gasLimit = await this.getGasLimitFromExternalAPI(\n        params.intendedChain as string,\n        params.bitgoFeeAddress as string,\n        params.walletContractAddress,\n        sendData,\n        params.apiKey\n      );\n      txBuilder.fee({\n        ...txFee,\n        gasLimit: gasLimit.toString(),\n      });\n    }\n\n    // Get the balance of bitgoFeeAddress to ensure funds are available to pay fees\n    await this.ensureSufficientBalance(bitgoFeeAddress, gasPrice, gasLimit, params.apiKey);\n\n    const tx = await txBuilder.build();\n\n    const txInfo = {\n      recipients: recipients,\n      expireTime: this.getDefaultExpireTime(),\n      contractSequenceId: sequenceId,\n      gasLimit: gasLimit.toString(10),\n      isEvmBasedCrossChainRecovery: true,\n    };\n\n    const response: OfflineVaultTxInfo = {\n      txHex: tx.toBroadcastFormat(),\n      userKey,\n      coin: this.getChain(),\n      gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n      gasLimit,\n      recipients: txInfo.recipients,\n      walletContractAddress: tx.toJson().to,\n      amount: batchExecutionInfo.totalAmount,\n      backupKeyNonce: bitgoFeeAddressNonce,\n      eip1559: params.eip1559,\n      ...(txBuilder.getWalletVersion() === 4 ? { walletVersion: txBuilder.getWalletVersion() } : {}),\n    };\n    _.extend(response, txInfo);\n    response.nextContractSequenceId = response.contractSequenceId;\n\n    if (params.walletPassphrase) {\n      const halfSignedTxn: HalfSignedTransaction = {\n        halfSigned: {\n          txHex: tx.toBroadcastFormat(),\n          recipients: txInfo.recipients,\n          expireTime: txInfo.expireTime,\n        },\n      };\n      _.extend(response, halfSignedTxn);\n\n      const feesUsed: FeesUsed = {\n        gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n        gasLimit: optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),\n      };\n      response['feesUsed'] = feesUsed;\n    }\n\n    return response;\n  }\n\n  /**\n   * Query explorer for the balance of an address for a token\n   * @param {string} tokenContractAddress - address where the token smart contract is hosted\n   * @param {string} walletContractAddress - address of the wallet\n   * @param {string} apiKey - optional API key to use instead of the one from the environment\n   * @returns {BigNumber} token balaance in base units\n   */\n  async queryAddressTokenBalance(\n    tokenContractAddress: string,\n    walletContractAddress: string,\n    apiKey?: string\n  ): Promise<any> {\n    if (!optionalDeps.ethUtil.isValidAddress(tokenContractAddress)) {\n      throw new Error('cannot get balance for invalid token address');\n    }\n    if (!optionalDeps.ethUtil.isValidAddress(walletContractAddress)) {\n      throw new Error('cannot get token balance for invalid wallet address');\n    }\n\n    const result = await this.recoveryBlockchainExplorerQuery(\n      {\n        chainid: this.getChainId().toString(),\n        module: 'account',\n        action: 'tokenbalance',\n        contractaddress: tokenContractAddress,\n        address: walletContractAddress,\n        tag: 'latest',\n      },\n      apiKey\n    );\n    // throw if the result does not exist or the result is not a valid number\n    if (!result || !result.result || isNaN(result.result)) {\n      throw new Error(\n        `Could not obtain token address balance for ${tokenContractAddress} from Etherscan, got: ${result.result}`\n      );\n    }\n    return new optionalDeps.ethUtil.BN(result.result, 10);\n  }\n\n  async recoverEthLikeTokenforEvmBasedRecovery(\n    params: RecoverOptions,\n    bitgoFeeAddressNonce: number,\n    gasLimit,\n    gasPrice,\n    userKey,\n    userSigningKey,\n    apiKey?: string\n  ) {\n    // get token balance of wallet\n    const txAmount = await this.queryAddressTokenBalance(\n      params.tokenContractAddress as string,\n      params.walletContractAddress,\n      apiKey\n    );\n\n    // build recipients object\n    const recipients: Recipient[] = [\n      {\n        address: params.recoveryDestination,\n        amount: new BigNumber(txAmount).toFixed(),\n      },\n    ];\n\n    // Get sequence ID using contract call\n    // we need to wait between making two explorer api calls to avoid getting banned\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const sequenceId = await this.querySequenceId(params.walletContractAddress, params.apiKey);\n\n    const txBuilder = this.getTransactionBuilder(params.common) as TransactionBuilder;\n    txBuilder.counter(bitgoFeeAddressNonce);\n    txBuilder.contract(params.walletContractAddress as string);\n    let txFee;\n    if (params.eip1559) {\n      txFee = {\n        eip1559: {\n          maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,\n          maxFeePerGas: params.eip1559.maxFeePerGas,\n        },\n      };\n    } else {\n      txFee = { fee: gasPrice.toString() };\n    }\n    txBuilder.fee({\n      ...txFee,\n      gasLimit: gasLimit.toString(),\n    });\n\n    const transferBuilder = txBuilder.transfer() as TransferBuilder;\n\n    const network = this.getNetwork();\n    const token = getToken(\n      params.tokenContractAddress as string,\n      network as EthLikeNetwork,\n      this.staticsCoin?.family as string\n    )?.name as string;\n\n    transferBuilder\n      .amount(txAmount)\n      .contractSequenceId(sequenceId)\n      .expirationTime(this.getDefaultExpireTime())\n      .to(params.recoveryDestination);\n\n    if (token) {\n      transferBuilder.coin(token);\n    } else {\n      transferBuilder\n        .coin(this.staticsCoin?.name as string)\n        .tokenContractAddress(params.tokenContractAddress as string);\n    }\n\n    if (params.walletPassphrase) {\n      txBuilder.transfer().key(userSigningKey);\n    }\n    // If the intended chain is arbitrum or optimism, we need to use wallet version 4\n    // since these contracts construct operationHash differently\n    if (params.intendedChain && ['arbeth', 'opeth'].includes(coins.get(params.intendedChain).family)) {\n      txBuilder.walletVersion(4);\n    }\n\n    if (!params.gasLimit && userKey && !userKey.startsWith('xpub')) {\n      const sendData = txBuilder.getSendData();\n      gasLimit = await this.getGasLimitFromExternalAPI(\n        params.intendedChain as string,\n        params.bitgoFeeAddress as string,\n        params.walletContractAddress,\n        sendData,\n        apiKey\n      );\n      txBuilder.fee({\n        ...txFee,\n        gasLimit: gasLimit.toString(),\n      });\n    }\n\n    // Get the balance of bitgoFeeAddress to ensure funds are available to pay fees\n    await this.ensureSufficientBalance(params.bitgoFeeAddress as string, gasPrice, gasLimit, params.apiKey);\n\n    const tx = await txBuilder.build();\n\n    const txInfo = {\n      recipients: recipients,\n      expireTime: this.getDefaultExpireTime(),\n      contractSequenceId: sequenceId,\n      gasLimit: gasLimit.toString(10),\n      isEvmBasedCrossChainRecovery: true,\n    };\n\n    const response: OfflineVaultTxInfo = {\n      txHex: tx.toBroadcastFormat(),\n      userKey,\n      coin: token ? token : this.getChain(),\n      gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n      gasLimit,\n      recipients: txInfo.recipients,\n      walletContractAddress: tx.toJson().to,\n      amount: txAmount.toString(),\n      backupKeyNonce: bitgoFeeAddressNonce,\n      eip1559: params.eip1559,\n      ...(txBuilder.getWalletVersion() === 4 ? { walletVersion: txBuilder.getWalletVersion() } : {}),\n    };\n    _.extend(response, txInfo);\n    response.nextContractSequenceId = response.contractSequenceId;\n\n    if (params.walletPassphrase) {\n      const halfSignedTxn: HalfSignedTransaction = {\n        halfSigned: {\n          txHex: tx.toBroadcastFormat(),\n          recipients: txInfo.recipients,\n          expireTime: txInfo.expireTime,\n        },\n      };\n      _.extend(response, halfSignedTxn);\n\n      const feesUsed: FeesUsed = {\n        gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n        gasLimit: optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),\n      };\n      response['feesUsed'] = feesUsed;\n    }\n\n    return response;\n  }\n\n  /**\n   * Validate evm based cross chain recovery params\n   * @param params {RecoverOptions}\n   * @returns {void}\n   */\n  validateEvmBasedRecoveryParams(params: RecoverOptions): void {\n    if (_.isUndefined(params.bitgoFeeAddress) || !this.isValidAddress(params.bitgoFeeAddress)) {\n      throw new Error('invalid bitgoFeeAddress');\n    }\n\n    if (_.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {\n      throw new Error('invalid walletContractAddress');\n    }\n\n    if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n  }\n\n  /**\n   * Return types, values, and total amount in wei to send in a batch transaction, using the method signature\n   * `distributeBatch(address[], uint256[])`\n   * @param {Recipient[]} recipients - transaction recipients\n   * @returns {GetBatchExecutionInfoRT} information needed to execute the batch transaction\n   */\n  getBatchExecutionInfo(recipients: Recipient[]): GetBatchExecutionInfoRT {\n    const addresses: string[] = [];\n    const amounts: string[] = [];\n    let sum = new BigNumber('0');\n    _.forEach(recipients, ({ address, amount }) => {\n      addresses.push(address);\n      amounts.push(amount as string);\n      sum = sum.plus(amount);\n    });\n\n    return {\n      values: [addresses, amounts],\n      totalAmount: sum.toFixed(),\n    };\n  }\n\n  /**\n   * Get the data required to make an ETH function call defined by the given types and values\n   *\n   * @param {string} functionName - The name of the function being called, e.g. transfer\n   * @param types The types of the function call in order\n   * @param values The values of the function call in order\n   * @return {Buffer} The combined data for the function call\n   */\n  getMethodCallData = (functionName, types, values) => {\n    return Buffer.concat([\n      // function signature\n      optionalDeps.ethAbi.methodID(functionName, types),\n      // function arguments\n      optionalDeps.ethAbi.rawEncode(types, values),\n    ]);\n  };\n\n  /**\n   * Build arguments to call the send method on the wallet contract\n   * @param txInfo\n   */\n  getSendMethodArgs(txInfo: GetSendMethodArgsOptions): SendMethodArgs[] {\n    // Method signature is\n    // sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)\n    return [\n      {\n        name: 'toAddress',\n        type: 'address',\n        value: txInfo.recipient.address,\n      },\n      {\n        name: 'value',\n        type: 'uint',\n        value: txInfo.recipient.amount,\n      },\n      {\n        name: 'data',\n        type: 'bytes',\n        value: optionalDeps.ethUtil.toBuffer(optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),\n      },\n      {\n        name: 'expireTime',\n        type: 'uint',\n        value: txInfo.expireTime,\n      },\n      {\n        name: 'sequenceId',\n        type: 'uint',\n        value: txInfo.contractSequenceId,\n      },\n      {\n        name: 'signature',\n        type: 'bytes',\n        value: optionalDeps.ethUtil.toBuffer(optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),\n      },\n    ];\n  }\n\n  /**\n   * Recovers a tx with TSS key shares\n   * same expected arguments as recover method, but with TSS key shares\n   */\n  protected async recoverTSS(\n    params: RecoverOptions\n  ): Promise<RecoveryInfo | OfflineVaultTxInfo | UnsignedSweepTxMPCv2> {\n    this.validateRecoveryParams(params);\n    // Clean up whitespace from entered values\n    const userPublicOrPrivateKeyShare = params.userKey.replace(/\\s/g, '');\n    const backupPrivateOrPublicKeyShare = params.backupKey.replace(/\\s/g, '');\n\n    if (\n      getIsUnsignedSweep({\n        userKey: userPublicOrPrivateKeyShare,\n        backupKey: backupPrivateOrPublicKeyShare,\n        isTss: params.isTss,\n      })\n    ) {\n      return this.buildUnsignedSweepTxnTSS(params);\n    } else {\n      const { userKeyShare, backupKeyShare, commonKeyChain } = await ECDSAUtils.getMpcV2RecoveryKeyShares(\n        userPublicOrPrivateKeyShare,\n        backupPrivateOrPublicKeyShare,\n        params.walletPassphrase\n      );\n\n      const { gasLimit, gasPrice } = await this.getGasValues(params);\n\n      const MPC = new Ecdsa();\n      const derivedCommonKeyChain = MPC.deriveUnhardened(commonKeyChain, 'm/0');\n      const backupKeyPair = new KeyPairLib({ pub: derivedCommonKeyChain.slice(0, 66) });\n      const baseAddress = backupKeyPair.getAddress();\n      const unsignedTx = (await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params)).tx;\n      const messageHash = unsignedTx.getMessageToSign(true);\n      const signature = await ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);\n      const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);\n      const signedTx = this.getSignedTxFromSignature(ethCommmon, unsignedTx, signature);\n\n      return {\n        id: addHexPrefix(signedTx.hash().toString('hex')),\n        tx: addHexPrefix(signedTx.serialize().toString('hex')),\n      };\n    }\n  }\n\n  private async getGasValues(params: RecoverOptions): Promise<{ gasLimit: number; gasPrice: Buffer }> {\n    const gasLimit = new optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));\n    const gasPrice = params.eip1559\n      ? new optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)\n      : new optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));\n    return { gasLimit, gasPrice };\n  }\n\n  protected async buildUnsignedSweepTxnTSS(params: RecoverOptions): Promise<OfflineVaultTxInfo | UnsignedSweepTxMPCv2> {\n    const userPublicOrPrivateKeyShare = params.userKey.replace(/\\s/g, '');\n    const backupPrivateOrPublicKeyShare = params.backupKey.replace(/\\s/g, '');\n\n    const { gasLimit, gasPrice } = await this.getGasValues(params);\n\n    const backupKeyPair = new KeyPairLib({ pub: backupPrivateOrPublicKeyShare });\n    const baseAddress = backupKeyPair.getAddress();\n    const { txInfo, tx, nonce } = await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params);\n    return this.formatForOfflineVaultTSS(\n      txInfo,\n      tx,\n      userPublicOrPrivateKeyShare,\n      backupPrivateOrPublicKeyShare,\n      gasPrice,\n      gasLimit,\n      nonce,\n      params.eip1559,\n      params.replayProtectionOptions\n    );\n  }\n\n  protected async buildUnsignedSweepTxnMPCv2(params: RecoverOptions): Promise<UnsignedSweepTxMPCv2> {\n    const { gasLimit, gasPrice } = await this.getGasValues(params);\n\n    const recoverParams = params as RecoverOptions;\n    this.validateUnsignedSweepTSSParams(recoverParams);\n\n    const derivationPath = recoverParams.derivationSeed ? getDerivationPath(recoverParams.derivationSeed) : 'm/0';\n    const MPC = new Ecdsa();\n    const derivedCommonKeyChain = MPC.deriveUnhardened(recoverParams.backupKey as string, derivationPath);\n    const backupKeyPair = new KeyPairLib({ pub: derivedCommonKeyChain.slice(0, 66) });\n    const baseAddress = backupKeyPair.getAddress();\n    const { txInfo, tx, nonce } = await this.buildTssRecoveryTxn(baseAddress, gasPrice, gasLimit, params);\n    return this.buildTxRequestForOfflineVaultMPCv2(\n      txInfo,\n      tx,\n      derivationPath,\n      nonce,\n      gasPrice,\n      gasLimit,\n      params.eip1559,\n      params.replayProtectionOptions,\n      recoverParams.backupKey as string\n    );\n  }\n\n  async createBroadcastableSweepTransaction(params: MPCSweepRecoveryOptions): Promise<MPCTxs> {\n    const req = params.signatureShares;\n    const broadcastableTransactions: MPCTx[] = [];\n    let lastScanIndex = 0;\n\n    for (let i = 0; i < req.length; i++) {\n      const transaction = req[i]?.txRequest?.transactions?.[0]?.unsignedTx as unknown as UnsignedTransactionTss;\n      if (!req[i].ovc || !req[i].ovc[0].ecdsaSignature) {\n        throw new Error('Missing signature(s)');\n      }\n      if (!transaction.signableHex) {\n        throw new Error('Missing signable hex');\n      }\n      const signature = req[i].ovc[0].ecdsaSignature;\n      if (!signature) {\n        throw new Error('Signature is undefined');\n      }\n      const shares: string[] = signature.toString().split(':');\n      if (shares.length !== 4) {\n        throw new Error('Invalid signature');\n      }\n      const finalSignature: ECDSAMethodTypes.Signature = {\n        recid: Number(shares[0]),\n        r: shares[1],\n        s: shares[2],\n        y: shares[3],\n      } as unknown as ECDSAMethodTypes.Signature;\n\n      if (!transaction.coinSpecific?.commonKeyChain) {\n        throw new Error(`Missing common keychain for transaction at index ${i}`);\n      }\n      const commonKeyChain = transaction.coinSpecific.commonKeyChain;\n      if (!transaction.derivationPath) {\n        throw new Error(`Missing derivation path for transaction at index ${i}`);\n      }\n      if (!commonKeyChain) {\n        throw new Error(`Missing common key chain for transaction at index ${i}`);\n      }\n\n      const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(\n        transaction.eip1559,\n        transaction.replayProtectionOptions\n      );\n      let unsignedTx: EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction;\n      if (transaction.eip1559) {\n        unsignedTx = FeeMarketEIP1559Transaction.fromSerializedTx(Buffer.from(transaction.serializedTxHex, 'hex'));\n      } else {\n        unsignedTx = LegacyTransaction.fromSerializedTx(Buffer.from(transaction.serializedTxHex, 'hex'));\n      }\n      const signedTx = this.getSignedTxFromSignature(ethCommmon, unsignedTx, finalSignature);\n      broadcastableTransactions.push({\n        serializedTx: addHexPrefix(signedTx.serialize().toString('hex')),\n      });\n\n      if (i === req.length - 1 && transaction.coinSpecific?.lastScanIndex) {\n        lastScanIndex = transaction.coinSpecific?.lastScanIndex as number;\n      }\n    }\n\n    return { transactions: broadcastableTransactions, lastScanIndex };\n  }\n\n  /**\n   * Method to validate recovery params\n   * @param {RecoverOptions} params\n   * @returns {void}\n   */\n  private async validateUnsignedSweepTSSParams(params: RecoverOptions): Promise<void> {\n    if (_.isUndefined(params.backupKey) && params.backupKey === '') {\n      throw new Error('missing commonKeyChain');\n    }\n    if (!_.isUndefined(params.derivationSeed) && typeof params.derivationSeed !== 'string') {\n      throw new Error('invalid derivationSeed');\n    }\n    if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('missing or invalid destinationAddress');\n    }\n  }\n\n  /**\n   * Helper function for recover()\n   * This transforms the unsigned transaction information into a format the BitGo offline vault expects\n   * @param {UnformattedTxInfo} txInfo - tx info\n   * @param {LegacyTransaction | FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object\n   * @param {string} derivationPath - the derivationPath\n   * @param {number} nonce - the nonce of the backup key address\n   * @param {Buffer} gasPrice - gas price for the tx\n   * @param {number} gasLimit - gas limit for the tx\n   * @param {EIP1559} eip1559 - eip1559 params\n   * @param replayProtectionOptions\n   * @param commonKeyChain\n   * @returns {Promise<OfflineVaultTxInfo>}\n   */\n  private buildTxRequestForOfflineVaultMPCv2(\n    txInfo: UnformattedTxInfo,\n    ethTx: LegacyTransaction | FeeMarketEIP1559Transaction,\n    derivationPath: string,\n    nonce: number,\n    gasPrice: Buffer,\n    gasLimit: number,\n    eip1559?: EIP1559,\n    replayProtectionOptions?: ReplayProtectionOptions,\n    commonKeyChain?: string\n  ): UnsignedSweepTxMPCv2 {\n    if (!ethTx.to) {\n      throw new Error('Eth tx must have a `to` address');\n    }\n\n    const fee = eip1559\n      ? gasLimit * eip1559.maxFeePerGas\n      : gasLimit * optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed();\n\n    const unsignedTx: UnsignedTransactionTss = {\n      serializedTxHex: ethTx.serialize().toString('hex'),\n      signableHex:\n        ethTx instanceof FeeMarketEIP1559Transaction\n          ? ethTx.getMessageToSign(false).toString('hex')\n          : Buffer.from(RLP.encode(bufArrToArr(ethTx.getMessageToSign(false)))).toString('hex'),\n      derivationPath: derivationPath,\n      feeInfo: {\n        fee: fee,\n        feeString: fee.toString(),\n      },\n      parsedTx: {\n        spendAmount: txInfo.recipient.amount,\n        outputs: [\n          {\n            coinName: this.getChain(),\n            address: txInfo.recipient.address,\n            valueString: txInfo.recipient.amount,\n          },\n        ],\n      },\n      coinSpecific: {\n        commonKeyChain: commonKeyChain,\n      },\n      eip1559: eip1559,\n      replayProtectionOptions: replayProtectionOptions,\n    };\n\n    return {\n      txRequests: [\n        {\n          walletCoin: this.getChain(),\n          transactions: [\n            {\n              unsignedTx: unsignedTx,\n              nonce: nonce,\n              signatureShares: [],\n            },\n          ],\n        },\n      ],\n    };\n  }\n\n  private async buildTssRecoveryTxn(baseAddress: string, gasPrice: any, gasLimit: any, params: RecoverOptions) {\n    const txAmount = await this.validateBalanceAndGetTxAmount(baseAddress, gasPrice, gasLimit, params.apiKey);\n    const nonce = await this.getAddressNonce(baseAddress, params.apiKey);\n    const recipients = [\n      {\n        address: params.recoveryDestination,\n        amount: txAmount.toString(10),\n      },\n    ];\n\n    const txInfo = {\n      recipient: recipients[0],\n      expireTime: this.getDefaultExpireTime(),\n      gasLimit: gasLimit.toString(10),\n    };\n\n    const txParams = {\n      to: params.recoveryDestination,\n      nonce: nonce,\n      value: txAmount,\n      gasPrice: gasPrice,\n      gasLimit: gasLimit,\n      data: Buffer.from('0x'),\n      eip1559: params.eip1559,\n      replayProtectionOptions: params.replayProtectionOptions,\n    };\n\n    const tx = AbstractEthLikeNewCoins.buildTransaction(txParams);\n    return { txInfo, tx, nonce };\n  }\n\n  async validateBalanceAndGetTxAmount(baseAddress: string, gasPrice: BN, gasLimit: BN, apiKey?: string) {\n    const baseAddressBalance = await this.queryAddressBalance(baseAddress, apiKey);\n    let totalGasNeeded = gasPrice.mul(gasLimit);\n    // On L2 chains with L1 data fees, add buffer for L1 fees\n    if (this.staticsCoin?.family !== undefined && coinFamiliesWithL1Fees.includes(this.staticsCoin.family)) {\n      totalGasNeeded = totalGasNeeded.add(new optionalDeps.ethUtil.BN(ethGasConfigs.l1GasFeeBuffer));\n    }\n    const weiToGwei = new BN(10 ** 9);\n    if (baseAddressBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Backup key address ${baseAddress} has balance ${baseAddressBalance.div(weiToGwei).toString()} Gwei.` +\n          `This address must have a balance of at least ${totalGasNeeded.div(weiToGwei).toString()}` +\n          ` Gwei to perform recoveries. Try sending some ETH to this address then retry.`\n      );\n    }\n    const txAmount = baseAddressBalance.sub(totalGasNeeded);\n    return txAmount;\n  }\n\n  /**\n   * Make a query to blockchain explorer for information such as balance, token balance, solidity calls\n   * @param query {Object} key-value pairs of parameters to append after /api\n   * @param apiKey {string} optional API key to use instead of the one from the environment\n   * @returns {Object} response from the blockchain explorer\n   */\n  async recoveryBlockchainExplorerQuery(query: Record<string, string>, apiKey?: string): Promise<any> {\n    throw new Error('method not implemented');\n  }\n\n  /**\n   * Creates the extra parameters needed to build a hop transaction\n   * @param buildParams The original build parameters\n   * @returns extra parameters object to merge with the original build parameters object and send to the platform\n   */\n  async createHopTransactionParams(buildParams: HopTransactionBuildOptions): Promise<HopParams> {\n    const wallet = buildParams.wallet;\n    const recipients = buildParams.recipients;\n    const walletPassphrase = buildParams.walletPassphrase;\n\n    const userKeychain = await this.keychains().get({ id: wallet.keyIds()[0] });\n    const userPrv = wallet.getUserPrv({ keychain: userKeychain, walletPassphrase });\n    const userPrvBuffer = bip32.fromBase58(userPrv).privateKey;\n    if (!userPrvBuffer) {\n      throw new Error('invalid userPrv');\n    }\n    if (!recipients || !Array.isArray(recipients)) {\n      throw new Error('expecting array of recipients');\n    }\n\n    // Right now we only support 1 recipient\n    if (recipients.length !== 1) {\n      throw new Error('must send to exactly 1 recipient');\n    }\n    const recipientAddress = recipients[0].address;\n    const recipientAmount = recipients[0].amount as string;\n    const feeEstimateParams = {\n      recipient: recipientAddress,\n      amount: recipientAmount,\n      hop: true,\n    };\n    const feeEstimate: FeeEstimate = await this.feeEstimate(feeEstimateParams);\n\n    const gasLimit = feeEstimate.gasLimitEstimate;\n    const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);\n    const gasPriceMax = gasPrice * 5;\n    // Payment id a random number so its different for every tx\n    const paymentId = Math.floor(Math.random() * 10000000000).toString();\n    const hopDigest: Buffer = AbstractEthLikeNewCoins.getHopDigest([\n      recipientAddress,\n      recipientAmount,\n      gasPriceMax.toString(),\n      gasLimit.toString(),\n      paymentId,\n    ]);\n\n    const userReqSig = optionalDeps.ethUtil.addHexPrefix(\n      Buffer.from(secp256k1.ecdsaSign(hopDigest, userPrvBuffer).signature).toString('hex')\n    );\n\n    return {\n      hopParams: {\n        gasPriceMax,\n        userReqSig,\n        paymentId,\n        gasLimit,\n      },\n    };\n  }\n\n  /**\n   * Validates that the hop prebuild from the HSM is valid and correct\n   * @param {IWallet} wallet - The wallet that the prebuild is for\n   * @param {HopPrebuild} hopPrebuild - The prebuild to validate\n   * @param {Object} originalParams - The original parameters passed to prebuildTransaction\n   * @param {Recipient[]} originalParams.recipients - The original recipients array\n   * @returns {void}\n   * @throws Error if The prebuild is invalid\n   */\n  async validateHopPrebuild(\n    wallet: IWallet,\n    hopPrebuild: HopPrebuild,\n    originalParams?: { recipients: Recipient[] }\n  ): Promise<void> {\n    const { tx, id, signature } = hopPrebuild;\n\n    // first, validate the HSM signature\n    const serverXpub = common.Environments[this.bitgo.getEnv()].hsmXpub;\n    const serverPubkeyBuffer: Buffer = bip32.fromBase58(serverXpub).publicKey;\n    const signatureBuffer: Buffer = Buffer.from(optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');\n    const messageBuffer: Buffer = Buffer.from(\n      optionalDeps.ethUtil.padToEven(optionalDeps.ethUtil.stripHexPrefix(id)),\n      'hex'\n    );\n\n    const sig = new Uint8Array(signatureBuffer.slice(1));\n    const isValidSignature: boolean = secp256k1.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);\n    if (!isValidSignature) {\n      throw new Error(\n        `Hop txid signature invalid - pub: ${serverXpub}, msg: ${messageBuffer?.toString()}, sig: ${signatureBuffer?.toString()}`\n      );\n    }\n\n    const builtHopTx = optionalDeps.EthTx.TransactionFactory.fromSerializedData(optionalDeps.ethUtil.toBuffer(tx));\n    // If original params are given, we can check them against the transaction prebuild params\n    if (!_.isNil(originalParams)) {\n      const { recipients } = originalParams;\n\n      // Then validate that the tx params actually equal the requested params\n      const originalAmount = new BigNumber(recipients[0].amount);\n      const originalDestination: string = recipients[0].address;\n\n      const hopAmount = new BigNumber(optionalDeps.ethUtil.bufferToHex(builtHopTx.value));\n      if (!builtHopTx.to) {\n        throw new Error(`Transaction does not have a destination address`);\n      }\n      const hopDestination = builtHopTx.to.toString();\n      if (!hopAmount.eq(originalAmount)) {\n        throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);\n      }\n      if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {\n        throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);\n      }\n    }\n\n    if (!builtHopTx.verifySignature()) {\n      // We dont want to continue at all in this case, at risk of ETH being stuck on the hop address\n      throw new Error(`Invalid hop transaction signature, txid: ${id}`);\n    }\n    if (optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {\n      throw new Error(`Signed hop txid does not equal actual txid`);\n    }\n  }\n\n  /**\n   * Gets the hop digest for the user to sign. This is validated in the HSM to prove that the user requested this tx\n   * @param {string[]} paramsArr - The parameters to hash together for the digest\n   * @returns {Buffer}\n   */\n  public static getHopDigest(paramsArr: string[]): Buffer {\n    const hash = Keccak('keccak256');\n    hash.update([AbstractEthLikeNewCoins.hopTransactionSalt, ...paramsArr].join('$'));\n    return hash.digest();\n  }\n\n  /**\n   * Modify prebuild before sending it to the server. Add things like hop transaction params\n   * @param {BuildOptions} buildParams - The whitelisted parameters for this prebuild\n   * @param {boolean} buildParams.hop - True if this should prebuild a hop tx, else false\n   * @param {Recipient[]} buildParams.recipients - The recipients array of this transaction\n   * @param {Wallet} buildParams.wallet - The wallet sending this tx\n   * @param {string} buildParams.walletPassphrase - the passphrase for this wallet\n   * @returns {Promise<BuildOptions>}\n   */\n  async getExtraPrebuildParams(buildParams: BuildOptions): Promise<BuildOptions> {\n    if (\n      !_.isUndefined(buildParams.hop) &&\n      buildParams.hop &&\n      !_.isUndefined(buildParams.wallet) &&\n      !_.isUndefined(buildParams.recipients) &&\n      !_.isUndefined(buildParams.walletPassphrase)\n    ) {\n      if (this instanceof EthLikeToken) {\n        throw new Error(\n          `Hop transactions are not enabled for ERC-20 tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`\n        );\n      }\n      return (await this.createHopTransactionParams({\n        wallet: buildParams.wallet,\n        recipients: buildParams.recipients,\n        walletPassphrase: buildParams.walletPassphrase,\n      })) as any;\n    }\n    return {};\n  }\n\n  /**\n   * Modify prebuild after receiving it from the server. Add things like nlocktime\n   * @param {TransactionPrebuild} params - The prebuild to modify\n   * @returns {TransactionPrebuild} The modified prebuild\n   */\n  async postProcessPrebuild(params: TransactionPrebuild): Promise<TransactionPrebuild> {\n    if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {\n      await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);\n    }\n    return params;\n  }\n\n  /**\n   * Coin-specific things done before signing a transaction, i.e. verification\n   * @param {PresignTransactionOptions} params\n   * @returns {Promise<PresignTransactionOptions>}\n   */\n  async presignTransaction(params: PresignTransactionOptions): Promise<PresignTransactionOptions> {\n    if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {\n      await this.validateHopPrebuild(params.wallet, params.hopTransaction);\n    }\n    return params;\n  }\n\n  /**\n   * Fetch fee estimate information from the server\n   * @param {Object} params - The params passed into the function\n   * @param {boolean} [params.hop] - True if we should estimate fee for a hop transaction\n   * @param {string} [params.recipient] - The recipient of the transaction to estimate a send to\n   * @param {string} [params.data] - The ETH tx data to estimate a send for\n   * @returns {Object} The fee info returned from the server\n   */\n  async feeEstimate(params: FeeEstimateOptions): Promise<FeeEstimate> {\n    const query: FeeEstimateOptions = {};\n    if (params && params.hop) {\n      query.hop = params.hop;\n    }\n    if (params && params.recipient) {\n      query.recipient = params.recipient;\n    }\n    if (params && params.data) {\n      query.data = params.data;\n    }\n    if (params && params.amount) {\n      query.amount = params.amount;\n    }\n\n    return await this.bitgo.get(this.url('/tx/fee')).query(query).result();\n  }\n\n  /**\n   * Generate secp256k1 key pair\n   *\n   * @param {Buffer} seed\n   * @returns {KeyPair} object with generated pub and prv\n   */\n  generateKeyPair(seed: Buffer): KeyPair {\n    if (!seed) {\n      // An extended private key has both a normal 256 bit private key and a 256\n      // bit chain code, both of which must be random. 512 bits is therefore the\n      // maximum entropy and gives us maximum security against cracking.\n      seed = randomBytes(512 / 8);\n    }\n    const extendedKey = bip32.fromSeed(seed);\n    const xpub = extendedKey.neutered().toBase58();\n    return {\n      pub: xpub,\n      prv: extendedKey.toBase58(),\n    };\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  /**\n   * Get forwarder factory and implementation addresses for deposit address verification.\n   * Forwarders are smart contracts that forward funds to the base wallet address.\n   *\n   * @param {number | undefined} forwarderVersion - The wallet version\n   * @returns {object} Factory and implementation addresses for forwarders\n   */\n  getForwarderFactoryAddressesAndForwarderImplementationAddress(forwarderVersion: number | undefined): {\n    forwarderFactoryAddress: string;\n    forwarderImplementationAddress: string;\n  } {\n    const ethNetwork = this.getNetwork();\n\n    switch (forwarderVersion) {\n      case 1:\n        if (!ethNetwork?.forwarderFactoryAddress || !ethNetwork?.forwarderImplementationAddress) {\n          throw new Error('Forwarder factory addresses not configured for this network');\n        }\n        return {\n          forwarderFactoryAddress: ethNetwork.forwarderFactoryAddress,\n          forwarderImplementationAddress: ethNetwork.forwarderImplementationAddress,\n        };\n      case 2:\n        if (!ethNetwork?.walletV2ForwarderFactoryAddress || !ethNetwork?.walletV2ForwarderImplementationAddress) {\n          throw new Error('Wallet v2 factory addresses not configured for this network');\n        }\n        return {\n          forwarderFactoryAddress: ethNetwork.walletV2ForwarderFactoryAddress,\n          forwarderImplementationAddress: ethNetwork.walletV2ForwarderImplementationAddress,\n        };\n      case 4:\n      case 5:\n        if (!ethNetwork?.walletV4ForwarderFactoryAddress || !ethNetwork?.walletV4ForwarderImplementationAddress) {\n          throw new Error(`Forwarder v${forwarderVersion} factory addresses not configured for this network`);\n        }\n        return {\n          forwarderFactoryAddress: ethNetwork.walletV4ForwarderFactoryAddress,\n          forwarderImplementationAddress: ethNetwork.walletV4ForwarderImplementationAddress,\n        };\n      default:\n        throw new Error(`Forwarder version ${forwarderVersion} not supported`);\n    }\n  }\n\n  /**\n   * Get wallet base address factory and implementation addresses.\n   * This is used for base address verification for V1, V2, V4, and V5 wallets.\n   * The base address is the main wallet contract deployed via CREATE2.\n   *\n   * @param {number} walletVersion - The wallet version (1, 2, 4, or 5)\n   * @returns {object} Factory and implementation addresses for the wallet base address\n   * @throws {Error} if wallet version addresses are not configured\n   */\n  getWalletAddressFactoryAddressesAndImplementationAddress(walletVersion: number): {\n    walletFactoryAddress: string;\n    walletImplementationAddress: string;\n  } {\n    const ethNetwork = this.getNetwork();\n\n    switch (walletVersion) {\n      case 1:\n        if (!ethNetwork?.walletFactoryAddress || !ethNetwork?.walletImplementationAddress) {\n          throw new Error('Wallet v1 factory addresses not configured for this network');\n        }\n        return {\n          walletFactoryAddress: ethNetwork.walletFactoryAddress,\n          walletImplementationAddress: ethNetwork.walletImplementationAddress,\n        };\n      case 2:\n        if (!ethNetwork?.walletV2FactoryAddress || !ethNetwork?.walletV2ImplementationAddress) {\n          throw new Error('Wallet v2 factory addresses not configured for this network');\n        }\n        return {\n          walletFactoryAddress: ethNetwork.walletV2FactoryAddress,\n          walletImplementationAddress: ethNetwork.walletV2ImplementationAddress,\n        };\n      case 4:\n      case 5:\n        if (!ethNetwork?.walletV4ForwarderFactoryAddress || !ethNetwork?.walletV4ForwarderImplementationAddress) {\n          throw new Error(`Wallet v${walletVersion} factory addresses not configured for this network`);\n        }\n        return {\n          walletFactoryAddress: ethNetwork.walletV4ForwarderFactoryAddress,\n          walletImplementationAddress: ethNetwork.walletV4ForwarderImplementationAddress,\n        };\n      default:\n        throw new Error(`Wallet version ${walletVersion} not supported`);\n    }\n  }\n\n  /**\n   * Helper method to create a salt buffer from hex string.\n   * Converts a hex salt string to a 32-byte buffer.\n   *\n   * @param {string} salt - The hex salt string\n   * @returns {Buffer} 32-byte salt buffer\n   */\n  private createSaltBuffer(salt: string): Buffer {\n    const ethUtil = optionalDeps.ethUtil;\n    return ethUtil.setLengthLeft(Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(salt || '')), 'hex'), 32);\n  }\n\n  /**\n   * Verify BIP32 wallet base address (V1, V2, V4).\n   * These wallets use a wallet factory to deploy base addresses with CREATE2.\n   * The address is derived from the keychains' ethAddresses and a salt.\n   *\n   * @param {VerifyBip32BaseAddressOptions} params - Verification parameters\n   * @returns {object} Expected and actual addresses for comparison\n   */\n  private verifyCreate2BaseAddress(params: VerifyContractBaseAddressOptions): boolean {\n    const { address, coinSpecific, keychains, walletVersion } = params;\n\n    if (!coinSpecific.salt) {\n      throw new Error(`missing salt for v${walletVersion} base address verification`);\n    }\n\n    // Get wallet factory and implementation addresses for the wallet version\n    const { walletFactoryAddress, walletImplementationAddress } =\n      this.getWalletAddressFactoryAddressesAndImplementationAddress(walletVersion);\n    const initcode = getProxyInitcode(walletImplementationAddress);\n\n    // Convert the wallet salt to a buffer, pad to 32 bytes\n    const saltBuffer = this.createSaltBuffer(coinSpecific.salt);\n\n    // Reconstruct calculationSalt using keychains' ethAddresses and wallet salt\n    const ethAddresses = keychains.map((kc) => {\n      if (!kc.ethAddress) {\n        throw new Error(`keychain missing ethAddress for v${walletVersion} base address verification`);\n      }\n      return kc.ethAddress;\n    });\n\n    const calculationSalt = optionalDeps.ethUtil.bufferToHex(\n      optionalDeps.ethAbi.soliditySHA3(['address[]', 'bytes32'], [ethAddresses, saltBuffer])\n    );\n\n    const expectedAddress = calculateForwarderV1Address(walletFactoryAddress, calculationSalt, initcode);\n\n    if (expectedAddress !== address) {\n      throw new UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);\n    }\n\n    return true;\n  }\n\n  /**\n   * Verify forwarder receive address (deposit address).\n   * Forwarder addresses are derived using CREATE2 from the base address and salt.\n   *\n   * @param {VerifyEthAddressOptions} params - Verification parameters\n   * @param {number} forwarderVersion - The forwarder version\n   * @returns {object} Expected and actual addresses for comparison\n   */\n  private verifyForwarderAddress(params: VerifyEthAddressOptions, forwarderVersion: number): boolean {\n    const { address, coinSpecific, baseAddress } = params;\n\n    const { forwarderFactoryAddress, forwarderImplementationAddress } =\n      this.getForwarderFactoryAddressesAndForwarderImplementationAddress(forwarderVersion);\n    const initcode = getProxyInitcode(forwarderImplementationAddress);\n    const saltBuffer = this.createSaltBuffer(coinSpecific.salt || '');\n\n    const { createForwarderParams, createForwarderTypes } =\n      forwarderVersion === 4\n        ? getCreateForwarderParamsAndTypes(baseAddress, saltBuffer, coinSpecific.feeAddress)\n        : getCreateForwarderParamsAndTypes(baseAddress, saltBuffer);\n\n    const calculationSalt = optionalDeps.ethUtil.bufferToHex(\n      optionalDeps.ethAbi.soliditySHA3(createForwarderTypes, createForwarderParams)\n    );\n\n    const expectedAddress = calculateForwarderV1Address(forwarderFactoryAddress, calculationSalt, initcode);\n\n    if (expectedAddress !== address) {\n      throw new UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);\n    }\n\n    return true;\n  }\n\n  /**\n   * Make sure an address is a wallet address and throw an error if it's not.\n   * @param {Object} params\n   * @param {string} params.address - The derived address string on the network\n   * @param {Object} params.coinSpecific - Coin-specific details for the address such as a forwarderVersion\n   * @param {string} params.baseAddress - The base address of the wallet on the network\n   * @throws {InvalidAddressError}\n   * @throws {InvalidAddressVerificationObjectPropertyError}\n   * @throws {UnexpectedAddressError}\n   * @returns {boolean} True iff address is a wallet address\n   */\n  async isWalletAddress(params: VerifyEthAddressOptions | TssVerifyEthAddressOptions): Promise<boolean> {\n    const { address, impliedForwarderVersion, coinSpecific, baseAddress } = params;\n    const forwarderVersion = impliedForwarderVersion ?? coinSpecific?.forwarderVersion;\n\n    // Validate address format\n    if (address && !this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    // Forwarder version 0 addresses cannot be verified because we do not store the nonce value required for address derivation.\n    if (forwarderVersion === 0) {\n      return true;\n    }\n\n    // Determine if we are verifying a base address\n    const isVerifyingBaseAddress = baseAddress && address === baseAddress;\n\n    // TSS/MPC wallet address verification (V3, V5, V6)\n    // V5 base addresses use TSS, but V5 forwarders use the regular forwarder verification\n    const isTssWalletVersion = params.walletVersion === 3 || params.walletVersion === 5 || params.walletVersion === 6;\n    const shouldUseTssVerification =\n      isTssVerifyAddressOptions(params) && isTssWalletVersion && (params.walletVersion !== 5 || isVerifyingBaseAddress);\n\n    if (shouldUseTssVerification) {\n      if (isVerifyingBaseAddress) {\n        const index = typeof params.index === 'string' ? parseInt(params.index, 10) : params.index;\n        if (index !== 0) {\n          throw new Error(\n            `Base address verification requires index 0, but got index ${params.index}. ` +\n              `The base address is always derived at index 0.`\n          );\n        }\n      }\n\n      return verifyMPCWalletAddress({ ...params, keyCurve: 'secp256k1' }, this.isValidAddress, (pubKey) => {\n        return new KeyPairLib({ pub: pubKey }).getAddress();\n      });\n    }\n\n    // From here on, we need baseAddress and coinSpecific for non-TSS verifications\n    if (_.isUndefined(baseAddress) || !this.isValidAddress(baseAddress)) {\n      throw new InvalidAddressError('invalid base address');\n    }\n\n    if (!_.isObject(coinSpecific)) {\n      throw new InvalidAddressVerificationObjectPropertyError(\n        'address validation failure: coinSpecific field must be an object'\n      );\n    }\n\n    // BIP32 wallet base address verification (V1, V2, V4)\n    if (isVerifyingBaseAddress && isVerifyContractBaseAddressOptions(params)) {\n      return this.verifyCreate2BaseAddress(params);\n    }\n\n    // Forwarder receive address verification (deposit addresses)\n    if (!isVerifyingBaseAddress) {\n      return this.verifyForwarderAddress(params, forwarderVersion);\n    }\n\n    // If we reach here, it's a base address verification for an unsupported wallet version\n    throw new Error(`Base address verification not supported for wallet version ${params.walletVersion}`);\n  }\n\n  /**\n   *\n   * @param {TransactionPrebuild} txPrebuild\n   * @returns {boolean}\n   */\n  verifyCoin(txPrebuild: TransactionPrebuild): boolean {\n    const nativeCoin = this.getChain().split(':')[0];\n    return txPrebuild.coin === nativeCoin;\n  }\n\n  /**\n   * Generate transaction explanation for error reporting\n   * @param txPrebuild - Transaction prebuild containing txHex and fee info\n   * @returns Stringified JSON explanation\n   */\n  private async getTxExplanation(txPrebuild?: TransactionPrebuild): Promise<string | undefined> {\n    if (!txPrebuild?.txHex || !txPrebuild?.gasPrice) {\n      return undefined;\n    }\n\n    try {\n      const explanation = await this.explainTransaction({\n        txHex: txPrebuild.txHex,\n        feeInfo: {\n          fee: txPrebuild.gasPrice.toString(),\n        },\n      });\n      return JSON.stringify(explanation, null, 2);\n    } catch (e) {\n      const errorDetails = {\n        error: 'Failed to parse transaction explanation',\n        txHex: txPrebuild.txHex,\n        details: e instanceof Error ? e.message : String(e),\n      };\n      return JSON.stringify(errorDetails, null, 2);\n    }\n  }\n\n  /**\n   * Verify if a tss transaction is valid\n   *\n   * @param {VerifyEthTransactionOptions} params\n   * @param {TransactionParams} params.txParams - params object passed to send\n   * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server\n   * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against\n   * @returns {boolean}\n   * @throws {TxIntentMismatchRecipientError} if transaction recipients don't match user intent\n   */\n  async verifyTssTransaction(params: VerifyEthTransactionOptions): Promise<boolean> {\n    const { txParams, txPrebuild, wallet } = params;\n\n    // Helper to throw TxIntentMismatchRecipientError with recipient details\n    const throwRecipientMismatch = async (message: string, mismatchedRecipients: Recipient[]): Promise<never> => {\n      const txExplanation = await this.getTxExplanation(txPrebuild);\n      throw new TxIntentMismatchRecipientError(\n        message,\n        undefined,\n        [txParams],\n        txPrebuild?.txHex,\n        mismatchedRecipients,\n        txExplanation\n      );\n    };\n\n    if (\n      !txParams?.recipients &&\n      !(\n        txParams.prebuildTx?.consolidateId ||\n        txPrebuild?.consolidateId ||\n        (txParams.type &&\n          ['acceleration', 'fillNonce', 'transferToken', 'tokenApproval', 'consolidate', 'bridgeFunds'].includes(\n            txParams.type\n          ))\n      )\n    ) {\n      throw new Error('missing txParams');\n    }\n    if (!wallet || !txPrebuild) {\n      throw new Error('missing params');\n    }\n    if (txParams.hop && txParams.recipients && txParams.recipients.length > 1) {\n      throw new Error('tx cannot be both a batch and hop transaction');\n    }\n\n    if (txParams.type && ['transfer'].includes(txParams.type)) {\n      if (txParams.recipients && txParams.recipients.length === 1) {\n        const recipients = txParams.recipients;\n        const expectedAmount = recipients[0].amount.toString();\n        const expectedDestination = recipients[0].address;\n\n        const txBuilder = this.getTransactionBuilder();\n        txBuilder.from(txPrebuild.txHex);\n        const tx = await txBuilder.build();\n        const txJson = tx.toJson();\n        if (txJson.data === '0x') {\n          if (expectedAmount !== txJson.value) {\n            await throwRecipientMismatch(\n              'the transaction amount in txPrebuild does not match the value given by client',\n              [{ address: txJson.to, amount: txJson.value }]\n            );\n          }\n          if (expectedDestination.toLowerCase() !== txJson.to.toLowerCase()) {\n            await throwRecipientMismatch('destination address does not match with the recipient address', [\n              { address: txJson.to, amount: txJson.value },\n            ]);\n          }\n        } else if (txJson.data.startsWith('0xa9059cbb')) {\n          const [recipientAddress, amount] = getRawDecoded(\n            ['address', 'uint256'],\n            getBufferedByteCode('0xa9059cbb', txJson.data)\n          );\n\n          // Check if recipients[0].data exists (WalletConnect flow)\n          let expectedRecipientAddress: string;\n          let expectedTokenAmount: string;\n          const recipientData = (recipients[0] as any).data;\n\n          if (recipientData && recipientData.startsWith('0xa9059cbb')) {\n            // WalletConnect: decode expected recipient and amount from recipients[0].data\n            const [expectedRecipient, expectedAmount] = getRawDecoded(\n              ['address', 'uint256'],\n              getBufferedByteCode('0xa9059cbb', recipientData)\n            );\n            expectedRecipientAddress = addHexPrefix(expectedRecipient.toString()).toLowerCase();\n            expectedTokenAmount = expectedAmount.toString();\n          } else {\n            // Normal flow: use recipients[0].address and recipients[0].amount\n            expectedRecipientAddress = expectedDestination.toLowerCase();\n            expectedTokenAmount = expectedAmount;\n          }\n\n          if (expectedTokenAmount !== amount.toString()) {\n            await throwRecipientMismatch(\n              'the transaction amount in txPrebuild does not match the value given by client',\n              [{ address: addHexPrefix(recipientAddress.toString()), amount: amount.toString() }]\n            );\n          }\n\n          if (expectedRecipientAddress !== addHexPrefix(recipientAddress.toString()).toLowerCase()) {\n            await throwRecipientMismatch('destination address does not match with the recipient address', [\n              { address: addHexPrefix(recipientAddress.toString()), amount: amount.toString() },\n            ]);\n          }\n        }\n      }\n    }\n\n    // Verify consolidation transactions send to base address\n    if (params.verification?.consolidationToBaseAddress) {\n      const coinSpecific = wallet.coinSpecific();\n      if (!coinSpecific || !coinSpecific.baseAddress) {\n        throw new Error('Unable to determine base address for consolidation');\n      }\n      const baseAddress = coinSpecific.baseAddress;\n\n      if (!txPrebuild.txHex) {\n        throw new Error('missing txHex in txPrebuild');\n      }\n\n      const txBuilder = this.getTransactionBuilder();\n      txBuilder.from(txPrebuild.txHex);\n      const tx = await txBuilder.build();\n      const txJson = tx.toJson();\n\n      // Verify the transaction recipient matches the base address\n      if (!txJson.to) {\n        throw new Error('Consolidation transaction is missing recipient address');\n      }\n\n      if (txJson.to.toLowerCase() !== baseAddress.toLowerCase()) {\n        await throwRecipientMismatch('Consolidation transaction recipient does not match wallet base address', [\n          { address: txJson.to, amount: txJson.value },\n        ]);\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Verify that a transaction prebuild complies with the original intention\n   *\n   * @param {VerifyEthTransactionOptions} params\n   * @param {TransactionParams} params.txParams - params object passed to send\n   * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server\n   * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against\n   * @returns {boolean}\n   * @throws {TxIntentMismatchError} if transaction validation fails\n   * @throws {TxIntentMismatchRecipientError} if transaction recipients don't match user intent\n   */\n  async verifyTransaction(params: VerifyEthTransactionOptions): Promise<boolean> {\n    const ethNetwork = this.getNetwork();\n    const { txParams, txPrebuild, wallet, walletType } = params;\n\n    if (walletType === 'tss') {\n      return this.verifyTssTransaction(params);\n    }\n\n    // Helper to throw TxIntentMismatchRecipientError with recipient details\n    const throwRecipientMismatch = async (message: string, mismatchedRecipients: Recipient[]): Promise<never> => {\n      const txExplanation = await this.getTxExplanation(txPrebuild);\n      throw new TxIntentMismatchRecipientError(\n        message,\n        undefined,\n        [txParams],\n        txPrebuild?.txHex,\n        mismatchedRecipients,\n        txExplanation\n      );\n    };\n\n    if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {\n      throw new Error('missing params');\n    }\n\n    const recipients = txParams.recipients!;\n\n    if (txParams.hop && recipients.length > 1) {\n      throw new Error('tx cannot be both a batch and hop transaction');\n    }\n    if (txPrebuild.recipients.length > 1) {\n      throw new Error(\n        `${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n      );\n    }\n    if (txParams.hop && txPrebuild.hopTransaction) {\n      // Check recipient amount for hop transaction\n      if (recipients.length !== 1) {\n        throw new Error(`hop transaction only supports 1 recipient but ${recipients.length} found`);\n      }\n\n      // Check tx sends to hop address\n      const decodedHopTx = optionalDeps.EthTx.TransactionFactory.fromSerializedData(\n        optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx)\n      );\n      const expectedHopAddress = optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());\n      const actualHopAddress = optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);\n      if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {\n        await throwRecipientMismatch('recipient address of txPrebuild does not match hop address', [\n          { address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() },\n        ]);\n      }\n\n      // Convert TransactionRecipient array to Recipient array\n      const hopRecipients: Recipient[] = recipients.map((r) => {\n        return {\n          address: r.address,\n          amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,\n        };\n      });\n\n      // Check destination address and amount\n      await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients: hopRecipients });\n    } else if (recipients.length > 1) {\n      // Check total amount for batch transaction\n      if (txParams.tokenName) {\n        const expectedTotalAmount = new BigNumber(0);\n        if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n          await throwRecipientMismatch(\n            'batch token transaction amount in txPrebuild should be zero for token transfers',\n            [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]\n          );\n        }\n      } else {\n        let expectedTotalAmount = new BigNumber(0);\n        for (let i = 0; i < recipients.length; i++) {\n          expectedTotalAmount = expectedTotalAmount.plus(recipients[i].amount);\n        }\n        if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n          await throwRecipientMismatch(\n            'batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client',\n            [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]\n          );\n        }\n      }\n\n      // Check batch transaction is sent to the batcher contract address for the chain\n      const batcherContractAddress = ethNetwork?.batcherContractAddress;\n      if (\n        !batcherContractAddress ||\n        batcherContractAddress.toLowerCase() !== txPrebuild.recipients[0].address.toLowerCase()\n      ) {\n        await throwRecipientMismatch('recipient address of txPrebuild does not match batcher address', [\n          { address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() },\n        ]);\n      }\n    } else {\n      // Check recipient address and amount for normal transaction\n      if (recipients.length !== 1) {\n        throw new Error(`normal transaction only supports 1 recipient but ${recipients.length} found`);\n      }\n      const expectedAmount = new BigNumber(recipients[0].amount);\n      if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n        await throwRecipientMismatch(\n          'normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client',\n          [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]\n        );\n      }\n      if (this.isETHAddress(recipients[0].address) && recipients[0].address !== txPrebuild.recipients[0].address) {\n        await throwRecipientMismatch(\n          'destination address in normal txPrebuild does not match that in txParams supplied by client',\n          [{ address: txPrebuild.recipients[0].address, amount: txPrebuild.recipients[0].amount.toString() }]\n        );\n      }\n    }\n    // Check coin is correct for all transaction types\n    if (!this.verifyCoin(txPrebuild)) {\n      const txExplanation = await this.getTxExplanation(txPrebuild);\n      throw new TxIntentMismatchError(\n        'coin in txPrebuild did not match that in txParams supplied by client',\n        undefined,\n        [txParams],\n        txPrebuild?.txHex,\n        txExplanation\n      );\n    }\n    return true;\n  }\n\n  /**\n   * Check if address is valid eth address\n   * @param address\n   * @returns {boolean}\n   */\n  private isETHAddress(address: string): boolean {\n    return !!address.match(/0x[a-fA-F0-9]{40}/);\n  }\n\n  /**\n   * Transform message to accommodate specific blockchain requirements.\n   * @param {string} message - the message to prepare\n   * @return {string} the prepared message as a hex encoded string.\n   */\n  encodeMessage(message: string): string {\n    const prefix = `\\u0019Ethereum Signed Message:\\n${message.length}`;\n    return Buffer.from(prefix.concat(message)).toString('hex');\n  }\n\n  /**\n   * Transform the Typed data to accomodate the blockchain requirements (EIP-712)\n   * @param {TypedData} typedData - the typed data to prepare\n   * @return {Buffer} a buffer of the result\n   */\n  encodeTypedData(typedData: TypedData): Buffer {\n    const version = typedData.version;\n    if (version === SignTypedDataVersion.V1) {\n      throw new Error('SignTypedData v1 is not supported due to security concerns');\n    }\n    const typedDataRaw = JSON.parse(typedData.typedDataRaw);\n    const sanitizedData = TypedDataUtils.sanitizeData(typedDataRaw as unknown as TypedMessage<any>);\n    const parts: Buffer[] = [Buffer.from('1901', 'hex')];\n    const eip712Domain = 'EIP712Domain';\n    parts.push(TypedDataUtils.hashStruct(eip712Domain, sanitizedData.domain, sanitizedData.types, version));\n\n    if (sanitizedData.primaryType !== eip712Domain) {\n      parts.push(\n        TypedDataUtils.hashStruct(\n          sanitizedData.primaryType as string,\n          sanitizedData.message,\n          sanitizedData.types,\n          version\n        )\n      );\n    }\n    return Buffer.concat(parts);\n  }\n\n  /**\n   * Build the data to transfer an ERC-721 or ERC-1155 token to another address\n   * @param params\n   */\n  buildNftTransferData(params: BuildNftTransferDataOptions): string {\n    const { tokenContractAddress, recipientAddress, fromAddress } = params;\n    switch (params.type) {\n      case 'ERC721': {\n        const tokenId = params.tokenId;\n        const contractData = new ERC721TransferBuilder()\n          .tokenContractAddress(tokenContractAddress)\n          .to(recipientAddress)\n          .from(fromAddress)\n          .tokenId(tokenId)\n          .build();\n        return contractData;\n      }\n\n      case 'ERC1155': {\n        const entries = params.entries;\n        const transferBuilder = new ERC1155TransferBuilder()\n          .tokenContractAddress(tokenContractAddress)\n          .to(recipientAddress)\n          .from(fromAddress);\n\n        for (const entry of entries) {\n          transferBuilder.entry(parseInt(entry.tokenId, 10), entry.amount);\n        }\n\n        return transferBuilder.build();\n      }\n\n      default:\n        throw new Error(`Unsupported NFT type: ${params.type}`);\n    }\n  }\n\n  /**\n   * Fetch the gas price from the explorer\n   * @param {string} wrongChainCoin - the coin that we're getting gas price for\n   * @param {string} apiKey - optional API key to use instead of the one from the environment\n   */\n  async getGasPriceFromExternalAPI(wrongChainCoin: string, apiKey?: string): Promise<BN> {\n    try {\n      const res = await this.recoveryBlockchainExplorerQuery(\n        {\n          chainid: this.getChainId().toString(),\n          module: 'proxy',\n          action: 'eth_gasPrice',\n        },\n        apiKey\n      );\n      const gasPrice = new BN(res.result.slice(2), 16);\n      console.log(` Got gas price: ${gasPrice}`);\n      return gasPrice;\n    } catch (e) {\n      throw new Error(`Failed to get gas price. Please make sure to use the api key of ${wrongChainCoin}`);\n    }\n  }\n\n  /**\n   * Fetch the gas limit from the explorer\n   * @param intendedChain\n   * @param from\n   * @param to\n   * @param data\n   * @param {string} apiKey - optional API key to use instead of the one from the environment\n   */\n  async getGasLimitFromExternalAPI(\n    intendedChain: string,\n    from: string,\n    to: string,\n    data: string,\n    apiKey?: string\n  ): Promise<BN> {\n    try {\n      const res = await this.recoveryBlockchainExplorerQuery(\n        {\n          chainid: this.getChainId().toString(),\n          module: 'proxy',\n          action: 'eth_estimateGas',\n          from,\n          to,\n          data,\n        },\n        apiKey\n      );\n      const gasLimit = new BN(res.result.slice(2), 16);\n      console.log(`Got gas limit: ${gasLimit}`);\n      return gasLimit;\n    } catch (e) {\n      throw new Error(\n        `Failed to get gas limit. Please make sure to use the privateKey aka userKey of ${intendedChain} wallet ${to}`\n      );\n    }\n  }\n\n  /**\n   * Get the balance of bitgoFeeAddress to ensure funds are available to pay fees\n   * @param bitgoFeeAddress\n   * @param gasPrice\n   * @param gasLimit\n   * @param apiKey - optional API key to use instead of the one from the environment\n   */\n  async ensureSufficientBalance(bitgoFeeAddress: string, gasPrice: BN, gasLimit: BN, apiKey?: string): Promise<void> {\n    const bitgoFeeAddressBalance = await this.queryAddressBalance(bitgoFeeAddress, apiKey);\n    const totalGasNeeded = Number(gasPrice.mul(gasLimit));\n    const weiToGwei = 10 ** 9;\n    if (bitgoFeeAddressBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Fee address ${bitgoFeeAddress} has balance ${(bitgoFeeAddressBalance / weiToGwei).toString()} Gwei.` +\n          `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +\n          ` Gwei to perform recoveries. Try sending some ${this.getChain()} to this address then retry.`\n      );\n    }\n  }\n}\n"]}