@bitgo-beta/abstract-eth 1.2.3-alpha.91 → 1.2.3-alpha.93

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.
@@ -9,8 +9,21 @@ exports.AbstractEthLikeNewCoins = exports.optionalDeps = void 0;
9
9
  */
10
10
  const debug_1 = __importDefault(require("debug"));
11
11
  const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
12
+ const bignumber_js_1 = require("bignumber.js");
13
+ const crypto_1 = require("crypto");
14
+ const keccak_1 = __importDefault(require("keccak"));
15
+ const lodash_1 = __importDefault(require("lodash"));
16
+ const secp256k1_1 = __importDefault(require("secp256k1"));
17
+ const bn_js_1 = __importDefault(require("bn.js"));
12
18
  const sdk_core_1 = require("@bitgo-beta/sdk-core");
19
+ const statics_1 = require("@bitgo-beta/statics");
20
+ const sdk_lib_mpc_1 = require("@bitgo-beta/sdk-lib-mpc");
21
+ const tx_1 = require("@ethereumjs/tx");
22
+ const ethereumjs_util_1 = require("ethereumjs-util");
23
+ const eth_sig_util_1 = require("@metamask/eth-sig-util");
24
+ const lib_1 = require("./lib");
13
25
  const abstractEthLikeCoin_1 = require("./abstractEthLikeCoin");
26
+ const ethLikeToken_1 = require("./ethLikeToken");
14
27
  const debug = debug_1.default('bitgo:v2:ethlike');
