@bitgo-beta/sdk-coin-stx 1.4.3-alpha.41 → 1.4.3-alpha.411

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. package/dist/src/index.d.ts +1 -0
  2. package/dist/src/index.d.ts.map +1 -1
  3. package/dist/src/index.js +24 -9
  4. package/dist/src/lib/abstractContractBuilder.js +2 -2
  5. package/dist/src/lib/constants.d.ts +2 -1
  6. package/dist/src/lib/constants.d.ts.map +1 -1
  7. package/dist/src/lib/constants.js +5 -3
  8. package/dist/src/lib/contractBuilder.js +10 -10
  9. package/dist/src/lib/fungibleTokenTransferBuilder.d.ts +62 -0
  10. package/dist/src/lib/fungibleTokenTransferBuilder.d.ts.map +1 -0
  11. package/dist/src/lib/fungibleTokenTransferBuilder.js +131 -0
  12. package/dist/src/lib/iface.d.ts +56 -0
  13. package/dist/src/lib/iface.d.ts.map +1 -1
  14. package/dist/src/lib/iface.js +1 -1
  15. package/dist/src/lib/index.js +23 -9
  16. package/dist/src/lib/keyPair.js +18 -19
  17. package/dist/src/lib/sendmanyBuilder.js +11 -11
  18. package/dist/src/lib/transaction.d.ts.map +1 -1
  19. package/dist/src/lib/transaction.js +31 -15
  20. package/dist/src/lib/transactionBuilder.js +12 -12
  21. package/dist/src/lib/transactionBuilderFactory.d.ts +2 -0
  22. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  23. package/dist/src/lib/transactionBuilderFactory.js +9 -2
  24. package/dist/src/lib/transferBuilder.js +4 -4
  25. package/dist/src/lib/utils.d.ts +25 -2
  26. package/dist/src/lib/utils.d.ts.map +1 -1
  27. package/dist/src/lib/utils.js +141 -62
  28. package/dist/src/register.d.ts.map +1 -1
  29. package/dist/src/register.js +5 -1
  30. package/dist/src/sip10Token.d.ts +22 -0
  31. package/dist/src/sip10Token.d.ts.map +1 -0
  32. package/dist/src/sip10Token.js +128 -0
  33. package/dist/src/stx.d.ts +146 -2
  34. package/dist/src/stx.d.ts.map +1 -1
  35. package/dist/src/stx.js +420 -11
  36. package/dist/src/tstx.js +1 -1
  37. package/dist/test/fixtures.d.ts +119 -0
  38. package/dist/test/fixtures.d.ts.map +1 -0
  39. package/dist/test/fixtures.js +147 -0
  40. package/dist/test/unit/keyPair.d.ts +2 -0
  41. package/dist/test/unit/keyPair.d.ts.map +1 -0
  42. package/dist/test/unit/keyPair.js +144 -0
  43. package/dist/test/unit/resources.d.ts +97 -0
  44. package/dist/test/unit/resources.d.ts.map +1 -0
  45. package/dist/test/unit/resources.js +144 -0
  46. package/dist/test/unit/sip10Token.d.ts +2 -0
  47. package/dist/test/unit/sip10Token.d.ts.map +1 -0
  48. package/dist/test/unit/sip10Token.js +374 -0
  49. package/dist/test/unit/stx.d.ts +2 -0
  50. package/dist/test/unit/stx.d.ts.map +1 -0
  51. package/dist/test/unit/stx.js +470 -0
  52. package/dist/test/unit/transaction.d.ts +2 -0
  53. package/dist/test/unit/transaction.d.ts.map +1 -0
  54. package/dist/test/unit/transaction.js +83 -0
  55. package/dist/test/unit/transactionBuilder/contractBuilder.d.ts +2 -0
  56. package/dist/test/unit/transactionBuilder/contractBuilder.d.ts.map +1 -0
  57. package/dist/test/unit/transactionBuilder/contractBuilder.js +421 -0
  58. package/dist/test/unit/transactionBuilder/fungibleTokenTransferBuilder.d.ts +2 -0
  59. package/dist/test/unit/transactionBuilder/fungibleTokenTransferBuilder.d.ts.map +1 -0
  60. package/dist/test/unit/transactionBuilder/fungibleTokenTransferBuilder.js +187 -0
  61. package/dist/test/unit/transactionBuilder/sendmanyBuilder.d.ts +2 -0
  62. package/dist/test/unit/transactionBuilder/sendmanyBuilder.d.ts.map +1 -0
  63. package/dist/test/unit/transactionBuilder/sendmanyBuilder.js +140 -0
  64. package/dist/test/unit/transactionBuilder/transferBuilder.d.ts +2 -0
  65. package/dist/test/unit/transactionBuilder/transferBuilder.d.ts.map +1 -0
  66. package/dist/test/unit/transactionBuilder/transferBuilder.js +347 -0
  67. package/dist/test/unit/util.d.ts +2 -0
  68. package/dist/test/unit/util.d.ts.map +1 -0
  69. package/dist/test/unit/util.js +326 -0
  70. package/dist/tsconfig.tsbuildinfo +1 -0
  71. package/package.json +13 -11
  72. package/.eslintignore +0 -5
  73. package/.mocharc.yml +0 -8
  74. package/CHANGELOG.md +0 -107
package/dist/src/stx.js CHANGED
@@ -1,9 +1,17 @@
1
1
  "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
2
5
  Object.defineProperty(exports, "__esModule", { value: true });
3
6
  exports.Stx = void 0;
4
7
  const sdk_core_1 = require("@bitgo-beta/sdk-core");
5
8
  const statics_1 = require("@bitgo-beta/statics");
9
+ const transactions_1 = require("@stacks/transactions");
10
+ const payload_1 = require("@stacks/transactions/dist/payload");
11
+ const bignumber_js_1 = __importDefault(require("bignumber.js"));
6
12
  const _1 = require(".");
