@bitgo-beta/abstract-substrate 1.0.1-beta.21 → 1.0.1-beta.210

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 (54) hide show
  1. package/CHANGELOG.md +87 -0
  2. package/dist/src/abstractSubstrateCoin.d.ts +55 -23
  3. package/dist/src/abstractSubstrateCoin.d.ts.map +1 -1
  4. package/dist/src/abstractSubstrateCoin.js +328 -62
  5. package/dist/src/lib/constants.d.ts +2 -0
  6. package/dist/src/lib/constants.d.ts.map +1 -0
  7. package/dist/src/lib/constants.js +5 -0
  8. package/dist/src/lib/errors.d.ts +8 -0
  9. package/dist/src/lib/errors.d.ts.map +1 -0
  10. package/dist/src/lib/errors.js +19 -0
  11. package/dist/src/lib/iface.d.ts +37 -234
  12. package/dist/src/lib/iface.d.ts.map +1 -1
  13. package/dist/src/lib/iface.js +4 -97
  14. package/dist/src/lib/index.d.ts +9 -2
  15. package/dist/src/lib/index.d.ts.map +1 -1
  16. package/dist/src/lib/index.js +22 -4
  17. package/dist/src/lib/keyPair.d.ts +6 -20
  18. package/dist/src/lib/keyPair.d.ts.map +1 -1
  19. package/dist/src/lib/keyPair.js +12 -66
  20. package/dist/src/lib/nativeTransferBuilder.d.ts +61 -0
  21. package/dist/src/lib/nativeTransferBuilder.d.ts.map +1 -0
  22. package/dist/src/lib/nativeTransferBuilder.js +147 -0
  23. package/dist/src/lib/singletonRegistry.d.ts +8 -0
  24. package/dist/src/lib/singletonRegistry.d.ts.map +1 -0
  25. package/dist/src/lib/singletonRegistry.js +20 -0
  26. package/dist/src/lib/transaction.d.ts +73 -0
  27. package/dist/src/lib/transaction.d.ts.map +1 -0
  28. package/dist/src/lib/transaction.js +424 -0
  29. package/dist/src/lib/transactionBuilder.d.ts +121 -0
  30. package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
  31. package/dist/src/lib/transactionBuilder.js +334 -0
  32. package/dist/src/lib/transferBuilder.d.ts +6 -0
  33. package/dist/src/lib/transferBuilder.d.ts.map +1 -0
  34. package/dist/src/lib/transferBuilder.js +11 -0
  35. package/dist/src/lib/txnSchema.d.ts +9 -0
  36. package/dist/src/lib/txnSchema.d.ts.map +1 -0
  37. package/dist/src/lib/txnSchema.js +52 -0
  38. package/dist/src/lib/utils.d.ts +7 -143
  39. package/dist/src/lib/utils.d.ts.map +1 -1
  40. package/dist/src/lib/utils.js +11 -251
  41. package/dist/tsconfig.tsbuildinfo +1 -1
  42. package/package.json +12 -7
  43. package/dist/src/lib/iface_utils.d.ts +0 -59
  44. package/dist/src/lib/iface_utils.d.ts.map +0 -1
  45. package/dist/src/lib/iface_utils.js +0 -92
  46. package/dist/src/resources/index.d.ts +0 -3
  47. package/dist/src/resources/index.d.ts.map +0 -1
  48. package/dist/src/resources/index.js +0 -19
  49. package/dist/src/resources/mainnet.d.ts +0 -2
  50. package/dist/src/resources/mainnet.d.ts.map +0 -1
  51. package/dist/src/resources/mainnet.js +0 -5
  52. package/dist/src/resources/westend.d.ts +0 -2
  53. package/dist/src/resources/westend.d.ts.map +0 -1
  54. package/dist/src/resources/westend.js +0 -5
@@ -1,56 +1,26 @@
1
1
  "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = 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);
9
- }) : (function(o, m, k, k2) {
10
- if (k2 === undefined) k2 = k;
11
- o[k2] = m[k];
12
- }));
13
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
- Object.defineProperty(o, "default", { enumerable: true, value: v });
15
- }) : function(o, v) {
16
- o["default"] = v;
17
- });
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
- })();
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
35
5
  Object.defineProperty(exports, "__esModule", { value: true });
36
6
  exports.SubstrateCoin = exports.DEFAULT_SCAN_FACTOR = void 0;
37
- const _ = __importStar(require("lodash"));
38
7
  const sdk_core_1 = require("@bitgo-beta/sdk-core");
39
8
  const lib_1 = require("./lib");
