@bitgo-beta/sdk-coin-sui 3.0.3-beta.98 → 3.0.3-beta.981

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 (136) hide show
  1. package/CHANGELOG.md +817 -0
  2. package/dist/src/index.d.ts +1 -0
  3. package/dist/src/index.d.ts.map +1 -1
  4. package/dist/src/index.js +7 -2
  5. package/dist/src/lib/compareTransactionBlocks.js +2 -3
  6. package/dist/src/lib/constants.d.ts +9 -2
  7. package/dist/src/lib/constants.d.ts.map +1 -1
  8. package/dist/src/lib/constants.js +18 -2
  9. package/dist/src/lib/customTransaction.d.ts +57 -0
  10. package/dist/src/lib/customTransaction.d.ts.map +1 -0
  11. package/dist/src/lib/customTransaction.js +159 -0
  12. package/dist/src/lib/customTransactionBuilder.d.ts +46 -0
  13. package/dist/src/lib/customTransactionBuilder.d.ts.map +1 -0
  14. package/dist/src/lib/customTransactionBuilder.js +117 -0
  15. package/dist/src/lib/iface.d.ts +76 -10
  16. package/dist/src/lib/iface.d.ts.map +1 -1
  17. package/dist/src/lib/iface.js +40 -5
  18. package/dist/src/lib/index.d.ts +8 -0
  19. package/dist/src/lib/index.d.ts.map +1 -1
  20. package/dist/src/lib/index.js +40 -10
  21. package/dist/src/lib/keyPair.js +24 -10
  22. package/dist/src/lib/mystenlab/builder/Inputs.d.ts +9 -9
  23. package/dist/src/lib/mystenlab/builder/Inputs.d.ts.map +1 -1
  24. package/dist/src/lib/mystenlab/builder/Inputs.js +18 -19
  25. package/dist/src/lib/mystenlab/builder/TransactionBlock.d.ts +40 -354
  26. package/dist/src/lib/mystenlab/builder/TransactionBlock.d.ts.map +1 -1
  27. package/dist/src/lib/mystenlab/builder/TransactionBlock.js +22 -25
  28. package/dist/src/lib/mystenlab/builder/TransactionDataBlock.d.ts +74 -74
  29. package/dist/src/lib/mystenlab/builder/TransactionDataBlock.d.ts.map +1 -1
  30. package/dist/src/lib/mystenlab/builder/TransactionDataBlock.js +41 -44
  31. package/dist/src/lib/mystenlab/builder/Transactions.d.ts +133 -188
  32. package/dist/src/lib/mystenlab/builder/Transactions.d.ts.map +1 -1
  33. package/dist/src/lib/mystenlab/builder/Transactions.js +52 -53
  34. package/dist/src/lib/mystenlab/builder/bcs.d.ts +1 -1
  35. package/dist/src/lib/mystenlab/builder/bcs.d.ts.map +1 -1
  36. package/dist/src/lib/mystenlab/builder/bcs.js +2 -2
  37. package/dist/src/lib/mystenlab/builder/index.js +6 -2
  38. package/dist/src/lib/mystenlab/builder/serializer.js +6 -8
  39. package/dist/src/lib/mystenlab/builder/utils.d.ts +1 -1
  40. package/dist/src/lib/mystenlab/builder/utils.d.ts.map +1 -1
  41. package/dist/src/lib/mystenlab/builder/utils.js +4 -4
  42. package/dist/src/lib/mystenlab/cryptography/hash.js +3 -4
  43. package/dist/src/lib/mystenlab/framework/framework.d.ts +6 -6
  44. package/dist/src/lib/mystenlab/framework/framework.d.ts.map +1 -1
  45. package/dist/src/lib/mystenlab/framework/framework.js +22 -25
  46. package/dist/src/lib/mystenlab/framework/index.js +6 -2
  47. package/dist/src/lib/mystenlab/framework/sui-system-state.js +2 -2
  48. package/dist/src/lib/mystenlab/txn-data-serializers/type-tag-serializer.js +2 -2
  49. package/dist/src/lib/mystenlab/types/coin.d.ts +10 -10
  50. package/dist/src/lib/mystenlab/types/coin.d.ts.map +1 -1
  51. package/dist/src/lib/mystenlab/types/coin.js +19 -19
  52. package/dist/src/lib/mystenlab/types/common.d.ts +8 -8
  53. package/dist/src/lib/mystenlab/types/common.d.ts.map +1 -1
  54. package/dist/src/lib/mystenlab/types/common.js +22 -22
  55. package/dist/src/lib/mystenlab/types/events.d.ts +14 -14
  56. package/dist/src/lib/mystenlab/types/events.d.ts.map +1 -1
  57. package/dist/src/lib/mystenlab/types/events.js +17 -17
  58. package/dist/src/lib/mystenlab/types/index.js +6 -2
  59. package/dist/src/lib/mystenlab/types/normalized.d.ts +21 -21
  60. package/dist/src/lib/mystenlab/types/normalized.d.ts.map +1 -1
  61. package/dist/src/lib/mystenlab/types/normalized.js +41 -41
  62. package/dist/src/lib/mystenlab/types/objects.d.ts +51 -51
  63. package/dist/src/lib/mystenlab/types/objects.d.ts.map +1 -1
  64. package/dist/src/lib/mystenlab/types/objects.js +96 -106
  65. package/dist/src/lib/mystenlab/types/option.d.ts +1 -1
  66. package/dist/src/lib/mystenlab/types/option.d.ts.map +1 -1
  67. package/dist/src/lib/mystenlab/types/option.js +2 -3
  68. package/dist/src/lib/mystenlab/types/sui-bcs.d.ts +8 -8
  69. package/dist/src/lib/mystenlab/types/sui-bcs.d.ts.map +1 -1
  70. package/dist/src/lib/mystenlab/types/sui-bcs.js +5 -5
  71. package/dist/src/lib/mystenlab/types/transactions.d.ts +625 -625
  72. package/dist/src/lib/mystenlab/types/transactions.d.ts.map +1 -1
  73. package/dist/src/lib/mystenlab/types/transactions.js +178 -194
  74. package/dist/src/lib/mystenlab/types/validator.d.ts +9 -9
  75. package/dist/src/lib/mystenlab/types/validator.d.ts.map +1 -1
  76. package/dist/src/lib/mystenlab/types/validator.js +124 -124
  77. package/dist/src/lib/resources/walrusConfig.d.ts +22 -0
  78. package/dist/src/lib/resources/walrusConfig.d.ts.map +1 -0
  79. package/dist/src/lib/resources/walrusConfig.js +37 -0
  80. package/dist/src/lib/rpcClient.d.ts +5 -0
  81. package/dist/src/lib/rpcClient.d.ts.map +1 -0
  82. package/dist/src/lib/rpcClient.js +74 -0
  83. package/dist/src/lib/stakingBuilder.d.ts.map +1 -1
  84. package/dist/src/lib/stakingBuilder.js +23 -7
  85. package/dist/src/lib/stakingTransaction.d.ts +1 -1
  86. package/dist/src/lib/stakingTransaction.d.ts.map +1 -1
  87. package/dist/src/lib/stakingTransaction.js +26 -15
  88. package/dist/src/lib/tokenTransferBuilder.d.ts +38 -0
  89. package/dist/src/lib/tokenTransferBuilder.d.ts.map +1 -0
  90. package/dist/src/lib/tokenTransferBuilder.js +132 -0
  91. package/dist/src/lib/tokenTransferTransaction.d.ts +57 -0
  92. package/dist/src/lib/tokenTransferTransaction.d.ts.map +1 -0
  93. package/dist/src/lib/tokenTransferTransaction.js +250 -0
  94. package/dist/src/lib/transaction.d.ts +12 -4
  95. package/dist/src/lib/transaction.d.ts.map +1 -1
  96. package/dist/src/lib/transaction.js +91 -18
  97. package/dist/src/lib/transactionBuilder.d.ts +2 -3
  98. package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
  99. package/dist/src/lib/transactionBuilder.js +4 -4
  100. package/dist/src/lib/transactionBuilderFactory.d.ts +14 -2
  101. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  102. package/dist/src/lib/transactionBuilderFactory.js +42 -1
  103. package/dist/src/lib/transferBuilder.d.ts.map +1 -1
  104. package/dist/src/lib/transferBuilder.js +21 -5
  105. package/dist/src/lib/transferTransaction.d.ts +1 -1
  106. package/dist/src/lib/transferTransaction.d.ts.map +1 -1
  107. package/dist/src/lib/transferTransaction.js +31 -8
  108. package/dist/src/lib/unstakingBuilder.js +6 -6
  109. package/dist/src/lib/unstakingTransaction.d.ts +1 -1
  110. package/dist/src/lib/unstakingTransaction.d.ts.map +1 -1
  111. package/dist/src/lib/unstakingTransaction.js +31 -17
  112. package/dist/src/lib/utils.d.ts +16 -4
  113. package/dist/src/lib/utils.d.ts.map +1 -1
  114. package/dist/src/lib/utils.js +269 -29
  115. package/dist/src/lib/walrusStakingBuilder.d.ts +66 -0
  116. package/dist/src/lib/walrusStakingBuilder.d.ts.map +1 -0
  117. package/dist/src/lib/walrusStakingBuilder.js +200 -0
  118. package/dist/src/lib/walrusStakingTransaction.d.ts +52 -0
  119. package/dist/src/lib/walrusStakingTransaction.d.ts.map +1 -0
  120. package/dist/src/lib/walrusStakingTransaction.js +269 -0
  121. package/dist/src/lib/walrusWithdrawStakeBuilder.d.ts +36 -0
  122. package/dist/src/lib/walrusWithdrawStakeBuilder.d.ts.map +1 -0
  123. package/dist/src/lib/walrusWithdrawStakeBuilder.js +173 -0
  124. package/dist/src/lib/walrusWithdrawStakeTransaction.d.ts +21 -0
  125. package/dist/src/lib/walrusWithdrawStakeTransaction.d.ts.map +1 -0
  126. package/dist/src/lib/walrusWithdrawStakeTransaction.js +190 -0
  127. package/dist/src/register.d.ts.map +1 -1
  128. package/dist/src/register.js +5 -1
  129. package/dist/src/sui.d.ts +46 -8
  130. package/dist/src/sui.d.ts.map +1 -1
  131. package/dist/src/sui.js +479 -32
  132. package/dist/src/suiToken.d.ts +22 -0
  133. package/dist/src/suiToken.d.ts.map +1 -0
  134. package/dist/src/suiToken.js +61 -0
  135. package/dist/src/tsui.js +1 -1
  136. package/package.json +10 -8