15
28
  exports.optionalDeps = {
16
29
  get ethAbi() {
@@ -57,63 +70,44 @@ exports.optionalDeps = {
57
70
  class AbstractEthLikeNewCoins extends abstractEthLikeCoin_1.AbstractEthLikeCoin {
58
71
  constructor(bitgo, staticsCoin) {
59
72
  super(bitgo, staticsCoin);
73
+ /**
74
+ * Get the data required to make an ETH function call defined by the given types and values
75
+ *
76
+ * @param {string} functionName - The name of the function being called, e.g. transfer
77
+ * @param types The types of the function call in order
78
+ * @param values The values of the function call in order
79
+ * @return {Buffer} The combined data for the function call
80
+ */
81
+ this.getMethodCallData = (functionName, types, values) => {
82
+ return Buffer.concat([
83
+ // function signature
84
+ exports.optionalDeps.ethAbi.methodID(functionName, types),
85
+ // function arguments
86
+ exports.optionalDeps.ethAbi.rawEncode(types, values),
87
+ ]);
88
+ };
60
89
  if (!staticsCoin) {
61
90
  throw new Error('missing required constructor parameter staticsCoin');
62
91
  }
63
- this._staticsCoin = staticsCoin;
64
- AbstractEthLikeNewCoins._ethLikeCoin = staticsCoin;
92
+ this.staticsCoin = staticsCoin;
65
93
  this.sendMethodName = 'sendMultiSig';
66
94
  }
67
- getChain() {
68
- return this._staticsCoin.name;
69
- }
70
95
  /**
71
- * Get the base chain that the coin exists on.
96
+ * Method to return the coin's network object
97
+ * @returns {EthLikeNetwork | undefined}
72
98
  */
73
- getBaseChain() {
74
- return this.getChain();
75
- }
76
- getFamily() {
77
- return this._staticsCoin.family;
78
- }
79
99
  getNetwork() {
80
100
  var _a;
81
- return (_a = this._staticsCoin) === null || _a === void 0 ? void 0 : _a.network;
82
- }
83
- getFullName() {
84
- return this._staticsCoin.fullName;
85
- }
86
- getBaseFactor() {
87
- return Math.pow(10, this._staticsCoin.decimalPlaces);
88
- }
89
- /** @inheritDoc */
90
- isEVM() {
91
- return true;
92
- }
93
- valuelessTransferAllowed() {
94
- return true;
101
+ return (_a = this.staticsCoin) === null || _a === void 0 ? void 0 : _a.network;
95
102
  }
96
103
  /**
97
104
  * Evaluates whether an address string is valid for this coin
98
- * @param address
105
+ * @param {string} address
106
+ * @returns {boolean} True if address is the valid ethlike adderss
99
107
  */
100
108
  isValidAddress(address) {
101
109
  return exports.optionalDeps.ethUtil.isValidAddress(exports.optionalDeps.ethUtil.addHexPrefix(address));
102
110
  }
103
- /**
104
- * Return boolean indicating whether input is valid public key for the coin.
105
- *
106
- * @param {String} pub the pub to be checked
107
- * @returns {Boolean} is it valid?
108
- */
109
- isValidPub(pub) {
110
- try {
111
- return utxo_lib_1.bip32.fromBase58(pub).isNeutered();
112
- }
113
- catch (e) {
114
- return false;
115
- }
116
- }
117
111
  /**
118
112
  * Flag for sending data along with transactions
119
113
  * @returns {boolean} True if okay to send tx data (ETH), false otherwise
@@ -142,6 +136,1663 @@ class AbstractEthLikeNewCoins extends abstractEthLikeCoin_1.AbstractEthLikeCoin
142
136
  getDefaultExpireTime() {
143
137
  return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
144
138
  }
139
+ /**
140
+ * Method to get the custom chain common object based on params from recovery
141
+ * @param {number} chainId - the chain id of the custom chain
142
+ * @returns {EthLikeCommon.default}
143
+ */
144
+ static getCustomChainCommon(chainId) {
145
+ const coinName = statics_1.CoinMap.coinNameFromChainId(chainId);
146
+ const coin = statics_1.coins.get(coinName);
147
+ const ethLikeCommon = lib_1.getCommon(coin.network);
148
+ return ethLikeCommon;
149
+ }
150
+ /**
151
+ * Gets correct Eth Common object based on params from either recovery or tx building
152
+ * @param {EIP1559} eip1559 - configs that specify whether we should construct an eip1559 tx
153
+ * @param {ReplayProtectionOptions} replayProtectionOptions - check if chain id supports replay protection
154
+ * @returns {EthLikeCommon.default}
155
+ */
156
+ static getEthLikeCommon(eip1559, replayProtectionOptions) {
157
+ var _a;
158
+ // if eip1559 params are specified, default to london hardfork, otherwise,
159
+ // default to tangerine whistle to avoid replay protection issues
160
+ const defaultHardfork = !!eip1559 ? 'london' : exports.optionalDeps.EthCommon.Hardfork.TangerineWhistle;
161
+ const ethLikeCommon = AbstractEthLikeNewCoins.getCustomChainCommon(replayProtectionOptions === null || replayProtectionOptions === void 0 ? void 0 : replayProtectionOptions.chain);
162
+ ethLikeCommon.setHardfork((_a = replayProtectionOptions === null || replayProtectionOptions === void 0 ? void 0 : replayProtectionOptions.hardfork) !== null && _a !== void 0 ? _a : defaultHardfork);
163
+ return ethLikeCommon;
164
+ }
165
+ /**
166
+ * Method to build the tx object
167
+ * @param {BuildTransactionParams} params - params to build transaction
168
+ * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}
169
+ */
170
+ static buildTransaction(params) {
171
+ // if eip1559 params are specified, default to london hardfork, otherwise,
172
+ // default to tangerine whistle to avoid replay protection issues
173
+ const ethLikeCommon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);
174
+ const baseParams = {
175
+ to: params.to,
176
+ nonce: params.nonce,
177
+ value: params.value,
178
+ data: params.data,
179
+ gasLimit: new exports.optionalDeps.ethUtil.BN(params.gasLimit),
180
+ };
181
+ const unsignedEthTx = !!params.eip1559
182
+ ? exports.optionalDeps.EthTx.FeeMarketEIP1559Transaction.fromTxData({
183
+ ...baseParams,
184
+ maxFeePerGas: new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas),
185
+ maxPriorityFeePerGas: new exports.optionalDeps.ethUtil.BN(params.eip1559.maxPriorityFeePerGas),
186
+ }, { common: ethLikeCommon })
187
+ : exports.optionalDeps.EthTx.Transaction.fromTxData({
188
+ ...baseParams,
189
+ gasPrice: new exports.optionalDeps.ethUtil.BN(params.gasPrice),
190
+ }, { common: ethLikeCommon });
191
+ return unsignedEthTx;
192
+ }
193
+ /**
194
+ * Query explorer for the balance of an address
195
+ * @param {String} address - the ETHLike address
196
+ * @returns {BigNumber} address balance
197
+ */
198
+ async queryAddressBalance(address) {
199
+ const result = await this.recoveryBlockchainExplorerQuery({
200
+ module: 'account',
201
+ action: 'balance',
202
+ address: address,
203
+ });
204
+ // throw if the result does not exist or the result is not a valid number
205
+ if (!result || !result.result || isNaN(result.result)) {
206
+ throw new Error(`Could not obtain address balance for ${address} from the explorer, got: ${result.result}`);
207
+ }
208
+ return new exports.optionalDeps.ethUtil.BN(result.result, 10);
209
+ }
210
+ /**
211
+ * @param {Recipient[]} recipients - the recipients of the transaction
212
+ * @param {number} expireTime - the expire time of the transaction
213
+ * @param {number} contractSequenceId - the contract sequence id of the transaction
214
+ * @returns {string}
215
+ */
216
+ getOperationSha3ForExecuteAndConfirm(recipients, expireTime, contractSequenceId) {
217
+ if (!recipients || !Array.isArray(recipients)) {
218
+ throw new Error('expecting array of recipients');
219
+ }
220
+ // Right now we only support 1 recipient
221
+ if (recipients.length !== 1) {
222
+ throw new Error('must send to exactly 1 recipient');
223
+ }
224
+ if (!lodash_1.default.isNumber(expireTime)) {
225
+ throw new Error('expireTime must be number of seconds since epoch');
226
+ }
227
+ if (!lodash_1.default.isNumber(contractSequenceId)) {
228
+ throw new Error('contractSequenceId must be number');
229
+ }
230
+ // Check inputs
231
+ recipients.forEach(function (recipient) {
232
+ if (!lodash_1.default.isString(recipient.address) ||
233
+ !exports.optionalDeps.ethUtil.isValidAddress(exports.optionalDeps.ethUtil.addHexPrefix(recipient.address))) {
234
+ throw new Error('Invalid address: ' + recipient.address);
235
+ }
236
+ let amount;
237
+ try {
238
+ amount = new bignumber_js_1.BigNumber(recipient.amount);
239
+ }
240
+ catch (e) {
241
+ throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');
242
+ }
243
+ recipient.amount = amount.toFixed(0);
244
+ if (recipient.data && !lodash_1.default.isString(recipient.data)) {
245
+ throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');
246
+ }
247
+ });
248
+ const recipient = recipients[0];
249
+ return exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId)));
250
+ }
251
+ /**
252
+ * Get transfer operation for coin
253
+ * @param {Recipient} recipient - recipient info
254
+ * @param {number} expireTime - expiry time
255
+ * @param {number} contractSequenceId - sequence id
256
+ * @returns {Array} operation array
257
+ */
258
+ getOperation(recipient, expireTime, contractSequenceId) {
259
+ const network = this.getNetwork();
260
+ return [
261
+ ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
262
+ [
263
+ network.nativeCoinOperationHashPrefix,
264
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
265
+ recipient.amount,
266
+ Buffer.from(exports.optionalDeps.ethUtil.stripHexPrefix(exports.optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),
267
+ expireTime,
268
+ contractSequenceId,
269
+ ],
270
+ ];
271
+ }
272
+ /**
273
+ * Queries the contract (via explorer API) for the next sequence ID
274
+ * @param {String} address - address of the contract
275
+ * @returns {Promise<Number>} sequence ID
276
+ */
277
+ async querySequenceId(address) {
278
+ // Get sequence ID using contract call
279
+ const sequenceIdMethodSignature = exports.optionalDeps.ethAbi.methodID('getNextSequenceId', []);
280
+ const sequenceIdArgs = exports.optionalDeps.ethAbi.rawEncode([], []);
281
+ const sequenceIdData = Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');
282
+ const result = await this.recoveryBlockchainExplorerQuery({
283
+ module: 'proxy',
284
+ action: 'eth_call',
285
+ to: address,
286
+ data: sequenceIdData,
287
+ tag: 'latest',
288
+ });
289
+ if (!result || !result.result) {
290
+ throw new Error('Could not obtain sequence ID from explorer, got: ' + result.result);
291
+ }
292
+ const sequenceIdHex = result.result;
293
+ return new exports.optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();
294
+ }
295
+ /**
296
+ * Recover an unsupported token from a BitGo multisig wallet
297
+ * This builds a half-signed transaction, for which there will be an admin route to co-sign and broadcast. Optionally
298
+ * the user can set params.broadcast = true and the half-signed tx will be sent to BitGo for cosigning and broadcasting
299
+ * @param {RecoverTokenOptions} params
300
+ * @param {Wallet} params.wallet - the wallet to recover the token from
301
+ * @param {string} params.tokenContractAddress - the contract address of the unsupported token
302
+ * @param {string} params.recipient - the destination address recovered tokens should be sent to
303
+ * @param {string} params.walletPassphrase - the wallet passphrase
304
+ * @param {string} params.prv - the xprv
305
+ * @param {boolean} params.broadcast - if true, we will automatically submit the half-signed tx to BitGo for cosigning and broadcasting
306
+ * @returns {Promise<RecoverTokenTransaction>}
307
+ */
308
+ async recoverToken(params) {
309
+ const network = this.getNetwork();
310
+ if (!lodash_1.default.isObject(params)) {
311
+ throw new Error(`recoverToken must be passed a params object. Got ${params} (type ${typeof params})`);
312
+ }
313
+ if (lodash_1.default.isUndefined(params.tokenContractAddress) || !lodash_1.default.isString(params.tokenContractAddress)) {
314
+ throw new Error(`tokenContractAddress must be a string, got ${params.tokenContractAddress} (type ${typeof params.tokenContractAddress})`);
315
+ }
316
+ if (!this.isValidAddress(params.tokenContractAddress)) {
317
+ throw new Error('tokenContractAddress not a valid address');
318
+ }
319
+ if (lodash_1.default.isUndefined(params.wallet) || !(params.wallet instanceof sdk_core_1.Wallet)) {
320
+ throw new Error(`wallet must be a wallet instance, got ${params.wallet} (type ${typeof params.wallet})`);
321
+ }
322
+ if (lodash_1.default.isUndefined(params.recipient) || !lodash_1.default.isString(params.recipient)) {
323
+ throw new Error(`recipient must be a string, got ${params.recipient} (type ${typeof params.recipient})`);
324
+ }
325
+ if (!this.isValidAddress(params.recipient)) {
326
+ throw new Error('recipient not a valid address');
327
+ }
328
+ if (!exports.optionalDeps.ethUtil.bufferToHex || !exports.optionalDeps.ethAbi.soliditySHA3) {
329
+ throw new Error('ethereum not fully supported in this environment');
330
+ }
331
+ // Get token balance from external API
332
+ const coinSpecific = params.wallet.coinSpecific();
333
+ if (!coinSpecific || !lodash_1.default.isString(coinSpecific.baseAddress)) {
334
+ throw new Error('missing required coin specific property baseAddress');
335
+ }
336
+ const recoveryAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, coinSpecific.baseAddress);
337
+ if (params.broadcast) {
338
+ // We're going to create a normal ETH transaction that sends an amount of 0 ETH to the
339
+ // tokenContractAddress and encode the unsupported-token-send data in the data field
340
+ // #tricksy
341
+ const sendMethodArgs = [
342
+ {
343
+ name: '_to',
344
+ type: 'address',
345
+ value: params.recipient,
346
+ },
347
+ {
348
+ name: '_value',
349
+ type: 'uint256',
350
+ value: recoveryAmount.toString(10),
351
+ },
352
+ ];
353
+ const methodSignature = exports.optionalDeps.ethAbi.methodID('transfer', lodash_1.default.map(sendMethodArgs, 'type'));
354
+ const encodedArgs = exports.optionalDeps.ethAbi.rawEncode(lodash_1.default.map(sendMethodArgs, 'type'), lodash_1.default.map(sendMethodArgs, 'value'));
355
+ const sendData = Buffer.concat([methodSignature, encodedArgs]);
356
+ const broadcastParams = {
357
+ address: params.tokenContractAddress,
358
+ amount: '0',
359
+ data: sendData.toString('hex'),
360
+ };
361
+ if (params.walletPassphrase) {
362
+ broadcastParams.walletPassphrase = params.walletPassphrase;
363
+ }
364
+ else if (params.prv) {
365
+ broadcastParams.prv = params.prv;
366
+ }
367
+ return await params.wallet.send(broadcastParams);
368
+ }
369
+ const recipient = {
370
+ address: params.recipient,
371
+ amount: recoveryAmount.toString(10),
372
+ };
373
+ // This signature will be valid for one week
374
+ const expireTime = Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
375
+ // Get sequence ID. We do this by building a 'fake' eth transaction, so the platform will increment and return us the new sequence id
376
+ // This _does_ require the user to have a non-zero wallet balance
377
+ const { nextContractSequenceId, gasPrice, gasLimit } = (await params.wallet.prebuildTransaction({
378
+ recipients: [
379
+ {
380
+ address: params.recipient,
381
+ amount: '1',
382
+ },
383
+ ],
384
+ }));
385
+ // these recoveries need to be processed by support, but if the customer sends any transactions before recovery is
386
+ // complete the sequence ID will be invalid. artificially inflate the sequence ID to allow more time for processing
387
+ const safeSequenceId = nextContractSequenceId + 1000;
388
+ // Build sendData for ethereum tx
389
+ const operationTypes = ['string', 'address', 'uint', 'address', 'uint', 'uint'];
390
+ const operationArgs = [
391
+ // Token operation has prefix has been added here so that ether operation hashes, signatures cannot be re-used for tokenSending
392
+ network.tokenOperationHashPrefix,
393
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
394
+ recipient.amount,
395
+ new exports.optionalDeps.ethUtil.BN(exports.optionalDeps.ethUtil.stripHexPrefix(params.tokenContractAddress), 16),
396
+ expireTime,
397
+ safeSequenceId,
398
+ ];
399
+ const operationHash = exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(operationTypes, operationArgs));
400
+ const userPrv = await params.wallet.getPrv({
401
+ prv: params.prv,
402
+ walletPassphrase: params.walletPassphrase,
403
+ });
404
+ const signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userPrv));
405
+ return {
406
+ halfSigned: {
407
+ recipient: recipient,
408
+ expireTime: expireTime,
409
+ contractSequenceId: safeSequenceId,
410
+ operationHash: operationHash,
411
+ signature: signature,
412
+ gasLimit: gasLimit,
413
+ gasPrice: gasPrice,
414
+ tokenContractAddress: params.tokenContractAddress,
415
+ walletId: params.wallet.id(),
416
+ },
417
+ };
418
+ }
419
+ /**
420
+ * Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key
421
+ * @param {PrecreateBitGoOptions} params
422
+ * @param {string} params.enterprise {String} the enterprise id to associate with this key
423
+ * @param {string} params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)
424
+ * @returns {void}
425
+ */
426
+ preCreateBitGo(params) {
427
+ // We always need params object, since either enterprise or newFeeAddress is required
428
+ if (!lodash_1.default.isObject(params)) {
429
+ throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);
430
+ }
431
+ if (lodash_1.default.isUndefined(params.enterprise) && lodash_1.default.isUndefined(params.newFeeAddress)) {
432
+ 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.');
433
+ }
434
+ // Check whether key should be an enterprise key or a BitGo key for a new fee address
435
+ if (!lodash_1.default.isUndefined(params.enterprise) && !lodash_1.default.isUndefined(params.newFeeAddress)) {
436
+ throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);
437
+ }
438
+ if (!lodash_1.default.isUndefined(params.enterprise) && !lodash_1.default.isString(params.enterprise)) {
439
+ throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);
440
+ }
441
+ if (!lodash_1.default.isUndefined(params.newFeeAddress) && !lodash_1.default.isBoolean(params.newFeeAddress)) {
442
+ throw new Error(`newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`);
443
+ }
444
+ }
445
+ /**
446
+ * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address
447
+ * @param {string} address
448
+ * @returns {Promise<number>}
449
+ */
450
+ async getAddressNonce(address) {
451
+ // Get nonce for backup key (should be 0)
452
+ let nonce = 0;
453
+ const result = await this.recoveryBlockchainExplorerQuery({
454
+ module: 'account',
455
+ action: 'txlist',
456
+ address,
457
+ });
458
+ if (!result || !Array.isArray(result.result)) {
459
+ throw new Error('Unable to find next nonce from Etherscan, got: ' + JSON.stringify(result));
460
+ }
461
+ const backupKeyTxList = result.result;
462
+ if (backupKeyTxList.length > 0) {
463
+ // Calculate last nonce used
464
+ const outgoingTxs = backupKeyTxList.filter((tx) => tx.from === address);
465
+ nonce = outgoingTxs.length;
466
+ }
467
+ return nonce;
468
+ }
469
+ /**
470
+ * Helper function for recover()
471
+ * This transforms the unsigned transaction information into a format the BitGo offline vault expects
472
+ * @param {UnformattedTxInfo} txInfo - tx info
473
+ * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object
474
+ * @param {string} userKey - the user's key
475
+ * @param {string} backupKey - the backup key
476
+ * @param {Buffer} gasPrice - gas price for the tx
477
+ * @param {number} gasLimit - gas limit for the tx
478
+ * @param {EIP1559} eip1559 - eip1559 params
479
+ * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options
480
+ * @returns {Promise<OfflineVaultTxInfo>}
481
+ */
482
+ async formatForOfflineVault(txInfo, ethTx, userKey, backupKey, gasPrice, gasLimit, eip1559, replayProtectionOptions) {
483
+ if (!ethTx.to) {
484
+ throw new Error('Eth tx must have a `to` address');
485
+ }
486
+ const backupHDNode = utxo_lib_1.bip32.fromBase58(backupKey);
487
+ const backupSigningKey = backupHDNode.publicKey;
488
+ const response = {
489
+ tx: ethTx.serialize().toString('hex'),
490
+ userKey,
491
+ backupKey,
492
+ coin: this.getChain(),
493
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
494
+ gasLimit,
495
+ recipients: [txInfo.recipient],
496
+ walletContractAddress: ethTx.to.toString(),
497
+ amount: txInfo.recipient.amount,
498
+ backupKeyNonce: await this.getAddressNonce(`0x${exports.optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`),
499
+ eip1559,
500
+ replayProtectionOptions,
501
+ };
502
+ lodash_1.default.extend(response, txInfo);
503
+ response.nextContractSequenceId = response.contractSequenceId;
504
+ return response;
505
+ }
506
+ /**
507
+ * Helper function for recover()
508
+ * This transforms the unsigned transaction information into a format the BitGo offline vault expects
509
+ * @param {UnformattedTxInfo} txInfo - tx info
510
+ * @param {EthLikeTxLib.Transaction | EthLikeTxLib.FeeMarketEIP1559Transaction} ethTx - the ethereumjs tx object
511
+ * @param {string} userKey - the user's key
512
+ * @param {string} backupKey - the backup key
513
+ * @param {Buffer} gasPrice - gas price for the tx
514
+ * @param {number} gasLimit - gas limit for the tx
515
+ * @param {number} backupKeyNonce - the nonce of the backup key address
516
+ * @param {EIP1559} eip1559 - eip1559 params
517
+ * @param {ReplayProtectionOptions} replayProtectionOptions - replay protection options
518
+ * @returns {Promise<OfflineVaultTxInfo>}
519
+ */
520
+ formatForOfflineVaultTSS(txInfo, ethTx, userKey, backupKey, gasPrice, gasLimit, backupKeyNonce, eip1559, replayProtectionOptions) {
521
+ if (!ethTx.to) {
522
+ throw new Error('Eth tx must have a `to` address');
523
+ }
524
+ const response = {
525
+ tx: ethTx.serialize().toString('hex'),
526
+ txHex: ethTx.getMessageToSign(false).toString(),
527
+ userKey,
528
+ backupKey,
529
+ coin: this.getChain(),
530
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
531
+ gasLimit,
532
+ recipients: [txInfo.recipient],
533
+ walletContractAddress: ethTx.to.toString(),
534
+ amount: txInfo.recipient.amount,
535
+ backupKeyNonce: backupKeyNonce,
536
+ eip1559,
537
+ replayProtectionOptions,
538
+ };
539
+ lodash_1.default.extend(response, txInfo);
540
+ return response;
541
+ }
542
+ /**
543
+ * Check whether the gas price passed in by user are within our max and min bounds
544
+ * If they are not set, set them to the defaults
545
+ * @param {number} userGasPrice - user defined gas price
546
+ * @returns {number} the gas price to use for this transaction
547
+ */
548
+ setGasPrice(userGasPrice) {
549
+ if (!userGasPrice) {
550
+ return statics_1.ethGasConfigs.defaultGasPrice;
551
+ }
552
+ const gasPriceMax = statics_1.ethGasConfigs.maximumGasPrice;
553
+ const gasPriceMin = statics_1.ethGasConfigs.minimumGasPrice;
554
+ if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {
555
+ throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);
556
+ }
557
+ return userGasPrice;
558
+ }
559
+ /**
560
+ * Check whether gas limit passed in by user are within our max and min bounds
561
+ * If they are not set, set them to the defaults
562
+ * @param {number} userGasLimit user defined gas limit
563
+ * @returns {number} the gas limit to use for this transaction
564
+ */
565
+ setGasLimit(userGasLimit) {
566
+ if (!userGasLimit) {
567
+ return statics_1.ethGasConfigs.defaultGasLimit;
568
+ }
569
+ const gasLimitMax = statics_1.ethGasConfigs.maximumGasLimit;
570
+ const gasLimitMin = statics_1.ethGasConfigs.minimumGasLimit;
571
+ if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {
572
+ throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);
573
+ }
574
+ return userGasLimit;
575
+ }
576
+ /**
577
+ * Helper function for signTransaction for the rare case that SDK is doing the second signature
578
+ * Note: we are expecting this to be called from the offline vault
579
+ * @param {SignFinalOptions.txPrebuild} params.txPrebuild
580
+ * @param {string} params.prv
581
+ * @returns {{txHex: string}}
582
+ */
583
+ async signFinalEthLike(params) {
584
+ var _a;
585
+ const signingKey = new lib_1.KeyPair({ prv: params.prv }).getKeys().prv;
586
+ if (lodash_1.default.isUndefined(signingKey)) {
587
+ throw new Error('missing private key');
588
+ }
589
+ const txBuilder = this.getTransactionBuilder();
590
+ try {
591
+ txBuilder.from((_a = params.txPrebuild.halfSigned) === null || _a === void 0 ? void 0 : _a.txHex);
592
+ }
593
+ catch (e) {
594
+ throw new Error('invalid half-signed transaction');
595
+ }
596
+ txBuilder.sign({ key: signingKey });
597
+ const tx = await txBuilder.build();
598
+ return {
599
+ txHex: tx.toBroadcastFormat(),
600
+ };
601
+ }
602
+ /**
603
+ * Assemble half-sign prebuilt transaction
604
+ * @param {SignTransactionOptions} params
605
+ */
606
+ async signTransaction(params) {
607
+ // Normally the SDK provides the first signature for an EthLike tx, but occasionally it provides the second and final one.
608
+ if (params.isLastSignature) {
609
+ // In this case when we're doing the second (final) signature, the logic is different.
610
+ return await this.signFinalEthLike(params);
611
+ }
612
+ const txBuilder = this.getTransactionBuilder();
613
+ txBuilder.from(params.txPrebuild.txHex);
614
+ txBuilder.transfer().key(new lib_1.KeyPair({ prv: params.prv }).getKeys().prv);
615
+ const transaction = await txBuilder.build();
616
+ const recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));
617
+ const txParams = {
618
+ eip1559: params.txPrebuild.eip1559,
619
+ txHex: transaction.toBroadcastFormat(),
620
+ recipients: recipients,
621
+ expiration: params.txPrebuild.expireTime,
622
+ hopTransaction: params.txPrebuild.hopTransaction,
623
+ custodianTransactionId: params.custodianTransactionId,
624
+ expireTime: params.expireTime,
625
+ contractSequenceId: params.txPrebuild.nextContractSequenceId,
626
+ sequenceId: params.sequenceId,
627
+ };
628
+ return { halfSigned: txParams };
629
+ }
630
+ /**
631
+ * Method to validate recovery params
632
+ * @param {RecoverOptions} params
633
+ * @returns {void}
634
+ */
635
+ validateRecoveryParams(params) {
636
+ if (lodash_1.default.isUndefined(params.userKey)) {
637
+ throw new Error('missing userKey');
638
+ }
639
+ if (lodash_1.default.isUndefined(params.backupKey)) {
640
+ throw new Error('missing backupKey');
641
+ }
642
+ if (lodash_1.default.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub') && !params.isTss) {
643
+ throw new Error('missing wallet passphrase');
644
+ }
645
+ if (lodash_1.default.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
646
+ throw new Error('invalid walletContractAddress');
647
+ }
648
+ if (lodash_1.default.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
649
+ throw new Error('invalid recoveryDestination');
650
+ }
651
+ }
652
+ /**
653
+ * Method to sign tss recovery transaction
654
+ * @param {ECDSA.KeyCombined} userKeyCombined
655
+ * @param {ECDSA.KeyCombined} backupKeyCombined
656
+ * @param {string} txHex
657
+ * @param {Object} options
658
+ * @param {EcdsaTypes.SerializedNtilde} options.rangeProofChallenge
659
+ * @returns {Promise<ECDSAMethodTypes.Signature>}
660
+ */
661
+ async signRecoveryTSS(userKeyCombined, backupKeyCombined, txHex, { rangeProofChallenge, } = {}) {
662
+ const MPC = new sdk_core_1.Ecdsa();
663
+ const signerOneIndex = userKeyCombined.xShare.i;
664
+ const signerTwoIndex = backupKeyCombined.xShare.i;
665
+ rangeProofChallenge =
666
+ rangeProofChallenge !== null && rangeProofChallenge !== void 0 ? rangeProofChallenge : sdk_lib_mpc_1.EcdsaTypes.serializeNtildeWithProofs(await sdk_lib_mpc_1.EcdsaRangeProof.generateNtilde());
667
+ const userToBackupPaillierChallenge = await sdk_lib_mpc_1.EcdsaPaillierProof.generateP(sdk_core_1.hexToBigInt(userKeyCombined.yShares[signerTwoIndex].n));
668
+ const backupToUserPaillierChallenge = await sdk_lib_mpc_1.EcdsaPaillierProof.generateP(sdk_core_1.hexToBigInt(backupKeyCombined.yShares[signerOneIndex].n));
669
+ const userXShare = MPC.appendChallenge(userKeyCombined.xShare, rangeProofChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: userToBackupPaillierChallenge }));
670
+ const userYShare = MPC.appendChallenge(userKeyCombined.yShares[signerTwoIndex], rangeProofChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: backupToUserPaillierChallenge }));
671
+ const backupXShare = MPC.appendChallenge(backupKeyCombined.xShare, rangeProofChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: backupToUserPaillierChallenge }));
672
+ const backupYShare = MPC.appendChallenge(backupKeyCombined.yShares[signerOneIndex], rangeProofChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: userToBackupPaillierChallenge }));
673
+ const signShares = await MPC.signShare(userXShare, userYShare);
674
+ const signConvertS21 = await MPC.signConvertStep1({
675
+ xShare: backupXShare,
676
+ yShare: backupYShare,
677
+ kShare: signShares.kShare,
678
+ });
679
+ const signConvertS12 = await MPC.signConvertStep2({
680
+ aShare: signConvertS21.aShare,
681
+ wShare: signShares.wShare,
682
+ });
683
+ const signConvertS21_2 = await MPC.signConvertStep3({
684
+ muShare: signConvertS12.muShare,
685
+ bShare: signConvertS21.bShare,
686
+ });
687
+ const [signCombineOne, signCombineTwo] = [
688
+ MPC.signCombine({
689
+ gShare: signConvertS12.gShare,
690
+ signIndex: {
691
+ i: signConvertS12.muShare.i,
692
+ j: signConvertS12.muShare.j,
693
+ },
694
+ }),
695
+ MPC.signCombine({
696
+ gShare: signConvertS21_2.gShare,
697
+ signIndex: {
698
+ i: signConvertS21_2.signIndex.i,
699
+ j: signConvertS21_2.signIndex.j,
700
+ },
701
+ }),
702
+ ];
703
+ const MESSAGE = Buffer.from(txHex, 'hex');
704
+ const [signA, signB] = [
705
+ MPC.sign(MESSAGE, signCombineOne.oShare, signCombineTwo.dShare, keccak_1.default('keccak256')),
706
+ MPC.sign(MESSAGE, signCombineTwo.oShare, signCombineOne.dShare, keccak_1.default('keccak256')),
707
+ ];
708
+ return MPC.constructSignature([signA, signB]);
709
+ }
710
+ /**
711
+ * Helper which combines key shares of user and backup
712
+ * @param {string} userPublicOrPrivateKeyShare
713
+ * @param {string} backupPrivateOrPublicKeyShare
714
+ * @param {string} walletPassphrase
715
+ * @returns {[ECDSAMethodTypes.KeyCombined, ECDSAMethodTypes.KeyCombined]}
716
+ */
717
+ getKeyCombinedFromTssKeyShares(userPublicOrPrivateKeyShare, backupPrivateOrPublicKeyShare, walletPassphrase) {
718
+ let backupPrv;
719
+ let userPrv;
720
+ try {
721
+ backupPrv = this.bitgo.decrypt({
722
+ input: backupPrivateOrPublicKeyShare,
723
+ password: walletPassphrase,
724
+ });
725
+ userPrv = this.bitgo.decrypt({
726
+ input: userPublicOrPrivateKeyShare,
727
+ password: walletPassphrase,
728
+ });
729
+ }
730
+ catch (e) {
731
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
732
+ }
733
+ const userSigningMaterial = JSON.parse(userPrv);
734
+ const backupSigningMaterial = JSON.parse(backupPrv);
735
+ if (!userSigningMaterial.backupNShare) {
736
+ throw new Error('Invalid user key - missing backupNShare');
737
+ }
738
+ if (!backupSigningMaterial.userNShare) {
739
+ throw new Error('Invalid backup key - missing userNShare');
740
+ }
741
+ const MPC = new sdk_core_1.Ecdsa();
742
+ const userKeyCombined = MPC.keyCombine(userSigningMaterial.pShare, [
743
+ userSigningMaterial.bitgoNShare,
744
+ userSigningMaterial.backupNShare,
745
+ ]);
746
+ const backupKeyCombined = MPC.keyCombine(backupSigningMaterial.pShare, [
747
+ backupSigningMaterial.bitgoNShare,
748
+ backupSigningMaterial.userNShare,
749
+ ]);
750
+ if (userKeyCombined.xShare.y !== backupKeyCombined.xShare.y ||
751
+ userKeyCombined.xShare.chaincode !== backupKeyCombined.xShare.chaincode) {
752
+ throw new Error('Common keychains do not match');
753
+ }
754
+ return [userKeyCombined, backupKeyCombined];
755
+ }
756
+ /**
757
+ * Helper which Adds signatures to tx object and re-serializes tx
758
+ * @param {EthLikeCommon.default} ethCommon
759
+ * @param {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction} tx
760
+ * @param {ECDSAMethodTypes.Signature} signature
761
+ * @returns {EthLikeTxLib.FeeMarketEIP1559Transaction | EthLikeTxLib.Transaction}
762
+ */
763
+ getSignedTxFromSignature(ethCommon, tx, signature) {
764
+ // get signed Tx from signature
765
+ const txData = tx.toJSON();
766
+ const yParity = signature.recid;
767
+ const baseParams = {
768
+ to: txData.to,
769
+ nonce: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.nonce), 'hex'),
770
+ value: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.value), 'hex'),
771
+ gasLimit: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.gasLimit), 'hex'),
772
+ data: txData.data,
773
+ r: ethereumjs_util_1.addHexPrefix(signature.r),
774
+ s: ethereumjs_util_1.addHexPrefix(signature.s),
775
+ };
776
+ let finalTx;
777
+ if (txData.maxFeePerGas && txData.maxPriorityFeePerGas) {
778
+ finalTx = tx_1.FeeMarketEIP1559Transaction.fromTxData({
779
+ ...baseParams,
780
+ maxPriorityFeePerGas: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.maxPriorityFeePerGas), 'hex'),
781
+ maxFeePerGas: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.maxFeePerGas), 'hex'),
782
+ v: new bn_js_1.default(yParity.toString()),
783
+ }, { common: ethCommon });
784
+ }
785
+ else if (txData.gasPrice) {
786
+ const v = BigInt(35) + BigInt(yParity) + BigInt(ethCommon.chainIdBN().toNumber()) * BigInt(2);
787
+ finalTx = tx_1.Transaction.fromTxData({
788
+ ...baseParams,
789
+ v: new bn_js_1.default(v.toString()),
790
+ gasPrice: new bn_js_1.default(ethereumjs_util_1.stripHexPrefix(txData.gasPrice.toString()), 'hex'),
791
+ }, { common: ethCommon });
792
+ }
793
+ return finalTx;
794
+ }
795
+ /**
796
+ * Builds a funds recovery transaction without BitGo
797
+ * @param params
798
+ * @param {string} params.userKey - [encrypted] xprv
799
+ * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
800
+ * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
801
+ * @param {string} params.walletContractAddress - the ETH address of the wallet contract
802
+ * @param {string} params.krsProvider - necessary if backup key is held by KRS
803
+ * @param {string} params.recoveryDestination - target address to send recovered funds to
804
+ * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn
805
+ * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn
806
+ */
807
+ async recover(params) {
808
+ if (params.isTss) {
809
+ return this.recoverTSS(params);
810
+ }
811
+ return this.recoverEthLike(params);
812
+ }
813
+ /**
814
+ * Builds a funds recovery transaction without BitGo for non-TSS transaction
815
+ * @param params
816
+ * @param {string} params.userKey [encrypted] xprv or xpub
817
+ * @param {string} params.backupKey [encrypted] xprv or xpub if the xprv is held by a KRS provider
818
+ * @param {string} params.walletPassphrase used to decrypt userKey and backupKey
819
+ * @param {string} params.walletContractAddress the EthLike address of the wallet contract
820
+ * @param {string} params.krsProvider necessary if backup key is held by KRS
821
+ * @param {string} params.recoveryDestination target address to send recovered funds to
822
+ * @param {string} params.bitgoFeeAddress wrong chain wallet fee address for evm based cross chain recovery txn
823
+ * @param {string} params.bitgoDestinationAddress target bitgo address where fee will be sent for evm based cross chain recovery txn
824
+ * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}
825
+ */
826
+ async recoverEthLike(params) {
827
+ // bitgoFeeAddress is only defined when it is a evm cross chain recovery
828
+ // as we use fee from this wrong chain address for the recovery txn on the correct chain.
829
+ if (params.bitgoFeeAddress) {
830
+ return this.recoverEthLikeforEvmBasedRecovery(params);
831
+ }
832
+ this.validateRecoveryParams(params);
833
+ const isUnsignedSweep = sdk_core_1.getIsUnsignedSweep(params);
834
+ // Clean up whitespace from entered values
835
+ let userKey = params.userKey.replace(/\s/g, '');
836
+ const backupKey = params.backupKey.replace(/\s/g, '');
837
+ const gasLimit = new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
838
+ const gasPrice = params.eip1559
839
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
840
+ : new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
841
+ if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
842
+ try {
843
+ userKey = this.bitgo.decrypt({
844
+ input: userKey,
845
+ password: params.walletPassphrase,
846
+ });
847
+ }
848
+ catch (e) {
849
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
850
+ }
851
+ }
852
+ let backupKeyAddress;
853
+ let backupSigningKey;
854
+ if (isUnsignedSweep) {
855
+ const backupHDNode = utxo_lib_1.bip32.fromBase58(backupKey);
856
+ backupSigningKey = backupHDNode.publicKey;
857
+ backupKeyAddress = `0x${exports.optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`;
858
+ }
859
+ else {
860
+ // Decrypt backup private key and get address
861
+ let backupPrv;
862
+ try {
863
+ backupPrv = this.bitgo.decrypt({
864
+ input: backupKey,
865
+ password: params.walletPassphrase,
866
+ });
867
+ }
868
+ catch (e) {
869
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
870
+ }
871
+ const keyPair = new lib_1.KeyPair({ prv: backupPrv });
872
+ backupSigningKey = keyPair.getKeys().prv;
873
+ if (!backupSigningKey) {
874
+ throw new Error('no private key');
875
+ }
876
+ backupKeyAddress = keyPair.getAddress();
877
+ }
878
+ const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);
879
+ // get balance of backupKey to ensure funds are available to pay fees
880
+ const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);
881
+ const totalGasNeeded = gasPrice.mul(gasLimit);
882
+ const weiToGwei = 10 ** 9;
883
+ if (backupKeyBalance.lt(totalGasNeeded)) {
884
+ throw new Error(`Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +
885
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
886
+ ` Gwei to perform recoveries. Try sending some MATIC to this address then retry.`);
887
+ }
888
+ // get balance of wallet
889
+ const txAmount = await this.queryAddressBalance(params.walletContractAddress);
890
+ // build recipients object
891
+ const recipients = [
892
+ {
893
+ address: params.recoveryDestination,
894
+ amount: txAmount.toString(10),
895
+ },
896
+ ];
897
+ // Get sequence ID using contract call
898
+ // we need to wait between making two explorer api calls to avoid getting banned
899
+ await new Promise((resolve) => setTimeout(resolve, 1000));
900
+ const sequenceId = await this.querySequenceId(params.walletContractAddress);
901
+ let operationHash, signature;
902
+ // Get operation hash and sign it
903
+ if (!isUnsignedSweep) {
904
+ operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);
905
+ signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userKey));
906
+ try {
907
+ sdk_core_1.Util.ecRecoverEthAddress(operationHash, signature);
908
+ }
909
+ catch (e) {
910
+ throw new Error('Invalid signature');
911
+ }
912
+ }
913
+ const txInfo = {
914
+ recipient: recipients[0],
915
+ expireTime: this.getDefaultExpireTime(),
916
+ contractSequenceId: sequenceId,
917
+ operationHash: operationHash,
918
+ signature: signature,
919
+ gasLimit: gasLimit.toString(10),
920
+ };
921
+ const txBuilder = this.getTransactionBuilder();
922
+ txBuilder.counter(backupKeyNonce);
923
+ txBuilder.contract(params.walletContractAddress);
924
+ let txFee;
925
+ if (params.eip1559) {
926
+ txFee = {
927
+ eip1559: {
928
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
929
+ maxFeePerGas: params.eip1559.maxFeePerGas,
930
+ },
931
+ };
932
+ }
933
+ else {
934
+ txFee = { fee: gasPrice.toString() };
935
+ }
936
+ txBuilder.fee({
937
+ ...txFee,
938
+ gasLimit: gasLimit.toString(),
939
+ });
940
+ const transferBuilder = txBuilder.transfer();
941
+ transferBuilder
942
+ .amount(recipients[0].amount)
943
+ .contractSequenceId(sequenceId)
944
+ .expirationTime(this.getDefaultExpireTime())
945
+ .to(params.recoveryDestination);
946
+ const tx = await txBuilder.build();
947
+ if (isUnsignedSweep) {
948
+ const response = {
949
+ txHex: tx.toBroadcastFormat(),
950
+ userKey,
951
+ backupKey,
952
+ coin: this.getChain(),
953
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
954
+ gasLimit,
955
+ recipients: [txInfo.recipient],
956
+ walletContractAddress: tx.toJson().to,
957
+ amount: txInfo.recipient.amount,
958
+ backupKeyNonce,
959
+ eip1559: params.eip1559,
960
+ };
961
+ lodash_1.default.extend(response, txInfo);
962
+ response.nextContractSequenceId = response.contractSequenceId;
963
+ return response;
964
+ }
965
+ txBuilder.transfer().key(new lib_1.KeyPair({ prv: userKey }).getKeys().prv);
966
+ txBuilder.sign({ key: backupSigningKey });
967
+ const signedTx = await txBuilder.build();
968
+ return {
969
+ id: signedTx.toJson().id,
970
+ tx: signedTx.toBroadcastFormat(),
971
+ };
972
+ }
973
+ /**
974
+ * Builds a unsigned (for cold, custody wallet) or
975
+ * half-signed (for hot wallet) evm cross chain recovery transaction with
976
+ * same expected arguments as recover method.
977
+ * This helps recover funds from evm based wrong chain.
978
+ * @param {RecoverOptions} params
979
+ * @returns {Promise<RecoveryInfo | OfflineVaultTxInfo>}
980
+ */
981
+ async recoverEthLikeforEvmBasedRecovery(params) {
982
+ var _a, _b, _c, _d, _e;
983
+ this.validateEvmBasedRecoveryParams(params);
984
+ // Clean up whitespace from entered values
985
+ const userKey = params.userKey.replace(/\s/g, '');
986
+ const bitgoFeeAddress = (_a = params.bitgoFeeAddress) === null || _a === void 0 ? void 0 : _a.replace(/\s/g, '');
987
+ const bitgoDestinationAddress = (_b = params.bitgoDestinationAddress) === null || _b === void 0 ? void 0 : _b.replace(/\s/g, '');
988
+ const recoveryDestination = (_c = params.recoveryDestination) === null || _c === void 0 ? void 0 : _c.replace(/\s/g, '');
989
+ const walletContractAddress = (_d = params.walletContractAddress) === null || _d === void 0 ? void 0 : _d.replace(/\s/g, '');
990
+ const tokenContractAddress = (_e = params.tokenContractAddress) === null || _e === void 0 ? void 0 : _e.replace(/\s/g, '');
991
+ let userSigningKey;
992
+ let userKeyPrv;
993
+ if (params.walletPassphrase) {
994
+ if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
995
+ try {
996
+ userKeyPrv = this.bitgo.decrypt({
997
+ input: userKey,
998
+ password: params.walletPassphrase,
999
+ });
1000
+ }
1001
+ catch (e) {
1002
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
1003
+ }
1004
+ }
1005
+ const keyPair = new lib_1.KeyPair({ prv: userKeyPrv });
1006
+ userSigningKey = keyPair.getKeys().prv;
1007
+ if (!userSigningKey) {
1008
+ throw new Error('no private key');
1009
+ }
1010
+ }
1011
+ const gasLimit = new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
1012
+ const gasPrice = params.eip1559
1013
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
1014
+ : new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
1015
+ const bitgoFeeAddressNonce = await this.getAddressNonce(bitgoFeeAddress);
1016
+ // get balance of bitgoFeeAddress to ensure funds are available to pay fees
1017
+ const bitgoFeeAddressBalance = await this.queryAddressBalance(bitgoFeeAddress);
1018
+ const totalGasNeeded = gasPrice.mul(gasLimit);
1019
+ const weiToGwei = 10 ** 9;
1020
+ if (bitgoFeeAddressBalance.lt(totalGasNeeded)) {
1021
+ throw new Error(`Fee address ${bitgoFeeAddressBalance} has balance ${(bitgoFeeAddressBalance / weiToGwei).toString()} Gwei.` +
1022
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
1023
+ ` Gwei to perform recoveries. Try sending some ${this.getChain()} to this address then retry.`);
1024
+ }
1025
+ if (tokenContractAddress) {
1026
+ return this.recoverEthLikeTokenforEvmBasedRecovery(params, bitgoFeeAddressNonce, gasLimit, gasPrice, userKey, userSigningKey);
1027
+ }
1028
+ // get balance of wallet
1029
+ const txAmount = await this.queryAddressBalance(walletContractAddress);
1030
+ const bitgoFeePercentage = 0; // TODO: BG-71912 can change the fee% here.
1031
+ const bitgoFeeAmount = txAmount * (bitgoFeePercentage / 100);
1032
+ // build recipients object
1033
+ const recipients = [
1034
+ {
1035
+ address: recoveryDestination,
1036
+ amount: new bignumber_js_1.BigNumber(txAmount).minus(bitgoFeeAmount).toFixed(),
1037
+ },
1038
+ ];
1039
+ if (bitgoFeePercentage > 0) {
1040
+ if (lodash_1.default.isUndefined(bitgoDestinationAddress) || !this.isValidAddress(bitgoDestinationAddress)) {
1041
+ throw new Error('invalid bitgoDestinationAddress');
1042
+ }
1043
+ recipients.push({
1044
+ address: bitgoDestinationAddress,
1045
+ amount: bitgoFeeAmount.toString(10),
1046
+ });
1047
+ }
1048
+ // calculate batch data
1049
+ const BATCH_METHOD_NAME = 'batch';
1050
+ const BATCH_METHOD_TYPES = ['address[]', 'uint256[]'];
1051
+ const batchExecutionInfo = this.getBatchExecutionInfo(recipients);
1052
+ const batchData = exports.optionalDeps.ethUtil.addHexPrefix(this.getMethodCallData(BATCH_METHOD_NAME, BATCH_METHOD_TYPES, batchExecutionInfo.values).toString('hex'));
1053
+ // Get sequence ID using contract call
1054
+ // we need to wait between making two explorer api calls to avoid getting banned
1055
+ await new Promise((resolve) => setTimeout(resolve, 1000));
1056
+ const sequenceId = await this.querySequenceId(walletContractAddress);
1057
+ const txInfo = {
1058
+ recipients: recipients,
1059
+ expireTime: this.getDefaultExpireTime(),
1060
+ contractSequenceId: sequenceId,
1061
+ gasLimit: gasLimit.toString(10),
1062
+ isEvmBasedCrossChainRecovery: true,
1063
+ };
1064
+ const network = this.getNetwork();
1065
+ const batcherContractAddress = network === null || network === void 0 ? void 0 : network.batcherContractAddress;
1066
+ const txBuilder = this.getTransactionBuilder();
1067
+ txBuilder.counter(bitgoFeeAddressNonce);
1068
+ txBuilder.contract(walletContractAddress);
1069
+ let txFee;
1070
+ if (params.eip1559) {
1071
+ txFee = {
1072
+ eip1559: {
1073
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
1074
+ maxFeePerGas: params.eip1559.maxFeePerGas,
1075
+ },
1076
+ };
1077
+ }
1078
+ else {
1079
+ txFee = { fee: gasPrice.toString() };
1080
+ }
1081
+ txBuilder.fee({
1082
+ ...txFee,
1083
+ gasLimit: gasLimit.toString(),
1084
+ });
1085
+ const transferBuilder = txBuilder.transfer();
1086
+ transferBuilder
1087
+ .amount(batchExecutionInfo.totalAmount)
1088
+ .contractSequenceId(sequenceId)
1089
+ .expirationTime(this.getDefaultExpireTime())
1090
+ .to(batcherContractAddress)
1091
+ .data(batchData);
1092
+ if (params.walletPassphrase) {
1093
+ txBuilder.transfer().key(userSigningKey);
1094
+ }
1095
+ const tx = await txBuilder.build();
1096
+ const response = {
1097
+ txHex: tx.toBroadcastFormat(),
1098
+ userKey,
1099
+ coin: this.getChain(),
1100
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1101
+ gasLimit,
1102
+ recipients: txInfo.recipients,
1103
+ walletContractAddress: tx.toJson().to,
1104
+ amount: batchExecutionInfo.totalAmount,
1105
+ backupKeyNonce: bitgoFeeAddressNonce,
1106
+ eip1559: params.eip1559,
1107
+ };
1108
+ lodash_1.default.extend(response, txInfo);
1109
+ response.nextContractSequenceId = response.contractSequenceId;
1110
+ if (params.walletPassphrase) {
1111
+ const halfSignedTxn = {
1112
+ halfSigned: {
1113
+ txHex: tx.toBroadcastFormat(),
1114
+ recipients: txInfo.recipients,
1115
+ expireTime: txInfo.expireTime,
1116
+ },
1117
+ };
1118
+ lodash_1.default.extend(response, halfSignedTxn);
1119
+ const feesUsed = {
1120
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1121
+ gasLimit: exports.optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),
1122
+ };
1123
+ response['feesUsed'] = feesUsed;
1124
+ }
1125
+ return response;
1126
+ }
1127
+ /**
1128
+ * Query explorer for the balance of an address for a token
1129
+ * @param {string} tokenContractAddress - address where the token smart contract is hosted
1130
+ * @param {string} walletContractAddress - address of the wallet
1131
+ * @returns {BigNumber} token balaance in base units
1132
+ */
1133
+ async queryAddressTokenBalance(tokenContractAddress, walletContractAddress) {
1134
+ if (!exports.optionalDeps.ethUtil.isValidAddress(tokenContractAddress)) {
1135
+ throw new Error('cannot get balance for invalid token address');
1136
+ }
1137
+ if (!exports.optionalDeps.ethUtil.isValidAddress(walletContractAddress)) {
1138
+ throw new Error('cannot get token balance for invalid wallet address');
1139
+ }
1140
+ const result = await this.recoveryBlockchainExplorerQuery({
1141
+ module: 'account',
1142
+ action: 'tokenbalance',
1143
+ contractaddress: tokenContractAddress,
1144
+ address: walletContractAddress,
1145
+ tag: 'latest',
1146
+ });
1147
+ // throw if the result does not exist or the result is not a valid number
1148
+ if (!result || !result.result || isNaN(result.result)) {
1149
+ throw new Error(`Could not obtain token address balance for ${tokenContractAddress} from Etherscan, got: ${result.result}`);
1150
+ }
1151
+ return new exports.optionalDeps.ethUtil.BN(result.result, 10);
1152
+ }
1153
+ async recoverEthLikeTokenforEvmBasedRecovery(params, bitgoFeeAddressNonce, gasLimit, gasPrice, userKey, userSigningKey) {
1154
+ var _a;
1155
+ // get token balance of wallet
1156
+ const txAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, params.walletContractAddress);
1157
+ // build recipients object
1158
+ const recipients = [
1159
+ {
1160
+ address: params.recoveryDestination,
1161
+ amount: new bignumber_js_1.BigNumber(txAmount).toFixed(),
1162
+ },
1163
+ ];
1164
+ // Get sequence ID using contract call
1165
+ // we need to wait between making two explorer api calls to avoid getting banned
1166
+ await new Promise((resolve) => setTimeout(resolve, 1000));
1167
+ const sequenceId = await this.querySequenceId(params.walletContractAddress);
1168
+ const txInfo = {
1169
+ recipients: recipients,
1170
+ expireTime: this.getDefaultExpireTime(),
1171
+ contractSequenceId: sequenceId,
1172
+ gasLimit: gasLimit.toString(10),
1173
+ isEvmBasedCrossChainRecovery: true,
1174
+ };
1175
+ const txBuilder = this.getTransactionBuilder();
1176
+ txBuilder.counter(bitgoFeeAddressNonce);
1177
+ txBuilder.contract(params.walletContractAddress);
1178
+ let txFee;
1179
+ if (params.eip1559) {
1180
+ txFee = {
1181
+ eip1559: {
1182
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
1183
+ maxFeePerGas: params.eip1559.maxFeePerGas,
1184
+ },
1185
+ };
1186
+ }
1187
+ else {
1188
+ txFee = { fee: gasPrice.toString() };
1189
+ }
1190
+ txBuilder.fee({
1191
+ ...txFee,
1192
+ gasLimit: gasLimit.toString(),
1193
+ });
1194
+ const transferBuilder = txBuilder.transfer();
1195
+ const network = this.getNetwork();
1196
+ const token = (_a = lib_1.getToken(params.tokenContractAddress, network)) === null || _a === void 0 ? void 0 : _a.name;
1197
+ transferBuilder
1198
+ .amount(txAmount)
1199
+ .contractSequenceId(sequenceId)
1200
+ .expirationTime(this.getDefaultExpireTime())
1201
+ .to(params.recoveryDestination)
1202
+ .coin(token);
1203
+ if (params.walletPassphrase) {
1204
+ txBuilder.transfer().key(userSigningKey);
1205
+ }
1206
+ const tx = await txBuilder.build();
1207
+ const response = {
1208
+ txHex: tx.toBroadcastFormat(),
1209
+ userKey,
1210
+ coin: token,
1211
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1212
+ gasLimit,
1213
+ recipients: txInfo.recipients,
1214
+ walletContractAddress: tx.toJson().to,
1215
+ amount: txAmount.toString(),
1216
+ backupKeyNonce: bitgoFeeAddressNonce,
1217
+ eip1559: params.eip1559,
1218
+ };
1219
+ lodash_1.default.extend(response, txInfo);
1220
+ response.nextContractSequenceId = response.contractSequenceId;
1221
+ if (params.walletPassphrase) {
1222
+ const halfSignedTxn = {
1223
+ halfSigned: {
1224
+ txHex: tx.toBroadcastFormat(),
1225
+ recipients: txInfo.recipients,
1226
+ expireTime: txInfo.expireTime,
1227
+ },
1228
+ };
1229
+ lodash_1.default.extend(response, halfSignedTxn);
1230
+ const feesUsed = {
1231
+ gasPrice: exports.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
1232
+ gasLimit: exports.optionalDeps.ethUtil.bufferToInt(gasLimit).toFixed(),
1233
+ };
1234
+ response['feesUsed'] = feesUsed;
1235
+ }
1236
+ return response;
1237
+ }
1238
+ /**
1239
+ * Validate evm based cross chain recovery params
1240
+ * @param params {RecoverOptions}
1241
+ * @returns {void}
1242
+ */
1243
+ validateEvmBasedRecoveryParams(params) {
1244
+ if (lodash_1.default.isUndefined(params.bitgoFeeAddress) || !this.isValidAddress(params.bitgoFeeAddress)) {
1245
+ throw new Error('invalid bitgoFeeAddress');
1246
+ }
1247
+ if (lodash_1.default.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
1248
+ throw new Error('invalid walletContractAddress');
1249
+ }
1250
+ if (lodash_1.default.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
1251
+ throw new Error('invalid recoveryDestination');
1252
+ }
1253
+ }
1254
+ /**
1255
+ * Return types, values, and total amount in wei to send in a batch transaction, using the method signature
1256
+ * `distributeBatch(address[], uint256[])`
1257
+ * @param {Recipient[]} recipients - transaction recipients
1258
+ * @returns {GetBatchExecutionInfoRT} information needed to execute the batch transaction
1259
+ */
1260
+ getBatchExecutionInfo(recipients) {
1261
+ const addresses = [];
1262
+ const amounts = [];
1263
+ let sum = new bignumber_js_1.BigNumber('0');
1264
+ lodash_1.default.forEach(recipients, ({ address, amount }) => {
1265
+ addresses.push(address);
1266
+ amounts.push(amount);
1267
+ sum = sum.plus(amount);
1268
+ });
1269
+ return {
1270
+ values: [addresses, amounts],
1271
+ totalAmount: sum.toFixed(),
1272
+ };
1273
+ }
1274
+ /**
1275
+ * Build arguments to call the send method on the wallet contract
1276
+ * @param txInfo
1277
+ */
1278
+ getSendMethodArgs(txInfo) {
1279
+ // Method signature is
1280
+ // sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)
1281
+ return [
1282
+ {
1283
+ name: 'toAddress',
1284
+ type: 'address',
1285
+ value: txInfo.recipient.address,
1286
+ },
1287
+ {
1288
+ name: 'value',
1289
+ type: 'uint',
1290
+ value: txInfo.recipient.amount,
1291
+ },
1292
+ {
1293
+ name: 'data',
1294
+ type: 'bytes',
1295
+ value: exports.optionalDeps.ethUtil.toBuffer(exports.optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),
1296
+ },
1297
+ {
1298
+ name: 'expireTime',
1299
+ type: 'uint',
1300
+ value: txInfo.expireTime,
1301
+ },
1302
+ {
1303
+ name: 'sequenceId',
1304
+ type: 'uint',
1305
+ value: txInfo.contractSequenceId,
1306
+ },
1307
+ {
1308
+ name: 'signature',
1309
+ type: 'bytes',
1310
+ value: exports.optionalDeps.ethUtil.toBuffer(exports.optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),
1311
+ },
1312
+ ];
1313
+ }
1314
+ /**
1315
+ * Recovers a tx with TSS key shares
1316
+ * same expected arguments as recover method, but with TSS key shares
1317
+ */
1318
+ async recoverTSS(params) {
1319
+ this.validateRecoveryParams(params);
1320
+ const isUnsignedSweep = sdk_core_1.getIsUnsignedSweep(params);
1321
+ // Clean up whitespace from entered values
1322
+ const userPublicOrPrivateKeyShare = params.userKey.replace(/\s/g, '');
1323
+ const backupPrivateOrPublicKeyShare = params.backupKey.replace(/\s/g, '');
1324
+ // Set new eth tx fees (using default config values from platform)
1325
+ const gasLimit = new exports.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
1326
+ const gasPrice = params.eip1559
1327
+ ? new exports.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
1328
+ : new exports.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
1329
+ const [backupKeyAddress, userKeyCombined, backupKeyCombined] = (() => {
1330
+ if (isUnsignedSweep) {
1331
+ const backupKeyPair = new lib_1.KeyPair({ pub: backupPrivateOrPublicKeyShare });
1332
+ return [backupKeyPair.getAddress(), undefined, undefined];
1333
+ }
1334
+ else {
1335
+ const [userKeyCombined, backupKeyCombined] = this.getKeyCombinedFromTssKeyShares(userPublicOrPrivateKeyShare, backupPrivateOrPublicKeyShare, params.walletPassphrase);
1336
+ const backupKeyPair = new lib_1.KeyPair({ pub: backupKeyCombined.xShare.y });
1337
+ return [backupKeyPair.getAddress(), userKeyCombined, backupKeyCombined];
1338
+ }
1339
+ })();
1340
+ const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);
1341
+ // get balance of backupKey to ensure funds are available to pay fees
1342
+ const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);
1343
+ const totalGasNeeded = gasPrice.mul(gasLimit);
1344
+ const weiToGwei = 10 ** 9;
1345
+ if (backupKeyBalance.lt(totalGasNeeded)) {
1346
+ throw new Error(`Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +
1347
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
1348
+ ` Gwei to perform recoveries. Try sending some ETH to this address then retry.`);
1349
+ }
1350
+ // get balance of wallet and deduct fees to get transaction amount, wallet contract address acts as base address for tss?
1351
+ const txAmount = backupKeyBalance.sub(totalGasNeeded);
1352
+ // build recipients object
1353
+ const recipients = [
1354
+ {
1355
+ address: params.recoveryDestination,
1356
+ amount: txAmount.toString(10),
1357
+ },
1358
+ ];
1359
+ const txInfo = {
1360
+ recipient: recipients[0],
1361
+ expireTime: this.getDefaultExpireTime(),
1362
+ gasLimit: gasLimit.toString(10),
1363
+ };
1364
+ const txParams = {
1365
+ to: params.recoveryDestination,
1366
+ nonce: backupKeyNonce,
1367
+ value: txAmount,
1368
+ gasPrice: gasPrice,
1369
+ gasLimit: gasLimit,
1370
+ data: Buffer.from('0x'),
1371
+ eip1559: params.eip1559,
1372
+ replayProtectionOptions: params.replayProtectionOptions,
1373
+ };
1374
+ let tx = AbstractEthLikeNewCoins.buildTransaction(txParams);
1375
+ if (isUnsignedSweep) {
1376
+ return this.formatForOfflineVaultTSS(txInfo, tx, userPublicOrPrivateKeyShare, backupPrivateOrPublicKeyShare, gasPrice, gasLimit, backupKeyNonce, params.eip1559, params.replayProtectionOptions);
1377
+ }
1378
+ const signableHex = tx.getMessageToSign(false).toString('hex');
1379
+ if (!userKeyCombined || !backupKeyCombined) {
1380
+ throw new Error('Missing key combined shares for user or backup');
1381
+ }
1382
+ const signature = await this.signRecoveryTSS(userKeyCombined, backupKeyCombined, signableHex);
1383
+ const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);
1384
+ tx = this.getSignedTxFromSignature(ethCommmon, tx, signature);
1385
+ return {
1386
+ id: ethereumjs_util_1.addHexPrefix(tx.hash().toString('hex')),
1387
+ tx: ethereumjs_util_1.addHexPrefix(tx.serialize().toString('hex')),
1388
+ };
1389
+ }
1390
+ async recoveryBlockchainExplorerQuery(query) {
1391
+ throw new Error('method not implemented');
1392
+ }
1393
+ /**
1394
+ * Creates the extra parameters needed to build a hop transaction
1395
+ * @param buildParams The original build parameters
1396
+ * @returns extra parameters object to merge with the original build parameters object and send to the platform
1397
+ */
1398
+ async createHopTransactionParams(buildParams) {
1399
+ const wallet = buildParams.wallet;
1400
+ const recipients = buildParams.recipients;
1401
+ const walletPassphrase = buildParams.walletPassphrase;
1402
+ const userKeychain = await this.keychains().get({ id: wallet.keyIds()[0] });
1403
+ const userPrv = wallet.getUserPrv({ keychain: userKeychain, walletPassphrase });
1404
+ const userPrvBuffer = utxo_lib_1.bip32.fromBase58(userPrv).privateKey;
1405
+ if (!userPrvBuffer) {
1406
+ throw new Error('invalid userPrv');
1407
+ }
1408
+ if (!recipients || !Array.isArray(recipients)) {
1409
+ throw new Error('expecting array of recipients');
1410
+ }
1411
+ // Right now we only support 1 recipient
1412
+ if (recipients.length !== 1) {
1413
+ throw new Error('must send to exactly 1 recipient');
1414
+ }
1415
+ const recipientAddress = recipients[0].address;
1416
+ const recipientAmount = recipients[0].amount;
1417
+ const feeEstimateParams = {
1418
+ recipient: recipientAddress,
1419
+ amount: recipientAmount,
1420
+ hop: true,
1421
+ };
1422
+ const feeEstimate = await this.feeEstimate(feeEstimateParams);
1423
+ const gasLimit = feeEstimate.gasLimitEstimate;
1424
+ const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);
1425
+ const gasPriceMax = gasPrice * 5;
1426
+ // Payment id a random number so its different for every tx
1427
+ const paymentId = Math.floor(Math.random() * 10000000000).toString();
1428
+ const hopDigest = AbstractEthLikeNewCoins.getHopDigest([
1429
+ recipientAddress,
1430
+ recipientAmount,
1431
+ gasPriceMax.toString(),
1432
+ gasLimit.toString(),
1433
+ paymentId,
1434
+ ]);
1435
+ const userReqSig = exports.optionalDeps.ethUtil.addHexPrefix(Buffer.from(secp256k1_1.default.ecdsaSign(hopDigest, userPrvBuffer).signature).toString('hex'));
1436
+ return {
1437
+ hopParams: {
1438
+ gasPriceMax,
1439
+ userReqSig,
1440
+ paymentId,
1441
+ },
1442
+ gasLimit,
1443
+ };
1444
+ }
1445
+ /**
1446
+ * Validates that the hop prebuild from the HSM is valid and correct
1447
+ * @param {IWallet} wallet - The wallet that the prebuild is for
1448
+ * @param {HopPrebuild} hopPrebuild - The prebuild to validate
1449
+ * @param {Object} originalParams - The original parameters passed to prebuildTransaction
1450
+ * @param {Recipient[]} originalParams.recipients - The original recipients array
1451
+ * @returns {void}
1452
+ * @throws Error if The prebuild is invalid
1453
+ */
1454
+ async validateHopPrebuild(wallet, hopPrebuild, originalParams) {
1455
+ const { tx, id, signature } = hopPrebuild;
1456
+ // first, validate the HSM signature
1457
+ const serverXpub = sdk_core_1.common.Environments[this.bitgo.getEnv()].hsmXpub;
1458
+ const serverPubkeyBuffer = utxo_lib_1.bip32.fromBase58(serverXpub).publicKey;
1459
+ const signatureBuffer = Buffer.from(exports.optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');
1460
+ const messageBuffer = Buffer.from(exports.optionalDeps.ethUtil.padToEven(exports.optionalDeps.ethUtil.stripHexPrefix(id)), 'hex');
1461
+ const sig = new Uint8Array(signatureBuffer.slice(1));
1462
+ const isValidSignature = secp256k1_1.default.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);
1463
+ if (!isValidSignature) {
1464
+ throw new Error(`Hop txid signature invalid - pub: ${serverXpub}, msg: ${messageBuffer === null || messageBuffer === void 0 ? void 0 : messageBuffer.toString()}, sig: ${signatureBuffer === null || signatureBuffer === void 0 ? void 0 : signatureBuffer.toString()}`);
1465
+ }
1466
+ const builtHopTx = exports.optionalDeps.EthTx.TransactionFactory.fromSerializedData(exports.optionalDeps.ethUtil.toBuffer(tx));
1467
+ // If original params are given, we can check them against the transaction prebuild params
1468
+ if (!lodash_1.default.isNil(originalParams)) {
1469
+ const { recipients } = originalParams;
1470
+ // Then validate that the tx params actually equal the requested params
1471
+ const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount);
1472
+ const originalDestination = recipients[0].address;
1473
+ const hopAmount = new bignumber_js_1.BigNumber(exports.optionalDeps.ethUtil.bufferToHex(builtHopTx.value));
1474
+ if (!builtHopTx.to) {
1475
+ throw new Error(`Transaction does not have a destination address`);
1476
+ }
1477
+ const hopDestination = builtHopTx.to.toString();
1478
+ if (!hopAmount.eq(originalAmount)) {
1479
+ throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
1480
+ }
1481
+ if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
1482
+ throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);
1483
+ }
1484
+ }
1485
+ if (!builtHopTx.verifySignature()) {
1486
+ // We dont want to continue at all in this case, at risk of ETH being stuck on the hop address
1487
+ throw new Error(`Invalid hop transaction signature, txid: ${id}`);
1488
+ }
1489
+ if (exports.optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {
1490
+ throw new Error(`Signed hop txid does not equal actual txid`);
1491
+ }
1492
+ }
1493
+ /**
1494
+ * Gets the hop digest for the user to sign. This is validated in the HSM to prove that the user requested this tx
1495
+ * @param {string[]} paramsArr - The parameters to hash together for the digest
1496
+ * @returns {Buffer}
1497
+ */
1498
+ static getHopDigest(paramsArr) {
1499
+ const hash = keccak_1.default('keccak256');
1500
+ hash.update([AbstractEthLikeNewCoins.hopTransactionSalt, ...paramsArr].join('$'));
1501
+ return hash.digest();
1502
+ }
1503
+ /**
1504
+ * Modify prebuild before sending it to the server. Add things like hop transaction params
1505
+ * @param {BuildOptions} buildParams - The whitelisted parameters for this prebuild
1506
+ * @param {boolean} buildParams.hop - True if this should prebuild a hop tx, else false
1507
+ * @param {Recipient[]} buildParams.recipients - The recipients array of this transaction
1508
+ * @param {Wallet} buildParams.wallet - The wallet sending this tx
1509
+ * @param {string} buildParams.walletPassphrase - the passphrase for this wallet
1510
+ * @returns {Promise<BuildOptions>}
1511
+ */
1512
+ async getExtraPrebuildParams(buildParams) {
1513
+ if (!lodash_1.default.isUndefined(buildParams.hop) &&
1514
+ buildParams.hop &&
1515
+ !lodash_1.default.isUndefined(buildParams.wallet) &&
1516
+ !lodash_1.default.isUndefined(buildParams.recipients) &&
1517
+ !lodash_1.default.isUndefined(buildParams.walletPassphrase)) {
1518
+ if (this instanceof ethLikeToken_1.EthLikeToken) {
1519
+ throw new Error(`Hop transactions are not enabled for ERC-20 tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`);
1520
+ }
1521
+ return (await this.createHopTransactionParams({
1522
+ wallet: buildParams.wallet,
1523
+ recipients: buildParams.recipients,
1524
+ walletPassphrase: buildParams.walletPassphrase,
1525
+ }));
1526
+ }
1527
+ return {};
1528
+ }
1529
+ /**
1530
+ * Modify prebuild after receiving it from the server. Add things like nlocktime
1531
+ * @param {TransactionPrebuild} params - The prebuild to modify
1532
+ * @returns {TransactionPrebuild} The modified prebuild
1533
+ */
1534
+ async postProcessPrebuild(params) {
1535
+ if (!lodash_1.default.isUndefined(params.hopTransaction) && !lodash_1.default.isUndefined(params.wallet) && !lodash_1.default.isUndefined(params.buildParams)) {
1536
+ await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);
1537
+ }
1538
+ return params;
1539
+ }
1540
+ /**
1541
+ * Coin-specific things done before signing a transaction, i.e. verification
1542
+ * @param {PresignTransactionOptions} params
1543
+ * @returns {Promise<PresignTransactionOptions>}
1544
+ */
1545
+ async presignTransaction(params) {
1546
+ if (!lodash_1.default.isUndefined(params.hopTransaction) && !lodash_1.default.isUndefined(params.wallet) && !lodash_1.default.isUndefined(params.buildParams)) {
1547
+ await this.validateHopPrebuild(params.wallet, params.hopTransaction);
1548
+ }
1549
+ return params;
1550
+ }
1551
+ /**
1552
+ * Fetch fee estimate information from the server
1553
+ * @param {Object} params - The params passed into the function
1554
+ * @param {boolean} [params.hop] - True if we should estimate fee for a hop transaction
1555
+ * @param {string} [params.recipient] - The recipient of the transaction to estimate a send to
1556
+ * @param {string} [params.data] - The ETH tx data to estimate a send for
1557
+ * @returns {Object} The fee info returned from the server
1558
+ */
1559
+ async feeEstimate(params) {
1560
+ const query = {};
1561
+ if (params && params.hop) {
1562
+ query.hop = params.hop;
1563
+ }
1564
+ if (params && params.recipient) {
1565
+ query.recipient = params.recipient;
1566
+ }
1567
+ if (params && params.data) {
1568
+ query.data = params.data;
1569
+ }
1570
+ if (params && params.amount) {
1571
+ query.amount = params.amount;
1572
+ }
1573
+ return await this.bitgo.get(this.url('/tx/fee')).query(query).result();
1574
+ }
1575
+ /**
1576
+ * Generate secp256k1 key pair
1577
+ *
1578
+ * @param {Buffer} seed
1579
+ * @returns {KeyPair} object with generated pub and prv
1580
+ */
1581
+ generateKeyPair(seed) {
1582
+ if (!seed) {
1583
+ // An extended private key has both a normal 256 bit private key and a 256
1584
+ // bit chain code, both of which must be random. 512 bits is therefore the
1585
+ // maximum entropy and gives us maximum security against cracking.
1586
+ seed = crypto_1.randomBytes(512 / 8);
1587
+ }
1588
+ const extendedKey = utxo_lib_1.bip32.fromSeed(seed);
1589
+ const xpub = extendedKey.neutered().toBase58();
1590
+ return {
1591
+ pub: xpub,
1592
+ prv: extendedKey.toBase58(),
1593
+ };
1594
+ }
1595
+ async parseTransaction(params) {
1596
+ return {};
1597
+ }
1598
+ /**
1599
+ * Make sure an address is a wallet address and throw an error if it's not.
1600
+ * @param {Object} params
1601
+ * @param {string} params.address - The derived address string on the network
1602
+ * @param {Object} params.coinSpecific - Coin-specific details for the address such as a forwarderVersion
1603
+ * @param {string} params.baseAddress - The base address of the wallet on the network
1604
+ * @throws {InvalidAddressError}
1605
+ * @throws {InvalidAddressVerificationObjectPropertyError}
1606
+ * @throws {UnexpectedAddressError}
1607
+ * @returns {boolean} True iff address is a wallet address
1608
+ */
1609
+ async isWalletAddress(params) {
1610
+ const ethUtil = exports.optionalDeps.ethUtil;
1611
+ let expectedAddress;
1612
+ let actualAddress;
1613
+ const { address, coinSpecific, baseAddress, impliedForwarderVersion = coinSpecific === null || coinSpecific === void 0 ? void 0 : coinSpecific.forwarderVersion } = params;
1614
+ if (address && !this.isValidAddress(address)) {
1615
+ throw new sdk_core_1.InvalidAddressError(`invalid address: ${address}`);
1616
+ }
1617
+ // base address is required to calculate the salt which is used in calculateForwarderV1Address method
1618
+ if (lodash_1.default.isUndefined(baseAddress) || !this.isValidAddress(baseAddress)) {
1619
+ throw new sdk_core_1.InvalidAddressError('invalid base address');
1620
+ }
1621
+ if (!lodash_1.default.isObject(coinSpecific)) {
1622
+ throw new sdk_core_1.InvalidAddressVerificationObjectPropertyError('address validation failure: coinSpecific field must be an object');
1623
+ }
1624
+ if (impliedForwarderVersion === 0 || impliedForwarderVersion === 3) {
1625
+ return true;
1626
+ }
1627
+ else {
1628
+ const ethNetwork = this.getNetwork();
1629
+ const forwarderFactoryAddress = ethNetwork === null || ethNetwork === void 0 ? void 0 : ethNetwork.forwarderFactoryAddress;
1630
+ const forwarderImplementationAddress = ethNetwork === null || ethNetwork === void 0 ? void 0 : ethNetwork.forwarderImplementationAddress;
1631
+ const initcode = lib_1.getProxyInitcode(forwarderImplementationAddress);
1632
+ const saltBuffer = ethUtil.setLengthLeft(Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(coinSpecific.salt || '')), 'hex'), 32);
1633
+ // Hash the wallet base address with the given salt, so the address directly relies on the base address
1634
+ const calculationSalt = exports.optionalDeps.ethUtil.bufferToHex(exports.optionalDeps.ethAbi.soliditySHA3(['address', 'bytes32'], [baseAddress, saltBuffer]));
1635
+ expectedAddress = lib_1.calculateForwarderV1Address(forwarderFactoryAddress, calculationSalt, initcode);
1636
+ actualAddress = address;
1637
+ }
1638
+ if (expectedAddress !== actualAddress) {
1639
+ throw new sdk_core_1.UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);
1640
+ }
1641
+ return true;
1642
+ }
1643
+ /**
1644
+ *
1645
+ * @param {TransactionPrebuild} txPrebuild
1646
+ * @returns {boolean}
1647
+ */
1648
+ verifyCoin(txPrebuild) {
1649
+ return txPrebuild.coin === this.getChain();
1650
+ }
1651
+ /**
1652
+ * Verify if a tss transaction is valid
1653
+ *
1654
+ * @param {VerifyEthTransactionOptions} params
1655
+ * @param {TransactionParams} params.txParams - params object passed to send
1656
+ * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server
1657
+ * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against
1658
+ * @returns {boolean}
1659
+ */
1660
+ verifyTssTransaction(params) {
1661
+ var _a;
1662
+ const { txParams, txPrebuild, wallet } = params;
1663
+ if (!(txParams === null || txParams === void 0 ? void 0 : txParams.recipients) &&
1664
+ !(((_a = txParams.prebuildTx) === null || _a === void 0 ? void 0 : _a.consolidateId) ||
1665
+ (txParams.type && ['acceleration', 'fillNonce', 'transferToken'].includes(txParams.type)))) {
1666
+ throw new Error(`missing txParams`);
1667
+ }
1668
+ if (!wallet || !txPrebuild) {
1669
+ throw new Error(`missing params`);
1670
+ }
1671
+ if (txParams.hop && txParams.recipients && txParams.recipients.length > 1) {
1672
+ throw new Error(`tx cannot be both a batch and hop transaction`);
1673
+ }
1674
+ return true;
1675
+ }
1676
+ /**
1677
+ * Verify that a transaction prebuild complies with the original intention
1678
+ *
1679
+ * @param {VerifyEthTransactionOptions} params
1680
+ * @param {TransactionParams} params.txParams - params object passed to send
1681
+ * @param {TransactionPrebuild} params.txPrebuild - prebuild object returned by server
1682
+ * @param {Wallet} params.wallet - Wallet object to obtain keys to verify against
1683
+ * @returns {boolean}
1684
+ */
1685
+ async verifyTransaction(params) {
1686
+ const ethNetwork = this.getNetwork();
1687
+ const { txParams, txPrebuild, wallet, walletType } = params;
1688
+ if (walletType === 'tss') {
1689
+ return this.verifyTssTransaction(params);
1690
+ }
1691
+ if (!(txParams === null || txParams === void 0 ? void 0 : txParams.recipients) || !(txPrebuild === null || txPrebuild === void 0 ? void 0 : txPrebuild.recipients) || !wallet) {
1692
+ throw new Error(`missing params`);
1693
+ }
1694
+ if (txParams.hop && txParams.recipients.length > 1) {
1695
+ throw new Error(`tx cannot be both a batch and hop transaction`);
1696
+ }
1697
+ if (txPrebuild.recipients.length !== 1) {
1698
+ throw new Error(`txPrebuild should only have 1 recipient but ${txPrebuild.recipients.length} found`);
1699
+ }
1700
+ if (txParams.hop && txPrebuild.hopTransaction) {
1701
+ // Check recipient amount for hop transaction
1702
+ if (txParams.recipients.length !== 1) {
1703
+ throw new Error(`hop transaction only supports 1 recipient but ${txParams.recipients.length} found`);
1704
+ }
1705
+ // Check tx sends to hop address
1706
+ const decodedHopTx = exports.optionalDeps.EthTx.TransactionFactory.fromSerializedData(exports.optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx));
1707
+ const expectedHopAddress = exports.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());
1708
+ const actualHopAddress = exports.optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);
1709
+ if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {
1710
+ throw new Error('recipient address of txPrebuild does not match hop address');
1711
+ }
1712
+ // Convert TransactionRecipient array to Recipient array
1713
+ const recipients = txParams.recipients.map((r) => {
1714
+ return {
1715
+ address: r.address,
1716
+ amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,
1717
+ };
1718
+ });
1719
+ // Check destination address and amount
1720
+ await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients });
1721
+ }
1722
+ else if (txParams.recipients.length > 1) {
1723
+ // Check total amount for batch transaction
1724
+ let expectedTotalAmount = new bignumber_js_1.BigNumber(0);
1725
+ for (let i = 0; i < txParams.recipients.length; i++) {
1726
+ expectedTotalAmount = expectedTotalAmount.plus(txParams.recipients[i].amount);
1727
+ }
1728
+ if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
1729
+ throw new Error('batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
1730
+ }
1731
+ // Check batch transaction is sent to the batcher contract address for the chain
1732
+ const batcherContractAddress = ethNetwork === null || ethNetwork === void 0 ? void 0 : ethNetwork.batcherContractAddress;
1733
+ if (!batcherContractAddress ||
1734
+ batcherContractAddress.toLowerCase() !== txPrebuild.recipients[0].address.toLowerCase()) {
1735
+ throw new Error('recipient address of txPrebuild does not match batcher address');
1736
+ }
1737
+ }
1738
+ else {
1739
+ // Check recipient address and amount for normal transaction
1740
+ if (txParams.recipients.length !== 1) {
1741
+ throw new Error(`normal transaction only supports 1 recipient but ${txParams.recipients.length} found`);
1742
+ }
1743
+ const expectedAmount = new bignumber_js_1.BigNumber(txParams.recipients[0].amount);
1744
+ if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
1745
+ throw new Error('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
1746
+ }
1747
+ if (this.isETHAddress(txParams.recipients[0].address) &&
1748
+ txParams.recipients[0].address !== txPrebuild.recipients[0].address) {
1749
+ throw new Error('destination address in normal txPrebuild does not match that in txParams supplied by client');
1750
+ }
1751
+ }
1752
+ // Check coin is correct for all transaction types
1753
+ if (!this.verifyCoin(txPrebuild)) {
1754
+ throw new Error(`coin in txPrebuild did not match that in txParams supplied by client`);
1755
+ }
1756
+ return true;
1757
+ }
1758
+ /**
1759
+ * Check if address is valid eth address
1760
+ * @param address
1761
+ * @returns {boolean}
1762
+ */
1763
+ isETHAddress(address) {
1764
+ return !!address.match(/0x[a-fA-F0-9]{40}/);
1765
+ }
1766
+ /**
1767
+ * Transform message to accommodate specific blockchain requirements.
1768
+ * @param {string} message - the message to prepare
1769
+ * @return {string} the prepared message.
1770
+ */
1771
+ encodeMessage(message) {
1772
+ const prefix = `\u0019Ethereum Signed Message:\n${message.length}`;
1773
+ return prefix.concat(message);
1774
+ }
1775
+ /**
1776
+ * Transform the Typed data to accomodate the blockchain requirements (EIP-712)
1777
+ * @param {TypedData} typedData - the typed data to prepare
1778
+ * @return {Buffer} a buffer of the result
1779
+ */
1780
+ encodeTypedData(typedData) {
1781
+ const version = typedData.version;
1782
+ if (version === eth_sig_util_1.SignTypedDataVersion.V1) {
1783
+ throw new Error('SignTypedData v1 is not supported due to security concerns');
1784
+ }
1785
+ const typedDataRaw = JSON.parse(typedData.typedDataRaw);
1786
+ const sanitizedData = eth_sig_util_1.TypedDataUtils.sanitizeData(typedDataRaw);
1787
+ const parts = [Buffer.from('1901', 'hex')];
1788
+ const eip712Domain = 'EIP712Domain';
1789
+ parts.push(eth_sig_util_1.TypedDataUtils.hashStruct(eip712Domain, sanitizedData.domain, sanitizedData.types, version));
1790
+ if (sanitizedData.primaryType !== eip712Domain) {
1791
+ parts.push(eth_sig_util_1.TypedDataUtils.hashStruct(sanitizedData.primaryType, sanitizedData.message, sanitizedData.types, version));
1792
+ }
1793
+ return Buffer.concat(parts);
1794
+ }
145
1795
  }