40
- const utils = lib_1.Utils.default;
41
- exports.DEFAULT_SCAN_FACTOR = 20; // default number of receive addresses to scan for funds
9
+ const constants_1 = require("./lib/constants");
10
+ const utils_1 = __importDefault(require("./lib/utils"));
11
+ const sdk_lib_mpc_1 = require("@bitgo-beta/sdk-lib-mpc");
12
+ const bignumber_js_1 = __importDefault(require("bignumber.js"));
13
+ exports.DEFAULT_SCAN_FACTOR = 20;
42
14
  class SubstrateCoin extends sdk_core_1.BaseCoin {
43
15
  constructor(bitgo, staticsCoin) {
44
16
  super(bitgo);
45
17
  this.MAX_VALIDITY_DURATION = 2400;
18
+ this.SWEEP_TXN_DURATION = 64;
46
19
  if (!staticsCoin) {
47
20
  throw new Error('missing required constructor parameter staticsCoin');
48
21
  }
49
22
  this._staticsCoin = staticsCoin;
50
23
  }
51
- static createInstance(bitgo, staticsCoin) {
52
- return new SubstrateCoin(bitgo, staticsCoin);
53
- }
54
24
  /**
55
25
  * Creates an instance of TransactionBuilderFactory for the coin specific sdk
56
26
  */
@@ -77,13 +47,17 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
77
47
  supportsTss() {
78
48
  return true;
79
49
  }
50
+ /** inherited doc */
51
+ getDefaultMultisigType() {
52
+ return sdk_core_1.multisigTypes.tss;
53
+ }
80
54
  /** @inheritDoc **/
81
55
  getMPCAlgorithm() {
82
56
  return 'eddsa';
83
57
  }
84
58
  /** @inheritDoc **/
85
59
  generateKeyPair(seed) {
86
- const keyPair = seed ? utils.keyPairFromSeed(new Uint8Array(seed)) : new lib_1.KeyPair();
60
+ const keyPair = seed ? new lib_1.KeyPair({ seed }) : new lib_1.KeyPair();
87
61
  const keys = keyPair.getKeys();
88
62
  if (!keys.prv) {
89
63
  throw new Error('Missing prv in key generation.');
@@ -95,15 +69,15 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
95
69
  }
96
70
  /** @inheritDoc **/
97
71
  isValidPub(pub) {
98
- return utils.isValidPublicKey(pub);
72
+ return utils_1.default.isValidPublicKey(pub);
99
73
  }
100
74
  /** @inheritDoc **/
101
75
  isWalletAddress(params) {
102
- throw new Error('Method not implemented');
76
+ throw new sdk_core_1.MethodNotImplementedError();
103
77
  }
104
78
  /** @inheritDoc **/
105
- parseTransaction(params) {
106
- throw new Error('Method not implemented');
79
+ async parseTransaction(params) {
80
+ return {};
107
81
  }
108
82
  /** @inheritDoc **/
109
83
  async verifyTransaction(params) {
@@ -115,25 +89,16 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
115
89
  }
116
90
  /** @inheritDoc **/
117
91
  isValidAddress(address) {
118
- return utils.isValidAddress(address);
92
+ return utils_1.default.isValidAddress(address);
119
93
  }
120
94
  verifySignTransactionParams(params) {
121
- const prv = params.prv;
122
- const txHex = params.txPrebuild.txHex;
123
- if (!txHex) {
124
- throw new Error('missing txPrebuild parameter');
125
- }
126
- if (!_.isString(txHex)) {
127
- throw new Error(`txPrebuild must be an object, got type ${typeof txHex}`);
128
- }
129
- if (!prv) {
130
- throw new Error('missing prv parameter to sign transaction');
131
- }
132
- if (!_.isString(prv)) {
133
- throw new Error(`prv must be a string, got type ${typeof prv}`);
95
+ const prv = params?.prv;
96
+ const txHex = params?.txPrebuild?.txHex;
97
+ if (typeof txHex !== 'string') {
98
+ throw new Error(`txHex must be string, got type ${typeof txHex}`);
134
99
  }
135
- if (!_.has(params, 'pubs')) {
136
- throw new Error('missing public key parameter to sign transaction');
100
+ if (typeof prv !== 'string') {
101
+ throw new Error(`prv must be string, got type ${typeof prv}`);
137
102
  }
138
103
  return { txHex, prv };
139
104
  }
@@ -145,7 +110,7 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
145
110
  const keyPair = new lib_1.KeyPair({ prv: prv });
146
111
  const { referenceBlock, blockNumber, transactionVersion, sender } = params.txPrebuild.transaction;
147
112
  txBuilder
148
- .validity({ firstValid: blockNumber, maxDuration: this.MAX_VALIDITY_DURATION })
113
+ .validity({ firstValid: blockNumber, maxDuration: this.getMaxValidityDurationBlocks() })
149
114
  .referenceBlock(referenceBlock)
150
115
  .version(transactionVersion)
151
116
  .sender({ address: sender })
@@ -157,6 +122,307 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
157
122
  const signedTxHex = transaction.toBroadcastFormat();
158
123
  return { txHex: signedTxHex };
159
124
  }
125
+ /**
126
+ * Retrieves the address format for the substrate coin.
127
+ *
128
+ * @returns {number} The address format as a number.
129
+ */
130
+ getAddressFormat() {
131
+ return constants_1.DEFAULT_SUBSTRATE_PREFIX;
132
+ }
133
+ /**
134
+ * Retrieves the maximum validity duration in blocks.
135
+ *
136
+ * This method is intended to be overridden by subclasses to provide the specific
137
+ * maximum validity duration for different types of Substrate-based coins.
138
+ *
139
+ * @returns {number} The maximum validity duration in blocks.
140
+ * @throws {Error} If the method is not implemented by the subclass.
141
+ */
142
+ getMaxValidityDurationBlocks() {
143
+ throw new Error('Method not implemented.');
144
+ }
145
+ getAddressFromPublicKey(Pubkey) {
146
+ return new lib_1.KeyPair({ pub: Pubkey }).getAddress(this.getAddressFormat());
147
+ }
148
+ async getInitializedNodeAPI() {
149
+ throw new Error('Method not implemented.');
150
+ }
151
+ async getAccountInfo(walletAddr) {
152
+ throw new Error('Method not implemented.');
153
+ }
154
+ async getFee(destAddr, srcAddr, amount) {
155
+ throw new Error('Method not implemented.');
156
+ }
157
+ async getHeaderInfo() {
158
+ throw new Error('Method not implemented.');
159
+ }
160
+ async getMaterial() {
161
+ throw new Error('Method not implemented.');
162
+ }
163
+ /**
164
+ * Builds a funds recovery transaction without BitGo
165
+ * @param {MPCRecoveryOptions} params parameters needed to construct and
166
+ * (maybe) sign the transaction
167
+ *
168
+ * @returns {MPCTx} the serialized transaction hex string and index
169
+ * of the address being swept
170
+ */
171
+ async recover(params) {
172
+ if (!params.bitgoKey) {
173
+ throw new Error('Missing bitgoKey');
174
+ }
175
+ if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
176
+ throw new Error('Invalid recovery destination address');
177
+ }
178
+ const bitgoKey = params.bitgoKey.replace(/\s/g, '');
179
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
180
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
181
+ const index = params.index || 0;
182
+ const currPath = params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) + `/${index}` : `m/${index}`;
183
+ const accountId = MPC.deriveUnhardened(bitgoKey, currPath).slice(0, 64);
184
+ const senderAddr = this.getAddressFromPublicKey(accountId);
185
+ const { nonce, freeBalance } = await this.getAccountInfo(senderAddr);
186
+ const destAddr = params.recoveryDestination;
187
+ const amount = freeBalance;
188
+ const partialFee = await this.getFee(destAddr, senderAddr, amount);
189
+ const value = new bignumber_js_1.default(freeBalance).minus(new bignumber_js_1.default(partialFee));
190
+ if (value.isLessThanOrEqualTo(0)) {
191
+ throw new Error('Did not find address with funds to recover');
192
+ }
193
+ const { headerNumber, headerHash } = await this.getHeaderInfo();
194
+ const material = await this.getMaterial();
195
+ const validityWindow = { firstValid: headerNumber, maxDuration: this.MAX_VALIDITY_DURATION };
196
+ const txBuilder = this.getBuilder().getTransferBuilder().material(material);
197
+ txBuilder
198
+ .sweep(false)
199
+ .to({ address: params.recoveryDestination })
200
+ .sender({ address: senderAddr })
201
+ .validity(validityWindow)
202
+ .referenceBlock(headerHash)
203
+ .sequenceId({ name: 'Nonce', keyword: 'Nonce', value: nonce })
204
+ .fee({ amount: 0, type: 'tip' });
205
+ const unsignedTransaction = (await txBuilder.build());
206
+ let serializedTx = unsignedTransaction.toBroadcastFormat();
207
+ if (!isUnsignedSweep) {
208
+ if (!params.userKey) {
209
+ throw new Error('missing userKey');
210
+ }
211
+ if (!params.backupKey) {
212
+ throw new Error('missing backupKey');
213
+ }
214
+ if (!params.walletPassphrase) {
215
+ throw new Error('missing wallet passphrase');
216
+ }
217
+ const userKey = params.userKey.replace(/\s/g, '');
218
+ const backupKey = params.backupKey.replace(/\s/g, '');
219
+ // Decrypt private keys from KeyCard values
220
+ let userPrv;
221
+ try {
222
+ userPrv = this.bitgo.decrypt({
223
+ input: userKey,
224
+ password: params.walletPassphrase,
225
+ });
226
+ }
227
+ catch (e) {
228
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
229
+ }
230
+ const userSigningMaterial = JSON.parse(userPrv);
231
+ let backupPrv;
232
+ try {
233
+ backupPrv = this.bitgo.decrypt({
234
+ input: backupKey,
235
+ password: params.walletPassphrase,
236
+ });
237
+ }
238
+ catch (e) {
239
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
240
+ }
241
+ const backupSigningMaterial = JSON.parse(backupPrv);
242
+ // add signature
243
+ const signatureHex = await sdk_core_1.EDDSAMethods.getTSSSignature(userSigningMaterial, backupSigningMaterial, currPath, unsignedTransaction);
244
+ const substrateKeyPair = new lib_1.KeyPair({ pub: accountId });
245
+ txBuilder.addSignature({ pub: substrateKeyPair.getKeys().pub }, signatureHex);
246
+ const signedTransaction = await txBuilder.build();
247
+ serializedTx = signedTransaction.toBroadcastFormat();
248
+ }
249
+ else {
250
+ const value = new bignumber_js_1.default(freeBalance);
251
+ const walletCoin = this.getChain();
252
+ const inputs = [
253
+ {
254
+ address: unsignedTransaction.inputs[0].address,
255
+ valueString: value.toString(),
256
+ value: value.toNumber(),
257
+ },
258
+ ];
259
+ const outputs = [
260
+ {
261
+ address: unsignedTransaction.outputs[0].address,
262
+ valueString: value.toString(),
263
+ coinName: walletCoin,
264
+ },
265
+ ];
266
+ const spendAmount = value.toString();
267
+ const parsedTx = { inputs: inputs, outputs: outputs, spendAmount: spendAmount, type: '' };
268
+ const feeInfo = { fee: 0, feeString: '0' };
269
+ const transaction = {
270
+ serializedTx: serializedTx,
271
+ scanIndex: index,
272
+ coin: walletCoin,
273
+ signableHex: unsignedTransaction.signablePayload.toString('hex'),
274
+ derivationPath: currPath,
275
+ parsedTx: parsedTx,
276
+ feeInfo: feeInfo,
277
+ coinSpecific: { ...validityWindow, commonKeychain: bitgoKey },
278
+ };
279
+ const unsignedTx = { unsignedTx: transaction, signatureShares: [] };
280
+ const transactions = [unsignedTx];
281
+ const txRequest = {
282
+ transactions: transactions,
283
+ walletCoin: walletCoin,
284
+ };
285
+ const txRequests = { txRequests: [txRequest] };
286
+ return txRequests;
287
+ }
288
+ const transaction = { serializedTx: serializedTx, scanIndex: index };
289
+ return transaction;
290
+ }
291
+ /**
292
+ * Builds native TAO recoveries of receive addresses in batch without BitGo.
293
+ * Funds will be recovered to base address first. You need to initiate another sweep txn after that.
294
+ *
295
+ * @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.
296
+ * @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).
297
+ * @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).
298
+ */
299
+ async recoverConsolidations(params) {
300
+ const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
301
+ const startIdx = params.startingScanIndex || 1;
302
+ const endIdx = params.endingScanIndex || startIdx + exports.DEFAULT_SCAN_FACTOR;
303
+ if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * exports.DEFAULT_SCAN_FACTOR) {
304
+ throw new Error(`Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`);
305
+ }
306
+ const bitgoKey = params.bitgoKey.replace(/\s/g, '');
307
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
308
+ const baseIndex = 0;
309
+ const basePath = params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) + `/${baseIndex}` : `m/${baseIndex}`;
310
+ const accountId = MPC.deriveUnhardened(bitgoKey, basePath).slice(0, 64);
311
+ const baseAddress = this.getAddressFromPublicKey(accountId);
312
+ const consolidationTransactions = [];
313
+ let lastScanIndex = startIdx;
314
+ for (let i = startIdx; i < endIdx; i++) {
315
+ const recoverParams = {
316
+ userKey: params.userKey,
317
+ backupKey: params.backupKey,
318
+ bitgoKey: params.bitgoKey,
319
+ walletPassphrase: params.walletPassphrase,
320
+ recoveryDestination: baseAddress,
321
+ seed: params.seed,
322
+ index: i,
323
+ };
324
+ let recoveryTransaction;
325
+ try {
326
+ recoveryTransaction = await this.recover(recoverParams);
327
+ }
328
+ catch (e) {
329
+ if (e.message === 'Did not find address with funds to recover') {
330
+ lastScanIndex = i;
331
+ continue;
332
+ }
333
+ throw e;
334
+ }
335
+ if (isUnsignedSweep) {
336
+ consolidationTransactions.push(recoveryTransaction.txRequests[0]);
337
+ }
338
+ else {
339
+ consolidationTransactions.push(recoveryTransaction);
340
+ }
341
+ lastScanIndex = i;
342
+ }
343
+ if (consolidationTransactions.length == 0) {
344
+ throw new Error('Did not find an address with funds to recover');
345
+ }
346
+ if (isUnsignedSweep) {
347
+ // lastScanIndex will be used to inform user the last address index scanned for available funds (so they can
348
+ // appropriately adjust the scan range on the next iteration of consolidation recoveries). In the case of unsigned
349
+ // sweep consolidations, this lastScanIndex will be provided in the coinSpecific of the last txn made.
350
+ const lastTransactionCoinSpecific = {
351
+ firstValid: consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific
352
+ .firstValid,
353
+ maxDuration: consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific
354
+ .maxDuration,
355
+ commonKeychain: consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific
356
+ .commonKeychain,
357
+ lastScanIndex: lastScanIndex,
358
+ };
359
+ consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific =
360
+ lastTransactionCoinSpecific;
361
+ const consolidationSweepTransactions = { txRequests: consolidationTransactions };
362
+ return consolidationSweepTransactions;
363
+ }
364
+ return { transactions: consolidationTransactions, lastScanIndex };
365
+ }
366
+ /** inherited doc */
367
+ async createBroadcastableSweepTransaction(params) {
368
+ const req = params.signatureShares;
369
+ const broadcastableTransactions = [];
370
+ let lastScanIndex = 0;
371
+ for (let i = 0; i < req.length; i++) {
372
+ const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
373
+ const transaction = req[i].txRequest.transactions[0].unsignedTx;
374
+ if (!req[i].ovc || !req[i].ovc[0].eddsaSignature) {
375
+ throw new Error('Missing signature(s)');
376
+ }
377
+ const signature = req[i].ovc[0].eddsaSignature;
378
+ if (!transaction.signableHex) {
379
+ throw new Error('Missing signable hex');
380
+ }
381
+ const messageBuffer = Buffer.from(transaction.signableHex, 'hex');
382
+ const result = MPC.verify(messageBuffer, signature);
383
+ if (!result) {
384
+ throw new Error('Invalid signature');
385
+ }
386
+ const signatureHex = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);
387
+ if (!transaction.coinSpecific ||
388
+ !transaction.coinSpecific?.firstValid ||
389
+ !transaction.coinSpecific?.maxDuration) {
390
+ throw new Error('missing validity window');
391
+ }
392
+ const validityWindow = {
393
+ firstValid: transaction.coinSpecific?.firstValid,
394
+ maxDuration: transaction.coinSpecific?.maxDuration,
395
+ };
396
+ const material = await this.getMaterial();
397
+ if (!transaction.coinSpecific?.commonKeychain) {
398
+ throw new Error('Missing common keychain');
399
+ }
400
+ const commonKeychain = transaction.coinSpecific.commonKeychain;
401
+ if (!transaction.derivationPath) {
402
+ throw new Error('Missing derivation path');
403
+ }
404
+ const derivationPath = transaction.derivationPath;
405
+ const accountId = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);
406
+ const senderAddr = this.getAddressFromPublicKey(accountId);
407
+ const txnBuilder = this.getBuilder()
408
+ .material(material)
409
+ .from(transaction.serializedTx)
410
+ .sender({ address: senderAddr })
411
+ .validity(validityWindow);
412
+ const substrateKeyPair = new lib_1.KeyPair({ pub: accountId });
413
+ txnBuilder.addSignature({ pub: substrateKeyPair.getKeys().pub }, signatureHex);
414
+ const signedTransaction = await txnBuilder.build();
415
+ const serializedTx = signedTransaction.toBroadcastFormat();
416
+ broadcastableTransactions.push({
417
+ serializedTx: serializedTx,
418
+ scanIndex: transaction.scanIndex,
419
+ });
420
+ if (i === req.length - 1 && transaction.coinSpecific.lastScanIndex) {
421
+ lastScanIndex = transaction.coinSpecific.lastScanIndex;
422
+ }
423
+ }
424
+ return { transactions: broadcastableTransactions, lastScanIndex };
425
+ }
160
426
  }