package/dist/src/sui.js CHANGED
@@ -1,7 +1,11 @@
1
1
  "use strict";
2
2
  var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
3
  if (k2 === undefined) k2 = k;
4
- Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
5
9
  }) : (function(o, m, k, k2) {
6
10
  if (k2 === undefined) k2 = k;
7
11
  o[k2] = m[k];
@@ -11,13 +15,23 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
11
15
  }) : function(o, v) {
12
16
  o["default"] = v;
13
17
  });
14
- var __importStar = (this && this.__importStar) || function (mod) {
15
- if (mod && mod.__esModule) return mod;
16
- var result = {};
17
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
18
- __setModuleDefault(result, mod);
19
- return result;
20
- };
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
21
35
  var __importDefault = (this && this.__importDefault) || function (mod) {
22
36
  return (mod && mod.__esModule) ? mod : { "default": mod };
23
37
  };
@@ -29,7 +43,9 @@ const bignumber_js_1 = __importDefault(require("bignumber.js"));
29
43
  const lib_1 = require("./lib");
30
44
  const utils_1 = __importDefault(require("./lib/utils"));
31
45
  const _ = __importStar(require("lodash"));
32
- const transferTransaction_1 = require("./lib/transferTransaction");
46
+ const iface_1 = require("./lib/iface");
47
+ const constants_1 = require("./lib/constants");
48
+ const sdk_lib_mpc_1 = require("@bitgo-beta/sdk-lib-mpc");
33
49
  class Sui extends sdk_core_1.BaseCoin {
34
50
  constructor(bitgo, staticsCoin) {
35
51
  super(bitgo);
@@ -56,10 +72,17 @@ class Sui extends sdk_core_1.BaseCoin {
56
72
  getFullName() {
57
73
  return 'Sui';
58
74
  }
75
+ getNetwork() {
76
+ return this._staticsCoin.network;
77
+ }
59
78
  /** @inheritDoc */
60
79
  supportsTss() {
61
80
  return true;
62
81
  }
82
+ /** inherited doc */
83
+ getDefaultMultisigType() {
84
+ return sdk_core_1.multisigTypes.tss;
85
+ }
63
86
  getMPCAlgorithm() {
64
87
  return 'eddsa';
65
88
  }
@@ -67,11 +90,10 @@ class Sui extends sdk_core_1.BaseCoin {
67
90
  return true;
68
91
  }
69
92
  async verifyTransaction(params) {
70
- var _a;
71
93
  let totalAmount = new bignumber_js_1.default(0);
72
94
  const coinConfig = statics_1.coins.get(this.getChain());
73
95
  const { txPrebuild: txPrebuild, txParams: txParams } = params;
74
- const transaction = new transferTransaction_1.TransferTransaction(coinConfig);
96
+ const transaction = new lib_1.TransferTransaction(coinConfig);
75
97
  const rawTx = txPrebuild.txHex;
76
98
  if (!rawTx) {
77
99
  throw new Error('missing required tx prebuild property txHex');
@@ -79,7 +101,7 @@ class Sui extends sdk_core_1.BaseCoin {
79
101
  transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));
80
102
  const explainedTx = transaction.explainTransaction();
81
103
  if (txParams.recipients && txParams.recipients.length > 0) {
82
- const filteredRecipients = (_a = txParams.recipients) === null || _a === void 0 ? void 0 : _a.map((recipient) => {
104
+ const filteredRecipients = txParams.recipients?.map((recipient) => {
83
105
  const filteredRecipient = _.pick(recipient, ['address', 'amount']);
84
106
  filteredRecipient.amount = new bignumber_js_1.default(filteredRecipient.amount).toFixed();
85
107
  return filteredRecipient;
@@ -102,23 +124,10 @@ class Sui extends sdk_core_1.BaseCoin {
102
124
  return true;
103
125
  }
104
126
  async isWalletAddress(params) {
105
- const { keychains, address: newAddress, index } = params;
127
+ const { address: newAddress } = params;
106
128
  if (!this.isValidAddress(newAddress)) {
107
129
  throw new sdk_core_1.InvalidAddressError(`invalid address: ${newAddress}`);
108
130
  }
109
- if (!keychains) {
110
- throw new Error('missing required param keychains');
111
- }
112
- for (const keychain of keychains) {
113
- const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
114
- const commonKeychain = keychain.commonKeychain;
115
- const derivationPath = 'm/' + index;
116
- const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);
117
- const expectedAddress = this.getAddressFromPublicKey(derivedPublicKey);
118
- if (newAddress !== expectedAddress) {
119
- return false;
120
- }
121
- }
122
131
  return true;
123
132
  }
124
133
  async parseTransaction(params) {
@@ -126,20 +135,24 @@ class Sui extends sdk_core_1.BaseCoin {
126
135
  if (!transactionExplanation) {
127
136
  throw new Error('Invalid transaction');
128
137
  }
138
+ let fee = new bignumber_js_1.default(0);
129
139
  const suiTransaction = transactionExplanation;
130
140
  if (suiTransaction.outputs.length <= 0) {
131
141
  return {
132
142
  inputs: [],
133
143
  outputs: [],
144
+ fee,
134
145
  };
135
146
  }
136
147
  const senderAddress = suiTransaction.outputs[0].address;
137
- const feeAmount = new bignumber_js_1.default(suiTransaction.fee.fee === '' ? '0' : suiTransaction.fee.fee);
148
+ if (suiTransaction.fee.fee !== '') {
149
+ fee = new bignumber_js_1.default(suiTransaction.fee.fee);
150
+ }
138
151
  // assume 1 sender, who is also the fee payer
139
152
  const inputs = [
140
153
  {
141
154
  address: senderAddress,
142
- amount: new bignumber_js_1.default(suiTransaction.outputAmount).plus(feeAmount).toFixed(),
155
+ amount: new bignumber_js_1.default(suiTransaction.outputAmount).plus(fee).toFixed(),
143
156
  },
144
157
  ];
145
158
  const outputs = suiTransaction.outputs.map((output) => {
@@ -151,6 +164,7 @@ class Sui extends sdk_core_1.BaseCoin {
151
164
  return {
152
165
  inputs,
153
166
  outputs,
167
+ fee,
154
168
  };
155
169
  }
156
170
  generateKeyPair(seed) {
@@ -164,16 +178,16 @@ class Sui extends sdk_core_1.BaseCoin {
164
178
  prv: keys.prv,
165
179
  };
166
180
  }
167
- isValidPub(pub) {
181
+ isValidPub(_) {
168
182
  throw new Error('Method not implemented.');
169
183
  }
170
- isValidPrv(prv) {
184
+ isValidPrv(_) {
171
185
  throw new Error('Method not implemented.');
172
186
  }
173
187
  isValidAddress(address) {
174
188
  return utils_1.default.isValidAddress(address);
175
189
  }
176
- signTransaction(params) {
190
+ signTransaction(_) {
177
191
  throw new Error('Method not implemented.');
178
192
  }
179
193
  /**
@@ -205,6 +219,439 @@ class Sui extends sdk_core_1.BaseCoin {
205
219
  const rebuiltTransaction = await factory.from(serializedTx).build();
206
220
  return rebuiltTransaction.signablePayload;
207
221
  }
222
+ getPublicNodeUrl() {
223
+ return sdk_core_1.Environments[this.bitgo.getEnv()].suiNodeUrl;
224
+ }
225
+ async getBalance(owner, coinType) {
226
+ const url = this.getPublicNodeUrl();
227
+ return await utils_1.default.getBalance(url, owner, coinType);
228
+ }
229
+ async getInputCoins(owner, coinType) {
230
+ const url = this.getPublicNodeUrl();
231
+ return await utils_1.default.getInputCoins(url, owner, coinType);
232
+ }
233
+ async getFeeEstimate(txHex) {
234
+ const url = this.getPublicNodeUrl();
235
+ return await utils_1.default.getFeeEstimate(url, txHex);
236
+ }
237
+ /**
238
+ * Builds funds recovery transaction(s) without BitGo
239
+ *
240
+ * @param {MPCRecoveryOptions} params parameters needed to construct and
241
+ * (maybe) sign the transaction
242
+ *
243
+ * @returns {MPCTx | MPCSweepTxs} array of the serialized transaction hex strings and indices
244
+ * of the addresses being swept
245
+ */
246
+ async recover(params) {
247
+ if (!params.bitgoKey) {
248
+ throw new Error('missing bitgoKey');
249
+ }
250
+ if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
251
+ throw new Error('invalid recoveryDestination');
252
+ }
253
+ const startIdx = utils_1.default.validateNonNegativeNumber(0, 'Invalid starting index to scan for addresses', params.startingScanIndex);
254
+ const numIterations = utils_1.default.validateNonNegativeNumber(20, 'Invalid scanning factor', params.scan);
255
+ const endIdx = startIdx + numIterations;
256
+ const bitgoKey = params.bitgoKey.replace(/\s/g, '');
257
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
258
+ for (let idx = startIdx; idx < endIdx; idx++) {
259
+ const derivationPath = (params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) : 'm') + `/${idx}`;
260
+ const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);
261
+ const senderAddress = this.getAddressFromPublicKey(derivedPublicKey);
262
+ let availableBalance = new bignumber_js_1.default(0);
263
+ try {
264
+ availableBalance = new bignumber_js_1.default(await this.getBalance(senderAddress));
265
+ }
266
+ catch (e) {
267
+ continue;
268
+ }
269
+ if (availableBalance.minus(constants_1.MAX_GAS_BUDGET).toNumber() <= 0) {
270
+ continue;
271
+ }
272
+ // check for possible token recovery, recover the token provide by user
273
+ if (params.tokenContractAddress) {
274
+ const token = utils_1.default.getSuiTokenFromAddress(params.tokenContractAddress, this.getNetwork());
275
+ if (!token) {
276
+ throw new Error(`Sui Token Package ID not supported.`);
277
+ }
278
+ const coinType = `${token.packageId}::${token.module}::${token.symbol}`;
279
+ try {
280
+ const availableTokenBalance = new bignumber_js_1.default(await this.getBalance(senderAddress, coinType));
281
+ if (availableTokenBalance.toNumber() <= 0) {
282
+ continue;
283
+ }
284
+ }
285
+ catch (e) {
286
+ continue;
287
+ }
288
+ return this.recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey);
289
+ }
290
+ let inputCoins = await this.getInputCoins(senderAddress);
291
+ inputCoins = inputCoins.sort((a, b) => {
292
+ return b.balance.minus(a.balance).toNumber();
293
+ });
294
+ if (inputCoins.length > constants_1.MAX_OBJECT_LIMIT) {
295
+ inputCoins = inputCoins.slice(0, constants_1.MAX_OBJECT_LIMIT);
296
+ }
297
+ let netAmount = inputCoins.reduce((acc, obj) => acc.plus(obj.balance), new bignumber_js_1.default(0));
298
+ netAmount = netAmount.minus(constants_1.MAX_GAS_BUDGET);
299
+ const recipients = [
300
+ {
301
+ address: params.recoveryDestination,
302
+ amount: netAmount.toString(),
303
+ },
304
+ ];
305
+ // first build the unsigned txn
306
+ const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
307
+ const txBuilder = factory
308
+ .getTransferBuilder()
309
+ .type(iface_1.SuiTransactionType.Transfer)
310
+ .sender(senderAddress)
311
+ .send(recipients)
312
+ .gasData({
313
+ owner: senderAddress,
314
+ price: constants_1.DEFAULT_GAS_PRICE,
315
+ budget: constants_1.MAX_GAS_BUDGET,
316
+ payment: inputCoins,
317
+ });
318
+ const tempTx = (await txBuilder.build());
319
+ const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());
320
+ const gasBudget = Math.trunc(feeEstimate.toNumber() * constants_1.DEFAULT_GAS_OVERHEAD);
321
+ netAmount = netAmount.plus(constants_1.MAX_GAS_BUDGET).minus(gasBudget);
322
+ recipients[0].amount = netAmount.toString();
323
+ txBuilder.send(recipients);
324
+ txBuilder.gasData({
325
+ owner: senderAddress,
326
+ price: constants_1.DEFAULT_GAS_PRICE,
327
+ budget: gasBudget,
328
+ payment: inputCoins,
329
+ });
330
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
331
+ if (isUnsignedSweep) {
332
+ return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath);
333
+ }
334
+ await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, false);
335
+ const tx = (await txBuilder.build());
336
+ return {
337
+ transactions: [
338
+ {
339
+ scanIndex: idx,
340
+ recoveryAmount: netAmount.toString(),
341
+ serializedTx: tx.toBroadcastFormat(),
342
+ signature: Buffer.from(tx.serializedSig).toString('base64'),
343
+ coin: this.getChain(),
344
+ },
345
+ ],
346
+ lastScanIndex: idx,
347
+ };
348
+ }
349
+ throw new Error(`Did not find an address with sufficient funds to recover. Please start the next scan at address index ${endIdx}. If it is token transaction, please keep sufficient Sui balance in the address for the transaction fee.`);
350
+ }
351
+ async recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey) {
352
+ const coinType = `${token.packageId}::${token.module}::${token.symbol}`;
353
+ let tokenObjects = await this.getInputCoins(senderAddress, coinType);
354
+ tokenObjects = tokenObjects.sort((a, b) => {
355
+ return b.balance.minus(a.balance).toNumber();
356
+ });
357
+ if (tokenObjects.length > constants_1.TOKEN_OBJECT_LIMIT) {
358
+ tokenObjects = tokenObjects.slice(0, constants_1.TOKEN_OBJECT_LIMIT);
359
+ }
360
+ const netAmount = tokenObjects.reduce((acc, obj) => acc.plus(obj.balance), new bignumber_js_1.default(0));
361
+ const recipients = [
362
+ {
363
+ address: params.recoveryDestination,
364
+ amount: netAmount.toString(),
365
+ },
366
+ ];
367
+ const gasAmount = new bignumber_js_1.default(constants_1.MAX_GAS_BUDGET);
368
+ let gasObjects = await this.getInputCoins(senderAddress);
369
+ gasObjects = utils_1.default.selectObjectsInDescOrderOfBalance(gasObjects, gasAmount);
370
+ if (gasObjects.length >= constants_1.MAX_GAS_OBJECTS) {
371
+ gasObjects = gasObjects.slice(0, constants_1.MAX_GAS_OBJECTS - 1);
372
+ }
373
+ // first build the unsigned txn
374
+ const factory = new lib_1.TransactionBuilderFactory(token);
375
+ const txBuilder = factory
376
+ .getTokenTransferBuilder()
377
+ .type(iface_1.SuiTransactionType.TokenTransfer)
378
+ .sender(senderAddress)
379
+ .send(recipients)
380
+ .inputObjects(tokenObjects)
381
+ .gasData({
382
+ owner: senderAddress,
383
+ price: constants_1.DEFAULT_GAS_PRICE,
384
+ budget: constants_1.MAX_GAS_BUDGET,
385
+ payment: gasObjects,
386
+ });
387
+ const tempTx = (await txBuilder.build());
388
+ const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());
389
+ const gasBudget = Math.trunc(feeEstimate.toNumber() * constants_1.DEFAULT_GAS_OVERHEAD);
390
+ txBuilder.gasData({
391
+ owner: senderAddress,
392
+ price: constants_1.DEFAULT_GAS_PRICE,
393
+ budget: gasBudget,
394
+ payment: gasObjects,
395
+ });
396
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
397
+ if (isUnsignedSweep) {
398
+ return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath, token);
399
+ }
400
+ await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, true);
401
+ const tx = (await txBuilder.build());
402
+ return {
403
+ transactions: [
404
+ {
405
+ scanIndex: idx,
406
+ recoveryAmount: netAmount.toString(),
407
+ serializedTx: tx.toBroadcastFormat(),
408
+ signature: Buffer.from(tx.serializedSig).toString('base64'),
409
+ coin: token.name,
410
+ },
411
+ ],
412
+ lastScanIndex: idx,
413
+ };
414
+ }
415
+ async buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, index, derivationPath, token) {
416
+ const isTokenTransaction = !!token;
417
+ const unsignedTransaction = isTokenTransaction
418
+ ? (await txBuilder.build())
419
+ : (await txBuilder.build());
420
+ const serializedTx = unsignedTransaction.toBroadcastFormat();
421
+ const serializedTxHex = Buffer.from(serializedTx, 'base64').toString('hex');
422
+ const parsedTx = await this.parseTransaction({ txHex: serializedTxHex });
423
+ const walletCoin = isTokenTransaction ? token.name : this.getChain();
424
+ const output = parsedTx.outputs[0];
425
+ const inputs = [
426
+ {
427
+ address: senderAddress,
428
+ valueString: output.amount,
429
+ value: new bignumber_js_1.default(output.amount),
430
+ },
431
+ ];
432
+ const outputs = [
433
+ {
434
+ address: output.address,
435
+ valueString: output.amount,
436
+ coinName: walletCoin,
437
+ },
438
+ ];
439
+ const spendAmount = output.amount;
440
+ const completedParsedTx = {
441
+ inputs: inputs,
442
+ outputs: outputs,
443
+ spendAmount: spendAmount,
444
+ type: isTokenTransaction ? iface_1.SuiTransactionType.TokenTransfer : iface_1.SuiTransactionType.Transfer,
445
+ };
446
+ const fee = parsedTx.fee;
447
+ const feeInfo = { fee: fee.toNumber(), feeString: fee.toString() };
448
+ const coinSpecific = { commonKeychain: bitgoKey };
449
+ const transaction = {
450
+ serializedTx: serializedTxHex,
451
+ scanIndex: index,
452
+ coin: walletCoin,
453
+ signableHex: unsignedTransaction.signablePayload.toString('hex'),
454
+ derivationPath,
455
+ parsedTx: completedParsedTx,
456
+ feeInfo: feeInfo,
457
+ coinSpecific: coinSpecific,
458
+ };
459
+ const unsignedTx = { unsignedTx: transaction, signatureShares: [] };
460
+ const transactions = [unsignedTx];
461
+ const txRequest = {
462
+ transactions: transactions,
463
+ walletCoin: walletCoin,
464
+ };
465
+ return { txRequests: [txRequest] };
466
+ }
467
+ async signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, isTokenTransaction) {
468
+ // TODO(BG-51092): This looks like a common part which can be extracted out too
469
+ const unsignedTx = isTokenTransaction
470
+ ? (await txBuilder.build())
471
+ : (await txBuilder.build());
472
+ if (!params.userKey) {
473
+ throw new Error('missing userKey');
474
+ }
475
+ if (!params.backupKey) {
476
+ throw new Error('missing backupKey');
477
+ }
478
+ if (!params.walletPassphrase) {
479
+ throw new Error('missing wallet passphrase');
480
+ }
481
+ // Clean up whitespace from entered values
482
+ const userKey = params.userKey.replace(/\s/g, '');
483
+ const backupKey = params.backupKey.replace(/\s/g, '');
484
+ // Decrypt private keys from KeyCard values
485
+ let userPrv;
486
+ try {
487
+ userPrv = this.bitgo.decrypt({
488
+ input: userKey,
489
+ password: params.walletPassphrase,
490
+ });
491
+ }
492
+ catch (e) {
493
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
494
+ }
495
+ /** TODO BG-52419 Implement Codec for parsing */
496
+ const userSigningMaterial = JSON.parse(userPrv);
497
+ let backupPrv;
498
+ try {
499
+ backupPrv = this.bitgo.decrypt({
500
+ input: backupKey,
501
+ password: params.walletPassphrase,
502
+ });
503
+ }
504
+ catch (e) {
505
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
506
+ }
507
+ const backupSigningMaterial = JSON.parse(backupPrv);
508
+ /* ********************** END ***********************************/
509
+ // add signature
510
+ const signatureHex = await sdk_core_1.EDDSAMethods.getTSSSignature(userSigningMaterial, backupSigningMaterial, derivationPath, unsignedTx);
511
+ txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);
512
+ }
513
+ async broadcastTransaction({ transactions, }) {
514
+ const txIds = [];
515
+ const url = this.getPublicNodeUrl();
516
+ let digest = '';
517
+ if (!!transactions) {
518
+ for (const txn of transactions) {
519
+ try {
520
+ digest = await utils_1.default.executeTransactionBlock(url, txn.serializedTx, [txn.signature]);
521
+ }
522
+ catch (e) {
523
+ throw new Error(`Failed to broadcast transaction, error: ${e.message}`);
524
+ }
525
+ txIds.push(digest);
526
+ }
527
+ }
528
+ return { txIds };
529
+ }
530
+ /** inherited doc */
531
+ async createBroadcastableSweepTransaction(params) {
532
+ const req = params.signatureShares;
533
+ const broadcastableTransactions = [];
534
+ let lastScanIndex = 0;
535
+ for (let i = 0; i < req.length; i++) {
536
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
537
+ const transaction = req[i].txRequest.transactions[0].unsignedTx;
538
+ if (!req[i].ovc || !req[i].ovc[0].eddsaSignature) {
539
+ throw new Error('Missing signature(s)');
540
+ }
541
+ const signature = req[i].ovc[0].eddsaSignature;
542
+ if (!transaction.signableHex) {
543
+ throw new Error('Missing signable hex');
544
+ }
545
+ const messageBuffer = Buffer.from(transaction.signableHex, 'hex');
546
+ const result = MPC.verify(messageBuffer, signature);
547
+ if (!result) {
548
+ throw new Error('Invalid signature');
549
+ }
550
+ const signatureHex = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);
551
+ const serializedTxBase64 = Buffer.from(transaction.serializedTx, 'hex').toString('base64');
552
+ const txBuilder = this.getBuilder().from(serializedTxBase64);
553
+ if (!transaction.coinSpecific?.commonKeychain) {
554
+ throw new Error('Missing common keychain');
555
+ }
556
+ const commonKeychain = transaction.coinSpecific.commonKeychain;
557
+ if (!transaction.derivationPath) {
558
+ throw new Error('Missing derivation path');
559
+ }
560
+ const derivationPath = transaction.derivationPath;
561
+ const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);
562
+ // add combined signature from ovc
563
+ txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);
564
+ const signedTransaction = (await txBuilder.build());
565
+ const serializedTx = signedTransaction.toBroadcastFormat();
566
+ const outputAmount = signedTransaction.explainTransaction().outputAmount;
567
+ broadcastableTransactions.push({
568
+ serializedTx: serializedTx,
569
+ scanIndex: transaction.scanIndex,
570
+ signature: Buffer.from(signedTransaction.serializedSig).toString('base64'),
571
+ recoveryAmount: outputAmount.toString(),
572
+ });
573
+ if (i === req.length - 1 && transaction.coinSpecific.lastScanIndex) {
574
+ lastScanIndex = transaction.coinSpecific.lastScanIndex;
575
+ }
576
+ }
577
+ return { transactions: broadcastableTransactions, lastScanIndex };
578
+ }
579
+ /**
580
+ * Builds native SUI recoveries of receive addresses in batch without BitGo.
581
+ * Funds will be recovered to base address first. You need to initiate another sweep txn after that.
582
+ *
583
+ * @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.
584
+ * @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).
585
+ * @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).
586
+ */
587
+ async recoverConsolidations(params) {
588
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
589
+ const startIdx = utils_1.default.validateNonNegativeNumber(1, 'Invalid starting index to scan for addresses', params.startingScanIndex);
590
+ const endIdx = utils_1.default.validateNonNegativeNumber(startIdx + constants_1.DEFAULT_SCAN_FACTOR, 'Invalid ending index to scan for addresses', params.endingScanIndex);
591
+ if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * constants_1.DEFAULT_SCAN_FACTOR) {
592
+ throw new Error(`Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`);
593
+ }
594
+ const bitgoKey = params.bitgoKey.replace(/\s/g, '');
595
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
596
+ const derivationPath = (params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) : 'm') + '/0';
597
+ const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);
598
+ const baseAddress = this.getAddressFromPublicKey(derivedPublicKey);
599
+ const consolidationTransactions = [];
600
+ let lastScanIndex = startIdx;
601
+ for (let idx = startIdx; idx < endIdx; idx++) {
602
+ const recoverParams = {
603
+ userKey: params.userKey,
604
+ backupKey: params.backupKey,
605
+ bitgoKey: params.bitgoKey,
606
+ walletPassphrase: params.walletPassphrase,
607
+ seed: params.seed,
608
+ tokenContractAddress: params.tokenContractAddress,
609
+ recoveryDestination: baseAddress,
610
+ startingScanIndex: idx,
611
+ scan: 1,
612
+ };
613
+ let recoveryTransaction;
614
+ try {
615
+ recoveryTransaction = await this.recover(recoverParams);
616
+ }
617
+ catch (e) {
618
+ if (e.message.startsWith('Did not find an address with sufficient funds to recover.')) {
619
+ lastScanIndex = idx;
620
+ continue;
621
+ }
622
+ throw e;
623
+ }
624
+ if (isUnsignedSweep) {
625
+ consolidationTransactions.push(recoveryTransaction.txRequests[0]);
626
+ }
627
+ else {
628
+ consolidationTransactions.push(recoveryTransaction.transactions[0]);
629
+ }
630
+ lastScanIndex = idx;
631
+ }
632
+ if (consolidationTransactions.length === 0) {
633
+ throw new Error(`Did not find an address with sufficient funds to recover. Please start the next scan at address index ${lastScanIndex + 1}.`);
634
+ }
635
+ if (isUnsignedSweep) {
636
+ // lastScanIndex will be used to inform user the last address index scanned for available funds (so they can
637
+ // appropriately adjust the scan range on the next iteration of consolidation recoveries). In the case of unsigned
638
+ // sweep consolidations, this lastScanIndex will be provided in the coinSpecific of the last txn made.
639
+ consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific.lastScanIndex = lastScanIndex;
640
+ return { txRequests: consolidationTransactions };
641
+ }
642
+ return { transactions: consolidationTransactions, lastScanIndex };
643
+ }
644
+ /** inherited doc */
645
+ setCoinSpecificFieldsInIntent(intent, params) {
646
+ intent.unspents = params.unspents;
647
+ }
648
+ /** inherited doc */
649
+ auditDecryptedKey({ publicKey, prv, multiSigType }) {
650
+ if (multiSigType !== 'tss') {
651
+ throw new Error('Unsupported multiSigType');
652
+ }
653
+ (0, sdk_lib_mpc_1.auditEddsaPrivateKey)(prv, publicKey ?? '');
654
+ }
208
655
  }
