@bitgo-beta/abstract-substrate 1.0.1-beta.84 → 1.0.1-beta.840
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.
- package/dist/src/abstractSubstrateCoin.d.ts +42 -2
- package/dist/src/abstractSubstrateCoin.d.ts.map +1 -1
- package/dist/src/abstractSubstrateCoin.js +299 -2
- package/dist/src/lib/iface.d.ts +123 -12
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +56 -8
- package/dist/src/lib/keyPair.d.ts +7 -0
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +48 -1
- package/dist/src/lib/nativeTransferBuilder.d.ts.map +1 -1
- package/dist/src/lib/nativeTransferBuilder.js +7 -4
- package/dist/src/lib/transaction.d.ts +9 -4
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +247 -2
- package/dist/src/lib/transactionBuilder.d.ts +9 -9
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +1 -1
- package/dist/src/lib/txnSchema.d.ts +3 -0
- package/dist/src/lib/txnSchema.d.ts.map +1 -1
- package/dist/src/lib/txnSchema.js +25 -6
- package/dist/src/lib/utils.d.ts +21 -2
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +65 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +11 -6
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -51
|
@@ -1,8 +1,12 @@
|
|
|
1
|
-
import { BaseCoin, BitGoBase, KeyPair, MPCAlgorithm, ParsedTransaction, ParseTransactionOptions, SignedTransaction, VerifyAddressOptions, VerifyTransactionOptions } from '@bitgo-beta/sdk-core';
|
|
1
|
+
import { AuditDecryptedKeyParams, BaseCoin, BitGoBase, KeyPair, MPCAlgorithm, MPCConsolidationRecoveryOptions, MPCRecoveryOptions, MPCSweepRecoveryOptions, MPCSweepTxs, MPCTx, MPCTxs, MultisigType, ParsedTransaction, ParseTransactionOptions, SignedTransaction, VerifyAddressOptions, VerifyTransactionOptions } from '@bitgo-beta/sdk-core';
|
|
2
2
|
import { CoinFamily, BaseCoin as StaticsBaseCoin } from '@bitgo-beta/statics';
|
|
3
|
-
import { SignTransactionOptions, VerifiedTransactionParameters } from './lib/iface';
|
|
3
|
+
import { SignTransactionOptions, VerifiedTransactionParameters, Material } from './lib/iface';
|
|
4
|
+
import { ApiPromise } from '@polkadot/api';
|
|
5
|
+
export declare const DEFAULT_SCAN_FACTOR = 20;
|
|
4
6
|
export declare class SubstrateCoin extends BaseCoin {
|
|
5
7
|
protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;
|
|
8
|
+
readonly MAX_VALIDITY_DURATION = 2400;
|
|
9
|
+
readonly SWEEP_TXN_DURATION = 64;
|
|
6
10
|
protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>);
|
|
7
11
|
/**
|
|
8
12
|
* Creates an instance of TransactionBuilderFactory for the coin specific sdk
|
|
@@ -18,6 +22,8 @@ export declare class SubstrateCoin extends BaseCoin {
|
|
|
18
22
|
getFullName(): string;
|
|
19
23
|
/** @inheritDoc */
|
|
20
24
|
supportsTss(): boolean;
|
|
25
|
+
/** inherited doc */
|
|
26
|
+
getDefaultMultisigType(): MultisigType;
|
|
21
27
|
/** @inheritDoc **/
|
|
22
28
|
getMPCAlgorithm(): MPCAlgorithm;
|
|
23
29
|
/** @inheritDoc **/
|
|
@@ -51,5 +57,39 @@ export declare class SubstrateCoin extends BaseCoin {
|
|
|
51
57
|
* @throws {Error} If the method is not implemented by the subclass.
|
|
52
58
|
*/
|
|
53
59
|
protected getMaxValidityDurationBlocks(): number;
|
|
60
|
+
protected getAddressFromPublicKey(Pubkey: string): string;
|
|
61
|
+
protected getInitializedNodeAPI(): Promise<ApiPromise>;
|
|
62
|
+
protected getAccountInfo(walletAddr: string): Promise<{
|
|
63
|
+
nonce: number;
|
|
64
|
+
freeBalance: number;
|
|
65
|
+
}>;
|
|
66
|
+
protected getFee(destAddr: string, srcAddr: string, amount: number): Promise<number>;
|
|
67
|
+
protected getHeaderInfo(): Promise<{
|
|
68
|
+
headerNumber: number;
|
|
69
|
+
headerHash: string;
|
|
70
|
+
}>;
|
|
71
|
+
protected getMaterial(): Promise<Material>;
|
|
72
|
+
/**
|
|
73
|
+
* Builds a funds recovery transaction without BitGo
|
|
74
|
+
* @param {MPCRecoveryOptions} params parameters needed to construct and
|
|
75
|
+
* (maybe) sign the transaction
|
|
76
|
+
*
|
|
77
|
+
* @returns {MPCTx} the serialized transaction hex string and index
|
|
78
|
+
* of the address being swept
|
|
79
|
+
*/
|
|
80
|
+
recover(params: MPCRecoveryOptions): Promise<MPCTx | MPCSweepTxs>;
|
|
81
|
+
/**
|
|
82
|
+
* Builds native TAO recoveries of receive addresses in batch without BitGo.
|
|
83
|
+
* Funds will be recovered to base address first. You need to initiate another sweep txn after that.
|
|
84
|
+
*
|
|
85
|
+
* @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.
|
|
86
|
+
* @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).
|
|
87
|
+
* @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).
|
|
88
|
+
*/
|
|
89
|
+
recoverConsolidations(params: MPCConsolidationRecoveryOptions): Promise<MPCTxs | MPCSweepTxs>;
|
|
90
|
+
/** inherited doc */
|
|
91
|
+
createBroadcastableSweepTransaction(params: MPCSweepRecoveryOptions): Promise<MPCTxs>;
|
|
92
|
+
/** inherited doc */
|
|
93
|
+
auditDecryptedKey({ publicKey, prv, multiSigType }: AuditDecryptedKeyParams): void;
|
|
54
94
|
}
|
|
55
95
|
//# sourceMappingURL=abstractSubstrateCoin.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"abstractSubstrateCoin.d.ts","sourceRoot":"","sources":["../../src/abstractSubstrateCoin.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,QAAQ,EACR,SAAS,
|
|
1
|
+
{"version":3,"file":"abstractSubstrateCoin.d.ts","sourceRoot":"","sources":["../../src/abstractSubstrateCoin.ts"],"names":[],"mappings":"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"}
|
|
@@ -3,14 +3,19 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
3
3
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.SubstrateCoin = void 0;
|
|
6
|
+
exports.SubstrateCoin = exports.DEFAULT_SCAN_FACTOR = void 0;
|
|
7
7
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
8
8
|
const lib_1 = require("./lib");
|
|
9
9
|
const constants_1 = require("./lib/constants");
|
|
10
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;
|
|
11
14
|
class SubstrateCoin extends sdk_core_1.BaseCoin {
|
|
12
15
|
constructor(bitgo, staticsCoin) {
|
|
13
16
|
super(bitgo);
|
|
17
|
+
this.MAX_VALIDITY_DURATION = 2400;
|
|
18
|
+
this.SWEEP_TXN_DURATION = 64;
|
|
14
19
|
if (!staticsCoin) {
|
|
15
20
|
throw new Error('missing required constructor parameter staticsCoin');
|
|
16
21
|
}
|
|
@@ -42,6 +47,10 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
|
|
|
42
47
|
supportsTss() {
|
|
43
48
|
return true;
|
|
44
49
|
}
|
|
50
|
+
/** inherited doc */
|
|
51
|
+
getDefaultMultisigType() {
|
|
52
|
+
return sdk_core_1.multisigTypes.tss;
|
|
53
|
+
}
|
|
45
54
|
/** @inheritDoc **/
|
|
46
55
|
getMPCAlgorithm() {
|
|
47
56
|
return 'eddsa';
|
|
@@ -133,6 +142,294 @@ class SubstrateCoin extends sdk_core_1.BaseCoin {
|
|
|
133
142
|
getMaxValidityDurationBlocks() {
|
|
134
143
|
throw new Error('Method not implemented.');
|
|
135
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
|
+
}
|
|
426
|
+
/** inherited doc */
|
|
427
|
+
auditDecryptedKey({ publicKey, prv, multiSigType }) {
|
|
428
|
+
if (multiSigType !== 'tss') {
|
|
429
|
+
throw new Error('Unsupported multiSigType');
|
|
430
|
+
}
|
|
431
|
+
(0, sdk_lib_mpc_1.auditEddsaPrivateKey)(prv, publicKey ?? '');
|
|
432
|
+
}
|
|
136
433
|
}
|
|
137
434
|
exports.SubstrateCoin = SubstrateCoin;
|
|
138
|
-
//# 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  KeyPair,\n  MethodNotImplementedError,\n  MPCAlgorithm,\n  ParsedTransaction,\n  ParseTransactionOptions,\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 } from './lib';\nimport { DEFAULT_SUBSTRATE_PREFIX } from './lib/constants';\nimport { SignTransactionOptions, VerifiedTransactionParameters } from './lib/iface';\nimport utils from './lib/utils';\n\nexport class SubstrateCoin extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\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  /** @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"]}
|
|
435
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractSubstrateCoin.js","sourceRoot":"","sources":["../../src/abstractSubstrateCoin.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AuditDecryptedKeyParams,\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 { auditEddsaPrivateKey, 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  /** inherited doc */\n  auditDecryptedKey({ publicKey, prv, multiSigType }: AuditDecryptedKeyParams) {\n    if (multiSigType !== 'tss') {\n      throw new Error('Unsupported multiSigType');\n    }\n    auditEddsaPrivateKey(prv, publicKey ?? '');\n  }\n}\n"]}
|