161
427
  exports.SubstrateCoin = SubstrateCoin;
162
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractSubstrateCoin.js","sourceRoot":"","sources":["../../src/abstractSubstrateCoin.ts"],"names":[],"mappings":"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","sourcesContent":["import * as _ from 'lodash';\nimport {\n  BaseCoin,\n  BitGoBase,\n  KeyPair,\n  MPCAlgorithm,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, CoinFamily } from '@bitgo-beta/statics';\nimport { Interface, KeyPair as SubstrateKeyPair, Utils } from './lib';\n\nconst utils = Utils.default;\n\nexport const DEFAULT_SCAN_FACTOR = 20; // default number of receive addresses to scan for funds\n\nexport interface SignTransactionOptions extends BaseSignTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  prv: string;\n}\n\nexport interface TransactionPrebuild {\n  txHex: string;\n  transaction: Interface.TxData;\n}\n\nexport interface ExplainTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  publicKey: string;\n  feeInfo: {\n    fee: string;\n  };\n}\n\nexport interface VerifiedTransactionParameters {\n  txHex: string;\n  prv: string;\n}\n\nexport class SubstrateCoin extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  readonly MAX_VALIDITY_DURATION = 2400;\n\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 SubstrateCoin(bitgo, staticsCoin);\n  }\n\n  /**\n   * Creates an instance of TransactionBuilderFactory for the coin specific sdk\n   */\n  getBuilder(): any {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritDoc **/\n  getBaseFactor(): string | number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  /** @inheritDoc **/\n  getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  /** @inheritDoc **/\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  /** @inheritDoc **/\n  getFullName(): string {\n    return this._staticsCoin.fullName;\n  }\n\n  /** @inheritDoc */\n  supportsTss(): boolean {\n    return true;\n  }\n\n  /** @inheritDoc **/\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  /** @inheritDoc **/\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? utils.keyPairFromSeed(new Uint8Array(seed)) : new SubstrateKeyPair();\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  /** @inheritDoc **/\n  isValidPub(pub: string): boolean {\n    return utils.isValidPublicKey(pub);\n  }\n\n  /** @inheritDoc **/\n  isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    throw new Error('Method not implemented');\n  }\n\n  /** @inheritDoc **/\n  parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    throw new Error('Method not implemented');\n  }\n\n  /** @inheritDoc **/\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  /** @inheritDoc **/\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  verifySignTransactionParams(params: SignTransactionOptions): VerifiedTransactionParameters {\n    const prv = params.prv;\n\n    const txHex = params.txPrebuild.txHex;\n\n    if (!txHex) {\n      throw new Error('missing txPrebuild parameter');\n    }\n\n    if (!_.isString(txHex)) {\n      throw new Error(`txPrebuild must be an object, got type ${typeof txHex}`);\n    }\n\n    if (!prv) {\n      throw new Error('missing prv parameter to sign transaction');\n    }\n\n    if (!_.isString(prv)) {\n      throw new Error(`prv must be a string, got type ${typeof prv}`);\n    }\n\n    if (!_.has(params, 'pubs')) {\n      throw new Error('missing public key parameter to sign transaction');\n    }\n\n    return { txHex, prv };\n  }\n\n  /** @inheritDoc **/\n  async signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    const { txHex, prv } = this.verifySignTransactionParams(params);\n    const factory = this.getBuilder();\n    const txBuilder = factory.from(txHex);\n    const keyPair = new SubstrateKeyPair({ prv: prv });\n    const { referenceBlock, blockNumber, transactionVersion, sender } = params.txPrebuild.transaction;\n\n    txBuilder\n      .validity({ firstValid: blockNumber, maxDuration: this.MAX_VALIDITY_DURATION })\n      .referenceBlock(referenceBlock)\n      .version(transactionVersion)\n      .sender({ address: sender })\n      .sign({ key: keyPair.getKeys().prv });\n    const transaction = await txBuilder.build();\n    if (!transaction) {\n      throw new Error('Invalid transaction');\n    }\n    const signedTxHex = transaction.toBroadcastFormat();\n    return { txHex: signedTxHex };\n  }\n}\n"]}
428
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractSubstrateCoin.js","sourceRoot":"","sources":["../../src/abstractSubstrateCoin.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseCoin,\n  BitGoBase,\n  EDDSAMethods,\n  EDDSAMethodTypes,\n  KeyPair,\n  MethodNotImplementedError,\n  MPCAlgorithm,\n  MPCConsolidationRecoveryOptions,\n  MPCRecoveryOptions,\n  MPCSweepRecoveryOptions,\n  MPCSweepTxs,\n  MPCTx,\n  MPCTxs,\n  MPCUnsignedTx,\n  MultisigType,\n  multisigTypes,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  RecoveryTxRequest,\n  SignedTransaction,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { CoinFamily, BaseCoin as StaticsBaseCoin } from '@bitgo-beta/statics';\nimport { KeyPair as SubstrateKeyPair, Transaction } from './lib';\nimport { DEFAULT_SUBSTRATE_PREFIX } from './lib/constants';\nimport { SignTransactionOptions, VerifiedTransactionParameters, Material } from './lib/iface';\nimport utils from './lib/utils';\nimport { getDerivationPath } from '@bitgo-beta/sdk-lib-mpc';\nimport BigNumber from 'bignumber.js';\nimport { ApiPromise } from '@polkadot/api';\n\nexport const DEFAULT_SCAN_FACTOR = 20;\n\nexport class SubstrateCoin extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  readonly MAX_VALIDITY_DURATION = 2400;\n  readonly SWEEP_TXN_DURATION = 64;\n\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  /**\n   * Creates an instance of TransactionBuilderFactory for the coin specific sdk\n   */\n  getBuilder(): any {\n    throw new Error('Method not implemented.');\n  }\n\n  /** @inheritDoc **/\n  getBaseFactor(): string | number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  /** @inheritDoc **/\n  getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  /** @inheritDoc **/\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  /** @inheritDoc **/\n  getFullName(): string {\n    return this._staticsCoin.fullName;\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  /** @inheritDoc **/\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  /** @inheritDoc **/\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new SubstrateKeyPair({ seed }) : new SubstrateKeyPair();\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  /** @inheritDoc **/\n  isValidPub(pub: string): boolean {\n    return utils.isValidPublicKey(pub);\n  }\n\n  /** @inheritDoc **/\n  isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    throw new MethodNotImplementedError();\n  }\n\n  /** @inheritDoc **/\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  /** @inheritDoc **/\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  /** @inheritDoc **/\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  verifySignTransactionParams(params: SignTransactionOptions): VerifiedTransactionParameters {\n    const prv = params?.prv;\n    const txHex = params?.txPrebuild?.txHex;\n\n    if (typeof txHex !== 'string') {\n      throw new Error(`txHex must be string, got type ${typeof txHex}`);\n    }\n\n    if (typeof prv !== 'string') {\n      throw new Error(`prv must be string, got type ${typeof prv}`);\n    }\n\n    return { txHex, prv };\n  }\n\n  /** @inheritDoc **/\n  async signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    const { txHex, prv } = this.verifySignTransactionParams(params);\n    const factory = this.getBuilder();\n    const txBuilder = factory.from(txHex);\n    const keyPair = new SubstrateKeyPair({ prv: prv });\n    const { referenceBlock, blockNumber, transactionVersion, sender } = params.txPrebuild.transaction;\n\n    txBuilder\n      .validity({ firstValid: blockNumber, maxDuration: this.getMaxValidityDurationBlocks() })\n      .referenceBlock(referenceBlock)\n      .version(transactionVersion)\n      .sender({ address: sender })\n      .sign({ key: keyPair.getKeys().prv });\n    const transaction = await txBuilder.build();\n    if (!transaction) {\n      throw new Error('Invalid transaction');\n    }\n    const signedTxHex = transaction.toBroadcastFormat();\n    return { txHex: signedTxHex };\n  }\n\n  /**\n   * Retrieves the address format for the substrate coin.\n   *\n   * @returns {number} The address format as a number.\n   */\n  protected getAddressFormat(): number {\n    return DEFAULT_SUBSTRATE_PREFIX;\n  }\n\n  /**\n   * Retrieves the maximum validity duration in blocks.\n   *\n   * This method is intended to be overridden by subclasses to provide the specific\n   * maximum validity duration for different types of Substrate-based coins.\n   *\n   * @returns {number} The maximum validity duration in blocks.\n   * @throws {Error} If the method is not implemented by the subclass.\n   */\n  protected getMaxValidityDurationBlocks(): number {\n    throw new Error('Method not implemented.');\n  }\n\n  protected getAddressFromPublicKey(Pubkey: string): string {\n    return new SubstrateKeyPair({ pub: Pubkey }).getAddress(this.getAddressFormat());\n  }\n\n  protected async getInitializedNodeAPI(): Promise<ApiPromise> {\n    throw new Error('Method not implemented.');\n  }\n\n  protected async getAccountInfo(walletAddr: string): Promise<{ nonce: number; freeBalance: number }> {\n    throw new Error('Method not implemented.');\n  }\n\n  protected async getFee(destAddr: string, srcAddr: string, amount: number): Promise<number> {\n    throw new Error('Method not implemented.');\n  }\n\n  protected async getHeaderInfo(): Promise<{ headerNumber: number; headerHash: string }> {\n    throw new Error('Method not implemented.');\n  }\n\n  protected async getMaterial(): Promise<Material> {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo\n   * @param {MPCRecoveryOptions} params parameters needed to construct and\n   * (maybe) sign the transaction\n   *\n   * @returns {MPCTx} the serialized transaction hex string and index\n   * of the address being swept\n   */\n  async recover(params: MPCRecoveryOptions): Promise<MPCTx | MPCSweepTxs> {\n    if (!params.bitgoKey) {\n      throw new Error('Missing bitgoKey');\n    }\n\n    if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('Invalid recovery destination address');\n    }\n\n    const bitgoKey = params.bitgoKey.replace(/\\s/g, '');\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n\n    const MPC = await EDDSAMethods.getInitializedMpcInstance();\n\n    const index = params.index || 0;\n    const currPath = params.seed ? getDerivationPath(params.seed) + `/${index}` : `m/${index}`;\n    const accountId = MPC.deriveUnhardened(bitgoKey, currPath).slice(0, 64);\n    const senderAddr = this.getAddressFromPublicKey(accountId);\n\n    const { nonce, freeBalance } = await this.getAccountInfo(senderAddr);\n    const destAddr = params.recoveryDestination;\n    const amount = freeBalance;\n    const partialFee = await this.getFee(destAddr, senderAddr, amount);\n\n    const value = new BigNumber(freeBalance).minus(new BigNumber(partialFee));\n    if (value.isLessThanOrEqualTo(0)) {\n      throw new Error('Did not find address with funds to recover');\n    }\n\n    const { headerNumber, headerHash } = await this.getHeaderInfo();\n    const material = await this.getMaterial();\n    const validityWindow = { firstValid: headerNumber, maxDuration: this.MAX_VALIDITY_DURATION };\n\n    const txBuilder = this.getBuilder().getTransferBuilder().material(material);\n    txBuilder\n      .sweep(false)\n      .to({ address: params.recoveryDestination })\n      .sender({ address: senderAddr })\n      .validity(validityWindow)\n      .referenceBlock(headerHash)\n      .sequenceId({ name: 'Nonce', keyword: 'Nonce', value: nonce })\n      .fee({ amount: 0, type: 'tip' });\n\n    const unsignedTransaction = (await txBuilder.build()) as Transaction;\n\n    let serializedTx = unsignedTransaction.toBroadcastFormat();\n    if (!isUnsignedSweep) {\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      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;\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      const userSigningMaterial = JSON.parse(userPrv) as EDDSAMethodTypes.UserSigningMaterial;\n\n      let backupPrv;\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\n      // add signature\n      const signatureHex = await EDDSAMethods.getTSSSignature(\n        userSigningMaterial,\n        backupSigningMaterial,\n        currPath,\n        unsignedTransaction\n      );\n\n      const substrateKeyPair = new SubstrateKeyPair({ pub: accountId });\n      txBuilder.addSignature({ pub: substrateKeyPair.getKeys().pub }, signatureHex);\n      const signedTransaction = await txBuilder.build();\n      serializedTx = signedTransaction.toBroadcastFormat();\n    } else {\n      const value = new BigNumber(freeBalance);\n      const walletCoin = this.getChain();\n      const inputs = [\n        {\n          address: unsignedTransaction.inputs[0].address,\n          valueString: value.toString(),\n          value: value.toNumber(),\n        },\n      ];\n      const outputs = [\n        {\n          address: unsignedTransaction.outputs[0].address,\n          valueString: value.toString(),\n          coinName: walletCoin,\n        },\n      ];\n      const spendAmount = value.toString();\n      const parsedTx = { inputs: inputs, outputs: outputs, spendAmount: spendAmount, type: '' };\n      const feeInfo = { fee: 0, feeString: '0' };\n      const transaction: MPCTx = {\n        serializedTx: serializedTx,\n        scanIndex: index,\n        coin: walletCoin,\n        signableHex: unsignedTransaction.signablePayload.toString('hex'),\n        derivationPath: currPath,\n        parsedTx: parsedTx,\n        feeInfo: feeInfo,\n        coinSpecific: { ...validityWindow, commonKeychain: bitgoKey },\n      };\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      const txRequests: MPCSweepTxs = { txRequests: [txRequest] };\n      return txRequests;\n    }\n\n    const transaction: MPCTx = { serializedTx: serializedTx, scanIndex: index };\n    return transaction;\n  }\n\n  /**\n   * Builds native TAO 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 = params.startingScanIndex || 1;\n    const endIdx = params.endingScanIndex || startIdx + DEFAULT_SCAN_FACTOR;\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 baseIndex = 0;\n    const basePath = params.seed ? getDerivationPath(params.seed) + `/${baseIndex}` : `m/${baseIndex}`;\n    const accountId = MPC.deriveUnhardened(bitgoKey, basePath).slice(0, 64);\n    const baseAddress = this.getAddressFromPublicKey(accountId);\n\n    const consolidationTransactions: any[] = [];\n    let lastScanIndex = startIdx;\n    for (let i = startIdx; i < endIdx; i++) {\n      const recoverParams = {\n        userKey: params.userKey,\n        backupKey: params.backupKey,\n        bitgoKey: params.bitgoKey,\n        walletPassphrase: params.walletPassphrase,\n        recoveryDestination: baseAddress,\n        seed: params.seed,\n        index: i,\n      };\n\n      let recoveryTransaction;\n      try {\n        recoveryTransaction = await this.recover(recoverParams);\n      } catch (e) {\n        if (e.message === 'Did not find address with funds to recover') {\n          lastScanIndex = i;\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);\n      }\n      lastScanIndex = i;\n    }\n\n    if (consolidationTransactions.length == 0) {\n      throw new Error('Did not find an address with funds to recover');\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      const lastTransactionCoinSpecific = {\n        firstValid:\n          consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific\n            .firstValid,\n        maxDuration:\n          consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific\n            .maxDuration,\n        commonKeychain:\n          consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific\n            .commonKeychain,\n        lastScanIndex: lastScanIndex,\n      };\n      consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific =\n        lastTransactionCoinSpecific;\n      const consolidationSweepTransactions: MPCSweepTxs = { txRequests: consolidationTransactions };\n      return consolidationSweepTransactions;\n    }\n\n    return { transactions: consolidationTransactions, lastScanIndex };\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      if (\n        !transaction.coinSpecific ||\n        !transaction.coinSpecific?.firstValid ||\n        !transaction.coinSpecific?.maxDuration\n      ) {\n        throw new Error('missing validity window');\n      }\n      const validityWindow = {\n        firstValid: transaction.coinSpecific?.firstValid,\n        maxDuration: transaction.coinSpecific?.maxDuration,\n      };\n      const material = await this.getMaterial();\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 accountId = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);\n      const senderAddr = this.getAddressFromPublicKey(accountId);\n\n      const txnBuilder = this.getBuilder()\n        .material(material)\n        .from(transaction.serializedTx as string)\n        .sender({ address: senderAddr })\n        .validity(validityWindow);\n\n      const substrateKeyPair = new SubstrateKeyPair({ pub: accountId });\n      txnBuilder.addSignature({ pub: substrateKeyPair.getKeys().pub }, signatureHex);\n      const signedTransaction = await txnBuilder.build();\n      const serializedTx = signedTransaction.toBroadcastFormat();\n\n      broadcastableTransactions.push({\n        serializedTx: serializedTx,\n        scanIndex: transaction.scanIndex,\n      });\n\n      if (i === req.length - 1 && transaction.coinSpecific!.lastScanIndex) {\n        lastScanIndex = transaction.coinSpecific!.lastScanIndex as number;\n      }\n    }\n    return { transactions: broadcastableTransactions, lastScanIndex };\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export declare const DEFAULT_SUBSTRATE_PREFIX = 42;
2
+ //# sourceMappingURL=constants.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../src/lib/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,wBAAwB,KAAK,CAAC"}
@@ -0,0 +1,5 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DEFAULT_SUBSTRATE_PREFIX = void 0;
4
+ exports.DEFAULT_SUBSTRATE_PREFIX = 42;
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpYi9jb25zdGFudHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSx3QkFBd0IsR0FBRyxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgREVGQVVMVF9TVUJTVFJBVEVfUFJFRklYID0gNDI7XG4iXX0=
@@ -0,0 +1,8 @@
1
+ import { BuildTransactionError } from '@bitgo-beta/sdk-core';
2
+ export declare class AddressValidationError extends BuildTransactionError {
3
+ constructor(malformedAddress: string);
4
+ }
5
+ export declare class InvalidFeeError extends BuildTransactionError {
6
+ constructor(type?: string, expectedType?: string);
7
+ }
8
+ //# sourceMappingURL=errors.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../../../src/lib/errors.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAE7D,qBAAa,sBAAuB,SAAQ,qBAAqB;gBACnD,gBAAgB,EAAE,MAAM;CAIrC;AAED,qBAAa,eAAgB,SAAQ,qBAAqB;gBAC5C,IAAI,CAAC,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM;CAIjD"}
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.InvalidFeeError = exports.AddressValidationError = void 0;
4
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
5
+ class AddressValidationError extends sdk_core_1.BuildTransactionError {
6
+ constructor(malformedAddress) {
7
+ super(`The address '${malformedAddress}' is not a well-formed dot address`);
8
+ this.name = AddressValidationError.name;
9
+ }
10
+ }
11
+ exports.AddressValidationError = AddressValidationError;
12
+ class InvalidFeeError extends sdk_core_1.BuildTransactionError {
13
+ constructor(type, expectedType) {
14
+ super(`The specified type: "${type}" is not valid. Please provide the type: "${expectedType}"`);
15
+ this.name = InvalidFeeError.name;
16
+ }
17
+ }
18
+ exports.InvalidFeeError = InvalidFeeError;
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpYi9lcnJvcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsbURBQTZEO0FBRTdELE1BQWEsc0JBQXVCLFNBQVEsZ0NBQXFCO0lBQy9ELFlBQVksZ0JBQXdCO1FBQ2xDLEtBQUssQ0FBQyxnQkFBZ0IsZ0JBQWdCLG9DQUFvQyxDQUFDLENBQUM7UUFDNUUsSUFBSSxDQUFDLElBQUksR0FBRyxzQkFBc0IsQ0FBQyxJQUFJLENBQUM7SUFDMUMsQ0FBQztDQUNGO0FBTEQsd0RBS0M7QUFFRCxNQUFhLGVBQWdCLFNBQVEsZ0NBQXFCO0lBQ3hELFlBQVksSUFBYSxFQUFFLFlBQXFCO1FBQzlDLEtBQUssQ0FBQyx3QkFBd0IsSUFBSSw2Q0FBNkMsWUFBWSxHQUFHLENBQUMsQ0FBQztRQUNoRyxJQUFJLENBQUMsSUFBSSxHQUFHLGVBQWUsQ0FBQyxJQUFJLENBQUM7SUFDbkMsQ0FBQztDQUNGO0FBTEQsMENBS0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBCdWlsZFRyYW5zYWN0aW9uRXJyb3IgfSBmcm9tICdAYml0Z28tYmV0YS9zZGstY29yZSc7XG5cbmV4cG9ydCBjbGFzcyBBZGRyZXNzVmFsaWRhdGlvbkVycm9yIGV4dGVuZHMgQnVpbGRUcmFuc2FjdGlvbkVycm9yIHtcbiAgY29uc3RydWN0b3IobWFsZm9ybWVkQWRkcmVzczogc3RyaW5nKSB7XG4gICAgc3VwZXIoYFRoZSBhZGRyZXNzICcke21hbGZvcm1lZEFkZHJlc3N9JyBpcyBub3QgYSB3ZWxsLWZvcm1lZCBkb3QgYWRkcmVzc2ApO1xuICAgIHRoaXMubmFtZSA9IEFkZHJlc3NWYWxpZGF0aW9uRXJyb3IubmFtZTtcbiAgfVxufVxuXG5leHBvcnQgY2xhc3MgSW52YWxpZEZlZUVycm9yIGV4dGVuZHMgQnVpbGRUcmFuc2FjdGlvbkVycm9yIHtcbiAgY29uc3RydWN0b3IodHlwZT86IHN0cmluZywgZXhwZWN0ZWRUeXBlPzogc3RyaW5nKSB7XG4gICAgc3VwZXIoYFRoZSBzcGVjaWZpZWQgdHlwZTogXCIke3R5cGV9XCIgaXMgbm90IHZhbGlkLiBQbGVhc2UgcHJvdmlkZSB0aGUgdHlwZTogXCIke2V4cGVjdGVkVHlwZX1cImApO1xuICAgIHRoaXMubmFtZSA9IEludmFsaWRGZWVFcnJvci5uYW1lO1xuICB9XG59XG4iXX0=