209
656
  exports.Sui = Sui;
210
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sui.js","sourceRoot":"","sources":["../../src/sui.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  EDDSAMethods,\n  InvalidAddressError,\n  KeyPair,\n  MPCAlgorithm,\n  ParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions,\n  TransactionExplanation,\n  TssVerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, coins } from '@bitgo-beta/statics';\nimport BigNumber from 'bignumber.js';\nimport { TransactionBuilderFactory, KeyPair as SuiKeyPair } from './lib';\nimport utils from './lib/utils';\nimport * as _ from 'lodash';\nimport { TransferTransaction } from './lib/transferTransaction';\n\nexport interface ExplainTransactionOptions {\n  txHex: string;\n}\n\nexport interface SuiParseTransactionOptions extends BaseParseTransactionOptions {\n  txHex: string;\n}\n\ninterface TransactionOutput {\n  address: string;\n  amount: string;\n}\n\ntype TransactionInput = TransactionOutput;\n\nexport interface SuiParsedTransaction extends ParsedTransaction {\n  // total assets being moved, including fees\n  inputs: TransactionInput[];\n\n  // where assets are moved to\n  outputs: TransactionOutput[];\n}\n\nexport type SuiTransactionExplanation = TransactionExplanation;\n\nexport class Sui extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected 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 Sui(bitgo, staticsCoin);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e9;\n  }\n\n  public getChain(): string {\n    return 'sui';\n  }\n\n  public getFamily(): string {\n    return 'sui';\n  }\n\n  public getFullName(): string {\n    return 'Sui';\n  }\n\n  /** @inheritDoc */\n  supportsTss(): boolean {\n    return true;\n  }\n\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const coinConfig = coins.get(this.getChain());\n    const { txPrebuild: txPrebuild, txParams: txParams } = params;\n    const transaction = new TransferTransaction(coinConfig);\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n\n    transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => {\n        const filteredRecipient = _.pick(recipient, ['address', 'amount']);\n        filteredRecipient.amount = new BigNumber(filteredRecipient.amount).toFixed();\n        return filteredRecipient;\n      });\n      const filteredOutputs = explainedTx.outputs.map((output) => {\n        const filteredOutput = _.pick(output, ['address', 'amount']);\n        filteredOutput.amount = new BigNumber(filteredOutput.amount).toFixed();\n        return filteredOutput;\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      for (const recipients of txParams.recipients) {\n        totalAmount = totalAmount.plus(recipients.amount);\n      }\n      if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n        throw new Error('Tx total amount does not match with expected total amount field');\n      }\n    }\n    return true;\n  }\n\n  async isWalletAddress(params: TssVerifyAddressOptions): Promise<boolean> {\n    const { keychains, address: newAddress, index } = params;\n\n    if (!this.isValidAddress(newAddress)) {\n      throw new InvalidAddressError(`invalid address: ${newAddress}`);\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    for (const keychain of keychains) {\n      const MPC = await EDDSAMethods.getInitializedMpcInstance();\n      const commonKeychain = keychain.commonKeychain as string;\n\n      const derivationPath = 'm/' + index;\n      const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);\n      const expectedAddress = this.getAddressFromPublicKey(derivedPublicKey);\n\n      if (newAddress !== expectedAddress) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  async parseTransaction(params: SuiParseTransactionOptions): Promise<SuiParsedTransaction> {\n    const transactionExplanation = await this.explainTransaction({ txHex: params.txHex });\n\n    if (!transactionExplanation) {\n      throw new Error('Invalid transaction');\n    }\n\n    const suiTransaction = transactionExplanation as SuiTransactionExplanation;\n    if (suiTransaction.outputs.length <= 0) {\n      return {\n        inputs: [],\n        outputs: [],\n      };\n    }\n\n    const senderAddress = suiTransaction.outputs[0].address;\n    const feeAmount = new BigNumber(suiTransaction.fee.fee === '' ? '0' : suiTransaction.fee.fee);\n\n    // assume 1 sender, who is also the fee payer\n    const inputs = [\n      {\n        address: senderAddress,\n        amount: new BigNumber(suiTransaction.outputAmount).plus(feeAmount).toFixed(),\n      },\n    ];\n\n    const outputs: TransactionOutput[] = suiTransaction.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: new BigNumber(output.amount).toFixed(),\n      };\n    });\n\n    return {\n      inputs,\n      outputs,\n    };\n  }\n\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new SuiKeyPair({ seed }) : new SuiKeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return {\n      pub: keys.pub,\n      prv: keys.prv,\n    };\n  }\n\n  isValidPub(pub: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidPrv(prv: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Explain a Sui transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<SuiTransactionExplanation> {\n    const factory = this.getBuilder();\n    let rebuiltTransaction: BaseTransaction;\n\n    try {\n      const transactionBuilder = factory.from(Buffer.from(params.txHex, 'hex').toString('base64'));\n      rebuiltTransaction = await transactionBuilder.build();\n    } catch {\n      throw new Error('Invalid transaction');\n    }\n\n    return rebuiltTransaction.explainTransaction();\n  }\n\n  private getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  private getAddressFromPublicKey(derivedPublicKey: string) {\n    // TODO(BG-59016) replace with account lib implementation\n    return utils.getAddressFromPublicKey(derivedPublicKey);\n  }\n\n  /** @inheritDoc */\n  async getSignablePayload(serializedTx: string): Promise<Buffer> {\n    const factory = this.getBuilder();\n    const rebuiltTransaction = await factory.from(serializedTx).build();\n    return rebuiltTransaction.signablePayload;\n  }\n}\n"]}
657
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sui.js","sourceRoot":"","sources":["../../src/sui.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseBroadcastTransactionOptions,\n  BaseBroadcastTransactionResult,\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  EDDSAMethods,\n  EDDSAMethodTypes,\n  Environments,\n  InvalidAddressError,\n  KeyPair,\n  MPCAlgorithm,\n  MPCRecoveryOptions,\n  MPCConsolidationRecoveryOptions,\n  MPCSweepRecoveryOptions,\n  MPCSweepTxs,\n  MPCTx,\n  MPCTxs,\n  MPCUnsignedTx,\n  ParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  RecoveryTxRequest,\n  SignedTransaction,\n  SignTransactionOptions,\n  TransactionExplanation,\n  TssVerifyAddressOptions,\n  VerifyTransactionOptions,\n  PopulatedIntent,\n  PrebuildTransactionWithIntentOptions,\n  MultisigType,\n  multisigTypes,\n  AuditDecryptedKeyParams,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, BaseNetwork, coins, SuiCoin } from '@bitgo-beta/statics';\nimport BigNumber from 'bignumber.js';\nimport {\n  KeyPair as SuiKeyPair,\n  TokenTransferTransaction,\n  TransactionBuilder,\n  TransactionBuilderFactory,\n  TransferTransaction,\n} from './lib';\nimport utils from './lib/utils';\nimport * as _ from 'lodash';\nimport { SuiObjectInfo, SuiTransactionType } from './lib/iface';\nimport {\n  DEFAULT_GAS_OVERHEAD,\n  DEFAULT_GAS_PRICE,\n  DEFAULT_SCAN_FACTOR,\n  MAX_GAS_BUDGET,\n  MAX_GAS_OBJECTS,\n  MAX_OBJECT_LIMIT,\n  TOKEN_OBJECT_LIMIT,\n} from './lib/constants';\nimport { auditEddsaPrivateKey, getDerivationPath } from '@bitgo-beta/sdk-lib-mpc';\n\nexport interface ExplainTransactionOptions {\n  txHex: string;\n}\n\nexport interface SuiParseTransactionOptions extends BaseParseTransactionOptions {\n  txHex: string;\n}\n\ninterface TransactionOutput {\n  address: string;\n  amount: string;\n}\n\ntype TransactionInput = TransactionOutput;\n\nexport interface SuiParsedTransaction extends ParsedTransaction {\n  // total assets being moved, including fees\n  inputs: TransactionInput[];\n\n  // where assets are moved to\n  outputs: TransactionOutput[];\n\n  fee: BigNumber;\n}\n\nexport type SuiTransactionExplanation = TransactionExplanation;\n\nexport class Sui extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected 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 Sui(bitgo, staticsCoin);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e9;\n  }\n\n  public getChain(): string {\n    return 'sui';\n  }\n\n  public getFamily(): string {\n    return 'sui';\n  }\n\n  public getFullName(): string {\n    return 'Sui';\n  }\n\n  getNetwork(): BaseNetwork {\n    return this._staticsCoin.network;\n  }\n\n  /** @inheritDoc */\n  supportsTss(): boolean {\n    return true;\n  }\n\n  /** inherited doc */\n  getDefaultMultisigType(): MultisigType {\n    return multisigTypes.tss;\n  }\n\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const coinConfig = coins.get(this.getChain());\n    const { txPrebuild: txPrebuild, txParams: txParams } = params;\n    const transaction = new TransferTransaction(coinConfig);\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n\n    transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => {\n        const filteredRecipient = _.pick(recipient, ['address', 'amount']);\n        filteredRecipient.amount = new BigNumber(filteredRecipient.amount).toFixed();\n        return filteredRecipient;\n      });\n      const filteredOutputs = explainedTx.outputs.map((output) => {\n        const filteredOutput = _.pick(output, ['address', 'amount']);\n        filteredOutput.amount = new BigNumber(filteredOutput.amount).toFixed();\n        return filteredOutput;\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      for (const recipients of txParams.recipients) {\n        totalAmount = totalAmount.plus(recipients.amount);\n      }\n      if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n        throw new Error('Tx total amount does not match with expected total amount field');\n      }\n    }\n    return true;\n  }\n\n  async isWalletAddress(params: TssVerifyAddressOptions): Promise<boolean> {\n    const { address: newAddress } = params;\n\n    if (!this.isValidAddress(newAddress)) {\n      throw new InvalidAddressError(`invalid address: ${newAddress}`);\n    }\n    return true;\n  }\n\n  async parseTransaction(params: SuiParseTransactionOptions): Promise<SuiParsedTransaction> {\n    const transactionExplanation = await this.explainTransaction({ txHex: params.txHex });\n\n    if (!transactionExplanation) {\n      throw new Error('Invalid transaction');\n    }\n\n    let fee = new BigNumber(0);\n\n    const suiTransaction = transactionExplanation as SuiTransactionExplanation;\n    if (suiTransaction.outputs.length <= 0) {\n      return {\n        inputs: [],\n        outputs: [],\n        fee,\n      };\n    }\n\n    const senderAddress = suiTransaction.outputs[0].address;\n    if (suiTransaction.fee.fee !== '') {\n      fee = new BigNumber(suiTransaction.fee.fee);\n    }\n\n    // assume 1 sender, who is also the fee payer\n    const inputs = [\n      {\n        address: senderAddress,\n        amount: new BigNumber(suiTransaction.outputAmount).plus(fee).toFixed(),\n      },\n    ];\n\n    const outputs: TransactionOutput[] = suiTransaction.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: new BigNumber(output.amount).toFixed(),\n      };\n    });\n\n    return {\n      inputs,\n      outputs,\n      fee,\n    };\n  }\n\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new SuiKeyPair({ seed }) : new SuiKeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return {\n      pub: keys.pub,\n      prv: keys.prv,\n    };\n  }\n\n  isValidPub(_: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidPrv(_: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  signTransaction(_: SignTransactionOptions): Promise<SignedTransaction> {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Explain a Sui transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<SuiTransactionExplanation> {\n    const factory = this.getBuilder();\n    let rebuiltTransaction: BaseTransaction;\n\n    try {\n      const transactionBuilder = factory.from(Buffer.from(params.txHex, 'hex').toString('base64'));\n      rebuiltTransaction = await transactionBuilder.build();\n    } catch {\n      throw new Error('Invalid transaction');\n    }\n\n    return rebuiltTransaction.explainTransaction();\n  }\n\n  private getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  private getAddressFromPublicKey(derivedPublicKey: string) {\n    // TODO(BG-59016) replace with account lib implementation\n    return utils.getAddressFromPublicKey(derivedPublicKey);\n  }\n\n  /** @inheritDoc */\n  async getSignablePayload(serializedTx: string): Promise<Buffer> {\n    const factory = this.getBuilder();\n    const rebuiltTransaction = await factory.from(serializedTx).build();\n    return rebuiltTransaction.signablePayload;\n  }\n\n  protected getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].suiNodeUrl;\n  }\n\n  protected async getBalance(owner: string, coinType?: string): Promise<string> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getBalance(url, owner, coinType);\n  }\n\n  protected async getInputCoins(owner: string, coinType?: string): Promise<SuiObjectInfo[]> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getInputCoins(url, owner, coinType);\n  }\n\n  protected async getFeeEstimate(txHex: string): Promise<BigNumber> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getFeeEstimate(url, txHex);\n  }\n\n  /**\n   * Builds funds recovery transaction(s) without BitGo\n   *\n   * @param {MPCRecoveryOptions} params parameters needed to construct and\n   * (maybe) sign the transaction\n   *\n   * @returns {MPCTx | MPCSweepTxs} array of the serialized transaction hex strings and indices\n   * of the addresses being swept\n   */\n  async recover(params: MPCRecoveryOptions): Promise<MPCTxs | MPCSweepTxs> {\n    if (!params.bitgoKey) {\n      throw new Error('missing bitgoKey');\n    }\n    if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    const startIdx = utils.validateNonNegativeNumber(\n      0,\n      'Invalid starting index to scan for addresses',\n      params.startingScanIndex\n    );\n    const numIterations = utils.validateNonNegativeNumber(20, 'Invalid scanning factor', params.scan);\n    const endIdx = startIdx + numIterations;\n    const bitgoKey = params.bitgoKey.replace(/\\s/g, '');\n    const MPC = await EDDSAMethods.getInitializedMpcInstance();\n\n    for (let idx = startIdx; idx < endIdx; idx++) {\n      const derivationPath = (params.seed ? getDerivationPath(params.seed) : 'm') + `/${idx}`;\n      const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);\n      const senderAddress = this.getAddressFromPublicKey(derivedPublicKey);\n      let availableBalance = new BigNumber(0);\n      try {\n        availableBalance = new BigNumber(await this.getBalance(senderAddress));\n      } catch (e) {\n        continue;\n      }\n      if (availableBalance.minus(MAX_GAS_BUDGET).toNumber() <= 0) {\n        continue;\n      }\n\n      // check for possible token recovery, recover the token provide by user\n      if (params.tokenContractAddress) {\n        const token = utils.getSuiTokenFromAddress(params.tokenContractAddress!, this.getNetwork()) as SuiCoin;\n        if (!token) {\n          throw new Error(`Sui Token Package ID not supported.`);\n        }\n        const coinType = `${token.packageId}::${token.module}::${token.symbol}`;\n        try {\n          const availableTokenBalance = new BigNumber(await this.getBalance(senderAddress, coinType));\n          if (availableTokenBalance.toNumber() <= 0) {\n            continue;\n          }\n        } catch (e) {\n          continue;\n        }\n        return this.recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey);\n      }\n\n      let inputCoins = await this.getInputCoins(senderAddress);\n      inputCoins = inputCoins.sort((a, b) => {\n        return b.balance.minus(a.balance).toNumber();\n      });\n      if (inputCoins.length > MAX_OBJECT_LIMIT) {\n        inputCoins = inputCoins.slice(0, MAX_OBJECT_LIMIT);\n      }\n      let netAmount = inputCoins.reduce((acc, obj) => acc.plus(obj.balance), new BigNumber(0));\n      netAmount = netAmount.minus(MAX_GAS_BUDGET);\n\n      const recipients = [\n        {\n          address: params.recoveryDestination,\n          amount: netAmount.toString(),\n        },\n      ];\n\n      // first build the unsigned txn\n      const factory = new TransactionBuilderFactory(coins.get(this.getChain()));\n      const txBuilder = factory\n        .getTransferBuilder()\n        .type(SuiTransactionType.Transfer)\n        .sender(senderAddress)\n        .send(recipients)\n        .gasData({\n          owner: senderAddress,\n          price: DEFAULT_GAS_PRICE,\n          budget: MAX_GAS_BUDGET,\n          payment: inputCoins,\n        });\n\n      const tempTx = (await txBuilder.build()) as TransferTransaction;\n      const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());\n      const gasBudget = Math.trunc(feeEstimate.toNumber() * DEFAULT_GAS_OVERHEAD);\n\n      netAmount = netAmount.plus(MAX_GAS_BUDGET).minus(gasBudget);\n      recipients[0].amount = netAmount.toString();\n      txBuilder.send(recipients);\n      txBuilder.gasData({\n        owner: senderAddress,\n        price: DEFAULT_GAS_PRICE,\n        budget: gasBudget,\n        payment: inputCoins,\n      });\n\n      const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n      if (isUnsignedSweep) {\n        return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath);\n      }\n\n      await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, false);\n      const tx = (await txBuilder.build()) as TransferTransaction;\n      return {\n        transactions: [\n          {\n            scanIndex: idx,\n            recoveryAmount: netAmount.toString(),\n            serializedTx: tx.toBroadcastFormat(),\n            signature: Buffer.from(tx.serializedSig).toString('base64'),\n            coin: this.getChain(),\n          },\n        ],\n        lastScanIndex: idx,\n      };\n    }\n\n    throw new Error(\n      `Did not find an address with sufficient funds to recover. Please start the next scan at address index ${endIdx}. If it is token transaction, please keep sufficient Sui balance in the address for the transaction fee.`\n    );\n  }\n\n  private async recoverSuiToken(\n    params: MPCRecoveryOptions,\n    token: SuiCoin,\n    senderAddress: string,\n    derivationPath: string,\n    derivedPublicKey: string,\n    idx: number,\n    bitgoKey: string\n  ): Promise<MPCTxs | MPCSweepTxs> {\n    const coinType = `${token.packageId}::${token.module}::${token.symbol}`;\n    let tokenObjects = await this.getInputCoins(senderAddress, coinType);\n    tokenObjects = tokenObjects.sort((a, b) => {\n      return b.balance.minus(a.balance).toNumber();\n    });\n    if (tokenObjects.length > TOKEN_OBJECT_LIMIT) {\n      tokenObjects = tokenObjects.slice(0, TOKEN_OBJECT_LIMIT);\n    }\n    const netAmount = tokenObjects.reduce((acc, obj) => acc.plus(obj.balance), new BigNumber(0));\n    const recipients = [\n      {\n        address: params.recoveryDestination,\n        amount: netAmount.toString(),\n      },\n    ];\n\n    const gasAmount = new BigNumber(MAX_GAS_BUDGET);\n    let gasObjects = await this.getInputCoins(senderAddress);\n    gasObjects = utils.selectObjectsInDescOrderOfBalance(gasObjects, gasAmount);\n    if (gasObjects.length >= MAX_GAS_OBJECTS) {\n      gasObjects = gasObjects.slice(0, MAX_GAS_OBJECTS - 1);\n    }\n\n    // first build the unsigned txn\n    const factory = new TransactionBuilderFactory(token);\n    const txBuilder = factory\n      .getTokenTransferBuilder()\n      .type(SuiTransactionType.TokenTransfer)\n      .sender(senderAddress)\n      .send(recipients)\n      .inputObjects(tokenObjects)\n      .gasData({\n        owner: senderAddress,\n        price: DEFAULT_GAS_PRICE,\n        budget: MAX_GAS_BUDGET,\n        payment: gasObjects,\n      });\n\n    const tempTx = (await txBuilder.build()) as TokenTransferTransaction;\n    const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());\n    const gasBudget = Math.trunc(feeEstimate.toNumber() * DEFAULT_GAS_OVERHEAD);\n\n    txBuilder.gasData({\n      owner: senderAddress,\n      price: DEFAULT_GAS_PRICE,\n      budget: gasBudget,\n      payment: gasObjects,\n    });\n\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n    if (isUnsignedSweep) {\n      return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath, token);\n    }\n\n    await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, true);\n    const tx = (await txBuilder.build()) as TokenTransferTransaction;\n    return {\n      transactions: [\n        {\n          scanIndex: idx,\n          recoveryAmount: netAmount.toString(),\n          serializedTx: tx.toBroadcastFormat(),\n          signature: Buffer.from(tx.serializedSig).toString('base64'),\n          coin: token.name,\n        },\n      ],\n      lastScanIndex: idx,\n    };\n  }\n\n  private async buildUnsignedSweepTransaction(\n    txBuilder: TransactionBuilder,\n    senderAddress: string,\n    bitgoKey: string,\n    index: number,\n    derivationPath: string,\n    token?: SuiCoin\n  ): Promise<MPCSweepTxs> {\n    const isTokenTransaction = !!token;\n    const unsignedTransaction = isTokenTransaction\n      ? ((await txBuilder.build()) as TokenTransferTransaction)\n      : ((await txBuilder.build()) as TransferTransaction);\n    const serializedTx = unsignedTransaction.toBroadcastFormat();\n    const serializedTxHex = Buffer.from(serializedTx, 'base64').toString('hex');\n    const parsedTx = await this.parseTransaction({ txHex: serializedTxHex });\n    const walletCoin = isTokenTransaction ? token.name : this.getChain();\n    const output = parsedTx.outputs[0];\n    const inputs = [\n      {\n        address: senderAddress,\n        valueString: output.amount,\n        value: new BigNumber(output.amount),\n      },\n    ];\n    const outputs = [\n      {\n        address: output.address,\n        valueString: output.amount,\n        coinName: walletCoin,\n      },\n    ];\n    const spendAmount = output.amount;\n    const completedParsedTx = {\n      inputs: inputs,\n      outputs: outputs,\n      spendAmount: spendAmount,\n      type: isTokenTransaction ? SuiTransactionType.TokenTransfer : SuiTransactionType.Transfer,\n    };\n    const fee = parsedTx.fee;\n    const feeInfo = { fee: fee.toNumber(), feeString: fee.toString() };\n    const coinSpecific = { commonKeychain: bitgoKey };\n    const transaction: MPCTx = {\n      serializedTx: serializedTxHex,\n      scanIndex: index,\n      coin: walletCoin,\n      signableHex: unsignedTransaction.signablePayload.toString('hex'),\n      derivationPath,\n      parsedTx: completedParsedTx,\n      feeInfo: feeInfo,\n      coinSpecific: coinSpecific,\n    };\n    const unsignedTx: MPCUnsignedTx = { unsignedTx: transaction, signatureShares: [] };\n    const transactions: MPCUnsignedTx[] = [unsignedTx];\n    const txRequest: RecoveryTxRequest = {\n      transactions: transactions,\n      walletCoin: walletCoin,\n    };\n    return { txRequests: [txRequest] };\n  }\n\n  private async signRecoveryTransaction(\n    txBuilder: TransactionBuilder,\n    params: MPCRecoveryOptions,\n    derivationPath: string,\n    derivedPublicKey: string,\n    isTokenTransaction: boolean\n  ) {\n    // TODO(BG-51092): This looks like a common part which can be extracted out too\n    const unsignedTx = isTokenTransaction\n      ? ((await txBuilder.build()) as TokenTransferTransaction)\n      : ((await txBuilder.build()) as TransferTransaction);\n    if (!params.userKey) {\n      throw new Error('missing userKey');\n    }\n    if (!params.backupKey) {\n      throw new Error('missing backupKey');\n    }\n    if (!params.walletPassphrase) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    // Clean up whitespace from entered values\n    const userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n\n    // Decrypt private keys from KeyCard values\n    let userPrv: string;\n    try {\n      userPrv = 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    /** TODO BG-52419 Implement Codec for parsing */\n    const userSigningMaterial = JSON.parse(userPrv) as EDDSAMethodTypes.UserSigningMaterial;\n\n    let backupPrv: string;\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    const backupSigningMaterial = JSON.parse(backupPrv) as EDDSAMethodTypes.BackupSigningMaterial;\n    /* ********************** END ***********************************/\n\n    // add signature\n    const signatureHex = await EDDSAMethods.getTSSSignature(\n      userSigningMaterial,\n      backupSigningMaterial,\n      derivationPath,\n      unsignedTx\n    );\n    txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);\n  }\n\n  async broadcastTransaction({\n    transactions,\n  }: BaseBroadcastTransactionOptions): Promise<BaseBroadcastTransactionResult> {\n    const txIds: string[] = [];\n    const url = this.getPublicNodeUrl();\n    let digest = '';\n    if (!!transactions) {\n      for (const txn of transactions) {\n        try {\n          digest = await utils.executeTransactionBlock(url, txn.serializedTx, [txn.signature!]);\n        } catch (e) {\n          throw new Error(`Failed to broadcast transaction, error: ${e.message}`);\n        }\n        txIds.push(digest);\n      }\n    }\n    return { txIds };\n  }\n\n  /** inherited doc */\n  async createBroadcastableSweepTransaction(params: MPCSweepRecoveryOptions): Promise<MPCTxs> {\n    const req = params.signatureShares;\n    const broadcastableTransactions: MPCTx[] = [];\n    let lastScanIndex = 0;\n\n    for (let i = 0; i < req.length; i++) {\n      const MPC = await EDDSAMethods.getInitializedMpcInstance();\n      const transaction = req[i].txRequest.transactions[0].unsignedTx;\n      if (!req[i].ovc || !req[i].ovc[0].eddsaSignature) {\n        throw new Error('Missing signature(s)');\n      }\n      const signature = req[i].ovc[0].eddsaSignature;\n      if (!transaction.signableHex) {\n        throw new Error('Missing signable hex');\n      }\n      const messageBuffer = Buffer.from(transaction.signableHex!, 'hex');\n      const result = MPC.verify(messageBuffer, signature);\n      if (!result) {\n        throw new Error('Invalid signature');\n      }\n      const signatureHex = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);\n      const serializedTxBase64 = Buffer.from(transaction.serializedTx, 'hex').toString('base64');\n      const txBuilder = this.getBuilder().from(serializedTxBase64);\n      if (!transaction.coinSpecific?.commonKeychain) {\n        throw new Error('Missing common keychain');\n      }\n      const commonKeychain = transaction.coinSpecific!.commonKeychain! as string;\n      if (!transaction.derivationPath) {\n        throw new Error('Missing derivation path');\n      }\n      const derivationPath = transaction.derivationPath as string;\n      const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);\n\n      // add combined signature from ovc\n      txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);\n      const signedTransaction = (await txBuilder.build()) as TransferTransaction;\n      const serializedTx = signedTransaction.toBroadcastFormat();\n      const outputAmount = signedTransaction.explainTransaction().outputAmount;\n\n      broadcastableTransactions.push({\n        serializedTx: serializedTx,\n        scanIndex: transaction.scanIndex,\n        signature: Buffer.from(signedTransaction.serializedSig).toString('base64'),\n        recoveryAmount: outputAmount.toString(),\n      });\n\n      if (i === req.length - 1 && transaction.coinSpecific!.lastScanIndex) {\n        lastScanIndex = transaction.coinSpecific!.lastScanIndex as number;\n      }\n    }\n\n    return { transactions: broadcastableTransactions, lastScanIndex };\n  }\n\n  /**\n   * Builds native SUI recoveries of receive addresses in batch without BitGo.\n   * Funds will be recovered to base address first. You need to initiate another sweep txn after that.\n   *\n   * @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.\n   * @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).\n   * @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).\n   */\n  async recoverConsolidations(params: MPCConsolidationRecoveryOptions): Promise<MPCTxs | MPCSweepTxs> {\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n    const startIdx = utils.validateNonNegativeNumber(\n      1,\n      'Invalid starting index to scan for addresses',\n      params.startingScanIndex\n    );\n    const endIdx = utils.validateNonNegativeNumber(\n      startIdx + DEFAULT_SCAN_FACTOR,\n      'Invalid ending index to scan for addresses',\n      params.endingScanIndex\n    );\n\n    if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * DEFAULT_SCAN_FACTOR) {\n      throw new Error(\n        `Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`\n      );\n    }\n\n    const bitgoKey = params.bitgoKey.replace(/\\s/g, '');\n    const MPC = await EDDSAMethods.getInitializedMpcInstance();\n    const derivationPath = (params.seed ? getDerivationPath(params.seed) : 'm') + '/0';\n    const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);\n    const baseAddress = this.getAddressFromPublicKey(derivedPublicKey);\n\n    const consolidationTransactions: any[] = [];\n    let lastScanIndex = startIdx;\n    for (let idx = startIdx; idx < endIdx; idx++) {\n      const recoverParams = {\n        userKey: params.userKey,\n        backupKey: params.backupKey,\n        bitgoKey: params.bitgoKey,\n        walletPassphrase: params.walletPassphrase,\n        seed: params.seed,\n        tokenContractAddress: params.tokenContractAddress,\n        recoveryDestination: baseAddress,\n        startingScanIndex: idx,\n        scan: 1,\n      };\n\n      let recoveryTransaction: MPCTxs | MPCSweepTxs;\n      try {\n        recoveryTransaction = await this.recover(recoverParams);\n      } catch (e) {\n        if (e.message.startsWith('Did not find an address with sufficient funds to recover.')) {\n          lastScanIndex = idx;\n          continue;\n        }\n        throw e;\n      }\n\n      if (isUnsignedSweep) {\n        consolidationTransactions.push((recoveryTransaction as MPCSweepTxs).txRequests[0]);\n      } else {\n        consolidationTransactions.push((recoveryTransaction as MPCTxs).transactions[0]);\n      }\n      lastScanIndex = idx;\n    }\n\n    if (consolidationTransactions.length === 0) {\n      throw new Error(\n        `Did not find an address with sufficient funds to recover. Please start the next scan at address index ${\n          lastScanIndex + 1\n        }.`\n      );\n    }\n\n    if (isUnsignedSweep) {\n      // lastScanIndex will be used to inform user the last address index scanned for available funds (so they can\n      // appropriately adjust the scan range on the next iteration of consolidation recoveries). In the case of unsigned\n      // sweep consolidations, this lastScanIndex will be provided in the coinSpecific of the last txn made.\n      consolidationTransactions[\n        consolidationTransactions.length - 1\n      ].transactions[0].unsignedTx.coinSpecific.lastScanIndex = lastScanIndex;\n      return { txRequests: consolidationTransactions };\n    }\n\n    return { transactions: consolidationTransactions, lastScanIndex };\n  }\n\n  /** inherited doc */\n  setCoinSpecificFieldsInIntent(intent: PopulatedIntent, params: PrebuildTransactionWithIntentOptions): void {\n    intent.unspents = params.unspents;\n  }\n\n  /** inherited doc */\n  auditDecryptedKey({ publicKey, prv, multiSigType }: AuditDecryptedKeyParams) {\n    if (multiSigType !== 'tss') {\n      throw new Error('Unsupported multiSigType');\n    }\n\n    auditEddsaPrivateKey(prv, publicKey ?? '');\n  }\n}\n"]}