146
1796
  exports.AbstractEthLikeNewCoins = AbstractEthLikeNewCoins;
147
- //# 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 debugLib from 'debug';\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport {\n  BitGoBase,\n  EthereumLibraryUnavailableError,\n  Recipient,\n  TransactionRecipient,\n  TransactionPrebuild as BaseTransactionPrebuild,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, CoinFamily, EthereumNetwork as EthLikeNetwork } from '@bitgo-beta/statics';\nimport type * as EthLikeTxLib from '@ethereumjs/tx';\nimport type * as EthLikeCommon from '@ethereumjs/common';\n\nimport { AbstractEthLikeCoin } from './abstractEthLikeCoin';\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\nexport interface TransactionPrebuild extends BaseTransactionPrebuild {\n  hopTransaction?: HopPrebuild;\n  buildParams: {\n    recipients: Recipient[];\n  };\n  recipients: TransactionRecipient[];\n  nextContractSequenceId: string;\n  gasPrice: number;\n  gasLimit: number;\n  isBatch: boolean;\n  coin: string;\n  token?: string;\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  protected readonly sendMethodName: 'sendMultiSig' | 'sendMultiSigToken';\n\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  private static _ethLikeCoin: 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    AbstractEthLikeNewCoins._ethLikeCoin = staticsCoin;\n    this.sendMethodName = 'sendMultiSig';\n  }\n\n  readonly staticsCoin?: Readonly<StaticsBaseCoin>;\n\n  getChain() {\n    return this._staticsCoin.name;\n  }\n\n  /**\n   * Get the base chain that the coin exists on.\n   */\n  getBaseChain() {\n    return this.getChain();\n  }\n\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  getNetwork(): EthLikeNetwork | undefined {\n    return this._staticsCoin?.network as EthLikeNetwork;\n  }\n\n  getFullName() {\n    return this._staticsCoin.fullName;\n  }\n\n  getBaseFactor() {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  /** @inheritDoc */\n  isEVM(): boolean {\n    return true;\n  }\n\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  /**\n   * Evaluates whether an address string is valid for this coin\n   * @param address\n   */\n  isValidAddress(address: string): boolean {\n    return optionalDeps.ethUtil.isValidAddress(optionalDeps.ethUtil.addHexPrefix(address));\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPub(pub: string): boolean {\n    try {\n      return bip32.fromBase58(pub).isNeutered();\n    } catch (e) {\n      return false;\n    }\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  /**\n   * Default gas price from platform\n   * @returns {BigNumber}\n   */\n  getRecoveryGasPrice(): any {\n    return new optionalDeps.ethUtil.BN('20000000000');\n  }\n\n  /**\n   * Default gas limit from platform\n   * @returns {BigNumber}\n   */\n  getRecoveryGasLimit(): any {\n    return new optionalDeps.ethUtil.BN('500000');\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"]}
1797
+ AbstractEthLikeNewCoins.hopTransactionSalt = 'bitgoHopAddressRequestSalt';
1798
+ //# 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 debugLib from 'debug';\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport { BigNumber } from 'bignumber.js';\nimport { randomBytes } from 'crypto';\nimport Keccak from 'keccak';\nimport _ from 'lodash';\nimport secp256k1 from 'secp256k1';\nimport BN from 'bn.js';\nimport {\n  AddressCoinSpecific,\n  BitGoBase,\n  common,\n  ECDSA,\n  Ecdsa,\n  ECDSAMethodTypes,\n  EthereumLibraryUnavailableError,\n  FeeEstimateOptions,\n  FullySignedTransaction,\n  getIsUnsignedSweep,\n  HalfSignedTransaction,\n  hexToBigInt,\n  InvalidAddressError,\n  InvalidAddressVerificationObjectPropertyError,\n  IWallet,\n  KeyPair,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  PrebuildTransactionResult,\n  PresignTransactionOptions as BasePresignTransactionOptions,\n  Recipient,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  TransactionParams,\n  TransactionRecipient,\n  TransactionPrebuild as BaseTransactionPrebuild,\n  TypedData,\n  UnexpectedAddressError,\n  Util,\n  VerifyAddressOptions as BaseVerifyAddressOptions,\n  VerifyTransactionOptions,\n  Wallet,\n} from '@bitgo-beta/sdk-core';\nimport {\n  BaseCoin as StaticsBaseCoin,\n  coins,\n  EthereumNetwork as EthLikeNetwork,\n  ethGasConfigs,\n  CoinMap,\n} from '@bitgo-beta/statics';\nimport type * as EthLikeTxLib from '@ethereumjs/tx';\nimport type * as EthLikeCommon from '@ethereumjs/common';\nimport { EcdsaPaillierProof, EcdsaRangeProof, EcdsaTypes } from '@bitgo-beta/sdk-lib-mpc';\nimport { FeeMarketEIP1559Transaction, Transaction as LegacyTransaction } from '@ethereumjs/tx';\nimport { addHexPrefix, stripHexPrefix } from 'ethereumjs-util';\nimport { SignTypedDataVersion, TypedDataUtils, TypedMessage } from '@metamask/eth-sig-util';\n\nimport {\n  calculateForwarderV1Address,\n  getCommon,\n  getProxyInitcode,\n  getToken,\n  KeyPair as KeyPairLib,\n  TransactionBuilder,\n  TransferBuilder,\n} from './lib';\nimport { AbstractEthLikeCoin } from './abstractEthLikeCoin';\nimport { EthLikeToken } from './ethLikeToken';\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  };\n  gasLimit: number;\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}\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}\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}\n\ninterface UnformattedTxInfo {\n  recipient: Recipient;\n}\n\nexport interface 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  isTss?: boolean;\n  bitgoFeeAddress?: string;\n  bitgoDestinationAddress?: string;\n  tokenContractAddress?: string;\n}\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}\n\ninterface 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}\n\nexport interface VerifyEthAddressOptions extends BaseVerifyAddressOptions {\n  baseAddress: string;\n  coinSpecific: EthAddressCoinSpecifics;\n  forwarderVersion: number;\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  /**\n   * Default gas price from platform\n   * @returns {BigNumber}\n   */\n  getRecoveryGasPrice(): any {\n    return new optionalDeps.ethUtil.BN('20000000000');\n  }\n\n  /**\n   * Default gas limit from platform\n   * @returns {BigNumber}\n   */\n  getRecoveryGasLimit(): any {\n    return new optionalDeps.ethUtil.BN('500000');\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 = CoinMap.coinNameFromChainId(chainId);\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 tangerine whistle to avoid replay protection issues\n    const defaultHardfork = !!eip1559 ? 'london' : optionalDeps.EthCommon.Hardfork.TangerineWhistle;\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   * @returns {BigNumber} address balance\n   */\n  async queryAddressBalance(address: string): Promise<any> {\n    const result = await this.recoveryBlockchainExplorerQuery({\n      module: 'account',\n      action: 'balance',\n      address: address,\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   * @returns {Promise<Number>} sequence ID\n   */\n  async querySequenceId(address: 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      module: 'proxy',\n      action: 'eth_call',\n      to: address,\n      data: sequenceIdData,\n      tag: 'latest',\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   * @returns {Promise<number>}\n   */\n  async getAddressNonce(address: string): Promise<number> {\n    // Get nonce for backup key (should be 0)\n    let nonce = 0;\n\n    const result = await this.recoveryBlockchainExplorerQuery({\n      module: 'account',\n      action: 'txlist',\n      address,\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   * @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  ): 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      ),\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();\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();\n    txBuilder.from(params.txPrebuild.txHex);\n    txBuilder.transfer().key(new KeyPairLib({ prv: params.prv }).getKeys().prv!);\n    const transaction = await txBuilder.build();\n\n    const recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));\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    };\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 (_.isUndefined(params.userKey)) {\n      throw new Error('missing userKey');\n    }\n\n    if (_.isUndefined(params.backupKey)) {\n      throw new Error('missing backupKey');\n    }\n\n    if (_.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub') && !params.isTss) {\n      throw new Error('missing wallet passphrase');\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   * Method to sign tss recovery transaction\n   * @param {ECDSA.KeyCombined} userKeyCombined\n   * @param {ECDSA.KeyCombined} backupKeyCombined\n   * @param {string} txHex\n   * @param {Object} options\n   * @param {EcdsaTypes.SerializedNtilde} options.rangeProofChallenge\n   * @returns {Promise<ECDSAMethodTypes.Signature>}\n   */\n  private async signRecoveryTSS(\n    userKeyCombined: ECDSA.KeyCombined,\n    backupKeyCombined: ECDSA.KeyCombined,\n    txHex: string,\n    {\n      rangeProofChallenge,\n    }: {\n      rangeProofChallenge?: EcdsaTypes.SerializedNtilde;\n    } = {}\n  ): Promise<ECDSAMethodTypes.Signature> {\n    const MPC = new Ecdsa();\n    const signerOneIndex = userKeyCombined.xShare.i;\n    const signerTwoIndex = backupKeyCombined.xShare.i;\n\n    rangeProofChallenge =\n      rangeProofChallenge ?? EcdsaTypes.serializeNtildeWithProofs(await EcdsaRangeProof.generateNtilde());\n\n    const userToBackupPaillierChallenge = await EcdsaPaillierProof.generateP(\n      hexToBigInt(userKeyCombined.yShares[signerTwoIndex].n)\n    );\n    const backupToUserPaillierChallenge = await EcdsaPaillierProof.generateP(\n      hexToBigInt(backupKeyCombined.yShares[signerOneIndex].n)\n    );\n\n    const userXShare = MPC.appendChallenge(\n      userKeyCombined.xShare,\n      rangeProofChallenge,\n      EcdsaTypes.serializePaillierChallenge({ p: userToBackupPaillierChallenge })\n    );\n    const userYShare = MPC.appendChallenge(\n      userKeyCombined.yShares[signerTwoIndex],\n      rangeProofChallenge,\n      EcdsaTypes.serializePaillierChallenge({ p: backupToUserPaillierChallenge })\n    );\n    const backupXShare = MPC.appendChallenge(\n      backupKeyCombined.xShare,\n      rangeProofChallenge,\n      EcdsaTypes.serializePaillierChallenge({ p: backupToUserPaillierChallenge })\n    );\n    const backupYShare = MPC.appendChallenge(\n      backupKeyCombined.yShares[signerOneIndex],\n      rangeProofChallenge,\n      EcdsaTypes.serializePaillierChallenge({ p: userToBackupPaillierChallenge })\n    );\n\n    const signShares: ECDSA.SignShareRT = await MPC.signShare(userXShare, userYShare);\n\n    const signConvertS21 = await MPC.signConvertStep1({\n      xShare: backupXShare,\n      yShare: backupYShare, // YShare corresponding to the other participant signerOne\n      kShare: signShares.kShare,\n    });\n    const signConvertS12 = await MPC.signConvertStep2({\n      aShare: signConvertS21.aShare,\n      wShare: signShares.wShare,\n    });\n    const signConvertS21_2 = await MPC.signConvertStep3({\n      muShare: signConvertS12.muShare,\n      bShare: signConvertS21.bShare,\n    });\n\n    const [signCombineOne, signCombineTwo] = [\n      MPC.signCombine({\n        gShare: signConvertS12.gShare,\n        signIndex: {\n          i: signConvertS12.muShare.i,\n          j: signConvertS12.muShare.j,\n        },\n      }),\n      MPC.signCombine({\n        gShare: signConvertS21_2.gShare,\n        signIndex: {\n          i: signConvertS21_2.signIndex.i,\n          j: signConvertS21_2.signIndex.j,\n        },\n      }),\n    ];\n\n    const MESSAGE = Buffer.from(txHex, 'hex');\n\n    const [signA, signB] = [\n      MPC.sign(MESSAGE, signCombineOne.oShare, signCombineTwo.dShare, Keccak('keccak256')),\n      MPC.sign(MESSAGE, signCombineTwo.oShare, signCombineOne.dShare, Keccak('keccak256')),\n    ];\n\n    return MPC.constructSignature([signA, signB]);\n  }\n\n  /**\n   * Helper which combines key shares of user and backup\n   * @param {string} userPublicOrPrivateKeyShare\n   * @param {string} backupPrivateOrPublicKeyShare\n   * @param {string} walletPassphrase\n   * @returns {[ECDSAMethodTypes.KeyCombined, ECDSAMethodTypes.KeyCombined]}\n   */\n  private getKeyCombinedFromTssKeyShares(\n    userPublicOrPrivateKeyShare: string,\n    backupPrivateOrPublicKeyShare: string,\n    walletPassphrase?: string\n  ): [ECDSAMethodTypes.KeyCombined, ECDSAMethodTypes.KeyCombined] {\n    let backupPrv;\n    let userPrv;\n    try {\n      backupPrv = this.bitgo.decrypt({\n        input: backupPrivateOrPublicKeyShare,\n        password: walletPassphrase,\n      });\n      userPrv = this.bitgo.decrypt({\n        input: userPublicOrPrivateKeyShare,\n        password: walletPassphrase,\n      });\n    } catch (e) {\n      throw new Error(`Error decrypting backup keychain: ${e.message}`);\n    }\n\n    const userSigningMaterial = JSON.parse(userPrv) as ECDSAMethodTypes.SigningMaterial;\n    const backupSigningMaterial = JSON.parse(backupPrv) as ECDSAMethodTypes.SigningMaterial;\n\n    if (!userSigningMaterial.backupNShare) {\n      throw new Error('Invalid user key - missing backupNShare');\n    }\n\n    if (!backupSigningMaterial.userNShare) {\n      throw new Error('Invalid backup key - missing userNShare');\n    }\n\n    const MPC = new Ecdsa();\n\n    const userKeyCombined = MPC.keyCombine(userSigningMaterial.pShare, [\n      userSigningMaterial.bitgoNShare,\n      userSigningMaterial.backupNShare,\n    ]);\n    const backupKeyCombined = MPC.keyCombine(backupSigningMaterial.pShare, [\n      backupSigningMaterial.bitgoNShare,\n      backupSigningMaterial.userNShare,\n    ]);\n\n    if (\n      userKeyCombined.xShare.y !== backupKeyCombined.xShare.y ||\n      userKeyCombined.xShare.chaincode !== backupKeyCombined.xShare.chaincode\n    ) {\n      throw new Error('Common keychains do not match');\n    }\n\n    return [userKeyCombined, backupKeyCombined];\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> {\n    if (params.isTss) {\n      return this.recoverTSS(params);\n    }\n    return this.recoverEthLike(params);\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 = 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);\n    // get balance of backupKey to ensure funds are available to pay fees\n    const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);\n    const totalGasNeeded = gasPrice.mul(gasLimit);\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 MATIC to this address then retry.`\n      );\n    }\n\n    // get balance of wallet\n    const txAmount = await this.queryAddressBalance(params.walletContractAddress);\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);\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() 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      .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.transfer().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   * 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, '') as string;\n    const bitgoDestinationAddress = params.bitgoDestinationAddress?.replace(/\\s/g, '') as string;\n    const recoveryDestination = params.recoveryDestination?.replace(/\\s/g, '') as string;\n    const walletContractAddress = params.walletContractAddress?.replace(/\\s/g, '') as string;\n    const tokenContractAddress = params.tokenContractAddress?.replace(/\\s/g, '') 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    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    const bitgoFeeAddressNonce = await this.getAddressNonce(bitgoFeeAddress);\n\n    // get balance of bitgoFeeAddress to ensure funds are available to pay fees\n    const bitgoFeeAddressBalance = await this.queryAddressBalance(bitgoFeeAddress);\n    const totalGasNeeded = gasPrice.mul(gasLimit);\n    const weiToGwei = 10 ** 9;\n    if (bitgoFeeAddressBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Fee address ${bitgoFeeAddressBalance} 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    if (tokenContractAddress) {\n      return this.recoverEthLikeTokenforEvmBasedRecovery(\n        params,\n        bitgoFeeAddressNonce,\n        gasLimit,\n        gasPrice,\n        userKey,\n        userSigningKey\n      );\n    }\n\n    // get balance of wallet\n    const txAmount = await this.queryAddressBalance(walletContractAddress);\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);\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 network = this.getNetwork();\n    const batcherContractAddress = network?.batcherContractAddress as string;\n\n    const txBuilder = this.getTransactionBuilder() 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\n    transferBuilder\n      .amount(batchExecutionInfo.totalAmount)\n      .contractSequenceId(sequenceId)\n      .expirationTime(this.getDefaultExpireTime())\n      .to(batcherContractAddress)\n      .data(batchData);\n\n    if (params.walletPassphrase) {\n      txBuilder.transfer().key(userSigningKey);\n    }\n\n    const tx = await txBuilder.build();\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    };\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   * @returns {BigNumber} token balaance in base units\n   */\n  async queryAddressTokenBalance(tokenContractAddress: string, walletContractAddress: string): 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      module: 'account',\n      action: 'tokenbalance',\n      contractaddress: tokenContractAddress,\n      address: walletContractAddress,\n      tag: 'latest',\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  ) {\n    // get token balance of wallet\n    const txAmount = await this.queryAddressTokenBalance(\n      params.tokenContractAddress as string,\n      params.walletContractAddress\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);\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 txBuilder = this.getTransactionBuilder() 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(params.tokenContractAddress as string, network as EthLikeNetwork)?.name as string;\n\n    transferBuilder\n      .amount(txAmount)\n      .contractSequenceId(sequenceId)\n      .expirationTime(this.getDefaultExpireTime())\n      .to(params.recoveryDestination)\n      .coin(token);\n\n    if (params.walletPassphrase) {\n      txBuilder.transfer().key(userSigningKey);\n    }\n\n    const tx = await txBuilder.build();\n\n    const response: OfflineVaultTxInfo = {\n      txHex: tx.toBroadcastFormat(),\n      userKey,\n      coin: token,\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    };\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(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    this.validateRecoveryParams(params);\n    const isUnsignedSweep = getIsUnsignedSweep(params);\n\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    // Set new eth tx fees (using default config values from platform)\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    const [backupKeyAddress, userKeyCombined, backupKeyCombined] = ((): [\n      string,\n      ECDSAMethodTypes.KeyCombined | undefined,\n      ECDSAMethodTypes.KeyCombined | undefined\n    ] => {\n      if (isUnsignedSweep) {\n        const backupKeyPair = new KeyPairLib({ pub: backupPrivateOrPublicKeyShare });\n        return [backupKeyPair.getAddress(), undefined, undefined];\n      } else {\n        const [userKeyCombined, backupKeyCombined] = this.getKeyCombinedFromTssKeyShares(\n          userPublicOrPrivateKeyShare,\n          backupPrivateOrPublicKeyShare,\n          params.walletPassphrase\n        );\n        const backupKeyPair = new KeyPairLib({ pub: backupKeyCombined.xShare.y });\n        return [backupKeyPair.getAddress(), userKeyCombined, backupKeyCombined];\n      }\n    })();\n\n    const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);\n\n    // get balance of backupKey to ensure funds are available to pay fees\n    const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);\n\n    const totalGasNeeded = gasPrice.mul(gasLimit);\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 ETH to this address then retry.`\n      );\n    }\n\n    // get balance of wallet and deduct fees to get transaction amount, wallet contract address acts as base address for tss?\n    const txAmount = backupKeyBalance.sub(totalGasNeeded);\n\n    // build recipients object\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, // no contract address, so this field should not be used anyways\n      nonce: backupKeyNonce,\n      value: txAmount,\n      gasPrice: gasPrice,\n      gasLimit: gasLimit,\n      data: Buffer.from('0x'), // no contract call\n      eip1559: params.eip1559,\n      replayProtectionOptions: params.replayProtectionOptions,\n    };\n\n    let tx = AbstractEthLikeNewCoins.buildTransaction(txParams);\n\n    if (isUnsignedSweep) {\n      return this.formatForOfflineVaultTSS(\n        txInfo,\n        tx,\n        userPublicOrPrivateKeyShare,\n        backupPrivateOrPublicKeyShare,\n        gasPrice,\n        gasLimit,\n        backupKeyNonce,\n        params.eip1559,\n        params.replayProtectionOptions\n      );\n    }\n\n    const signableHex = tx.getMessageToSign(false).toString('hex');\n    if (!userKeyCombined || !backupKeyCombined) {\n      throw new Error('Missing key combined shares for user or backup');\n    }\n    const signature = await this.signRecoveryTSS(userKeyCombined, backupKeyCombined, signableHex);\n    const ethCommmon = AbstractEthLikeNewCoins.getEthLikeCommon(params.eip1559, params.replayProtectionOptions);\n    tx = this.getSignedTxFromSignature(ethCommmon, tx, signature);\n\n    return {\n      id: addHexPrefix(tx.hash().toString('hex')),\n      tx: addHexPrefix(tx.serialize().toString('hex')),\n    };\n  }\n\n  async recoveryBlockchainExplorerQuery(query: Record<string, 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      },\n      gasLimit,\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  private 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   * 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): Promise<boolean> {\n    const ethUtil = optionalDeps.ethUtil;\n\n    let expectedAddress;\n    let actualAddress;\n\n    const { address, coinSpecific, baseAddress, impliedForwarderVersion = coinSpecific?.forwarderVersion } = params;\n\n    if (address && !this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    // base address is required to calculate the salt which is used in calculateForwarderV1Address method\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    if (impliedForwarderVersion === 0 || impliedForwarderVersion === 3) {\n      return true;\n    } else {\n      const ethNetwork = this.getNetwork();\n      const forwarderFactoryAddress = ethNetwork?.forwarderFactoryAddress as string;\n      const forwarderImplementationAddress = ethNetwork?.forwarderImplementationAddress as string;\n\n      const initcode = getProxyInitcode(forwarderImplementationAddress);\n      const saltBuffer = ethUtil.setLengthLeft(\n        Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(coinSpecific.salt || '')), 'hex'),\n        32\n      );\n\n      // Hash the wallet base address with the given salt, so the address directly relies on the base address\n      const calculationSalt = optionalDeps.ethUtil.bufferToHex(\n        optionalDeps.ethAbi.soliditySHA3(['address', 'bytes32'], [baseAddress, saltBuffer])\n      );\n\n      expectedAddress = calculateForwarderV1Address(forwarderFactoryAddress, calculationSalt, initcode);\n      actualAddress = address;\n    }\n\n    if (expectedAddress !== actualAddress) {\n      throw new UnexpectedAddressError(`address validation failure: expected ${expectedAddress} but got ${address}`);\n    }\n\n    return true;\n  }\n\n  /**\n   *\n   * @param {TransactionPrebuild} txPrebuild\n   * @returns {boolean}\n   */\n  verifyCoin(txPrebuild: TransactionPrebuild): boolean {\n    return txPrebuild.coin === this.getChain();\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   */\n  verifyTssTransaction(params: VerifyEthTransactionOptions): boolean {\n    const { txParams, txPrebuild, wallet } = params;\n    if (\n      !txParams?.recipients &&\n      !(\n        txParams.prebuildTx?.consolidateId ||\n        (txParams.type && ['acceleration', 'fillNonce', 'transferToken'].includes(txParams.type))\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    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   */\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    if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {\n      throw new Error(`missing params`);\n    }\n    if (txParams.hop && txParams.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(`txPrebuild should only have 1 recipient but ${txPrebuild.recipients.length} found`);\n    }\n    if (txParams.hop && txPrebuild.hopTransaction) {\n      // Check recipient amount for hop transaction\n      if (txParams.recipients.length !== 1) {\n        throw new Error(`hop transaction only supports 1 recipient but ${txParams.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        throw new Error('recipient address of txPrebuild does not match hop address');\n      }\n\n      // Convert TransactionRecipient array to Recipient array\n      const recipients: Recipient[] = txParams.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 });\n    } else if (txParams.recipients.length > 1) {\n      // Check total amount for batch transaction\n      let expectedTotalAmount = new BigNumber(0);\n      for (let i = 0; i < txParams.recipients.length; i++) {\n        expectedTotalAmount = expectedTotalAmount.plus(txParams.recipients[i].amount);\n      }\n      if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n        throw new Error(\n          'batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\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        throw new Error('recipient address of txPrebuild does not match batcher address');\n      }\n    } else {\n      // Check recipient address and amount for normal transaction\n      if (txParams.recipients.length !== 1) {\n        throw new Error(`normal transaction only supports 1 recipient but ${txParams.recipients.length} found`);\n      }\n      const expectedAmount = new BigNumber(txParams.recipients[0].amount);\n      if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n        throw new Error(\n          'normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n      }\n      if (\n        this.isETHAddress(txParams.recipients[0].address) &&\n        txParams.recipients[0].address !== txPrebuild.recipients[0].address\n      ) {\n        throw new Error('destination address in normal txPrebuild does not match that in txParams supplied by client');\n      }\n    }\n    // Check coin is correct for all transaction types\n    if (!this.verifyCoin(txPrebuild)) {\n      throw new Error(`coin in txPrebuild did not match that in txParams supplied by client`);\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.\n   */\n  encodeMessage(message: string): string {\n    const prefix = `\\u0019Ethereum Signed Message:\\n${message.length}`;\n    return prefix.concat(message);\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.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"]}