13
+ const lib_1 = require("./lib");
14
+ const utils_1 = require("./lib/utils");
7
15
  class Stx extends sdk_core_1.BaseCoin {
8
16
  constructor(bitgo, staticsCoin) {
9
17
  super(bitgo);
@@ -27,8 +35,18 @@ class Stx extends sdk_core_1.BaseCoin {
27
35
  getBaseFactor() {
28
36
  return Math.pow(10, this._staticsCoin.decimalPlaces);
29
37
  }
38
+ getTransaction(coinConfig) {
39
+ return new lib_1.TransactionBuilderFactory(coinConfig).getTransferBuilder();
40
+ }
41
+ /** inherited doc */
42
+ getDefaultMultisigType() {
43
+ return sdk_core_1.multisigTypes.onchain;
44
+ }
30
45
  async verifyTransaction(params) {
31
- // TODO: Implement when available on the SDK.
46
+ const { txParams } = params;
47
+ if (Array.isArray(txParams.recipients) && txParams.recipients.length > 1) {
48
+ throw new Error(`${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
49
+ }
32
50
  return true;
33
51
  }
34
52
  /**
@@ -117,9 +135,10 @@ class Stx extends sdk_core_1.BaseCoin {
117
135
  if (!transaction) {
118
136
  throw new Error('Invalid message passed to signMessage');
119
137
  }
120
- return {
138
+ const txHex = {
121
139
  txHex: transaction.toBroadcastFormat(),
122
140
  };
141
+ return transaction.signature.length >= 2 ? txHex : { halfSigned: txHex };
123
142
  }
124
143
  async parseTransaction(params) {
125
144
  return {};
@@ -145,23 +164,43 @@ class Stx extends sdk_core_1.BaseCoin {
145
164
  const tx = await txBuilder.build();
146
165
  const txJson = tx.toJson();
147
166
  if (tx.type === sdk_core_1.TransactionType.Send) {
148
- const outputs = [
149
- {
167
+ // check if it is a token transaction or native coin transaction
168
+ let transactionRecipient;
169
+ let outputAmount;
170
+ let memo;
171
+ if (txJson.payload.contractAddress && txJson.payload.functionArgs.length >= 3) {
172
+ outputAmount = (0, transactions_1.cvToValue)(txJson.payload.functionArgs[0]).toString();
173
+ transactionRecipient = {
174
+ address: (0, transactions_1.cvToString)(txJson.payload.functionArgs[2]),
175
+ amount: outputAmount,
176
+ tokenName: (0, utils_1.findTokenNameByContract)(txJson.payload.contractAddress, txJson.payload.contractName),
177
+ };
178
+ if (txJson.payload.functionArgs.length === 4 &&
179
+ txJson.payload.functionArgs[3].type === transactions_1.ClarityType.OptionalSome) {
180
+ memo = Buffer.from(txJson.payload.functionArgs[3].value.buffer).toString();
181
+ transactionRecipient['memo'] = memo;
182
+ }
183
+ }
184
+ else {
185
+ outputAmount = txJson.payload.amount;
186
+ memo = txJson.payload.memo;
187
+ transactionRecipient = {
150
188
  address: txJson.payload.to,
151
- amount: txJson.payload.amount,
152
- memo: txJson.payload.memo,
153
- },
154
- ];
189
+ amount: outputAmount,
190
+ memo: memo,
191
+ };
192
+ }
193
+ const outputs = [transactionRecipient];
155
194
  const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'memo', 'type'];
156
195
  return {
157
196
  displayOrder,
158
197
  id: txJson.id,
159
- outputAmount: txJson.payload.amount.toString(),
198
+ outputAmount: outputAmount.toString(),
160
199
  changeAmount: '0',
161
200
  outputs,
162
201
  changeOutputs: [],
163
202
  fee: txJson.fee,
164
- memo: txJson.payload.memo,
203
+ memo: memo,
165
204
  type: tx.type,
166
205
  };
167
206
  }
@@ -191,6 +230,376 @@ class Stx extends sdk_core_1.BaseCoin {
191
230
  };
192
231
  }
193
232
  }
233
+ /**
234
+ * Get URLs of some active public nodes
235
+ * @returns {String} node url
236
+ */
237
+ getPublicNodeUrl() {
238
+ return sdk_core_1.Environments[this.bitgo.getEnv()].stxNodeUrl;
239
+ }
240
+ /**
241
+ * Get native stacks balance for an account
242
+ * @param {String} address - stacks address
243
+ * @returns {Promise<NativeStxBalance>}
244
+ */
245
+ async getNativeStxBalanceFromNode({ address }) {
246
+ const endpoint = `${this.getPublicNodeUrl()}/extended/v2/addresses/${address}/balances/stx`;
247
+ try {
248
+ const response = await this.bitgo.get(endpoint);
249
+ if (response.statusCode !== 200) {
250
+ throw new Error(`request failed with status ${response.statusCode}`);
251
+ }
252
+ const body = response.body;
253
+ return body;
254
+ }
255
+ catch (e) {
256
+ throw new Error(`unable to get native stx balance from node: ${e.message}`);
257
+ }
258
+ }
259
+ /**
260
+ * Get single fungible token balance for an account
261
+ * @param {String} address - stacks address
262
+ * @param {String} assetId - fungible token asset id
263
+ * @returns {Promise<SingleFungibleTokenBalance>}
264
+ */
265
+ async getSingleFungibleTokenBalanceFromNode({ address, assetId, }) {
266
+ const endpoint = `${this.getPublicNodeUrl()}/extended/v2/addresses/${address}/balances/ft/${assetId}`;
267
+ try {
268
+ const response = await this.bitgo.get(endpoint);
269
+ if (response.statusCode !== 200) {
270
+ throw new Error(`request failed with status ${response.statusCode}`);
271
+ }
272
+ const body = response.body;
273
+ return body;
274
+ }
275
+ catch (e) {
276
+ throw new Error(`unable to get native stx balance from node: ${e.message}`);
277
+ }
278
+ }
279
+ /**
280
+ * Get nonce data specific to an account from a public node
281
+ * @param {String} address - stacks address
282
+ * @returns {Promise<StxNonceResponse>}
283
+ */
284
+ async getAccountNonceFromNode({ address }) {
285
+ const endpoint = `${this.getPublicNodeUrl()}/extended/v1/address/${address}/nonces`;
286
+ try {
287
+ const response = await this.bitgo.get(endpoint);
288
+ if (response.statusCode !== 200) {
289
+ throw new Error(`request failed with status ${response.statusCode}`);
290
+ }
291
+ const body = response.body;
292
+ return body;
293
+ }
294
+ catch (e) {
295
+ throw new Error(`unable to get account nonce from node: ${e.message}`);
296
+ }
297
+ }
298
+ /**
299
+ * Get stacks transaction estimated fee
300
+ * @param {String} txHex - hex of stacks transaction payload
301
+ * @param {Number} txHexLength - length of built serialized transaction
302
+ * @returns {Promise<Number>} - fee estimate (taking the lowest)
303
+ */
304
+ async getTransactionFeeEstimation({ txHex, txHexLength, }) {
305
+ const endpoint = `${this.getPublicNodeUrl()}/v2/fees/transaction`;
306
+ const requestBody = {
307
+ transaction_payload: txHex,
308
+ estimated_len: txHexLength,
309
+ };
310
+ try {
311
+ const response = await this.bitgo.post(endpoint).send(requestBody);
312
+ if (response.statusCode !== 200) {
313
+ throw new Error(`request failed with status ${response.statusCode}`);
314
+ }
315
+ const body = response.body;
316
+ if (body.estimations.length !== 3) {
317
+ throw new Error('Invalid response estimation length');
318
+ }
319
+ return body.estimations[0].fee;
320
+ }
321
+ catch (e) {
322
+ throw new Error(`unable to get transaction fee estimation: ${e.message}`);
323
+ }
324
+ }
325
+ /**
326
+ * Format for offline vault signing
327
+ * @param {BaseTransaction} tx - base transaction
328
+ * @returns {Promise<RecoveryInfo>}
329
+ */
330
+ async formatForOfflineVault(tx) {
331
+ const txJson = tx.toJson();
332
+ const transactionExplanation = (await this.explainTransaction({
333
+ txHex: tx.toBroadcastFormat(),
334
+ feeInfo: { fee: txJson.fee },
335
+ }));
336
+ transactionExplanation.coin = this.getChain();
337
+ transactionExplanation.feeInfo = { fee: txJson.fee };
338
+ transactionExplanation.txHex = tx.toBroadcastFormat();
339
+ return transactionExplanation;
340
+ }
341
+ /**
342
+ * Get the recoverable amount & fee after subtracting the txn fee
343
+ * @param {String} serializedHex - serialized txn hex
344
+ * @param {Number} txHexLength - deserialized txn length
345
+ * @param {String} balance - total account balance
346
+ * @param {String} tokenBalance - total token balance
347
+ * @returns {Promise<Record<string, string>>}
348
+ */
349
+ async getRecoverableAmountAndFee(serializedHex, txHexLength, balance, tokenBalance) {
350
+ const estimatedFee = await this.getTransactionFeeEstimation({
351
+ txHex: serializedHex,
352
+ txHexLength: txHexLength,
353
+ });
354
+ const balanceBN = new bignumber_js_1.default(balance);
355
+ const feeBN = new bignumber_js_1.default(estimatedFee);
356
+ if (balanceBN.isLessThan(feeBN)) {
357
+ throw new Error('insufficient balance to build the transaction');
358
+ }
359
+ return {
360
+ recoverableAmount: tokenBalance ?? balanceBN.minus(feeBN).toString(),
361
+ fee: feeBN.toString(),
362
+ };
363
+ }
364
+ /**
365
+ * Method to find the right builder for token or native coin transfer
366
+ * @param {String} contractAddress - token contract address
367
+ * @param {String} contractName - token contract name
368
+ * @returns {TransferBuilder|FungibleTokenTransferBuilder}
369
+ */
370
+ getTokenOrNativeTransferBuilder(contractAddress, contractName) {
371
+ const isToken = !!contractAddress && !!contractName;
372
+ let factory;
373
+ if (isToken) {
374
+ const tokenName = (0, utils_1.findTokenNameByContract)(contractAddress, contractName);
375
+ if (!tokenName) {
376
+ throw new Error('invalid contract address or contract name, not supported');
377
+ }
378
+ factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(tokenName));
379
+ }
380
+ else {
381
+ factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
382
+ }
383
+ let builder;
384
+ if (isToken) {
385
+ builder = factory.getFungibleTokenTransferBuilder();
386
+ }
387
+ else {
388
+ builder = factory.getTransferBuilder();
389
+ }
390
+ return builder;
391
+ }
392
+ /**
393
+ * Method to build fungible token transfer transaction
394
+ * @param {FungibleTokenTransferBuilder} builder - fungible token transfer builder
395
+ * @param {String} contractAddress - token contract address
396
+ * @param {String} contractName - token contract name
397
+ * @param {String[]} pubs - account public keys
398
+ * @param {Number} nonce - account nonce
399
+ * @param {AddressDetails} rootAddressDetails - root address details
400
+ * @param {AddressDetails} destinationAddressDetails - receive address details
401
+ * @param {String} stxBalance - native stx balance
402
+ * @returns {Promise<BaseTransaction>} - built transaction
403
+ */
404
+ async buildTokenTransferTransaction({ builder, contractAddress, contractName, pubs, nonce, rootAddressDetails, destinationAddressDetails, stxBalance, }) {
405
+ const txBuilder = builder;
406
+ const contractTokenName = (0, utils_1.findContractTokenNameUsingContract)(contractAddress, contractName);
407
+ if (!contractTokenName) {
408
+ throw new Error('invalid contract address or contract name, not supported');
409
+ }
410
+ const assetId = `${contractAddress}.${contractName}::${contractTokenName}`;
411
+ // fetch the token balance
412
+ const tokenBalanceData = await this.getSingleFungibleTokenBalanceFromNode({
413
+ address: rootAddressDetails.address,
414
+ assetId,
415
+ });
416
+ const tokenBalance = tokenBalanceData?.balance;
417
+ if (!Number(tokenBalance) || isNaN(Number(tokenBalance))) {
418
+ throw new Error(`no token balance found to recover for address: ${rootAddressDetails.address}, token: ${assetId}`);
419
+ }
420
+ txBuilder.fee({ fee: '200' });
421
+ txBuilder.numberSignatures(2);
422
+ txBuilder.fromPubKey(pubs);
423
+ txBuilder.nonce(nonce);
424
+ txBuilder.contractAddress(contractAddress);
425
+ txBuilder.contractName(contractName);
426
+ if (contractTokenName) {
427
+ txBuilder.tokenName(contractTokenName);
428
+ }
429
+ txBuilder.functionName('transfer');
430
+ const functionArgs = [
431
+ (0, transactions_1.uintCV)(tokenBalance),
432
+ (0, transactions_1.standardPrincipalCV)(rootAddressDetails.address),
433
+ (0, transactions_1.standardPrincipalCV)(destinationAddressDetails.address),
434
+ ];
435
+ if (destinationAddressDetails.memoId) {
436
+ functionArgs.push((0, transactions_1.someCV)((0, transactions_1.bufferCVFromString)(destinationAddressDetails.memoId)));
437
+ }
438
+ else {
439
+ functionArgs.push((0, transactions_1.noneCV)());
440
+ }
441
+ txBuilder.functionArgs(functionArgs);
442
+ const baseTxn = await txBuilder.build();
443
+ const txBroadcastFormat = baseTxn.toBroadcastFormat();
444
+ const txDeserialized = (0, transactions_1.deserializeTransaction)(txBroadcastFormat);
445
+ const serializedHex = (0, payload_1.serializePayload)(txDeserialized.payload).toString('hex');
446
+ const { recoverableAmount, fee } = await this.getRecoverableAmountAndFee(serializedHex, txBroadcastFormat.length, stxBalance, tokenBalance);
447
+ functionArgs[0] = (0, transactions_1.uintCV)(recoverableAmount);
448
+ txBuilder.functionArgs(functionArgs);
449
+ txBuilder.fee({ fee: fee });
450
+ return await txBuilder.build();
451
+ }
452
+ /**
453
+ * Method to build native transfer transaction
454
+ * @param {TransferBuilder} builder - transfer builder
455
+ * @param {String[]} pubs - account public keys
456
+ * @param {Number} nonce - account nonce
457
+ * @param {AddressDetails} destinationAddressDetails - receive address details
458
+ * @param {String} stxBalance - native stx balance
459
+ * @returns {Promise<BaseTransaction>} - built transaction
460
+ */
461
+ async buildNativeTransferTransaction({ builder, pubs, nonce, destinationAddressDetails, stxBalance, }) {
462
+ const txBuilder = builder;
463
+ txBuilder.fee({ fee: '200' });
464
+ txBuilder.numberSignatures(2);
465
+ txBuilder.fromPubKey(pubs);
466
+ txBuilder.nonce(nonce);
467
+ txBuilder.to(destinationAddressDetails.address);
468
+ txBuilder.amount(stxBalance);
469
+ if (destinationAddressDetails.memoId) {
470
+ txBuilder.memo(destinationAddressDetails.memoId);
471
+ }
472
+ const baseTxn = await txBuilder.build();
473
+ const txBroadcastFormat = baseTxn.toBroadcastFormat();
474
+ const txDeserialized = (0, transactions_1.deserializeTransaction)(txBroadcastFormat);
475
+ const serializedHex = (0, payload_1.serializePayload)(txDeserialized.payload).toString('hex');
476
+ const { recoverableAmount, fee } = await this.getRecoverableAmountAndFee(serializedHex, txBroadcastFormat.length, stxBalance);
477
+ txBuilder.amount(recoverableAmount);
478
+ txBuilder.fee({ fee: fee });
479
+ return await txBuilder.build();
480
+ }
481
+ /**
482
+ * Method that uses appropriate builder and builds transaction depending on token or native coin
483
+ * @param {String[]} pubs - public keys
484
+ * @param {AddressDetails} rootAddressDetails - sender address detail
485
+ * @param {AddressDetails} destinationAddressDetails - receiver address detail
486
+ * @param {Number} nonce - wallet nonce
487
+ * @param {String} balance - wallet balance
488
+ * @param {String | undefined} contractAddress - token contract address
489
+ * @param {String | undefined} contractName - token contract name
490
+ * @returns {Promise<BaseTransaction>} built transaction
491
+ */
492
+ async getNativeOrTokenTransaction({ pubs, rootAddressDetails, destinationAddressDetails, nonce, stxBalance, contractAddressInput, contractName, }) {
493
+ const builder = this.getTokenOrNativeTransferBuilder(contractAddressInput, contractName);
494
+ const contractAddress = contractAddressInput?.toUpperCase();
495
+ const isToken = !!contractAddress && !!contractName;
496
+ let finalTx;
497
+ if (isToken) {
498
+ finalTx = await this.buildTokenTransferTransaction({
499
+ builder: builder,
500
+ contractAddress,
501
+ contractName,
502
+ pubs,
503
+ nonce,
504
+ rootAddressDetails,
505
+ destinationAddressDetails,
506
+ stxBalance,
507
+ });
508
+ }
509
+ else {
510
+ finalTx = await this.buildNativeTransferTransaction({
511
+ builder: builder,
512
+ pubs,
513
+ nonce,
514
+ destinationAddressDetails,
515
+ stxBalance,
516
+ });
517
+ }
518
+ return {
519
+ tx: finalTx,
520
+ builder: builder,
521
+ };
522
+ }
523
+ /**
524
+ * Method to recover native stx or sip10 tokens from bitgo hot & cold wallets
525
+ * @param {String} params.backupKey - encrypted wallet backup key (public or private)
526
+ * @param {String} params.userKey - encrypted wallet user key (public or private)
527
+ * @param {String} params.rootAddress - wallet root address
528
+ * @param {String} params.recoveryDestination - receive address
529
+ * @param {String} params.bitgoKey - encrypted bitgo public key
530
+ * @param {String} params.walletPassphrase - wallet password
531
+ * @param {String} params.contractId - contract id of the token (mandatory for token recovery)
532
+ * @returns {Promise<RecoveryInfo|RecoveryTransaction>} RecoveryTransaction.txHex - hex of serialized transaction (signed or unsigned)
533
+ */
534
+ async recover(params) {
535
+ if (!this.isValidAddress(params.rootAddress)) {
536
+ throw new Error('invalid root address!');
537
+ }
538
+ if (!this.isValidAddress(params.recoveryDestination)) {
539
+ throw new Error('invalid destination address!');
540
+ }
541
+ let contractAddress;
542
+ let contractName;
543
+ if (params.contractId) {
544
+ [contractAddress, contractName] = params.contractId.split('.');
545
+ if ((contractAddress && !contractName) || (contractName && !contractAddress)) {
546
+ throw new Error('invalid contract id, please provide it in the form (contractAddress.contractName)');
547
+ }
548
+ }
549
+ const isUnsignedSweep = (0, sdk_core_1.getIsUnsignedSweep)(params);
550
+ const keys = (0, sdk_core_1.getBip32Keys)(this.bitgo, params, { requireBitGoXpub: true });
551
+ const rootAddressDetails = (0, utils_1.getAddressDetails)(params.rootAddress);
552
+ const [accountBalanceData, accountNonceData] = await Promise.all([
553
+ this.getNativeStxBalanceFromNode({ address: rootAddressDetails.address }),
554
+ this.getAccountNonceFromNode({ address: rootAddressDetails.address }),
555
+ ]);
556
+ const balance = Number(accountBalanceData.balance);
557
+ if (!balance || isNaN(balance)) {
558
+ throw new Error('could not find any balance to recover for ' + params.rootAddress);
559
+ }
560
+ const userPub = (0, transactions_1.publicKeyFromBuffer)(keys[0].publicKey);
561
+ const backupPub = (0, transactions_1.publicKeyFromBuffer)(keys[1].publicKey);
562
+ const bitgoPubKey = (0, transactions_1.publicKeyFromBuffer)(keys[2].publicKey);
563
+ const pubs = [(0, transactions_1.publicKeyToString)(userPub), (0, transactions_1.publicKeyToString)(backupPub), (0, transactions_1.publicKeyToString)(bitgoPubKey)];
564
+ const destinationAddressDetails = (0, utils_1.getAddressDetails)(params.recoveryDestination);
565
+ const nonce = typeof accountNonceData?.last_executed_tx_nonce === 'number' ? accountNonceData.last_executed_tx_nonce + 1 : 0;
566
+ const { tx, builder } = await this.getNativeOrTokenTransaction({
567
+ pubs,
568
+ rootAddressDetails,
569
+ destinationAddressDetails,
570
+ nonce,
571
+ stxBalance: accountBalanceData.balance,
572
+ contractAddressInput: contractAddress,
573
+ contractName: contractName,
574
+ });
575
+ if (isUnsignedSweep) {
576
+ return await this.formatForOfflineVault(tx);
577
+ }
578
+ // check the private key & sign
579
+ if (!keys[0].privateKey) {
580
+ throw new Error(`userKey is not a private key`);
581
+ }
582
+ const userKey = (0, transactions_1.createStacksPrivateKey)(keys[0].privateKey);
583
+ builder.sign({ key: (0, transactions_1.privateKeyToString)(userKey) });
584
+ const halfSignedTx = await builder.build();
585
+ const txHexHalfSigned = halfSignedTx.toBroadcastFormat();
586
+ const builder2 = this.getTokenOrNativeTransferBuilder(contractAddress, contractName);
587
+ builder2.from(txHexHalfSigned);
588
+ if (!keys[1].privateKey) {
589
+ throw new Error(`backupKey is not a private key`);
590
+ }
591
+ const backupKey = (0, transactions_1.createStacksPrivateKey)(keys[1].privateKey);
592
+ builder2.sign({ key: (0, transactions_1.privateKeyToString)(backupKey) });
593
+ const fullySignedTx = await builder2.build();
594
+ const fullySignedTxHex = fullySignedTx.toBroadcastFormat();
595
+ return {
596
+ txHex: fullySignedTxHex,
597
+ };
598
+ }
599
+ /** @inheritDoc */
600
+ auditDecryptedKey(params) {
601
+ throw new sdk_core_1.MethodNotImplementedError();
602
+ }
194
603
  }
195
604
  exports.Stx = Stx;
196
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stx.js","sourceRoot":"","sources":["../../src/stx.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseCoin,\n  BitGoBase,\n  KeyPair,\n  SignedTransaction,\n  TransactionRecipient,\n  TransactionType,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, CoinFamily, coins } from '@bitgo-beta/statics';\nimport { ExplainTransactionOptions, StxSignTransactionOptions, StxTransactionExplanation } from './types';\nimport { StxLib } from '.';\n\nexport class Stx extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n\n  constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Stx(bitgo, staticsCoin);\n  }\n\n  getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  getFullName(): string {\n    return this._staticsCoin.fullName;\n  }\n\n  getBaseFactor(): string | number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    // TODO: Implement when available on the SDK.\n    return true;\n  }\n\n  /**\n   * Check if address is valid, then make sure it matches the base address.\n   *\n   * @param {VerifyAddressOptions} params\n   * @param {String} params.address - the address to verify\n   * @param {String} params.baseAddress - the base address from the wallet\n   */\n  async isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    const { address, keychains } = params;\n    if (!keychains || keychains.length !== 3) {\n      throw new Error('Invalid keychains');\n    }\n    const pubs = keychains.map((keychain) => StxLib.Utils.xpubToSTXPubkey(keychain.pub));\n    const addressVersion = StxLib.Utils.getAddressVersion(address);\n    const baseAddress = StxLib.Utils.getSTXAddressFromPubKeys(pubs, addressVersion).address;\n    return StxLib.Utils.isSameBaseAddress(address, baseAddress);\n  }\n\n  /**\n   * Generate Stacks key pair\n   *\n   * @param {Buffer} seed - Seed from which the new keypair should be generated, otherwise a random seed is used\n   * @returns {Object} object with generated pub and prv\n   */\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new StxLib.KeyPair({ seed }) : new StxLib.KeyPair();\n    const keys = keyPair.getExtendedKeys();\n\n    if (!keys.xprv) {\n      throw new Error('Missing xprv in key generation.');\n    }\n\n    return {\n      pub: keys.xpub,\n      prv: keys.xprv,\n    };\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin\n   *\n   * @param {string} pub the prv to be checked\n   * @returns is it valid?\n   */\n  isValidPub(pub: string): boolean {\n    try {\n      return StxLib.Utils.isValidPublicKey(pub);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Return boolean indicating whether input is valid private key for the coin\n   *\n   * @param {string} prv the prv to be checked\n   * @returns is it valid?\n   */\n  isValidPrv(prv: string): boolean {\n    try {\n      return StxLib.Utils.isValidPrivateKey(prv);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  isValidAddress(address: string): boolean {\n    try {\n      return StxLib.Utils.isValidAddressWithPaymentId(address);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Signs stacks transaction\n   * @param params\n   */\n  async signTransaction(params: StxSignTransactionOptions): Promise<SignedTransaction> {\n    const factory = new StxLib.TransactionBuilderFactory(coins.get(this.getChain()));\n    const txBuilder = factory.from(params.txPrebuild.txHex);\n    const prvKeys = params.prv instanceof Array ? params.prv : [params.prv];\n    prvKeys.forEach((prv) => txBuilder.sign({ key: prv }));\n    if (params.pubKeys) txBuilder.fromPubKey(params.pubKeys);\n    // if (params.numberSignature) txBuilder.numberSignatures(params.numberSignature);\n    const transaction = await txBuilder.build();\n\n    if (!transaction) {\n      throw new Error('Invalid message passed to signMessage');\n    }\n\n    return {\n      txHex: transaction.toBroadcastFormat(),\n    };\n  }\n\n  async parseTransaction(params: any): Promise<any> {\n    return {};\n  }\n\n  /**\n   * Explain a Stacks transaction from txHex\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<StxTransactionExplanation | undefined> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex || !params.feeInfo) {\n      throw new Error('missing explain tx parameters');\n    }\n\n    const factory = new StxLib.TransactionBuilderFactory(coins.get(this.getChain()));\n    const txBuilder = factory.from(txHex);\n\n    if (params.publicKeys !== undefined) {\n      txBuilder.fromPubKey(params.publicKeys);\n      if (params.publicKeys.length === 1) {\n        // definitely a single sig tx\n        txBuilder.numberSignatures(1);\n      }\n    }\n\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson();\n\n    if (tx.type === TransactionType.Send) {\n      const outputs: TransactionRecipient[] = [\n        {\n          address: txJson.payload.to,\n          amount: txJson.payload.amount,\n          memo: txJson.payload.memo,\n        },\n      ];\n\n      const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'memo', 'type'];\n      return {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: txJson.payload.amount.toString(),\n        changeAmount: '0',\n        outputs,\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: txJson.payload.memo,\n        type: tx.type,\n      };\n    }\n\n    if (tx.type === TransactionType.ContractCall) {\n      const displayOrder = [\n        'id',\n        'fee',\n        'type',\n        'contractAddress',\n        'contractName',\n        'contractFunction',\n        'contractFunctionArgs',\n      ];\n      return {\n        displayOrder,\n        id: txJson.id,\n        changeAmount: '0',\n        outputAmount: '',\n        outputs: [],\n        changeOutputs: [],\n        fee: txJson.fee,\n        type: tx.type,\n        contractAddress: txJson.payload.contractAddress,\n        contractName: txJson.payload.contractName,\n        contractFunction: txJson.payload.functionName,\n        contractFunctionArgs: txJson.payload.functionArgs,\n      };\n    }\n  }\n}\n"]}
605
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stx.js","sourceRoot":"","sources":["../../src/stx.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AuditDecryptedKeyParams,\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  Environments,\n  getBip32Keys,\n  getIsUnsignedSweep,\n  KeyPair,\n  MethodNotImplementedError,\n  MultisigType,\n  multisigTypes,\n  SignedTransaction,\n  TransactionRecipient,\n  TransactionType,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, CoinFamily, coins } from '@bitgo-beta/statics';\nimport {\n  bufferCVFromString,\n  ClarityType,\n  ClarityValue,\n  createStacksPrivateKey,\n  cvToString,\n  cvToValue,\n  deserializeTransaction,\n  noneCV,\n  privateKeyToString,\n  publicKeyFromBuffer,\n  publicKeyToString,\n  someCV,\n  standardPrincipalCV,\n  uintCV,\n} from '@stacks/transactions';\nimport { serializePayload } from '@stacks/transactions/dist/payload';\nimport BigNumber from 'bignumber.js';\n\nimport { ExplainTransactionOptions, StxSignTransactionOptions, StxTransactionExplanation } from './types';\nimport { StxLib } from '.';\nimport { TransactionBuilderFactory } from './lib';\nimport { TransactionBuilder } from './lib/transactionBuilder';\nimport { findContractTokenNameUsingContract, findTokenNameByContract, getAddressDetails } from './lib/utils';\nimport {\n  AddressDetails,\n  NativeStxBalance,\n  RecoveryInfo,\n  RecoveryOptions,\n  RecoveryTransaction,\n  SingleFungibleTokenBalance,\n  StxNonceResponse,\n  StxTxnFeeEstimationResponse,\n  TxData,\n} from './lib/iface';\nimport { TransferBuilder } from './lib/transferBuilder';\nimport { FungibleTokenTransferBuilder } from './lib/fungibleTokenTransferBuilder';\n\nexport class Stx extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n\n  constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Stx(bitgo, staticsCoin);\n  }\n\n  getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  getFullName(): string {\n    return this._staticsCoin.fullName;\n  }\n\n  getBaseFactor(): string | number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  getTransaction(coinConfig: Readonly<StaticsBaseCoin>): TransactionBuilder {\n    return new TransactionBuilderFactory(coinConfig).getTransferBuilder();\n  }\n\n  /** inherited doc */\n  getDefaultMultisigType(): MultisigType {\n    return multisigTypes.onchain;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    const { txParams } = params;\n    if (Array.isArray(txParams.recipients) && txParams.recipients.length > 1) {\n      throw new Error(\n        `${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n      );\n    }\n    return true;\n  }\n\n  /**\n   * Check if address is valid, then make sure it matches the base address.\n   *\n   * @param {VerifyAddressOptions} params\n   * @param {String} params.address - the address to verify\n   * @param {String} params.baseAddress - the base address from the wallet\n   */\n  async isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    const { address, keychains } = params;\n    if (!keychains || keychains.length !== 3) {\n      throw new Error('Invalid keychains');\n    }\n    const pubs = keychains.map((keychain) => StxLib.Utils.xpubToSTXPubkey(keychain.pub));\n    const addressVersion = StxLib.Utils.getAddressVersion(address);\n    const baseAddress = StxLib.Utils.getSTXAddressFromPubKeys(pubs, addressVersion).address;\n    return StxLib.Utils.isSameBaseAddress(address, baseAddress);\n  }\n\n  /**\n   * Generate Stacks key pair\n   *\n   * @param {Buffer} seed - Seed from which the new keypair should be generated, otherwise a random seed is used\n   * @returns {Object} object with generated pub and prv\n   */\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new StxLib.KeyPair({ seed }) : new StxLib.KeyPair();\n    const keys = keyPair.getExtendedKeys();\n\n    if (!keys.xprv) {\n      throw new Error('Missing xprv in key generation.');\n    }\n\n    return {\n      pub: keys.xpub,\n      prv: keys.xprv,\n    };\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin\n   *\n   * @param {string} pub the prv to be checked\n   * @returns is it valid?\n   */\n  isValidPub(pub: string): boolean {\n    try {\n      return StxLib.Utils.isValidPublicKey(pub);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Return boolean indicating whether input is valid private key for the coin\n   *\n   * @param {string} prv the prv to be checked\n   * @returns is it valid?\n   */\n  isValidPrv(prv: string): boolean {\n    try {\n      return StxLib.Utils.isValidPrivateKey(prv);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  isValidAddress(address: string): boolean {\n    try {\n      return StxLib.Utils.isValidAddressWithPaymentId(address);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Signs stacks transaction\n   * @param params\n   */\n  async signTransaction(params: StxSignTransactionOptions): Promise<SignedTransaction> {\n    const factory = new StxLib.TransactionBuilderFactory(coins.get(this.getChain()));\n    const txBuilder = factory.from(params.txPrebuild.txHex);\n    const prvKeys = params.prv instanceof Array ? params.prv : [params.prv];\n    prvKeys.forEach((prv) => txBuilder.sign({ key: prv }));\n    if (params.pubKeys) txBuilder.fromPubKey(params.pubKeys);\n    // if (params.numberSignature) txBuilder.numberSignatures(params.numberSignature);\n    const transaction = await txBuilder.build();\n\n    if (!transaction) {\n      throw new Error('Invalid message passed to signMessage');\n    }\n\n    const txHex = {\n      txHex: transaction.toBroadcastFormat(),\n    };\n\n    return transaction.signature.length >= 2 ? txHex : { halfSigned: txHex };\n  }\n\n  async parseTransaction(params: any): Promise<any> {\n    return {};\n  }\n\n  /**\n   * Explain a Stacks transaction from txHex\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<StxTransactionExplanation | undefined> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex || !params.feeInfo) {\n      throw new Error('missing explain tx parameters');\n    }\n\n    const factory = new StxLib.TransactionBuilderFactory(coins.get(this.getChain()));\n    const txBuilder = factory.from(txHex);\n\n    if (params.publicKeys !== undefined) {\n      txBuilder.fromPubKey(params.publicKeys);\n      if (params.publicKeys.length === 1) {\n        // definitely a single sig tx\n        txBuilder.numberSignatures(1);\n      }\n    }\n\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson();\n\n    if (tx.type === TransactionType.Send) {\n      // check if it is a token transaction or native coin transaction\n      let transactionRecipient: TransactionRecipient;\n      let outputAmount: string;\n      let memo: string | undefined;\n      if (txJson.payload.contractAddress && txJson.payload.functionArgs.length >= 3) {\n        outputAmount = cvToValue(txJson.payload.functionArgs[0]).toString();\n        transactionRecipient = {\n          address: cvToString(txJson.payload.functionArgs[2]),\n          amount: outputAmount,\n          tokenName: findTokenNameByContract(txJson.payload.contractAddress, txJson.payload.contractName),\n        };\n        if (\n          txJson.payload.functionArgs.length === 4 &&\n          txJson.payload.functionArgs[3].type === ClarityType.OptionalSome\n        ) {\n          memo = Buffer.from(txJson.payload.functionArgs[3].value.buffer).toString();\n          transactionRecipient['memo'] = memo;\n        }\n      } else {\n        outputAmount = txJson.payload.amount;\n        memo = txJson.payload.memo;\n        transactionRecipient = {\n          address: txJson.payload.to,\n          amount: outputAmount,\n          memo: memo,\n        };\n      }\n      const outputs: TransactionRecipient[] = [transactionRecipient];\n\n      const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'memo', 'type'];\n      return {\n        displayOrder,\n        id: txJson.id,\n        outputAmount: outputAmount.toString(),\n        changeAmount: '0',\n        outputs,\n        changeOutputs: [],\n        fee: txJson.fee,\n        memo: memo,\n        type: tx.type,\n      };\n    }\n\n    if (tx.type === TransactionType.ContractCall) {\n      const displayOrder = [\n        'id',\n        'fee',\n        'type',\n        'contractAddress',\n        'contractName',\n        'contractFunction',\n        'contractFunctionArgs',\n      ];\n      return {\n        displayOrder,\n        id: txJson.id,\n        changeAmount: '0',\n        outputAmount: '',\n        outputs: [],\n        changeOutputs: [],\n        fee: txJson.fee,\n        type: tx.type,\n        contractAddress: txJson.payload.contractAddress,\n        contractName: txJson.payload.contractName,\n        contractFunction: txJson.payload.functionName,\n        contractFunctionArgs: txJson.payload.functionArgs,\n      };\n    }\n  }\n\n  /**\n   * Get URLs of some active public nodes\n   * @returns {String} node url\n   */\n  getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].stxNodeUrl;\n  }\n\n  /**\n   * Get native stacks balance for an account\n   * @param {String} address - stacks address\n   * @returns {Promise<NativeStxBalance>}\n   */\n  protected async getNativeStxBalanceFromNode({ address }: { address: string }): Promise<NativeStxBalance> {\n    const endpoint = `${this.getPublicNodeUrl()}/extended/v2/addresses/${address}/balances/stx`;\n    try {\n      const response = await this.bitgo.get(endpoint);\n      if (response.statusCode !== 200) {\n        throw new Error(`request failed with status ${response.statusCode}`);\n      }\n      const body: NativeStxBalance = response.body;\n      return body;\n    } catch (e) {\n      throw new Error(`unable to get native stx balance from node: ${e.message}`);\n    }\n  }\n\n  /**\n   * Get single fungible token balance for an account\n   * @param {String} address - stacks address\n   * @param {String} assetId - fungible token asset id\n   * @returns {Promise<SingleFungibleTokenBalance>}\n   */\n  protected async getSingleFungibleTokenBalanceFromNode({\n    address,\n    assetId,\n  }: {\n    address: string;\n    assetId: string;\n  }): Promise<SingleFungibleTokenBalance> {\n    const endpoint = `${this.getPublicNodeUrl()}/extended/v2/addresses/${address}/balances/ft/${assetId}`;\n    try {\n      const response = await this.bitgo.get(endpoint);\n      if (response.statusCode !== 200) {\n        throw new Error(`request failed with status ${response.statusCode}`);\n      }\n      const body: SingleFungibleTokenBalance = response.body;\n      return body;\n    } catch (e) {\n      throw new Error(`unable to get native stx balance from node: ${e.message}`);\n    }\n  }\n\n  /**\n   * Get nonce data specific to an account from a public node\n   * @param {String} address - stacks address\n   * @returns {Promise<StxNonceResponse>}\n   */\n  protected async getAccountNonceFromNode({ address }: { address: string }): Promise<StxNonceResponse> {\n    const endpoint = `${this.getPublicNodeUrl()}/extended/v1/address/${address}/nonces`;\n    try {\n      const response = await this.bitgo.get(endpoint);\n      if (response.statusCode !== 200) {\n        throw new Error(`request failed with status ${response.statusCode}`);\n      }\n      const body: StxNonceResponse = response.body;\n      return body;\n    } catch (e) {\n      throw new Error(`unable to get account nonce from node: ${e.message}`);\n    }\n  }\n\n  /**\n   * Get stacks transaction estimated fee\n   * @param {String} txHex - hex of stacks transaction payload\n   * @param {Number} txHexLength - length of built serialized transaction\n   * @returns {Promise<Number>} - fee estimate (taking the lowest)\n   */\n  protected async getTransactionFeeEstimation({\n    txHex,\n    txHexLength,\n  }: {\n    txHex: string;\n    txHexLength: number;\n  }): Promise<number> {\n    const endpoint = `${this.getPublicNodeUrl()}/v2/fees/transaction`;\n    const requestBody = {\n      transaction_payload: txHex,\n      estimated_len: txHexLength,\n    };\n    try {\n      const response = await this.bitgo.post(endpoint).send(requestBody);\n      if (response.statusCode !== 200) {\n        throw new Error(`request failed with status ${response.statusCode}`);\n      }\n      const body: StxTxnFeeEstimationResponse = response.body;\n      if (body.estimations.length !== 3) {\n        throw new Error('Invalid response estimation length');\n      }\n      return body.estimations[0].fee;\n    } catch (e) {\n      throw new Error(`unable to get transaction fee estimation: ${e.message}`);\n    }\n  }\n\n  /**\n   * Format for offline vault signing\n   * @param {BaseTransaction} tx - base transaction\n   * @returns {Promise<RecoveryInfo>}\n   */\n  protected async formatForOfflineVault(tx: BaseTransaction): Promise<RecoveryInfo> {\n    const txJson: TxData = tx.toJson();\n    const transactionExplanation: RecoveryInfo = (await this.explainTransaction({\n      txHex: tx.toBroadcastFormat(),\n      feeInfo: { fee: txJson.fee },\n    })) as RecoveryInfo;\n    transactionExplanation.coin = this.getChain();\n    transactionExplanation.feeInfo = { fee: txJson.fee };\n    transactionExplanation.txHex = tx.toBroadcastFormat();\n    return transactionExplanation;\n  }\n\n  /**\n   * Get the recoverable amount & fee after subtracting the txn fee\n   * @param {String} serializedHex - serialized txn hex\n   * @param {Number} txHexLength - deserialized txn length\n   * @param {String} balance - total account balance\n   * @param {String} tokenBalance - total token balance\n   * @returns {Promise<Record<string, string>>}\n   */\n  protected async getRecoverableAmountAndFee(\n    serializedHex: string,\n    txHexLength: number,\n    balance: string,\n    tokenBalance?: string\n  ): Promise<Record<string, string>> {\n    const estimatedFee = await this.getTransactionFeeEstimation({\n      txHex: serializedHex,\n      txHexLength: txHexLength,\n    });\n    const balanceBN = new BigNumber(balance);\n    const feeBN = new BigNumber(estimatedFee);\n    if (balanceBN.isLessThan(feeBN)) {\n      throw new Error('insufficient balance to build the transaction');\n    }\n    return {\n      recoverableAmount: tokenBalance ?? balanceBN.minus(feeBN).toString(),\n      fee: feeBN.toString(),\n    };\n  }\n\n  /**\n   * Method to find the right builder for token or native coin transfer\n   * @param {String} contractAddress - token contract address\n   * @param {String} contractName - token contract name\n   * @returns {TransferBuilder|FungibleTokenTransferBuilder}\n   */\n  protected getTokenOrNativeTransferBuilder(\n    contractAddress?: string,\n    contractName?: string\n  ): TransferBuilder | FungibleTokenTransferBuilder {\n    const isToken = !!contractAddress && !!contractName;\n    let factory: TransactionBuilderFactory;\n    if (isToken) {\n      const tokenName = findTokenNameByContract(contractAddress, contractName);\n      if (!tokenName) {\n        throw new Error('invalid contract address or contract name, not supported');\n      }\n      factory = new TransactionBuilderFactory(coins.get(tokenName));\n    } else {\n      factory = new TransactionBuilderFactory(coins.get(this.getChain()));\n    }\n    let builder: TransferBuilder | FungibleTokenTransferBuilder;\n    if (isToken) {\n      builder = factory.getFungibleTokenTransferBuilder();\n    } else {\n      builder = factory.getTransferBuilder();\n    }\n    return builder;\n  }\n\n  /**\n   * Method to build fungible token transfer transaction\n   * @param {FungibleTokenTransferBuilder} builder - fungible token transfer builder\n   * @param {String} contractAddress - token contract address\n   * @param {String} contractName - token contract name\n   * @param {String[]} pubs - account public keys\n   * @param {Number} nonce - account nonce\n   * @param {AddressDetails} rootAddressDetails - root address details\n   * @param {AddressDetails} destinationAddressDetails - receive address details\n   * @param {String} stxBalance - native stx balance\n   * @returns {Promise<BaseTransaction>} - built transaction\n   */\n  protected async buildTokenTransferTransaction({\n    builder,\n    contractAddress,\n    contractName,\n    pubs,\n    nonce,\n    rootAddressDetails,\n    destinationAddressDetails,\n    stxBalance,\n  }: {\n    builder: FungibleTokenTransferBuilder;\n    contractAddress: string;\n    contractName: string;\n    pubs: string[];\n    nonce: number;\n    rootAddressDetails: AddressDetails;\n    destinationAddressDetails: AddressDetails;\n    stxBalance: string;\n  }): Promise<BaseTransaction> {\n    const txBuilder = builder as FungibleTokenTransferBuilder;\n    const contractTokenName = findContractTokenNameUsingContract(contractAddress, contractName);\n    if (!contractTokenName) {\n      throw new Error('invalid contract address or contract name, not supported');\n    }\n    const assetId = `${contractAddress}.${contractName}::${contractTokenName}`;\n    // fetch the token balance\n    const tokenBalanceData = await this.getSingleFungibleTokenBalanceFromNode({\n      address: rootAddressDetails.address,\n      assetId,\n    });\n    const tokenBalance = tokenBalanceData?.balance;\n    if (!Number(tokenBalance) || isNaN(Number(tokenBalance))) {\n      throw new Error(\n        `no token balance found to recover for address: ${rootAddressDetails.address}, token: ${assetId}`\n      );\n    }\n    txBuilder.fee({ fee: '200' });\n    txBuilder.numberSignatures(2);\n    txBuilder.fromPubKey(pubs);\n    txBuilder.nonce(nonce);\n    txBuilder.contractAddress(contractAddress);\n    txBuilder.contractName(contractName);\n    if (contractTokenName) {\n      txBuilder.tokenName(contractTokenName);\n    }\n    txBuilder.functionName('transfer');\n    const functionArgs: ClarityValue[] = [\n      uintCV(tokenBalance),\n      standardPrincipalCV(rootAddressDetails.address),\n      standardPrincipalCV(destinationAddressDetails.address),\n    ];\n    if (destinationAddressDetails.memoId) {\n      functionArgs.push(someCV(bufferCVFromString(destinationAddressDetails.memoId)));\n    } else {\n      functionArgs.push(noneCV());\n    }\n    txBuilder.functionArgs(functionArgs);\n    const baseTxn = await txBuilder.build();\n    const txBroadcastFormat = baseTxn.toBroadcastFormat();\n    const txDeserialized = deserializeTransaction(txBroadcastFormat);\n    const serializedHex = serializePayload(txDeserialized.payload).toString('hex');\n    const { recoverableAmount, fee } = await this.getRecoverableAmountAndFee(\n      serializedHex,\n      txBroadcastFormat.length,\n      stxBalance,\n      tokenBalance\n    );\n    functionArgs[0] = uintCV(recoverableAmount);\n    txBuilder.functionArgs(functionArgs);\n    txBuilder.fee({ fee: fee });\n    return await txBuilder.build();\n  }\n\n  /**\n   * Method to build native transfer transaction\n   * @param {TransferBuilder} builder - transfer builder\n   * @param {String[]} pubs - account public keys\n   * @param {Number} nonce - account nonce\n   * @param {AddressDetails} destinationAddressDetails - receive address details\n   * @param {String} stxBalance - native stx balance\n   * @returns {Promise<BaseTransaction>} - built transaction\n   */\n  protected async buildNativeTransferTransaction({\n    builder,\n    pubs,\n    nonce,\n    destinationAddressDetails,\n    stxBalance,\n  }: {\n    builder: TransferBuilder;\n    pubs: string[];\n    nonce: number;\n    destinationAddressDetails: AddressDetails;\n    stxBalance: string;\n  }): Promise<BaseTransaction> {\n    const txBuilder = builder as TransferBuilder;\n    txBuilder.fee({ fee: '200' });\n    txBuilder.numberSignatures(2);\n    txBuilder.fromPubKey(pubs);\n    txBuilder.nonce(nonce);\n    txBuilder.to(destinationAddressDetails.address);\n    txBuilder.amount(stxBalance);\n    if (destinationAddressDetails.memoId) {\n      txBuilder.memo(destinationAddressDetails.memoId);\n    }\n    const baseTxn = await txBuilder.build();\n    const txBroadcastFormat = baseTxn.toBroadcastFormat();\n    const txDeserialized = deserializeTransaction(txBroadcastFormat);\n    const serializedHex = serializePayload(txDeserialized.payload).toString('hex');\n    const { recoverableAmount, fee } = await this.getRecoverableAmountAndFee(\n      serializedHex,\n      txBroadcastFormat.length,\n      stxBalance\n    );\n    txBuilder.amount(recoverableAmount);\n    txBuilder.fee({ fee: fee });\n    return await txBuilder.build();\n  }\n\n  /**\n   * Method that uses appropriate builder and builds transaction depending on token or native coin\n   * @param {String[]} pubs - public keys\n   * @param {AddressDetails} rootAddressDetails - sender address detail\n   * @param {AddressDetails} destinationAddressDetails - receiver address detail\n   * @param {Number} nonce - wallet nonce\n   * @param {String} balance - wallet balance\n   * @param {String | undefined} contractAddress - token contract address\n   * @param {String | undefined} contractName - token contract name\n   * @returns {Promise<BaseTransaction>} built transaction\n   */\n  protected async getNativeOrTokenTransaction({\n    pubs,\n    rootAddressDetails,\n    destinationAddressDetails,\n    nonce,\n    stxBalance,\n    contractAddressInput,\n    contractName,\n  }: {\n    pubs: string[];\n    rootAddressDetails: AddressDetails;\n    destinationAddressDetails: AddressDetails;\n    nonce: number;\n    stxBalance: string;\n    contractAddressInput?: string;\n    contractName?: string;\n  }): Promise<{ tx: BaseTransaction; builder: TransferBuilder | FungibleTokenTransferBuilder }> {\n    const builder = this.getTokenOrNativeTransferBuilder(contractAddressInput, contractName);\n    const contractAddress = contractAddressInput?.toUpperCase();\n    const isToken = !!contractAddress && !!contractName;\n    let finalTx: BaseTransaction;\n    if (isToken) {\n      finalTx = await this.buildTokenTransferTransaction({\n        builder: builder as FungibleTokenTransferBuilder,\n        contractAddress,\n        contractName,\n        pubs,\n        nonce,\n        rootAddressDetails,\n        destinationAddressDetails,\n        stxBalance,\n      });\n    } else {\n      finalTx = await this.buildNativeTransferTransaction({\n        builder: builder as TransferBuilder,\n        pubs,\n        nonce,\n        destinationAddressDetails,\n        stxBalance,\n      });\n    }\n    return {\n      tx: finalTx,\n      builder: builder,\n    };\n  }\n\n  /**\n   * Method to recover native stx or sip10 tokens from bitgo hot & cold wallets\n   * @param {String} params.backupKey - encrypted wallet backup key (public or private)\n   * @param {String} params.userKey - encrypted wallet user key (public or private)\n   * @param {String} params.rootAddress - wallet root address\n   * @param {String} params.recoveryDestination - receive address\n   * @param {String} params.bitgoKey - encrypted bitgo public key\n   * @param {String} params.walletPassphrase - wallet password\n   * @param {String} params.contractId - contract id of the token (mandatory for token recovery)\n   * @returns {Promise<RecoveryInfo|RecoveryTransaction>} RecoveryTransaction.txHex - hex of serialized transaction (signed or unsigned)\n   */\n  async recover(params: RecoveryOptions): Promise<RecoveryInfo | RecoveryTransaction> {\n    if (!this.isValidAddress(params.rootAddress)) {\n      throw new Error('invalid root address!');\n    }\n    if (!this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid destination address!');\n    }\n    let contractAddress: string | undefined;\n    let contractName: string | undefined;\n    if (params.contractId) {\n      [contractAddress, contractName] = params.contractId.split('.');\n      if ((contractAddress && !contractName) || (contractName && !contractAddress)) {\n        throw new Error('invalid contract id, please provide it in the form (contractAddress.contractName)');\n      }\n    }\n    const isUnsignedSweep = getIsUnsignedSweep(params);\n    const keys = getBip32Keys(this.bitgo, params, { requireBitGoXpub: true });\n    const rootAddressDetails = getAddressDetails(params.rootAddress);\n    const [accountBalanceData, accountNonceData] = await Promise.all([\n      this.getNativeStxBalanceFromNode({ address: rootAddressDetails.address }),\n      this.getAccountNonceFromNode({ address: rootAddressDetails.address }),\n    ]);\n    const balance = Number(accountBalanceData.balance);\n    if (!balance || isNaN(balance)) {\n      throw new Error('could not find any balance to recover for ' + params.rootAddress);\n    }\n    const userPub = publicKeyFromBuffer(keys[0].publicKey);\n    const backupPub = publicKeyFromBuffer(keys[1].publicKey);\n    const bitgoPubKey = publicKeyFromBuffer(keys[2].publicKey);\n    const pubs = [publicKeyToString(userPub), publicKeyToString(backupPub), publicKeyToString(bitgoPubKey)];\n\n    const destinationAddressDetails = getAddressDetails(params.recoveryDestination);\n    const nonce =\n      typeof accountNonceData?.last_executed_tx_nonce === 'number' ? accountNonceData.last_executed_tx_nonce + 1 : 0;\n\n    const { tx, builder } = await this.getNativeOrTokenTransaction({\n      pubs,\n      rootAddressDetails,\n      destinationAddressDetails,\n      nonce,\n      stxBalance: accountBalanceData.balance,\n      contractAddressInput: contractAddress,\n      contractName: contractName,\n    });\n\n    if (isUnsignedSweep) {\n      return await this.formatForOfflineVault(tx);\n    }\n    // check the private key & sign\n    if (!keys[0].privateKey) {\n      throw new Error(`userKey is not a private key`);\n    }\n    const userKey = createStacksPrivateKey(keys[0].privateKey);\n    builder.sign({ key: privateKeyToString(userKey) });\n\n    const halfSignedTx = await builder.build();\n    const txHexHalfSigned = halfSignedTx.toBroadcastFormat();\n    const builder2 = this.getTokenOrNativeTransferBuilder(contractAddress, contractName);\n    builder2.from(txHexHalfSigned);\n\n    if (!keys[1].privateKey) {\n      throw new Error(`backupKey is not a private key`);\n    }\n    const backupKey = createStacksPrivateKey(keys[1].privateKey);\n    builder2.sign({ key: privateKeyToString(backupKey) });\n\n    const fullySignedTx = await builder2.build();\n    const fullySignedTxHex = fullySignedTx.toBroadcastFormat();\n\n    return {\n      txHex: fullySignedTxHex,\n    };\n  }\n\n  /** @inheritDoc */\n  auditDecryptedKey(params: AuditDecryptedKeyParams) {\n    throw new MethodNotImplementedError();\n  }\n}\n"]}