@bitgo-beta/abstract-utxo 1.6.1-alpha.21 → 1.6.1-alpha.211
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/CHANGELOG.md +861 -0
- package/dist/src/abstractUtxoCoin.d.ts +147 -34
- package/dist/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/src/abstractUtxoCoin.js +347 -187
- package/dist/src/descriptor/NamedDescriptor.d.ts +7 -0
- package/dist/src/descriptor/NamedDescriptor.d.ts.map +1 -0
- package/dist/src/descriptor/NamedDescriptor.js +9 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.d.ts +4 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.d.ts.map +1 -0
- package/dist/src/descriptor/assertDescriptorWalletAddress.js +37 -0
- package/dist/src/descriptor/index.d.ts +4 -0
- package/dist/src/descriptor/index.d.ts.map +1 -0
- package/dist/src/descriptor/index.js +11 -0
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +8 -2
- package/dist/src/parseOutput.d.ts.map +1 -1
- package/dist/src/parseOutput.js +38 -1
- package/dist/src/recovery/RecoveryProvider.d.ts +1 -1
- package/dist/src/recovery/RecoveryProvider.d.ts.map +1 -1
- package/dist/src/recovery/RecoveryProvider.js +1 -2
- package/dist/src/recovery/backupKeyRecovery.d.ts +39 -11
- package/dist/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/src/recovery/backupKeyRecovery.js +124 -83
- package/dist/src/recovery/baseApi.d.ts +2 -2
- package/dist/src/recovery/baseApi.d.ts.map +1 -1
- package/dist/src/recovery/crossChainRecovery.d.ts +14 -3
- package/dist/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/src/recovery/crossChainRecovery.js +55 -11
- package/dist/src/recovery/index.d.ts +0 -1
- package/dist/src/recovery/index.d.ts.map +1 -1
- package/dist/src/recovery/index.js +6 -3
- package/dist/src/recovery/mempoolApi.d.ts.map +1 -1
- package/dist/src/recovery/mempoolApi.js +6 -3
- package/dist/src/sign.d.ts +29 -5
- package/dist/src/sign.d.ts.map +1 -1
- package/dist/src/sign.js +73 -7
- package/dist/src/transaction.d.ts +36 -0
- package/dist/src/transaction.d.ts.map +1 -0
- package/dist/src/transaction.js +278 -0
- package/dist/tsconfig.tsbuildinfo +1 -7797
- package/package.json +12 -10
- package/dist/src/recovery/smartbitApi.d.ts +0 -11
- package/dist/src/recovery/smartbitApi.d.ts.map +0 -1
- package/dist/src/recovery/smartbitApi.js +0 -36
|
@@ -1,17 +1,17 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.AbstractUtxoCoin = exports.AbstractUtxoCoinWallet = void 0;
|
|
3
|
+
exports.AbstractUtxoCoin = exports.AbstractUtxoCoinWallet = exports.isWalletOutput = void 0;
|
|
4
4
|
/**
|
|
5
5
|
* @prettier
|
|
6
6
|
*/
|
|
7
7
|
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
8
8
|
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
9
|
+
const assert = require("assert");
|
|
9
10
|
const bitcoinMessage = require("bitcoinjs-message");
|
|
10
11
|
const crypto_1 = require("crypto");
|
|
11
12
|
const debugLib = require("debug");
|
|
12
13
|
const _ = require("lodash");
|
|
13
14
|
const bignumber_js_1 = require("bignumber.js");
|
|
14
|
-
const backupKeyRecovery_1 = require("./recovery/backupKeyRecovery");
|
|
15
15
|
const recovery_1 = require("./recovery");
|
|
16
16
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
17
17
|
const parseOutput_1 = require("./parseOutput");
|
|
@@ -19,7 +19,13 @@ const debug = debugLib('bitgo:v2:utxo');
|
|
|
19
19
|
const replayProtection_1 = require("./replayProtection");
|
|
20
20
|
const sign_1 = require("./sign");
|
|
21
21
|
const config_1 = require("./config");
|
|
22
|
-
const
|
|
22
|
+
const transaction_1 = require("./transaction");
|
|
23
|
+
const assertDescriptorWalletAddress_1 = require("./descriptor/assertDescriptorWalletAddress");
|
|
24
|
+
const { getExternalChainCode, isChainCode, scriptTypeForChain, outputScripts } = utxo_lib_1.bitgo;
|
|
25
|
+
function isWalletOutput(output) {
|
|
26
|
+
return output.chain !== undefined && output.index !== undefined;
|
|
27
|
+
}
|
|
28
|
+
exports.isWalletOutput = isWalletOutput;
|
|
23
29
|
class AbstractUtxoCoinWallet extends sdk_core_1.Wallet {
|
|
24
30
|
constructor(bitgo, baseCoin, walletData) {
|
|
25
31
|
super(bitgo, baseCoin, walletData);
|
|
@@ -70,8 +76,8 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
70
76
|
}
|
|
71
77
|
const formats = param && param.anyFormat ? undefined : ['default'];
|
|
72
78
|
try {
|
|
73
|
-
utxolib.addressFormat.toOutputScriptTryFormats(address, this.network, formats);
|
|
74
|
-
return
|
|
79
|
+
const script = utxolib.addressFormat.toOutputScriptTryFormats(address, this.network, formats);
|
|
80
|
+
return address === utxolib.address.fromOutputScript(script, this.network);
|
|
75
81
|
}
|
|
76
82
|
catch (e) {
|
|
77
83
|
return false;
|
|
@@ -110,22 +116,23 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
110
116
|
if (_.isUndefined(prebuild.txHex)) {
|
|
111
117
|
throw new Error('missing required txPrebuild property txHex');
|
|
112
118
|
}
|
|
113
|
-
const
|
|
119
|
+
const tx = utxo_lib_1.bitgo.isPsbt(prebuild.txHex)
|
|
120
|
+
? utxo_lib_1.bitgo.createPsbtFromHex(prebuild.txHex, this.network)
|
|
121
|
+
: this.createTransactionFromHex(prebuild.txHex);
|
|
114
122
|
if (_.isUndefined(prebuild.blockHeight)) {
|
|
115
123
|
prebuild.blockHeight = (await this.getLatestBlockHeight());
|
|
116
124
|
}
|
|
117
|
-
return _.extend({}, prebuild, { txHex:
|
|
125
|
+
return _.extend({}, prebuild, { txHex: tx.toHex() });
|
|
118
126
|
}
|
|
119
127
|
/**
|
|
120
|
-
*
|
|
121
|
-
* @param
|
|
122
|
-
* @
|
|
123
|
-
* @returns {Array}
|
|
128
|
+
* @param first
|
|
129
|
+
* @param second
|
|
130
|
+
* @returns {Array} All outputs that are in the first array but not in the second
|
|
124
131
|
*/
|
|
125
|
-
static
|
|
132
|
+
static outputDifference(first, second) {
|
|
126
133
|
const keyFunc = ({ address, amount }) => `${address}:${amount}`;
|
|
127
|
-
const groupedOutputs = _.groupBy(
|
|
128
|
-
|
|
134
|
+
const groupedOutputs = _.groupBy(first, keyFunc);
|
|
135
|
+
second.forEach((output) => {
|
|
129
136
|
const group = groupedOutputs[keyFunc(output)];
|
|
130
137
|
if (group) {
|
|
131
138
|
group.pop();
|
|
@@ -155,7 +162,7 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
155
162
|
}
|
|
156
163
|
const disableNetworking = verification.disableNetworking;
|
|
157
164
|
const fetchKeychains = async (wallet) => {
|
|
158
|
-
return sdk_core_1.promiseProps({
|
|
165
|
+
return (0, sdk_core_1.promiseProps)({
|
|
159
166
|
user: this.keychains().get({ id: wallet.keyIds()[sdk_core_1.KeyIndices.USER], reqId }),
|
|
160
167
|
backup: this.keychains().get({ id: wallet.keyIds()[sdk_core_1.KeyIndices.BACKUP], reqId }),
|
|
161
168
|
bitgo: this.keychains().get({ id: wallet.keyIds()[sdk_core_1.KeyIndices.BITGO], reqId }),
|
|
@@ -184,11 +191,32 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
184
191
|
pubs: keychainArray.map((k) => k.pub),
|
|
185
192
|
});
|
|
186
193
|
const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
194
|
+
let expectedOutputs;
|
|
195
|
+
if (txParams.rbfTxIds) {
|
|
196
|
+
assert(txParams.rbfTxIds.length === 1);
|
|
197
|
+
const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
|
|
198
|
+
expectedOutputs = txToBeReplaced.outputs
|
|
199
|
+
.filter((output) => output.wallet !== wallet.id()) // For self-sends, the walletId will be the same as the wallet's id
|
|
200
|
+
.map((output) => {
|
|
201
|
+
return { amount: BigInt(output.valueString), address: this.canonicalAddress(output.address) };
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
// verify that each recipient from txParams has their own output
|
|
206
|
+
expectedOutputs = _.get(txParams, 'recipients', []).map((output) => {
|
|
207
|
+
return { ...output, address: this.canonicalAddress(output.address) };
|
|
208
|
+
});
|
|
209
|
+
if (params.txParams.allowExternalChangeAddress && params.txParams.changeAddress) {
|
|
210
|
+
// when an external change address is explicitly specified, count all outputs going towards that
|
|
211
|
+
// address in the expected outputs (regardless of the output amount)
|
|
212
|
+
expectedOutputs.push(...allOutputs
|
|
213
|
+
.map((output) => {
|
|
214
|
+
return { ...output, address: this.canonicalAddress(output.address) };
|
|
215
|
+
})
|
|
216
|
+
.filter((output) => output.address === this.canonicalAddress(params.txParams.changeAddress)));
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
const missingOutputs = AbstractUtxoCoin.outputDifference(expectedOutputs, allOutputs);
|
|
192
220
|
// get the keychains from the custom change wallet if needed
|
|
193
221
|
let customChange;
|
|
194
222
|
const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};
|
|
@@ -222,7 +250,7 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
222
250
|
* or external spends by setting the "external" property to true or false on the output object.
|
|
223
251
|
*/
|
|
224
252
|
const allOutputDetails = await Promise.all(allOutputs.map((currentOutput) => {
|
|
225
|
-
return parseOutput_1.parseOutput({
|
|
253
|
+
return (0, parseOutput_1.parseOutput)({
|
|
226
254
|
currentOutput,
|
|
227
255
|
coin: this,
|
|
228
256
|
txPrebuild,
|
|
@@ -234,11 +262,12 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
234
262
|
reqId,
|
|
235
263
|
});
|
|
236
264
|
}));
|
|
237
|
-
const needsCustomChangeKeySignatureVerification = allOutputDetails.some((output) => output.needsCustomChangeKeySignatureVerification);
|
|
265
|
+
const needsCustomChangeKeySignatureVerification = allOutputDetails.some((output) => output === null || output === void 0 ? void 0 : output.needsCustomChangeKeySignatureVerification);
|
|
238
266
|
const changeOutputs = _.filter(allOutputDetails, { external: false });
|
|
239
267
|
// these are all the outputs that were not originally explicitly specified in recipients
|
|
240
|
-
|
|
241
|
-
const
|
|
268
|
+
// ideally change outputs or a paygo output that might have been added
|
|
269
|
+
const implicitOutputs = AbstractUtxoCoin.outputDifference(allOutputDetails, expectedOutputs);
|
|
270
|
+
const explicitOutputs = AbstractUtxoCoin.outputDifference(allOutputDetails, implicitOutputs);
|
|
242
271
|
// these are all the non-wallet outputs that had been originally explicitly specified in recipients
|
|
243
272
|
const explicitExternalOutputs = _.filter(explicitOutputs, { external: true });
|
|
244
273
|
// this is the sum of all the originally explicitly specified non-wallet output values
|
|
@@ -281,17 +310,10 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
281
310
|
throw new Error('user keychain is required');
|
|
282
311
|
}
|
|
283
312
|
const userPub = userKeychain.pub;
|
|
284
|
-
// decrypt the user private key so we can verify that the claimed public key is a match
|
|
313
|
+
// decrypt the user private key, so we can verify that the claimed public key is a match
|
|
285
314
|
let userPrv = userKeychain.prv;
|
|
286
|
-
if (
|
|
287
|
-
|
|
288
|
-
if (encryptedPrv && !_.isEmpty(encryptedPrv)) {
|
|
289
|
-
// if the decryption fails, it will throw an error
|
|
290
|
-
userPrv = this.bitgo.decrypt({
|
|
291
|
-
input: encryptedPrv,
|
|
292
|
-
password: txParams.walletPassphrase,
|
|
293
|
-
});
|
|
294
|
-
}
|
|
315
|
+
if (!userPrv && txParams.walletPassphrase) {
|
|
316
|
+
userPrv = (0, sdk_core_1.decryptKeychainPrivateKey)(this.bitgo, userKeychain, txParams.walletPassphrase);
|
|
295
317
|
}
|
|
296
318
|
if (!userPrv) {
|
|
297
319
|
const errorMessage = 'user private key unavailable for verification';
|
|
@@ -334,10 +356,19 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
334
356
|
throw new Error('key signature is required');
|
|
335
357
|
}
|
|
336
358
|
// verify the signature against the user public key
|
|
359
|
+
assert(userKeychain.pub);
|
|
337
360
|
const publicKey = utxo_lib_1.bip32.fromBase58(userKeychain.pub).publicKey;
|
|
338
|
-
|
|
361
|
+
// Due to interface of `bitcoinMessage`, we need to convert the public key to an address.
|
|
362
|
+
// Note that this address has no relationship to on-chain transactions. We are
|
|
363
|
+
// only interested in the address as a representation of the public key.
|
|
364
|
+
const signingAddress = utxolib.address.toBase58Check(utxolib.crypto.hash160(publicKey), utxolib.networks.bitcoin.pubKeyHash,
|
|
365
|
+
// we do not pass `this.network` here because it would fail for zcash
|
|
366
|
+
// the bitcoinMessage library decodes the address and throws away the first byte
|
|
367
|
+
// because zcash has a two-byte prefix, verify() decodes zcash addresses to an invalid pubkey hash
|
|
368
|
+
utxolib.networks.bitcoin);
|
|
339
369
|
// BG-5703: use BTC mainnet prefix for all key signature operations
|
|
340
370
|
// (this means do not pass a prefix parameter, and let it use the default prefix instead)
|
|
371
|
+
assert(keychainToVerify.pub);
|
|
341
372
|
try {
|
|
342
373
|
return bitcoinMessage.verify(keychainToVerify.pub, signingAddress, Buffer.from(keySignature, 'hex'));
|
|
343
374
|
}
|
|
@@ -369,7 +400,11 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
369
400
|
if (!keySignature) {
|
|
370
401
|
throw new Error(`missing required custom change ${sdk_core_1.KeyIndices[keyIndex].toLowerCase()} keychain signature`);
|
|
371
402
|
}
|
|
372
|
-
if (!this.verifyKeySignature({
|
|
403
|
+
if (!this.verifyKeySignature({
|
|
404
|
+
userKeychain: userKeychain,
|
|
405
|
+
keychainToVerify: keychainToVerify,
|
|
406
|
+
keySignature,
|
|
407
|
+
})) {
|
|
373
408
|
debug('failed to verify custom change %s key signature!', sdk_core_1.KeyIndices[keyIndex].toLowerCase());
|
|
374
409
|
return false;
|
|
375
410
|
}
|
|
@@ -405,7 +440,12 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
405
440
|
* @returns {boolean}
|
|
406
441
|
*/
|
|
407
442
|
async verifyTransaction(params) {
|
|
443
|
+
var _a;
|
|
408
444
|
const { txParams, txPrebuild, wallet, verification = { allowPaygoOutput: true }, reqId } = params;
|
|
445
|
+
const isPsbt = txPrebuild.txHex && utxo_lib_1.bitgo.isPsbt(txPrebuild.txHex);
|
|
446
|
+
if (isPsbt && ((_a = txPrebuild.txInfo) === null || _a === void 0 ? void 0 : _a.unspents)) {
|
|
447
|
+
throw new Error('should not have unspents in txInfo for psbt');
|
|
448
|
+
}
|
|
409
449
|
const disableNetworking = !!verification.disableNetworking;
|
|
410
450
|
const parsedTransaction = await this.parseTransaction({
|
|
411
451
|
txParams,
|
|
@@ -427,7 +467,16 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
427
467
|
// let's verify these keychains
|
|
428
468
|
const keySignatures = parsedTransaction.keySignatures;
|
|
429
469
|
if (!_.isEmpty(keySignatures)) {
|
|
430
|
-
const verify = (key, pub) =>
|
|
470
|
+
const verify = (key, pub) => {
|
|
471
|
+
if (!keychains.user || !keychains.user.pub) {
|
|
472
|
+
throw new Error('missing user keychain');
|
|
473
|
+
}
|
|
474
|
+
return this.verifyKeySignature({
|
|
475
|
+
userKeychain: keychains.user,
|
|
476
|
+
keychainToVerify: key,
|
|
477
|
+
keySignature: pub,
|
|
478
|
+
});
|
|
479
|
+
};
|
|
431
480
|
const isBackupKeySignatureValid = verify(keychains.backup, keySignatures.backupPub);
|
|
432
481
|
const isBitgoKeySignatureValid = verify(keychains.bitgo, keySignatures.bitgoPub);
|
|
433
482
|
if (!isBackupKeySignatureValid || !isBitgoKeySignatureValid) {
|
|
@@ -469,46 +518,31 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
469
518
|
// get all the additional external outputs the server added and calculate their values
|
|
470
519
|
const nonChangeAmount = new bignumber_js_1.default(parsedTransaction.implicitExternalSpendAmount.toString());
|
|
471
520
|
debug('Intended spend is %s, Non-change amount is %s, paygo limit is %s', intendedExternalSpend.toString(), nonChangeAmount.toString(), payAsYouGoLimit.toString());
|
|
472
|
-
//
|
|
521
|
+
// There are two instances where we will get into this point here
|
|
473
522
|
if (nonChangeAmount.gt(payAsYouGoLimit)) {
|
|
474
|
-
|
|
475
|
-
|
|
523
|
+
if (isPsbt && parsedTransaction.customChange) {
|
|
524
|
+
// In the case that we have a custom change address on a wallet and we are building the transaction
|
|
525
|
+
// with a PSBT, we do not have the metadata to verify the address from the custom change wallet, nor
|
|
526
|
+
// can we fetch that information from the other wallet because we may not have the credentials. Therefore,
|
|
527
|
+
// we will not throw an error here, but we will log a warning.
|
|
528
|
+
debug(`cannot verify some of the addresses because it belongs to a separate wallet`);
|
|
529
|
+
}
|
|
530
|
+
else {
|
|
531
|
+
// the additional external outputs can only be BitGo's pay-as-you-go fee, but we cannot verify the wallet address
|
|
532
|
+
// there are some addresses that are outside the scope of intended recipients that are not change addresses
|
|
533
|
+
throw new Error('prebuild attempts to spend to unintended external recipients');
|
|
534
|
+
}
|
|
476
535
|
}
|
|
477
536
|
const allOutputs = parsedTransaction.outputs;
|
|
478
537
|
if (!txPrebuild.txHex) {
|
|
479
538
|
throw new Error(`txPrebuild.txHex not set`);
|
|
480
539
|
}
|
|
481
|
-
const
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
if (txHex) {
|
|
488
|
-
const localTx = this.createTransactionFromHex(txHex);
|
|
489
|
-
if (localTx.getId() !== transactionId) {
|
|
490
|
-
throw new Error('input transaction hex does not match id');
|
|
491
|
-
}
|
|
492
|
-
const currentOutput = localTx.outs[currentInput.index];
|
|
493
|
-
const address = utxolib.address.fromOutputScript(currentOutput.script, this.network);
|
|
494
|
-
return {
|
|
495
|
-
address,
|
|
496
|
-
value: currentOutput.value,
|
|
497
|
-
valueString: currentOutput.value.toString(),
|
|
498
|
-
};
|
|
499
|
-
}
|
|
500
|
-
else if (!transactionCache[transactionId]) {
|
|
501
|
-
if (disableNetworking) {
|
|
502
|
-
throw new Error('attempting to retrieve transaction details externally with networking disabled');
|
|
503
|
-
}
|
|
504
|
-
if (reqId) {
|
|
505
|
-
this.bitgo.setRequestTracer(reqId);
|
|
506
|
-
}
|
|
507
|
-
transactionCache[transactionId] = await this.bitgo.get(this.url(`/public/tx/${transactionId}`)).result();
|
|
508
|
-
}
|
|
509
|
-
const transactionDetails = transactionCache[transactionId];
|
|
510
|
-
return transactionDetails.outputs[currentInput.index];
|
|
511
|
-
}));
|
|
540
|
+
const inputs = isPsbt
|
|
541
|
+
? (0, transaction_1.getPsbtTxInputs)(txPrebuild.txHex, this.network).map((v) => ({
|
|
542
|
+
...v,
|
|
543
|
+
value: utxo_lib_1.bitgo.toTNumber(v.value, this.amountType),
|
|
544
|
+
}))
|
|
545
|
+
: await (0, transaction_1.getTxInputs)({ txPrebuild, bitgo: this.bitgo, coin: this, disableNetworking, reqId });
|
|
512
546
|
// coins (doge) that can exceed number limits (and thus will use bigint) will have the `valueString` field
|
|
513
547
|
const inputAmount = inputs.reduce((sum, i) => sum + BigInt(this.amountType === 'bigint' ? i.valueString : i.value), BigInt(0));
|
|
514
548
|
const outputAmount = allOutputs.reduce((sum, o) => sum + BigInt(o.amount), BigInt(0));
|
|
@@ -530,17 +564,21 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
530
564
|
* @throws {InvalidAddressDerivationPropertyError}
|
|
531
565
|
* @throws {UnexpectedAddressError}
|
|
532
566
|
*/
|
|
533
|
-
async isWalletAddress(params) {
|
|
534
|
-
const { address, addressType, keychains,
|
|
567
|
+
async isWalletAddress(params, wallet) {
|
|
568
|
+
const { address, addressType, keychains, chain, index } = params;
|
|
535
569
|
if (!this.isValidAddress(address)) {
|
|
536
570
|
throw new sdk_core_1.InvalidAddressError(`invalid address: ${address}`);
|
|
537
571
|
}
|
|
572
|
+
if (wallet) {
|
|
573
|
+
const walletCoinSpecific = wallet.coinSpecific();
|
|
574
|
+
if (walletCoinSpecific && 'descriptors' in walletCoinSpecific) {
|
|
575
|
+
(0, assertDescriptorWalletAddress_1.assertDescriptorWalletAddress)(this.network, params, walletCoinSpecific.descriptors);
|
|
576
|
+
return true;
|
|
577
|
+
}
|
|
578
|
+
}
|
|
538
579
|
if ((_.isUndefined(chain) && _.isUndefined(index)) || !(_.isFinite(chain) && _.isFinite(index))) {
|
|
539
580
|
throw new sdk_core_1.InvalidAddressDerivationPropertyError(`address validation failure: invalid chain (${chain}) or index (${index})`);
|
|
540
581
|
}
|
|
541
|
-
if (!_.isObject(coinSpecific)) {
|
|
542
|
-
throw new sdk_core_1.InvalidAddressVerificationObjectPropertyError('address validation failure: coinSpecific field must be an object');
|
|
543
|
-
}
|
|
544
582
|
if (!keychains) {
|
|
545
583
|
throw new Error('missing required param keychains');
|
|
546
584
|
}
|
|
@@ -594,7 +632,11 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
594
632
|
* @returns {{chain: number, index: number, coin: number, coinSpecific: {outputScript, redeemScript}}}
|
|
595
633
|
*/
|
|
596
634
|
generateAddress(params) {
|
|
597
|
-
|
|
635
|
+
let derivationIndex = 0;
|
|
636
|
+
if (_.isInteger(params.index) && params.index > 0) {
|
|
637
|
+
derivationIndex = params.index;
|
|
638
|
+
}
|
|
639
|
+
const { keychains, threshold, chain, segwit = false, bech32 = false } = params;
|
|
598
640
|
let derivationChain = getExternalChainCode('p2sh');
|
|
599
641
|
if (_.isNumber(chain) && _.isInteger(chain) && isChainCode(chain)) {
|
|
600
642
|
derivationChain = chain;
|
|
@@ -627,6 +669,8 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
627
669
|
throw new sdk_core_1.P2wshUnsupportedError();
|
|
628
670
|
case 'p2tr':
|
|
629
671
|
throw new sdk_core_1.P2trUnsupportedError();
|
|
672
|
+
case 'p2trMusig2':
|
|
673
|
+
throw new sdk_core_1.P2trMusig2UnsupportedError();
|
|
630
674
|
default:
|
|
631
675
|
throw new sdk_core_1.UnsupportedAddressTypeError();
|
|
632
676
|
}
|
|
@@ -641,13 +685,9 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
641
685
|
throw new Error('threshold cannot exceed number of keys');
|
|
642
686
|
}
|
|
643
687
|
}
|
|
644
|
-
let derivationIndex = 0;
|
|
645
|
-
if (_.isInteger(index) && index > 0) {
|
|
646
|
-
derivationIndex = index;
|
|
647
|
-
}
|
|
648
688
|
const path = '0/0/' + derivationChain + '/' + derivationIndex;
|
|
649
689
|
const hdNodes = keychains.map(({ pub }) => utxo_lib_1.bip32.fromBase58(pub));
|
|
650
|
-
const derivedKeys = hdNodes.map((hdNode) => hdNode.derivePath(sdk_core_1.sanitizeLegacyPath(path)).publicKey);
|
|
690
|
+
const derivedKeys = hdNodes.map((hdNode) => hdNode.derivePath((0, sdk_core_1.sanitizeLegacyPath)(path)).publicKey);
|
|
651
691
|
const { outputScript, redeemScript, witnessScript, address } = this.createMultiSigAddress(addressType, signatureThreshold, derivedKeys);
|
|
652
692
|
return {
|
|
653
693
|
address,
|
|
@@ -662,58 +702,194 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
662
702
|
addressType,
|
|
663
703
|
};
|
|
664
704
|
}
|
|
705
|
+
/**
|
|
706
|
+
* @returns input psbt added with deterministic MuSig2 nonce for bitgo key for each MuSig2 inputs.
|
|
707
|
+
* @param psbtHex all MuSig2 inputs should contain user MuSig2 nonce
|
|
708
|
+
* @param walletId
|
|
709
|
+
*/
|
|
710
|
+
async signPsbt(psbtHex, walletId) {
|
|
711
|
+
const params = { psbt: psbtHex };
|
|
712
|
+
return await this.bitgo
|
|
713
|
+
.post(this.url('/wallet/' + walletId + '/tx/signpsbt'))
|
|
714
|
+
.send(params)
|
|
715
|
+
.result();
|
|
716
|
+
}
|
|
717
|
+
/**
|
|
718
|
+
* @returns input psbt added with deterministic MuSig2 nonce for bitgo key for each MuSig2 inputs from OVC.
|
|
719
|
+
* @param ovcJson JSON object provided by OVC with fields psbtHex and walletId
|
|
720
|
+
*/
|
|
721
|
+
async signPsbtFromOVC(ovcJson) {
|
|
722
|
+
assert(ovcJson['psbtHex'], 'ovcJson must contain psbtHex');
|
|
723
|
+
assert(ovcJson['walletId'], 'ovcJson must contain walletId');
|
|
724
|
+
const psbt = (await this.signPsbt(ovcJson['psbtHex'], ovcJson['walletId'])).psbt;
|
|
725
|
+
assert(psbt, 'psbt not found');
|
|
726
|
+
return _.extend(ovcJson, { txHex: psbt });
|
|
727
|
+
}
|
|
665
728
|
/**
|
|
666
729
|
* Assemble keychain and half-sign prebuilt transaction
|
|
667
730
|
* @param params - {@see SignTransactionOptions}
|
|
668
731
|
* @returns {Promise<SignedTransaction | HalfSignedUtxoTransaction>}
|
|
669
732
|
*/
|
|
670
733
|
async signTransaction(params) {
|
|
671
|
-
var _a;
|
|
734
|
+
var _a, _b, _c;
|
|
672
735
|
const txPrebuild = params.txPrebuild;
|
|
673
|
-
const userPrv = params.prv;
|
|
674
736
|
if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {
|
|
675
737
|
if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {
|
|
676
738
|
throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);
|
|
677
739
|
}
|
|
678
740
|
throw new Error('missing txPrebuild parameter');
|
|
679
741
|
}
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
742
|
+
let tx = utxo_lib_1.bitgo.isPsbt(txPrebuild.txHex)
|
|
743
|
+
? utxo_lib_1.bitgo.createPsbtFromHex(txPrebuild.txHex, this.network)
|
|
744
|
+
: this.createTransactionFromHex(txPrebuild.txHex);
|
|
745
|
+
const isTxWithKeyPathSpendInput = tx instanceof utxo_lib_1.bitgo.UtxoPsbt && utxo_lib_1.bitgo.isTransactionWithKeyPathSpendInput(tx);
|
|
684
746
|
let isLastSignature = false;
|
|
685
747
|
if (_.isBoolean(params.isLastSignature)) {
|
|
748
|
+
// We can only be the first signature on a transaction with taproot key path spend inputs because
|
|
749
|
+
// we require the secret nonce in the cache of the first signer, which is impossible to retrieve if
|
|
750
|
+
// deserialized from a hex.
|
|
751
|
+
if (params.isLastSignature && isTxWithKeyPathSpendInput) {
|
|
752
|
+
throw new Error('Cannot be last signature on a transaction with key path spend inputs');
|
|
753
|
+
}
|
|
686
754
|
// if build is called instead of buildIncomplete, no signature placeholders are left in the sig script
|
|
687
755
|
isLastSignature = params.isLastSignature;
|
|
688
756
|
}
|
|
689
|
-
|
|
690
|
-
|
|
691
|
-
|
|
757
|
+
const getSignerKeychain = () => {
|
|
758
|
+
const userPrv = params.prv;
|
|
759
|
+
if (_.isUndefined(userPrv) || !_.isString(userPrv)) {
|
|
760
|
+
if (!_.isUndefined(userPrv)) {
|
|
761
|
+
throw new Error(`prv must be a string, got type ${typeof userPrv}`);
|
|
762
|
+
}
|
|
763
|
+
throw new Error('missing prv parameter to sign transaction');
|
|
764
|
+
}
|
|
765
|
+
const signerKeychain = utxo_lib_1.bip32.fromBase58(userPrv, utxolib.networks.bitcoin);
|
|
766
|
+
if (signerKeychain.isNeutered()) {
|
|
767
|
+
throw new Error('expected user private key but received public key');
|
|
768
|
+
}
|
|
769
|
+
debug(`Here is the public key of the xprv you used to sign: ${signerKeychain.neutered().toBase58()}`);
|
|
770
|
+
return signerKeychain;
|
|
771
|
+
};
|
|
772
|
+
const setSignerMusigNonceWithOverride = (psbt, signerKeychain, nonSegwitOverride) => {
|
|
773
|
+
utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.setAllInputsMusig2NonceHD(signerKeychain), nonSegwitOverride);
|
|
774
|
+
};
|
|
775
|
+
let signerKeychain;
|
|
776
|
+
if (tx instanceof utxo_lib_1.bitgo.UtxoPsbt && isTxWithKeyPathSpendInput) {
|
|
777
|
+
switch (params.signingStep) {
|
|
778
|
+
case 'signerNonce':
|
|
779
|
+
signerKeychain = getSignerKeychain();
|
|
780
|
+
setSignerMusigNonceWithOverride(tx, signerKeychain, !!params.allowNonSegwitSigningWithoutPrevTx);
|
|
781
|
+
AbstractUtxoCoin.PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);
|
|
782
|
+
return { txHex: tx.toHex() };
|
|
783
|
+
case 'cosignerNonce':
|
|
784
|
+
assert(txPrebuild.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
785
|
+
return { txHex: (await this.signPsbt(tx.toHex(), txPrebuild.walletId)).psbt };
|
|
786
|
+
case 'signerSignature':
|
|
787
|
+
const txId = tx.getUnsignedTx().getId();
|
|
788
|
+
const psbt = AbstractUtxoCoin.PSBT_CACHE.get(txId);
|
|
789
|
+
assert(psbt, `Psbt is missing from txCache (cache size ${AbstractUtxoCoin.PSBT_CACHE.size}).
|
|
790
|
+
This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
|
|
791
|
+
AbstractUtxoCoin.PSBT_CACHE.delete(txId);
|
|
792
|
+
tx = psbt.combine(tx);
|
|
793
|
+
break;
|
|
794
|
+
default:
|
|
795
|
+
// this instance is not an external signer
|
|
796
|
+
assert(txPrebuild.walletId, 'walletId is required for MuSig2 bitgo nonce');
|
|
797
|
+
signerKeychain = getSignerKeychain();
|
|
798
|
+
setSignerMusigNonceWithOverride(tx, signerKeychain, !!params.allowNonSegwitSigningWithoutPrevTx);
|
|
799
|
+
const response = await this.signPsbt(tx.toHex(), txPrebuild.walletId);
|
|
800
|
+
tx.combine(utxo_lib_1.bitgo.createPsbtFromHex(response.psbt, this.network));
|
|
801
|
+
break;
|
|
802
|
+
}
|
|
803
|
+
}
|
|
804
|
+
else {
|
|
805
|
+
switch (params.signingStep) {
|
|
806
|
+
case 'signerNonce':
|
|
807
|
+
case 'cosignerNonce':
|
|
808
|
+
/**
|
|
809
|
+
* In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
|
|
810
|
+
* Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
|
|
811
|
+
*/
|
|
812
|
+
return { txHex: tx.toHex() };
|
|
692
813
|
}
|
|
693
|
-
throw new Error('missing prv parameter to sign transaction');
|
|
694
814
|
}
|
|
695
|
-
if (
|
|
696
|
-
|
|
815
|
+
if (signerKeychain === undefined) {
|
|
816
|
+
signerKeychain = getSignerKeychain();
|
|
697
817
|
}
|
|
698
|
-
|
|
699
|
-
if (
|
|
700
|
-
|
|
818
|
+
let signedTransaction;
|
|
819
|
+
if (tx instanceof utxo_lib_1.bitgo.UtxoPsbt) {
|
|
820
|
+
signedTransaction = (0, sign_1.signAndVerifyPsbt)(tx, signerKeychain, {
|
|
821
|
+
isLastSignature,
|
|
822
|
+
allowNonSegwitSigningWithoutPrevTx: params.allowNonSegwitSigningWithoutPrevTx,
|
|
823
|
+
});
|
|
824
|
+
}
|
|
825
|
+
else {
|
|
826
|
+
if (tx.ins.length !== ((_b = (_a = txPrebuild.txInfo) === null || _a === void 0 ? void 0 : _a.unspents) === null || _b === void 0 ? void 0 : _b.length)) {
|
|
827
|
+
throw new Error('length of unspents array should equal to the number of transaction inputs');
|
|
828
|
+
}
|
|
829
|
+
if (!params.pubs || !(0, sdk_core_1.isTriple)(params.pubs)) {
|
|
830
|
+
throw new Error(`must provide xpub array`);
|
|
831
|
+
}
|
|
832
|
+
const keychains = params.pubs.map((pub) => utxo_lib_1.bip32.fromBase58(pub));
|
|
833
|
+
const cosignerPub = (_c = params.cosignerPub) !== null && _c !== void 0 ? _c : params.pubs[2];
|
|
834
|
+
const cosignerKeychain = utxo_lib_1.bip32.fromBase58(cosignerPub);
|
|
835
|
+
const walletSigner = new utxo_lib_1.bitgo.WalletUnspentSigner(keychains, signerKeychain, cosignerKeychain);
|
|
836
|
+
signedTransaction = (0, sign_1.signAndVerifyWalletTransaction)(tx, txPrebuild.txInfo.unspents, walletSigner, {
|
|
837
|
+
isLastSignature,
|
|
838
|
+
});
|
|
701
839
|
}
|
|
702
|
-
debug(`Here is the public key of the xprv you used to sign: ${signerKeychain.neutered().toBase58()}`);
|
|
703
|
-
const cosignerPub = (_a = params.cosignerPub) !== null && _a !== void 0 ? _a : params.pubs[2];
|
|
704
|
-
const keychains = params.pubs.map((pub) => utxo_lib_1.bip32.fromBase58(pub));
|
|
705
|
-
const cosignerKeychain = utxo_lib_1.bip32.fromBase58(cosignerPub);
|
|
706
|
-
const signedTransaction = sign_1.signAndVerifyWalletTransaction(transaction, txPrebuild.txInfo.unspents, new utxo_lib_1.bitgo.WalletUnspentSigner(keychains, signerKeychain, cosignerKeychain), { isLastSignature });
|
|
707
840
|
return {
|
|
708
841
|
txHex: signedTransaction.toBuffer().toString('hex'),
|
|
709
842
|
};
|
|
710
843
|
}
|
|
844
|
+
/**
|
|
845
|
+
* Sign a transaction with a custom signing function. Example use case is express external signer
|
|
846
|
+
* @param customSigningFunction custom signing function that returns a single signed transaction
|
|
847
|
+
* @param signTransactionParams parameters for custom signing function. Includes txPrebuild and pubs (for legacy tx only).
|
|
848
|
+
*
|
|
849
|
+
* @returns signed transaction as hex string
|
|
850
|
+
*/
|
|
851
|
+
async signWithCustomSigningFunction(customSigningFunction, signTransactionParams) {
|
|
852
|
+
const txHex = signTransactionParams.txPrebuild.txHex;
|
|
853
|
+
assert(txHex, 'missing txHex parameter');
|
|
854
|
+
const tx = utxo_lib_1.bitgo.isPsbt(txHex)
|
|
855
|
+
? utxo_lib_1.bitgo.createPsbtFromHex(txHex, this.network)
|
|
856
|
+
: this.createTransactionFromHex(txHex);
|
|
857
|
+
const isTxWithKeyPathSpendInput = tx instanceof utxo_lib_1.bitgo.UtxoPsbt && utxo_lib_1.bitgo.isTransactionWithKeyPathSpendInput(tx);
|
|
858
|
+
if (!isTxWithKeyPathSpendInput) {
|
|
859
|
+
return await customSigningFunction({ ...signTransactionParams, coin: this });
|
|
860
|
+
}
|
|
861
|
+
const getTxHex = (v) => {
|
|
862
|
+
if ('txHex' in v) {
|
|
863
|
+
return v.txHex;
|
|
864
|
+
}
|
|
865
|
+
throw new Error('txHex not found in signTransaction result');
|
|
866
|
+
};
|
|
867
|
+
const signerNonceTx = await customSigningFunction({
|
|
868
|
+
...signTransactionParams,
|
|
869
|
+
signingStep: 'signerNonce',
|
|
870
|
+
coin: this,
|
|
871
|
+
});
|
|
872
|
+
const { pubs } = signTransactionParams;
|
|
873
|
+
assert(pubs === undefined || (0, sdk_core_1.isTriple)(pubs));
|
|
874
|
+
const cosignerNonceTx = await this.signTransaction({
|
|
875
|
+
...signTransactionParams,
|
|
876
|
+
pubs,
|
|
877
|
+
txPrebuild: { ...signTransactionParams.txPrebuild, txHex: getTxHex(signerNonceTx) },
|
|
878
|
+
signingStep: 'cosignerNonce',
|
|
879
|
+
});
|
|
880
|
+
return await customSigningFunction({
|
|
881
|
+
...signTransactionParams,
|
|
882
|
+
txPrebuild: { ...signTransactionParams.txPrebuild, txHex: getTxHex(cosignerNonceTx) },
|
|
883
|
+
signingStep: 'signerSignature',
|
|
884
|
+
coin: this,
|
|
885
|
+
});
|
|
886
|
+
}
|
|
711
887
|
/**
|
|
712
888
|
* @param unspent
|
|
713
889
|
* @returns {boolean}
|
|
714
890
|
*/
|
|
715
891
|
isBitGoTaintedUnspent(unspent) {
|
|
716
|
-
return replayProtection_1.isReplayProtectionUnspent(unspent, this.network);
|
|
892
|
+
return (0, replayProtection_1.isReplayProtectionUnspent)(unspent, this.network);
|
|
717
893
|
}
|
|
718
894
|
/**
|
|
719
895
|
* @deprecated - use utxolib.bitgo.getDefaultSigHash(network) instead
|
|
@@ -735,89 +911,16 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
735
911
|
});
|
|
736
912
|
}
|
|
737
913
|
/**
|
|
738
|
-
* Decompose a raw transaction into useful information, such as the total amounts,
|
|
914
|
+
* Decompose a raw psbt/transaction into useful information, such as the total amounts,
|
|
739
915
|
* change amounts, and transaction outputs.
|
|
740
916
|
* @param params
|
|
741
917
|
*/
|
|
742
918
|
async explainTransaction(params) {
|
|
743
|
-
|
|
744
|
-
|
|
745
|
-
if (!txHex || !_.isString(txHex) || !txHex.match(/^([a-f0-9]{2})+$/i)) {
|
|
919
|
+
const { txHex } = params;
|
|
920
|
+
if (typeof txHex !== 'string' || !txHex.match(/^([a-f0-9]{2})+$/i)) {
|
|
746
921
|
throw new Error('invalid transaction hex, must be a valid hex string');
|
|
747
922
|
}
|
|
748
|
-
|
|
749
|
-
try {
|
|
750
|
-
transaction = this.createTransactionFromHex(txHex);
|
|
751
|
-
}
|
|
752
|
-
catch (e) {
|
|
753
|
-
throw new Error('failed to parse transaction hex');
|
|
754
|
-
}
|
|
755
|
-
const id = transaction.getId();
|
|
756
|
-
let spendAmount = utxolib.bitgo.toTNumber(0, this.amountType);
|
|
757
|
-
let changeAmount = utxolib.bitgo.toTNumber(0, this.amountType);
|
|
758
|
-
const explanation = {
|
|
759
|
-
displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'],
|
|
760
|
-
id: id,
|
|
761
|
-
outputs: [],
|
|
762
|
-
changeOutputs: [],
|
|
763
|
-
};
|
|
764
|
-
const { changeAddresses = [], unspents = [] } = (_a = params.txInfo) !== null && _a !== void 0 ? _a : {};
|
|
765
|
-
transaction.outs.forEach((currentOutput) => {
|
|
766
|
-
const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, this.network);
|
|
767
|
-
const currentAmount = currentOutput.value;
|
|
768
|
-
if (changeAddresses.includes(currentAddress)) {
|
|
769
|
-
// this is change
|
|
770
|
-
changeAmount += currentAmount;
|
|
771
|
-
explanation.changeOutputs.push({
|
|
772
|
-
address: currentAddress,
|
|
773
|
-
amount: currentAmount.toString(),
|
|
774
|
-
});
|
|
775
|
-
return;
|
|
776
|
-
}
|
|
777
|
-
spendAmount += currentAmount;
|
|
778
|
-
explanation.outputs.push({
|
|
779
|
-
address: currentAddress,
|
|
780
|
-
amount: currentAmount.toString(),
|
|
781
|
-
});
|
|
782
|
-
});
|
|
783
|
-
explanation.outputAmount = spendAmount.toString();
|
|
784
|
-
explanation.changeAmount = changeAmount.toString();
|
|
785
|
-
// add fee info if available
|
|
786
|
-
if (params.feeInfo) {
|
|
787
|
-
explanation.displayOrder.push('fee');
|
|
788
|
-
explanation.fee = params.feeInfo;
|
|
789
|
-
}
|
|
790
|
-
if (_.isInteger(transaction.locktime) && transaction.locktime > 0) {
|
|
791
|
-
explanation.locktime = transaction.locktime;
|
|
792
|
-
explanation.displayOrder.push('locktime');
|
|
793
|
-
}
|
|
794
|
-
const prevOutputs = (_b = params.txInfo) === null || _b === void 0 ? void 0 : _b.unspents.map((u) => toOutput(u, this.network));
|
|
795
|
-
// if keys are provided, prepare the keys for input signature checking
|
|
796
|
-
const keys = (_c = params.pubs) === null || _c === void 0 ? void 0 : _c.map((xpub) => utxo_lib_1.bip32.fromBase58(xpub));
|
|
797
|
-
const walletKeys = keys && keys.length === 3 ? new utxo_lib_1.bitgo.RootWalletKeys(keys) : undefined;
|
|
798
|
-
// get the number of signatures per input
|
|
799
|
-
const inputSignatureCounts = transaction.ins.map((input, idx) => {
|
|
800
|
-
if (unspents.length !== transaction.ins.length) {
|
|
801
|
-
return 0;
|
|
802
|
-
}
|
|
803
|
-
if (!prevOutputs) {
|
|
804
|
-
throw new Error(`invalid state`);
|
|
805
|
-
}
|
|
806
|
-
if (!walletKeys) {
|
|
807
|
-
// no pub keys or incorrect number of pub keys
|
|
808
|
-
return 0;
|
|
809
|
-
}
|
|
810
|
-
try {
|
|
811
|
-
return verifySignatureWithUnspent(transaction, idx, unspents, walletKeys).filter((v) => v).length;
|
|
812
|
-
}
|
|
813
|
-
catch (e) {
|
|
814
|
-
// some other error occurred and we can't validate the signatures
|
|
815
|
-
return 0;
|
|
816
|
-
}
|
|
817
|
-
});
|
|
818
|
-
explanation.inputSignatures = inputSignatureCounts;
|
|
819
|
-
explanation.signatures = _.max(inputSignatureCounts);
|
|
820
|
-
return explanation;
|
|
923
|
+
return utxolib.bitgo.isPsbt(txHex) ? (0, transaction_1.explainPsbt)(params, this.network) : (0, transaction_1.explainTx)(params, this);
|
|
821
924
|
}
|
|
822
925
|
/**
|
|
823
926
|
* Create a multisig address of a given type from a list of keychains and a signing threshold
|
|
@@ -840,7 +943,13 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
840
943
|
* @param params - {@see backupKeyRecovery}
|
|
841
944
|
*/
|
|
842
945
|
async recover(params) {
|
|
843
|
-
return
|
|
946
|
+
return (0, recovery_1.backupKeyRecovery)(this, this.bitgo, params);
|
|
947
|
+
}
|
|
948
|
+
async recoverV1(params) {
|
|
949
|
+
return (0, recovery_1.v1BackupKeyRecovery)(this, this.bitgo, params);
|
|
950
|
+
}
|
|
951
|
+
async sweepV1(params) {
|
|
952
|
+
return (0, recovery_1.v1Sweep)(this, this.bitgo, params);
|
|
844
953
|
}
|
|
845
954
|
/**
|
|
846
955
|
* Recover coin that was sent to wrong chain
|
|
@@ -852,10 +961,11 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
852
961
|
* @param params.signed return a half-signed transaction (default=true)
|
|
853
962
|
* @param params.walletPassphrase the wallet passphrase
|
|
854
963
|
* @param params.xprv the unencrypted xprv (used instead of wallet passphrase)
|
|
964
|
+
* @param params.apiKey for utxo coins other than [BTC,TBTC] this is a Block Chair api key
|
|
855
965
|
* @returns {*}
|
|
856
966
|
*/
|
|
857
967
|
async recoverFromWrongChain(params) {
|
|
858
|
-
const { txid, recoveryAddress, wallet, walletPassphrase, xprv } = params;
|
|
968
|
+
const { txid, recoveryAddress, wallet, walletPassphrase, xprv, apiKey } = params;
|
|
859
969
|
// params.recoveryCoin used to be params.coin, backwards compatibility
|
|
860
970
|
const recoveryCoin = params.coin || params.recoveryCoin;
|
|
861
971
|
if (!recoveryCoin) {
|
|
@@ -869,7 +979,7 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
869
979
|
if (_.isUndefined(supportedRecoveryCoins) || !supportedRecoveryCoins.includes(recoveryCoinFamily)) {
|
|
870
980
|
throw new Error(`Recovery of ${sourceCoinFamily} balances from ${recoveryCoinFamily} wallets is not supported.`);
|
|
871
981
|
}
|
|
872
|
-
return await recovery_1.recoverCrossChain(this.bitgo, {
|
|
982
|
+
return await (0, recovery_1.recoverCrossChain)(this.bitgo, {
|
|
873
983
|
sourceCoin: this,
|
|
874
984
|
recoveryCoin,
|
|
875
985
|
walletId: wallet,
|
|
@@ -877,6 +987,7 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
877
987
|
recoveryAddress,
|
|
878
988
|
walletPassphrase: signed ? walletPassphrase : undefined,
|
|
879
989
|
xprv: signed ? xprv : undefined,
|
|
990
|
+
apiKey,
|
|
880
991
|
});
|
|
881
992
|
}
|
|
882
993
|
/**
|
|
@@ -890,7 +1001,7 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
890
1001
|
// An extended private key has both a normal 256 bit private key and a 256
|
|
891
1002
|
// bit chain code, both of which must be random. 512 bits is therefore the
|
|
892
1003
|
// maximum entropy and gives us maximum security against cracking.
|
|
893
|
-
seed = crypto_1.randomBytes(512 / 8);
|
|
1004
|
+
seed = (0, crypto_1.randomBytes)(512 / 8);
|
|
894
1005
|
}
|
|
895
1006
|
const extendedKey = utxo_lib_1.bip32.fromSeed(seed);
|
|
896
1007
|
return {
|
|
@@ -898,13 +1009,53 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
898
1009
|
prv: extendedKey.toBase58(),
|
|
899
1010
|
};
|
|
900
1011
|
}
|
|
1012
|
+
shouldDefaultToPsbtTxFormat(buildParams) {
|
|
1013
|
+
const walletFlagMusigKp = buildParams.wallet.flag('musigKp') === 'true';
|
|
1014
|
+
const isHotWallet = buildParams.wallet.type() === 'hot';
|
|
1015
|
+
// if not txFormat is already specified figure out if we should default to psbt format
|
|
1016
|
+
return (buildParams.txFormat === undefined &&
|
|
1017
|
+
(buildParams.wallet.subType() === 'distributedCustody' ||
|
|
1018
|
+
// default to testnet for all utxo coins except zcash
|
|
1019
|
+
((0, utxo_lib_1.isTestnet)(this.network) &&
|
|
1020
|
+
// FIXME(BTC-1322): fix zcash PSBT support
|
|
1021
|
+
(0, utxo_lib_1.getMainnet)(this.network) !== utxolib.networks.zcash &&
|
|
1022
|
+
isHotWallet) ||
|
|
1023
|
+
// if mainnet, only default to psbt for btc hot wallets
|
|
1024
|
+
((0, utxo_lib_1.isMainnet)(this.network) && (0, utxo_lib_1.getMainnet)(this.network) === utxolib.networks.bitcoin && isHotWallet) ||
|
|
1025
|
+
// default to psbt if it has the wallet flag
|
|
1026
|
+
walletFlagMusigKp));
|
|
1027
|
+
}
|
|
901
1028
|
async getExtraPrebuildParams(buildParams) {
|
|
902
|
-
|
|
1029
|
+
let txFormat = buildParams.txFormat;
|
|
1030
|
+
let changeAddressType = buildParams.changeAddressType;
|
|
1031
|
+
if (this.shouldDefaultToPsbtTxFormat(buildParams)) {
|
|
1032
|
+
txFormat = 'psbt';
|
|
1033
|
+
}
|
|
1034
|
+
// if the addressType is not specified, we need to default to p2trMusig2 for testnet hot wallets for staged rollout of p2trMusig2
|
|
1035
|
+
if (buildParams.addressType === undefined && // addressType is deprecated and replaced by `changeAddress`
|
|
1036
|
+
buildParams.changeAddressType === undefined &&
|
|
1037
|
+
buildParams.changeAddress === undefined &&
|
|
1038
|
+
buildParams.wallet.type() === 'hot') {
|
|
1039
|
+
changeAddressType = ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'];
|
|
1040
|
+
}
|
|
1041
|
+
return {
|
|
1042
|
+
txFormat,
|
|
1043
|
+
changeAddressType,
|
|
1044
|
+
};
|
|
903
1045
|
}
|
|
904
1046
|
preCreateBitGo(params) {
|
|
905
1047
|
return;
|
|
906
1048
|
}
|
|
907
1049
|
async presignTransaction(params) {
|
|
1050
|
+
var _a, _b;
|
|
1051
|
+
// In the case that we have a 'psbt-lite' transaction format, we want to indicate in signing to not fail
|
|
1052
|
+
const txHex = ((_a = params.txHex) !== null && _a !== void 0 ? _a : (_b = params.txPrebuild) === null || _b === void 0 ? void 0 : _b.txHex);
|
|
1053
|
+
if (txHex &&
|
|
1054
|
+
utxolib.bitgo.isPsbt(txHex) &&
|
|
1055
|
+
utxolib.bitgo.isPsbtLite(utxolib.bitgo.createPsbtFromHex(txHex, this.network)) &&
|
|
1056
|
+
params.allowNonSegwitSigningWithoutPrevTx === undefined) {
|
|
1057
|
+
return { ...params, allowNonSegwitSigningWithoutPrevTx: true };
|
|
1058
|
+
}
|
|
908
1059
|
return params;
|
|
909
1060
|
}
|
|
910
1061
|
async supplementGenerateWallet(walletParams, keychains) {
|
|
@@ -917,8 +1068,17 @@ class AbstractUtxoCoin extends sdk_core_1.BaseCoin {
|
|
|
917
1068
|
return false;
|
|
918
1069
|
}
|
|
919
1070
|
getRecoveryProvider(apiToken) {
|
|
920
|
-
return recovery_1.forCoin(this.getChain(), apiToken);
|
|
1071
|
+
return (0, recovery_1.forCoin)(this.getChain(), apiToken);
|
|
921
1072
|
}
|
|
922
1073
|
}
|
|
923
1074
|
exports.AbstractUtxoCoin = AbstractUtxoCoin;
|
|
924
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractUtxoCoin.js","sourceRoot":"","sources":["../../src/abstractUtxoCoin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip32, BIP32Interface, bitgo } from '@bitgo-beta/utxo-lib';\nimport * as bitcoinMessage from 'bitcoinjs-message';\nimport { randomBytes } from 'crypto';\nimport * as debugLib from 'debug';\nimport * as _ from 'lodash';\nimport BigNumber from 'bignumber.js';\n\nimport { backupKeyRecovery, RecoverParams } from './recovery/backupKeyRecovery';\nimport {\n  CrossChainRecoverySigned,\n  CrossChainRecoveryUnsigned,\n  forCoin,\n  recoverCrossChain,\n  RecoveryProvider,\n} from './recovery';\n\nimport {\n  AddressCoinSpecific,\n  AddressTypeChainMismatchError,\n  BaseCoin,\n  BitGoBase,\n  ExtraPrebuildParamsOptions,\n  HalfSignedUtxoTransaction,\n  IBaseCoin,\n  InvalidAddressDerivationPropertyError,\n  InvalidAddressError,\n  InvalidAddressVerificationObjectPropertyError,\n  IRequestTracer,\n  ITransactionExplanation as BaseTransactionExplanation,\n  IWallet,\n  Keychain,\n  KeychainsTriplet,\n  KeyIndices,\n  P2shP2wshUnsupportedError,\n  P2trUnsupportedError,\n  P2wshUnsupportedError,\n  ParsedTransaction as BaseParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  PrecreateBitGoOptions,\n  PresignTransactionOptions,\n  promiseProps,\n  RequestTracer,\n  sanitizeLegacyPath,\n  SignedTransaction,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  SupplementGenerateWalletOptions,\n  TransactionParams as BaseTransactionParams,\n  TransactionPrebuild as BaseTransactionPrebuild,\n  TransactionRecipient,\n  Triple,\n  UnexpectedAddressError,\n  UnsupportedAddressTypeError,\n  VerificationOptions,\n  VerifyAddressOptions as BaseVerifyAddressOptions,\n  VerifyTransactionOptions as BaseVerifyTransactionOptions,\n  Wallet,\n  WalletData,\n} from '@bitgo-beta/sdk-core';\nimport { CustomChangeOptions, parseOutput } from './parseOutput';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nimport ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\nimport { isReplayProtectionUnspent } from './replayProtection';\nimport { signAndVerifyWalletTransaction } from './sign';\nimport { supportedCrossChainRecoveries } from './config';\n\nconst { getExternalChainCode, isChainCode, scriptTypeForChain, outputScripts, toOutput, verifySignatureWithUnspent } =\n  bitgo;\ntype Unspent<TNumber extends number | bigint = number> = bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = bitgo.RootWalletKeys;\nexport interface VerifyAddressOptions extends BaseVerifyAddressOptions {\n  chain: number;\n  index: number;\n}\n\nexport interface Output {\n  address: string;\n  amount: string | number;\n  external?: boolean;\n  needsCustomChangeKeySignatureVerification?: boolean;\n}\n\nexport interface TransactionExplanation extends BaseTransactionExplanation<string, string> {\n  locktime: number;\n  outputs: Output[];\n  changeOutputs: Output[];\n\n  /**\n   * Number of input signatures per input.\n   */\n  inputSignatures: number[];\n\n  /**\n   * Highest input signature count for the transaction\n   */\n  signatures: number;\n}\n\nexport interface TransactionInfo<TNumber extends number | bigint = number> {\n  /** Maps txid to txhex. Required for offline signing. */\n  txHexes?: Record<string, string>;\n  changeAddresses?: string[];\n  unspents: Unspent<TNumber>[];\n}\n\nexport interface ExplainTransactionOptions<TNumber extends number | bigint = number> {\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  feeInfo?: string;\n  pubs?: Triple<string>;\n}\n\nexport type UtxoNetwork = utxolib.Network;\n\nexport interface TransactionPrebuild<TNumber extends number | bigint = number> extends BaseTransactionPrebuild {\n  txInfo?: TransactionInfo<TNumber>;\n  blockHeight?: number;\n}\n\nexport interface TransactionParams extends BaseTransactionParams {\n  walletPassphrase?: string;\n  changeAddress?: string;\n}\n\n// parseTransactions' return type makes use of WalletData's type but with customChangeKeySignatures as required.\nexport interface AbstractUtxoCoinWalletData extends WalletData {\n  customChangeKeySignatures: {\n    user: string;\n    backup: string;\n    bitgo: string;\n  };\n}\n\nexport class AbstractUtxoCoinWallet extends Wallet {\n  public _wallet: AbstractUtxoCoinWalletData;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin, walletData: any) {\n    super(bitgo, baseCoin, walletData);\n  }\n}\n\nexport interface ParseTransactionOptions<TNumber extends number | bigint = number> extends BaseParseTransactionOptions {\n  txParams: TransactionParams;\n  txPrebuild: TransactionPrebuild<TNumber>;\n  wallet: AbstractUtxoCoinWallet;\n  verification?: VerificationOptions;\n  reqId?: IRequestTracer;\n}\n\nexport interface ParsedTransaction<TNumber extends number | bigint = number> extends BaseParsedTransaction {\n  keychains: {\n    user?: Keychain;\n    backup?: Keychain;\n    bitgo?: Keychain;\n  };\n  keySignatures: {\n    backupPub?: string;\n    bitgoPub?: string;\n  };\n  outputs: Output[];\n  missingOutputs: Output[];\n  explicitExternalOutputs: Output[];\n  implicitExternalOutputs: Output[];\n  changeOutputs: Output[];\n  explicitExternalSpendAmount: TNumber;\n  implicitExternalSpendAmount: TNumber;\n  needsCustomChangeKeySignatureVerification: boolean;\n  customChange?: CustomChangeOptions;\n}\n\nexport interface GenerateAddressOptions {\n  addressType?: ScriptType2Of3;\n  keychains: {\n    pub: string;\n    aspKeyId?: string;\n  }[];\n  threshold?: number;\n  chain?: number;\n  index?: number;\n  segwit?: boolean;\n  bech32?: boolean;\n}\n\nexport interface AddressDetails {\n  address: string;\n  chain: number;\n  index: number;\n  coin: string;\n  coinSpecific: AddressCoinSpecific;\n  addressType?: string;\n}\n\nexport interface SignTransactionOptions<TNumber extends number | bigint = number> extends BaseSignTransactionOptions {\n  /** Transaction prebuild from bitgo server */\n  txPrebuild: {\n    txHex: string;\n    txInfo: TransactionInfo<TNumber>;\n  };\n  /** xprv of user key or backup key */\n  prv: string;\n  /** xpubs triple for wallet (user, backup, bitgo) */\n  pubs: Triple<string>;\n  /** xpub for cosigner (defaults to bitgo) */\n  cosignerPub?: string;\n  /**\n   * When true, creates full-signed transaction without placeholder signatures.\n   * When false, creates half-signed transaction with placeholder signatures.\n   */\n  isLastSignature?: boolean;\n}\n\nexport interface MultiSigAddress {\n  outputScript: Buffer;\n  redeemScript?: Buffer;\n  witnessScript?: Buffer;\n  address: string;\n}\n\nexport interface RecoverFromWrongChainOptions {\n  txid: string;\n  recoveryAddress: string;\n  wallet: string;\n  walletPassphrase?: string;\n  xprv?: string;\n  /** @deprecated */\n  coin?: AbstractUtxoCoin;\n  recoveryCoin?: AbstractUtxoCoin;\n  signed?: boolean;\n}\n\nexport interface VerifyKeySignaturesOptions {\n  userKeychain?: Keychain;\n  keychainToVerify?: Keychain;\n  keySignature?: string;\n}\n\nexport interface VerifyUserPublicKeyOptions {\n  userKeychain?: Keychain;\n  disableNetworking: boolean;\n  txParams: TransactionParams;\n}\n\nexport interface VerifyTransactionOptions<TNumber extends number | bigint = number>\n  extends BaseVerifyTransactionOptions {\n  txPrebuild: TransactionPrebuild<TNumber>;\n  wallet: AbstractUtxoCoinWallet;\n}\n\nexport abstract class AbstractUtxoCoin extends BaseCoin {\n  public altScriptHash?: number;\n  public supportAltScriptDestination?: boolean;\n  public readonly amountType: 'number' | 'bigint';\n  private readonly _network: utxolib.Network;\n\n  protected constructor(bitgo: BitGoBase, network: utxolib.Network, amountType: 'number' | 'bigint' = 'number') {\n    super(bitgo);\n    if (!utxolib.isValidNetwork(network)) {\n      throw new Error(\n        'invalid network: please make sure to use the same version of ' +\n          '@bitgo-beta/utxo-lib as this library when initializing an instance of this class'\n      );\n    }\n    this.amountType = amountType;\n    this._network = network;\n  }\n\n  get network() {\n    return this._network;\n  }\n\n  sweepWithSendMany(): boolean {\n    return true;\n  }\n\n  /** @deprecated */\n  static get validAddressTypes(): ScriptType2Of3[] {\n    return [...outputScripts.scriptTypes2Of3];\n  }\n\n  /**\n   * Returns the factor between the base unit and its smallest subdivison\n   * @return {number}\n   */\n  getBaseFactor() {\n    return 1e8;\n  }\n\n  /**\n   * @deprecated\n   */\n  getCoinLibrary() {\n    return utxolib;\n  }\n\n  /**\n   * Check if an address is valid\n   * @param address\n   * @param param\n   */\n  isValidAddress(address: string, param?: { anyFormat: boolean } | /* legacy parameter */ boolean): boolean {\n    if (typeof param === 'boolean' && param) {\n      throw new Error('deprecated');\n    }\n\n    const formats = param && param.anyFormat ? undefined : ['default' as const];\n    try {\n      utxolib.addressFormat.toOutputScriptTryFormats(address, this.network, formats);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPub(pub: string) {\n    try {\n      return bip32.fromBase58(pub).isNeutered();\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Get the latest block height\n   * @param reqId\n   */\n  async getLatestBlockHeight(reqId?: RequestTracer): Promise<number> {\n    if (reqId) {\n      this.bitgo.setRequestTracer(reqId);\n    }\n    const chainhead = await this.bitgo.get(this.url('/public/block/latest')).result();\n    return (chainhead as any).height;\n  }\n\n  /**\n   * Run custom coin logic after a transaction prebuild has been received from BitGo\n   * @param prebuild\n   */\n  async postProcessPrebuild<TNumber extends number | bigint>(\n    prebuild: TransactionPrebuild<TNumber>\n  ): Promise<TransactionPrebuild<TNumber>> {\n    if (_.isUndefined(prebuild.txHex)) {\n      throw new Error('missing required txPrebuild property txHex');\n    }\n    const transaction = this.createTransactionFromHex<TNumber>(prebuild.txHex);\n    if (_.isUndefined(prebuild.blockHeight)) {\n      prebuild.blockHeight = (await this.getLatestBlockHeight()) as number;\n    }\n    return _.extend({}, prebuild, { txHex: transaction.toHex() });\n  }\n\n  /**\n   * Find outputs that are within expected outputs but not within actual outputs, including duplicates\n   * @param expectedOutputs\n   * @param actualOutputs\n   * @returns {Array}\n   */\n  protected static findMissingOutputs(expectedOutputs: Output[], actualOutputs: Output[]): Output[] {\n    const keyFunc = ({ address, amount }: Output): string => `${address}:${amount}`;\n    const groupedOutputs = _.groupBy(expectedOutputs, keyFunc);\n\n    actualOutputs.forEach((output) => {\n      const group = groupedOutputs[keyFunc(output)];\n      if (group) {\n        group.pop();\n      }\n    });\n\n    return _.flatten(_.values(groupedOutputs));\n  }\n\n  /**\n   * Determine an address' type based on its witness and redeem script presence\n   * @param addressDetails\n   */\n  static inferAddressType(addressDetails: { chain: number }): ScriptType2Of3 | null {\n    return isChainCode(addressDetails.chain) ? scriptTypeForChain(addressDetails.chain) : null;\n  }\n\n  createTransactionFromHex<TNumber extends number | bigint = number>(\n    hex: string\n  ): utxolib.bitgo.UtxoTransaction<TNumber> {\n    return utxolib.bitgo.createTransactionFromHex<TNumber>(hex, this.network, this.amountType);\n  }\n\n  /**\n   * Extract and fill transaction details such as internal/change spend, external spend (explicit vs. implicit), etc.\n   * @param params\n   * @returns {*}\n   */\n  async parseTransaction<TNumber extends number | bigint = number>(\n    params: ParseTransactionOptions<TNumber>\n  ): Promise<ParsedTransaction<TNumber>> {\n    const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;\n\n    if (!_.isUndefined(verification.disableNetworking) && !_.isBoolean(verification.disableNetworking)) {\n      throw new Error('verification.disableNetworking must be a boolean');\n    }\n    const disableNetworking = verification.disableNetworking;\n\n    const fetchKeychains = async (wallet: IWallet): Promise<VerificationOptions['keychains']> => {\n      return promiseProps({\n        user: this.keychains().get({ id: wallet.keyIds()[KeyIndices.USER], reqId }),\n        backup: this.keychains().get({ id: wallet.keyIds()[KeyIndices.BACKUP], reqId }),\n        bitgo: this.keychains().get({ id: wallet.keyIds()[KeyIndices.BITGO], reqId }),\n      });\n    };\n\n    // obtain the keychains and key signatures\n    let keychains: VerificationOptions['keychains'] | undefined = verification.keychains;\n    if (!keychains) {\n      if (disableNetworking) {\n        throw new Error('cannot fetch keychains without networking');\n      }\n      keychains = await fetchKeychains(wallet);\n    }\n\n    if (!keychains || !keychains.user || !keychains.backup || !keychains.bitgo) {\n      throw new Error('keychains are required, but could not be fetched');\n    }\n\n    const keychainArray: Triple<Keychain> = [keychains.user, keychains.backup, keychains.bitgo];\n\n    const keySignatures = _.get(wallet, '_wallet.keySignatures', {});\n\n    if (_.isUndefined(txPrebuild.txHex)) {\n      throw new Error('missing required txPrebuild property txHex');\n    }\n    // obtain all outputs\n    const explanation: TransactionExplanation = await this.explainTransaction<TNumber>({\n      txHex: txPrebuild.txHex,\n      txInfo: txPrebuild.txInfo,\n      pubs: keychainArray.map((k) => k.pub) as Triple<string>,\n    });\n\n    const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];\n\n    // verify that each recipient from txParams has their own output\n    const expectedOutputs = _.get(txParams, 'recipients', [] as TransactionRecipient[]).map((output) => {\n      return { ...output, address: this.canonicalAddress(output.address) };\n    });\n\n    const missingOutputs = AbstractUtxoCoin.findMissingOutputs(expectedOutputs, allOutputs);\n\n    // get the keychains from the custom change wallet if needed\n    let customChange: CustomChangeOptions | undefined;\n    const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};\n    if (customChangeWalletId) {\n      // fetch keychains from custom change wallet for deriving addresses.\n      // These keychains should be signed and this should be verified in verifyTransaction\n      const customChangeKeySignatures = wallet._wallet.customChangeKeySignatures;\n      const customChangeWallet: Wallet = await this.wallets().get({ id: customChangeWalletId });\n      const customChangeKeys = await fetchKeychains(customChangeWallet);\n\n      if (!customChangeKeys) {\n        throw new Error('failed to fetch keychains for custom change wallet');\n      }\n\n      if (customChangeKeys.user && customChangeKeys.backup && customChangeKeys.bitgo && customChangeWallet) {\n        const customChangeKeychains: [Keychain, Keychain, Keychain] = [\n          customChangeKeys.user,\n          customChangeKeys.backup,\n          customChangeKeys.bitgo,\n        ];\n\n        customChange = {\n          keys: customChangeKeychains,\n          signatures: [\n            customChangeKeySignatures.user,\n            customChangeKeySignatures.backup,\n            customChangeKeySignatures.bitgo,\n          ],\n        };\n      }\n    }\n\n    /**\n     * Loop through all the outputs and classify each of them as either internal spends\n     * or external spends by setting the \"external\" property to true or false on the output object.\n     */\n    const allOutputDetails: Output[] = await Promise.all(\n      allOutputs.map((currentOutput) => {\n        return parseOutput({\n          currentOutput,\n          coin: this,\n          txPrebuild,\n          verification,\n          keychainArray,\n          wallet,\n          txParams,\n          customChange,\n          reqId,\n        });\n      })\n    );\n\n    const needsCustomChangeKeySignatureVerification = allOutputDetails.some(\n      (output) => output.needsCustomChangeKeySignatureVerification\n    );\n\n    const changeOutputs = _.filter(allOutputDetails, { external: false });\n\n    // these are all the outputs that were not originally explicitly specified in recipients\n    const implicitOutputs = AbstractUtxoCoin.findMissingOutputs(allOutputDetails, expectedOutputs);\n\n    const explicitOutputs = AbstractUtxoCoin.findMissingOutputs(allOutputDetails, implicitOutputs);\n\n    // these are all the non-wallet outputs that had been originally explicitly specified in recipients\n    const explicitExternalOutputs = _.filter(explicitOutputs, { external: true });\n\n    // this is the sum of all the originally explicitly specified non-wallet output values\n    const explicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n      explicitExternalOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0)) as bigint,\n      this.amountType\n    );\n\n    /**\n     * The calculation of the implicit external spend amount pertains to verifying the pay-as-you-go-fee BitGo\n     * automatically applies to transactions sending money out of the wallet. The logic is fairly straightforward\n     * in that we compare the external spend amount that was specified explicitly by the user to the portion\n     * that was specified implicitly. To protect customers from people tampering with the transaction outputs, we\n     * define a threshold for the maximum percentage of the implicit external spend in relation to the explicit\n     * external spend.\n     */\n\n    // make sure that all the extra addresses are change addresses\n    // get all the additional external outputs the server added and calculate their values\n    const implicitExternalOutputs = _.filter(implicitOutputs, { external: true });\n    const implicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n      implicitExternalOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0)) as bigint,\n      this.amountType\n    );\n\n    return {\n      keychains,\n      keySignatures,\n      outputs: allOutputDetails,\n      missingOutputs,\n      explicitExternalOutputs,\n      implicitExternalOutputs,\n      changeOutputs,\n      explicitExternalSpendAmount,\n      implicitExternalSpendAmount,\n      needsCustomChangeKeySignatureVerification,\n      customChange,\n    };\n  }\n\n  /**\n   * Decrypt the wallet's user private key and verify that the claimed public key matches\n   * @param {VerifyUserPublicKeyOptions} params\n   * @return {boolean}\n   * @protected\n   */\n  protected verifyUserPublicKey(params: VerifyUserPublicKeyOptions): boolean {\n    const { userKeychain, txParams, disableNetworking } = params;\n    if (!userKeychain) {\n      throw new Error('user keychain is required');\n    }\n\n    const userPub = userKeychain.pub;\n\n    // decrypt the user private key so we can verify that the claimed public key is a match\n    let userPrv = userKeychain.prv;\n    if (_.isEmpty(userPrv)) {\n      const encryptedPrv = userKeychain.encryptedPrv;\n      if (encryptedPrv && !_.isEmpty(encryptedPrv)) {\n        // if the decryption fails, it will throw an error\n        userPrv = this.bitgo.decrypt({\n          input: encryptedPrv,\n          password: txParams.walletPassphrase,\n        });\n      }\n    }\n\n    if (!userPrv) {\n      const errorMessage = 'user private key unavailable for verification';\n      if (disableNetworking) {\n        console.log(errorMessage);\n        return false;\n      } else {\n        throw new Error(errorMessage);\n      }\n    } else {\n      const userPrivateKey = bip32.fromBase58(userPrv);\n      if (userPrivateKey.toBase58() === userPrivateKey.neutered().toBase58()) {\n        throw new Error('user private key is only public');\n      }\n      if (userPrivateKey.neutered().toBase58() !== userPub) {\n        throw new Error('user private key does not match public key');\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Verify signatures produced by the user key over the backup and bitgo keys.\n   *\n   * If set, these signatures ensure that the wallet keys cannot be changed after the wallet has been created.\n   * @param {VerifyKeySignaturesOptions} params\n   * @return {{backup: boolean, bitgo: boolean}}\n   */\n  protected verifyKeySignature(params: VerifyKeySignaturesOptions): boolean {\n    // first, let's verify the integrity of the user key, whose public key is used for subsequent verifications\n    const { userKeychain, keychainToVerify, keySignature } = params;\n    if (!userKeychain) {\n      throw new Error('user keychain is required');\n    }\n\n    if (!keychainToVerify) {\n      throw new Error('keychain to verify is required');\n    }\n\n    if (!keySignature) {\n      throw new Error('key signature is required');\n    }\n\n    // verify the signature against the user public key\n    const publicKey = bip32.fromBase58(userKeychain.pub).publicKey;\n    const signingAddress = utxolib.address.toBase58Check(\n      utxolib.crypto.hash160(publicKey),\n      utxolib.networks.bitcoin.pubKeyHash,\n      this.network\n    );\n\n    // BG-5703: use BTC mainnet prefix for all key signature operations\n    // (this means do not pass a prefix parameter, and let it use the default prefix instead)\n    try {\n      return bitcoinMessage.verify(keychainToVerify.pub, signingAddress, Buffer.from(keySignature, 'hex'));\n    } catch (e) {\n      debug('error thrown from bitcoinmessage while verifying key signature', e);\n      return false;\n    }\n  }\n\n  /**\n   * Verify signatures against the user private key over the change wallet extended keys\n   * @param {ParsedTransaction} tx\n   * @param {Keychain} userKeychain\n   * @return {boolean}\n   * @protected\n   */\n  protected verifyCustomChangeKeySignatures<TNumber extends number | bigint>(\n    tx: ParsedTransaction<TNumber>,\n    userKeychain: Keychain\n  ): boolean {\n    if (!tx.customChange) {\n      throw new Error('parsed transaction is missing required custom change verification data');\n    }\n\n    if (!Array.isArray(tx.customChange.keys) || !Array.isArray(tx.customChange.signatures)) {\n      throw new Error('customChange property is missing keys or signatures');\n    }\n\n    for (const keyIndex of [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO]) {\n      const keychainToVerify = tx.customChange.keys[keyIndex];\n      const keySignature = tx.customChange.signatures[keyIndex];\n      if (!keychainToVerify) {\n        throw new Error(`missing required custom change ${KeyIndices[keyIndex].toLowerCase()} keychain public key`);\n      }\n      if (!keySignature) {\n        throw new Error(`missing required custom change ${KeyIndices[keyIndex].toLowerCase()} keychain signature`);\n      }\n      if (!this.verifyKeySignature({ userKeychain, keychainToVerify, keySignature })) {\n        debug('failed to verify custom change %s key signature!', KeyIndices[keyIndex].toLowerCase());\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Get the maximum percentage limit for pay-as-you-go outputs\n   *\n   * @protected\n   */\n  protected getPayGoLimit(allowPaygoOutput?: boolean): number {\n    // allowing paygo outputs needs to be the default behavior, so only disallow paygo outputs if the\n    // relevant verification option is both set and false\n    if (!_.isNil(allowPaygoOutput) && !allowPaygoOutput) {\n      return 0;\n    }\n    // 150 basis points is the absolute permitted maximum if paygo outputs are allowed\n    return 0.015;\n  }\n\n  /**\n   * Verify that a transaction prebuild complies with the original intention\n   *\n   * @param params\n   * @param params.txParams params object passed to send\n   * @param params.txPrebuild prebuild object returned by server\n   * @param params.txPrebuild.txHex prebuilt transaction's txHex form\n   * @param params.wallet Wallet object to obtain keys to verify against\n   * @param params.verification Object specifying some verification parameters\n   * @param params.verification.disableNetworking Disallow fetching any data from the internet for verification purposes\n   * @param params.verification.keychains Pass keychains manually rather than fetching them by id\n   * @param params.verification.addresses Address details to pass in for out-of-band verification\n   * @returns {boolean}\n   */\n  async verifyTransaction<TNumber extends number | bigint = number>(\n    params: VerifyTransactionOptions<TNumber>\n  ): Promise<boolean> {\n    const { txParams, txPrebuild, wallet, verification = { allowPaygoOutput: true }, reqId } = params;\n    const disableNetworking = !!verification.disableNetworking;\n    const parsedTransaction: ParsedTransaction<TNumber> = await this.parseTransaction<TNumber>({\n      txParams,\n      txPrebuild,\n      wallet,\n      verification,\n      reqId,\n    });\n\n    const keychains = parsedTransaction.keychains;\n\n    // verify that the claimed user public key corresponds to the wallet's user private key\n    let userPublicKeyVerified = false;\n    try {\n      // verify the user public key matches the private key - this will throw if there is no match\n      userPublicKeyVerified = this.verifyUserPublicKey({ userKeychain: keychains.user, disableNetworking, txParams });\n    } catch (e) {\n      debug('failed to verify user public key!', e);\n    }\n\n    // let's verify these keychains\n    const keySignatures = parsedTransaction.keySignatures;\n    if (!_.isEmpty(keySignatures)) {\n      const verify = (key, pub) =>\n        this.verifyKeySignature({ userKeychain: keychains.user, keychainToVerify: key, keySignature: pub });\n      const isBackupKeySignatureValid = verify(keychains.backup, keySignatures.backupPub);\n      const isBitgoKeySignatureValid = verify(keychains.bitgo, keySignatures.bitgoPub);\n      if (!isBackupKeySignatureValid || !isBitgoKeySignatureValid) {\n        throw new Error('secondary public key signatures invalid');\n      }\n      debug('successfully verified backup and bitgo key signatures');\n    } else if (!disableNetworking) {\n      // these keys were obtained online and their signatures were not verified\n      // this could be dangerous\n      console.log('unsigned keys obtained online are being used for address verification');\n    }\n\n    if (parsedTransaction.needsCustomChangeKeySignatureVerification) {\n      if (!keychains.user || !userPublicKeyVerified) {\n        throw new Error('transaction requires verification of user public key, but it was unable to be verified');\n      }\n      const customChangeKeySignaturesVerified = this.verifyCustomChangeKeySignatures(parsedTransaction, keychains.user);\n      if (!customChangeKeySignaturesVerified) {\n        throw new Error(\n          'transaction requires verification of custom change key signatures, but they were unable to be verified'\n        );\n      }\n      debug('successfully verified user public key and custom change key signatures');\n    }\n\n    const missingOutputs = parsedTransaction.missingOutputs;\n    if (missingOutputs.length !== 0) {\n      // there are some outputs in the recipients list that have not made it into the actual transaction\n      throw new Error('expected outputs missing in transaction prebuild');\n    }\n\n    const intendedExternalSpend = parsedTransaction.explicitExternalSpendAmount;\n\n    // this is a limit we impose for the total value that is amended to the transaction beyond what was originally intended\n    const payAsYouGoLimit = new BigNumber(this.getPayGoLimit(verification.allowPaygoOutput)).multipliedBy(\n      intendedExternalSpend.toString()\n    );\n\n    /*\n    Some explanation for why we're doing what we're doing:\n    Some customers will have an output to BitGo's PAYGo wallet added to their transaction, and we need to account for\n    it here. To protect someone tampering with the output to make it send more than it should to BitGo, we define a\n    threshold for the output's value above which we'll throw an error, because the paygo output should never be that\n    high.\n     */\n\n    // make sure that all the extra addresses are change addresses\n    // get all the additional external outputs the server added and calculate their values\n    const nonChangeAmount = new BigNumber(parsedTransaction.implicitExternalSpendAmount.toString());\n\n    debug(\n      'Intended spend is %s, Non-change amount is %s, paygo limit is %s',\n      intendedExternalSpend.toString(),\n      nonChangeAmount.toString(),\n      payAsYouGoLimit.toString()\n    );\n\n    // the additional external outputs can only be BitGo's pay-as-you-go fee, but we cannot verify the wallet address\n    if (nonChangeAmount.gt(payAsYouGoLimit)) {\n      // there are some addresses that are outside the scope of intended recipients that are not change addresses\n      throw new Error('prebuild attempts to spend to unintended external recipients');\n    }\n\n    const allOutputs = parsedTransaction.outputs;\n    if (!txPrebuild.txHex) {\n      throw new Error(`txPrebuild.txHex not set`);\n    }\n    const transaction = this.createTransactionFromHex<TNumber>(txPrebuild.txHex);\n    const transactionCache = {};\n    const inputs = await Promise.all(\n      transaction.ins.map(async (currentInput) => {\n        const transactionId = (Buffer.from(currentInput.hash).reverse() as Buffer).toString('hex');\n        const txHex = txPrebuild.txInfo?.txHexes?.[transactionId];\n        if (txHex) {\n          const localTx = this.createTransactionFromHex<TNumber>(txHex);\n          if (localTx.getId() !== transactionId) {\n            throw new Error('input transaction hex does not match id');\n          }\n          const currentOutput = localTx.outs[currentInput.index];\n          const address = utxolib.address.fromOutputScript(currentOutput.script, this.network);\n          return {\n            address,\n            value: currentOutput.value,\n            valueString: currentOutput.value.toString(),\n          };\n        } else if (!transactionCache[transactionId]) {\n          if (disableNetworking) {\n            throw new Error('attempting to retrieve transaction details externally with networking disabled');\n          }\n          if (reqId) {\n            this.bitgo.setRequestTracer(reqId);\n          }\n          transactionCache[transactionId] = await this.bitgo.get(this.url(`/public/tx/${transactionId}`)).result();\n        }\n        const transactionDetails = transactionCache[transactionId];\n        return transactionDetails.outputs[currentInput.index];\n      })\n    );\n\n    // coins (doge) that can exceed number limits (and thus will use bigint) will have the `valueString` field\n    const inputAmount = inputs.reduce(\n      (sum: bigint, i) => sum + BigInt(this.amountType === 'bigint' ? i.valueString : i.value),\n      BigInt(0)\n    );\n    const outputAmount = allOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0));\n    const fee = inputAmount - outputAmount;\n\n    if (fee < 0) {\n      throw new Error(\n        `attempting to spend ${outputAmount} satoshis, which exceeds the input amount (${inputAmount} satoshis) by ${-fee}`\n      );\n    }\n\n    return true;\n  }\n\n  /**\n   * Make sure an address is valid and throw an error if it's not.\n   * @param params.address The address string on the network\n   * @param params.addressType\n   * @param params.keychains Keychain objects with xpubs\n   * @param params.coinSpecific Coin-specific details for the address such as a witness script\n   * @param params.chain Derivation chain\n   * @param params.index Derivation index\n   * @throws {InvalidAddressError}\n   * @throws {InvalidAddressDerivationPropertyError}\n   * @throws {UnexpectedAddressError}\n   */\n  async isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    const { address, addressType, keychains, coinSpecific, chain, index } = params;\n\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    if ((_.isUndefined(chain) && _.isUndefined(index)) || !(_.isFinite(chain) && _.isFinite(index))) {\n      throw new InvalidAddressDerivationPropertyError(\n        `address validation failure: invalid chain (${chain}) or index (${index})`\n      );\n    }\n\n    if (!_.isObject(coinSpecific)) {\n      throw new InvalidAddressVerificationObjectPropertyError(\n        'address validation failure: coinSpecific field must be an object'\n      );\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    const expectedAddress = this.generateAddress({\n      addressType: addressType as ScriptType2Of3,\n      keychains,\n      threshold: 2,\n      chain,\n      index,\n    });\n\n    if (expectedAddress.address !== address) {\n      throw new UnexpectedAddressError(\n        `address validation failure: expected ${expectedAddress.address} but got ${address}`\n      );\n    }\n\n    return true;\n  }\n\n  /**\n   * Indicates whether coin supports a block target\n   * @returns {boolean}\n   */\n  supportsBlockTarget() {\n    return true;\n  }\n\n  /**\n   * @param addressType\n   * @returns true iff coin supports spending from unspentType\n   */\n  supportsAddressType(addressType: ScriptType2Of3): boolean {\n    return utxolib.bitgo.outputScripts.isSupportedScriptType(this.network, addressType);\n  }\n\n  /**\n   * @param chain\n   * @return true iff coin supports spending from chain\n   */\n  supportsAddressChain(chain: number): boolean {\n    return isChainCode(chain) && this.supportsAddressType(utxolib.bitgo.scriptTypeForChain(chain));\n  }\n\n  keyIdsForSigning(): number[] {\n    return [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO];\n  }\n\n  /**\n   * TODO(BG-11487): Remove addressType, segwit, and bech32 params in SDKv6\n   * Generate an address for a wallet based on a set of configurations\n   * @param params.addressType {string}   Deprecated\n   * @param params.keychains   {[object]} Array of objects with xpubs\n   * @param params.threshold   {number}   Minimum number of signatures\n   * @param params.chain       {number}   Derivation chain (see https://github.com/BitGo/unspents/blob/master/src/codes.ts for\n   *                                                 the corresponding address type of a given chain code)\n   * @param params.index       {number}   Derivation index\n   * @param params.segwit      {boolean}  Deprecated\n   * @param params.bech32      {boolean}  Deprecated\n   * @returns {{chain: number, index: number, coin: number, coinSpecific: {outputScript, redeemScript}}}\n   */\n  generateAddress(params: GenerateAddressOptions): AddressDetails {\n    const { keychains, threshold, chain, index, segwit = false, bech32 = false } = params;\n    let derivationChain = getExternalChainCode('p2sh');\n    if (_.isNumber(chain) && _.isInteger(chain) && isChainCode(chain)) {\n      derivationChain = chain;\n    }\n\n    function convertFlagsToAddressType(): ScriptType2Of3 {\n      if (isChainCode(chain)) {\n        return utxolib.bitgo.scriptTypeForChain(chain);\n      }\n      if (_.isBoolean(segwit) && segwit) {\n        return 'p2shP2wsh';\n      } else if (_.isBoolean(bech32) && bech32) {\n        return 'p2wsh';\n      } else {\n        return 'p2sh';\n      }\n    }\n\n    const addressType = params.addressType || convertFlagsToAddressType();\n\n    if (addressType !== utxolib.bitgo.scriptTypeForChain(derivationChain)) {\n      throw new AddressTypeChainMismatchError(addressType, derivationChain);\n    }\n\n    if (!this.supportsAddressType(addressType)) {\n      switch (addressType) {\n        case 'p2sh':\n          throw new Error(`internal error: p2sh should always be supported`);\n        case 'p2shP2wsh':\n          throw new P2shP2wshUnsupportedError();\n        case 'p2wsh':\n          throw new P2wshUnsupportedError();\n        case 'p2tr':\n          throw new P2trUnsupportedError();\n        default:\n          throw new UnsupportedAddressTypeError();\n      }\n    }\n\n    let signatureThreshold = 2;\n    if (_.isInteger(threshold)) {\n      signatureThreshold = threshold as number;\n      if (signatureThreshold <= 0) {\n        throw new Error('threshold has to be positive');\n      }\n      if (signatureThreshold > keychains.length) {\n        throw new Error('threshold cannot exceed number of keys');\n      }\n    }\n\n    let derivationIndex = 0;\n    if (_.isInteger(index) && (index as number) > 0) {\n      derivationIndex = index as number;\n    }\n\n    const path = '0/0/' + derivationChain + '/' + derivationIndex;\n    const hdNodes = keychains.map(({ pub }) => bip32.fromBase58(pub));\n    const derivedKeys = hdNodes.map((hdNode) => hdNode.derivePath(sanitizeLegacyPath(path)).publicKey);\n\n    const { outputScript, redeemScript, witnessScript, address } = this.createMultiSigAddress(\n      addressType,\n      signatureThreshold,\n      derivedKeys\n    );\n\n    return {\n      address,\n      chain: derivationChain,\n      index: derivationIndex,\n      coin: this.getChain(),\n      coinSpecific: {\n        outputScript: outputScript.toString('hex'),\n        redeemScript: redeemScript && redeemScript.toString('hex'),\n        witnessScript: witnessScript && witnessScript.toString('hex'),\n      },\n      addressType,\n    };\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   * @param params - {@see SignTransactionOptions}\n   * @returns {Promise<SignedTransaction | HalfSignedUtxoTransaction>}\n   */\n  async signTransaction<TNumber extends number | bigint = number>(\n    params: SignTransactionOptions<TNumber>\n  ): Promise<SignedTransaction | HalfSignedUtxoTransaction> {\n    const txPrebuild = params.txPrebuild;\n    const userPrv = params.prv;\n\n    if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {\n      if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {\n        throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);\n      }\n      throw new Error('missing txPrebuild parameter');\n    }\n    const transaction = this.createTransactionFromHex<TNumber>(txPrebuild.txHex);\n\n    if (transaction.ins.length !== txPrebuild.txInfo.unspents.length) {\n      throw new Error('length of unspents array should equal to the number of transaction inputs');\n    }\n\n    let isLastSignature = false;\n    if (_.isBoolean(params.isLastSignature)) {\n      // if build is called instead of buildIncomplete, no signature placeholders are left in the sig script\n      isLastSignature = params.isLastSignature;\n    }\n\n    if (_.isUndefined(userPrv) || !_.isString(userPrv)) {\n      if (!_.isUndefined(userPrv)) {\n        throw new Error(`prv must be a string, got type ${typeof userPrv}`);\n      }\n      throw new Error('missing prv parameter to sign transaction');\n    }\n\n    if (!params.pubs || params.pubs.length !== 3) {\n      throw new Error(`must provide xpub array`);\n    }\n\n    const signerKeychain = bip32.fromBase58(userPrv, utxolib.networks.bitcoin);\n    if (signerKeychain.isNeutered()) {\n      throw new Error('expected user private key but received public key');\n    }\n    debug(`Here is the public key of the xprv you used to sign: ${signerKeychain.neutered().toBase58()}`);\n\n    const cosignerPub = params.cosignerPub ?? params.pubs[2];\n    const keychains = params.pubs.map((pub) => bip32.fromBase58(pub)) as Triple<BIP32Interface>;\n    const cosignerKeychain = bip32.fromBase58(cosignerPub);\n\n    const signedTransaction = signAndVerifyWalletTransaction(\n      transaction,\n      txPrebuild.txInfo.unspents,\n      new bitgo.WalletUnspentSigner<RootWalletKeys>(keychains, signerKeychain, cosignerKeychain),\n      { isLastSignature }\n    );\n\n    return {\n      txHex: signedTransaction.toBuffer().toString('hex'),\n    };\n  }\n\n  /**\n   * @param unspent\n   * @returns {boolean}\n   */\n  isBitGoTaintedUnspent<TNumber extends number | bigint>(unspent: Unspent<TNumber>): boolean {\n    return isReplayProtectionUnspent<TNumber>(unspent, this.network);\n  }\n\n  /**\n   * @deprecated - use utxolib.bitgo.getDefaultSigHash(network) instead\n   * @returns {number}\n   */\n  get defaultSigHashType(): number {\n    return utxolib.bitgo.getDefaultSigHash(this.network);\n  }\n\n  /**\n   * @deprecated - use utxolib.bitcoin.verifySignature() instead\n   */\n  verifySignature(\n    transaction: any,\n    inputIndex: number,\n    amount: number,\n    verificationSettings: {\n      signatureIndex?: number;\n      publicKey?: string;\n    } = {}\n  ): boolean {\n    if (transaction.network !== this.network) {\n      throw new Error(`network mismatch`);\n    }\n    return utxolib.bitgo.verifySignature(transaction, inputIndex, amount, {\n      signatureIndex: verificationSettings.signatureIndex,\n      publicKey: verificationSettings.publicKey ? Buffer.from(verificationSettings.publicKey, 'hex') : undefined,\n    });\n  }\n\n  /**\n   * Decompose a raw transaction into useful information, such as the total amounts,\n   * change amounts, and transaction outputs.\n   * @param params\n   */\n  async explainTransaction<TNumber extends number | bigint = number>(\n    params: ExplainTransactionOptions<TNumber>\n  ): Promise<TransactionExplanation> {\n    const txHex = _.get(params, 'txHex');\n    if (!txHex || !_.isString(txHex) || !txHex.match(/^([a-f0-9]{2})+$/i)) {\n      throw new Error('invalid transaction hex, must be a valid hex string');\n    }\n\n    let transaction;\n    try {\n      transaction = this.createTransactionFromHex(txHex);\n    } catch (e) {\n      throw new Error('failed to parse transaction hex');\n    }\n\n    const id = transaction.getId();\n    let spendAmount = utxolib.bitgo.toTNumber<TNumber>(0, this.amountType);\n    let changeAmount = utxolib.bitgo.toTNumber<TNumber>(0, this.amountType);\n    const explanation = {\n      displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'],\n      id: id,\n      outputs: [] as Output[],\n      changeOutputs: [] as Output[],\n    } as TransactionExplanation;\n\n    const { changeAddresses = [], unspents = [] } = params.txInfo ?? {};\n\n    transaction.outs.forEach((currentOutput) => {\n      const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, this.network);\n      const currentAmount = currentOutput.value;\n\n      if (changeAddresses.includes(currentAddress)) {\n        // this is change\n        changeAmount += currentAmount;\n        explanation.changeOutputs.push({\n          address: currentAddress,\n          amount: currentAmount.toString(),\n        });\n        return;\n      }\n\n      spendAmount += currentAmount;\n      explanation.outputs.push({\n        address: currentAddress,\n        amount: currentAmount.toString(),\n      });\n    });\n    explanation.outputAmount = spendAmount.toString();\n    explanation.changeAmount = changeAmount.toString();\n\n    // add fee info if available\n    if (params.feeInfo) {\n      explanation.displayOrder.push('fee');\n      explanation.fee = params.feeInfo;\n    }\n\n    if (_.isInteger(transaction.locktime) && transaction.locktime > 0) {\n      explanation.locktime = transaction.locktime;\n      explanation.displayOrder.push('locktime');\n    }\n\n    const prevOutputs = params.txInfo?.unspents.map((u) => toOutput<TNumber>(u, this.network));\n\n    // if keys are provided, prepare the keys for input signature checking\n    const keys = params.pubs?.map((xpub) => bip32.fromBase58(xpub));\n    const walletKeys = keys && keys.length === 3 ? new bitgo.RootWalletKeys(keys as Triple<BIP32Interface>) : undefined;\n\n    // get the number of signatures per input\n    const inputSignatureCounts = transaction.ins.map((input, idx): number => {\n      if (unspents.length !== transaction.ins.length) {\n        return 0;\n      }\n\n      if (!prevOutputs) {\n        throw new Error(`invalid state`);\n      }\n\n      if (!walletKeys) {\n        // no pub keys or incorrect number of pub keys\n        return 0;\n      }\n\n      try {\n        return verifySignatureWithUnspent<TNumber>(transaction, idx, unspents, walletKeys).filter((v) => v).length;\n      } catch (e) {\n        // some other error occurred and we can't validate the signatures\n        return 0;\n      }\n    });\n\n    explanation.inputSignatures = inputSignatureCounts;\n    explanation.signatures = _.max(inputSignatureCounts) as number;\n    return explanation;\n  }\n\n  /**\n   * Create a multisig address of a given type from a list of keychains and a signing threshold\n   * @param addressType\n   * @param signatureThreshold\n   * @param keys\n   */\n  createMultiSigAddress(addressType: ScriptType2Of3, signatureThreshold: number, keys: Buffer[]): MultiSigAddress {\n    const {\n      scriptPubKey: outputScript,\n      redeemScript,\n      witnessScript,\n    } = utxolib.bitgo.outputScripts.createOutputScript2of3(keys, addressType);\n\n    return {\n      outputScript,\n      redeemScript,\n      witnessScript,\n      address: utxolib.address.fromOutputScript(outputScript, this.network),\n    };\n  }\n\n  /**\n   * @deprecated - use {@see backupKeyRecovery}\n   * Builds a funds recovery transaction without BitGo\n   * @param params - {@see backupKeyRecovery}\n   */\n  async recover(params: RecoverParams): ReturnType<typeof backupKeyRecovery> {\n    return backupKeyRecovery(this, this.bitgo, params);\n  }\n\n  /**\n   * Recover coin that was sent to wrong chain\n   * @param params\n   * @param params.txid The txid of the faulty transaction\n   * @param params.recoveryAddress address to send recovered funds to\n   * @param params.wallet the wallet that received the funds\n   * @param params.recoveryCoin the coin type of the wallet that received the funds\n   * @param params.signed return a half-signed transaction (default=true)\n   * @param params.walletPassphrase the wallet passphrase\n   * @param params.xprv the unencrypted xprv (used instead of wallet passphrase)\n   * @returns {*}\n   */\n  async recoverFromWrongChain<TNumber extends number | bigint = number>(\n    params: RecoverFromWrongChainOptions\n  ): Promise<CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>> {\n    const { txid, recoveryAddress, wallet, walletPassphrase, xprv } = params;\n\n    // params.recoveryCoin used to be params.coin, backwards compatibility\n    const recoveryCoin = params.coin || params.recoveryCoin;\n    if (!recoveryCoin) {\n      throw new Error('missing required object recoveryCoin');\n    }\n    // signed should default to true, and only be disabled if explicitly set to false (not undefined)\n    const signed = params.signed !== false;\n\n    const sourceCoinFamily = this.getFamily();\n    const recoveryCoinFamily = recoveryCoin.getFamily();\n    const supportedRecoveryCoins = supportedCrossChainRecoveries[sourceCoinFamily];\n\n    if (_.isUndefined(supportedRecoveryCoins) || !supportedRecoveryCoins.includes(recoveryCoinFamily)) {\n      throw new Error(`Recovery of ${sourceCoinFamily} balances from ${recoveryCoinFamily} wallets is not supported.`);\n    }\n\n    return await recoverCrossChain<TNumber>(this.bitgo, {\n      sourceCoin: this,\n      recoveryCoin,\n      walletId: wallet,\n      txid,\n      recoveryAddress,\n      walletPassphrase: signed ? walletPassphrase : undefined,\n      xprv: signed ? xprv : undefined,\n    });\n  }\n\n  /**\n   * Generate bip32 key pair\n   *\n   * @param seed\n   * @returns {Object} object with generated pub and prv\n   */\n  generateKeyPair(seed: Buffer): { pub: string; prv: string } {\n    if (!seed) {\n      // An extended private key has both a normal 256 bit private key and a 256\n      // bit chain code, both of which must be random. 512 bits is therefore the\n      // maximum entropy and gives us maximum security against cracking.\n      seed = randomBytes(512 / 8);\n    }\n    const extendedKey = bip32.fromSeed(seed);\n    return {\n      pub: extendedKey.neutered().toBase58(),\n      prv: extendedKey.toBase58(),\n    };\n  }\n\n  async getExtraPrebuildParams(buildParams: ExtraPrebuildParamsOptions): Promise<any> {\n    return {};\n  }\n\n  preCreateBitGo(params: PrecreateBitGoOptions): void {\n    return;\n  }\n\n  async presignTransaction(params: PresignTransactionOptions): Promise<any> {\n    return params;\n  }\n\n  async supplementGenerateWallet(\n    walletParams: SupplementGenerateWalletOptions,\n    keychains: KeychainsTriplet\n  ): Promise<any> {\n    return walletParams;\n  }\n\n  transactionDataAllowed(): boolean {\n    return false;\n  }\n\n  valuelessTransferAllowed(): boolean {\n    return false;\n  }\n\n  getRecoveryProvider(apiToken?: string): RecoveryProvider {\n    return forCoin(this.getChain(), apiToken);\n  }\n}\n"]}
|
|
1075
|
+
/**
|
|
1076
|
+
* Key Value: Unsigned tx id => PSBT
|
|
1077
|
+
* It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
|
|
1078
|
+
* Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.
|
|
1079
|
+
* For more info, check SignTransactionOptions.signingStep
|
|
1080
|
+
*
|
|
1081
|
+
* TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.
|
|
1082
|
+
*/
|
|
1083
|
+
AbstractUtxoCoin.PSBT_CACHE = new Map();
|
|
1084
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractUtxoCoin.js","sourceRoot":"","sources":["../../src/abstractUtxoCoin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip32, BIP32Interface, bitgo, getMainnet, isMainnet, isTestnet } from '@bitgo-beta/utxo-lib';\nimport * as assert from 'assert';\nimport * as bitcoinMessage from 'bitcoinjs-message';\nimport { randomBytes } from 'crypto';\nimport * as debugLib from 'debug';\nimport * as _ from 'lodash';\nimport BigNumber from 'bignumber.js';\n\nimport {\n  CrossChainRecoverySigned,\n  CrossChainRecoveryUnsigned,\n  forCoin,\n  recoverCrossChain,\n  RecoveryProvider,\n  backupKeyRecovery,\n  RecoverParams,\n  V1RecoverParams,\n  v1BackupKeyRecovery,\n  V1SweepParams,\n  v1Sweep,\n} from './recovery';\n\nimport {\n  AddressCoinSpecific,\n  AddressTypeChainMismatchError,\n  BaseCoin,\n  BitGoBase,\n  decryptKeychainPrivateKey,\n  ExtraPrebuildParamsOptions,\n  HalfSignedUtxoTransaction,\n  IBaseCoin,\n  InvalidAddressDerivationPropertyError,\n  InvalidAddressError,\n  IRequestTracer,\n  isTriple,\n  ITransactionExplanation as BaseTransactionExplanation,\n  IWallet,\n  Keychain,\n  KeychainsTriplet,\n  KeyIndices,\n  P2shP2wshUnsupportedError,\n  P2trMusig2UnsupportedError,\n  P2trUnsupportedError,\n  P2wshUnsupportedError,\n  ParsedTransaction as BaseParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  PrecreateBitGoOptions,\n  PresignTransactionOptions,\n  promiseProps,\n  RequestTracer,\n  sanitizeLegacyPath,\n  SignedTransaction,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  SupplementGenerateWalletOptions,\n  TransactionParams as BaseTransactionParams,\n  TransactionPrebuild as BaseTransactionPrebuild,\n  TransactionRecipient,\n  Triple,\n  UnexpectedAddressError,\n  UnsupportedAddressTypeError,\n  VerificationOptions,\n  VerifyAddressOptions as BaseVerifyAddressOptions,\n  VerifyTransactionOptions as BaseVerifyTransactionOptions,\n  Wallet,\n  WalletData,\n} from '@bitgo-beta/sdk-core';\nimport { CustomChangeOptions, parseOutput } from './parseOutput';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nimport ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\nimport { isReplayProtectionUnspent } from './replayProtection';\nimport { signAndVerifyPsbt, signAndVerifyWalletTransaction } from './sign';\nimport { supportedCrossChainRecoveries } from './config';\nimport { explainPsbt, explainTx, getPsbtTxInputs, getTxInputs } from './transaction';\nimport { assertDescriptorWalletAddress } from './descriptor/assertDescriptorWalletAddress';\n\ntype UtxoCustomSigningFunction<TNumber extends number | bigint> = {\n  (params: {\n    coin: IBaseCoin;\n    txPrebuild: TransactionPrebuild<TNumber>;\n    pubs?: string[];\n    /**\n     * signingStep flag becomes applicable when both of the following conditions are met:\n     * 1) When the external express signer is activated\n     * 2) When the PSBT includes at least one taprootKeyPathSpend input.\n     *\n     * The signing process of a taprootKeyPathSpend input is a 4-step sequence:\n     * i) user nonce generation - signerNonce - this is the first call to external express signer signTransaction\n     * ii) bitgo nonce generation - cosignerNonce - this is the first and only call to local signTransaction\n     * iii) user signature - signerSignature - this is the second call to external express signer signTransaction\n     * iv) bitgo signature - not in signTransaction method’s scope\n     *\n     * In the absence of this flag, the aforementioned first three sequence is executed in a single signTransaction call.\n     *\n     * NOTE: We make a strong assumption that the external express signer and its caller uses sticky sessions,\n     * since PSBTs are cached in step 1 to be used in step 3 for MuSig2 user secure nonce access.\n     */\n    signingStep?: 'signerNonce' | 'signerSignature' | 'cosignerNonce';\n  }): Promise<SignedTransaction>;\n};\n\nconst { getExternalChainCode, isChainCode, scriptTypeForChain, outputScripts } = bitgo;\ntype Unspent<TNumber extends number | bigint = number> = bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = bitgo.RootWalletKeys;\n\nexport type UtxoCoinSpecific = AddressCoinSpecific | DescriptorAddressCoinSpecific;\n\nexport interface VerifyAddressOptions<TCoinSpecific extends UtxoCoinSpecific> extends BaseVerifyAddressOptions {\n  chain?: number;\n  index: number;\n  coinSpecific?: TCoinSpecific;\n}\n\nexport interface BaseOutput {\n  address: string;\n  amount: string | number;\n  // Even though this external flag is redundant with the chain property, it is necessary for backwards compatibility\n  // with legacy transaction format.\n  external?: boolean;\n}\n\nexport interface WalletOutput extends BaseOutput {\n  needsCustomChangeKeySignatureVerification?: boolean;\n  chain: number;\n  index: number;\n}\n\nexport type Output = BaseOutput | WalletOutput;\n\nexport function isWalletOutput(output: Output): output is WalletOutput {\n  return (output as WalletOutput).chain !== undefined && (output as WalletOutput).index !== undefined;\n}\n\nexport interface TransactionExplanation extends BaseTransactionExplanation<string, string> {\n  locktime: number;\n  outputs: Output[];\n  changeOutputs: Output[];\n\n  /**\n   * Number of input signatures per input.\n   */\n  inputSignatures: number[];\n\n  /**\n   * Highest input signature count for the transaction\n   */\n  signatures: number;\n}\n\nexport interface TransactionInfo<TNumber extends number | bigint = number> {\n  /** Maps txid to txhex. Required for offline signing. */\n  txHexes?: Record<string, string>;\n  changeAddresses?: string[];\n  /** psbt does not require unspents. */\n  unspents?: Unspent<TNumber>[];\n}\n\nexport interface ExplainTransactionOptions<TNumber extends number | bigint = number> {\n  txHex: string;\n  txInfo?: TransactionInfo<TNumber>;\n  feeInfo?: string;\n  pubs?: Triple<string>;\n}\n\nexport interface DecoratedExplainTransactionOptions<TNumber extends number | bigint = number>\n  extends ExplainTransactionOptions<TNumber> {\n  changeInfo?: { address: string; chain: number; index: number }[];\n}\n\nexport type UtxoNetwork = utxolib.Network;\n\nexport interface TransactionPrebuild<TNumber extends number | bigint = number> extends BaseTransactionPrebuild {\n  txInfo?: TransactionInfo<TNumber>;\n  blockHeight?: number;\n}\n\nexport interface TransactionParams extends BaseTransactionParams {\n  walletPassphrase?: string;\n  allowExternalChangeAddress?: boolean;\n  changeAddress?: string;\n  rbfTxIds?: string[];\n}\n\n// parseTransactions' return type makes use of WalletData's type but with customChangeKeySignatures as required.\nexport interface AbstractUtxoCoinWalletData extends WalletData {\n  customChangeKeySignatures: {\n    user: string;\n    backup: string;\n    bitgo: string;\n  };\n}\n\nexport class AbstractUtxoCoinWallet extends Wallet {\n  public _wallet: AbstractUtxoCoinWalletData;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin, walletData: any) {\n    super(bitgo, baseCoin, walletData);\n  }\n}\n\nexport interface ParseTransactionOptions<TNumber extends number | bigint = number> extends BaseParseTransactionOptions {\n  txParams: TransactionParams;\n  txPrebuild: TransactionPrebuild<TNumber>;\n  wallet: AbstractUtxoCoinWallet;\n  verification?: VerificationOptions;\n  reqId?: IRequestTracer;\n}\n\nexport interface ParsedTransaction<TNumber extends number | bigint = number> extends BaseParsedTransaction {\n  keychains: {\n    user?: Keychain;\n    backup?: Keychain;\n    bitgo?: Keychain;\n  };\n  keySignatures: {\n    backupPub?: string;\n    bitgoPub?: string;\n  };\n  outputs: Output[];\n  missingOutputs: Output[];\n  explicitExternalOutputs: Output[];\n  implicitExternalOutputs: Output[];\n  changeOutputs: Output[];\n  explicitExternalSpendAmount: TNumber;\n  implicitExternalSpendAmount: TNumber;\n  needsCustomChangeKeySignatureVerification: boolean;\n  customChange?: CustomChangeOptions;\n}\n\nexport interface GenerateAddressOptions {\n  addressType?: ScriptType2Of3;\n  threshold?: number;\n  chain?: number;\n  index?: number;\n  segwit?: boolean;\n  bech32?: boolean;\n}\n\nexport interface GenerateFixedScriptAddressOptions extends GenerateAddressOptions {\n  keychains: {\n    pub: string;\n    aspKeyId?: string;\n  }[];\n}\n\nexport interface AddressDetails {\n  address: string;\n  chain: number;\n  index: number;\n  coin: string;\n  coinSpecific: AddressCoinSpecific | DescriptorAddressCoinSpecific;\n  addressType?: string;\n}\n\nexport interface DescriptorAddressCoinSpecific extends AddressCoinSpecific {\n  descriptorName: string;\n  descriptorChecksum: string;\n}\n\ntype UtxoBaseSignTransactionOptions<TNumber extends number | bigint = number> = BaseSignTransactionOptions & {\n  /** Transaction prebuild from bitgo server */\n  txPrebuild: {\n    /**\n     * walletId is required in following 2 scenarios.\n     * 1. External signer express mode is used.\n     * 2. bitgo MuSig2 nonce is requested\n     */\n    walletId?: string;\n    txHex: string;\n    txInfo?: TransactionInfo<TNumber>;\n  };\n  /** xpubs triple for wallet (user, backup, bitgo). Required only when txPrebuild.txHex is not a PSBT */\n  pubs?: Triple<string>;\n  /** xpub for cosigner (defaults to bitgo) */\n  cosignerPub?: string;\n  /**\n   * When true, creates full-signed transaction without placeholder signatures.\n   * When false, creates half-signed transaction with placeholder signatures.\n   */\n  isLastSignature?: boolean;\n  /**\n   * If true, allows signing a non-segwit input with a witnessUtxo instead requiring a previous\n   * transaction (nonWitnessUtxo)\n   */\n  allowNonSegwitSigningWithoutPrevTx?: boolean;\n};\n\nexport type SignTransactionOptions<TNumber extends number | bigint = number> = UtxoBaseSignTransactionOptions<TNumber> &\n  (\n    | {\n        prv: string;\n        signingStep?: 'signerNonce' | 'signerSignature';\n      }\n    | {\n        signingStep: 'cosignerNonce';\n      }\n  );\n\nexport interface MultiSigAddress {\n  outputScript: Buffer;\n  redeemScript?: Buffer;\n  witnessScript?: Buffer;\n  address: string;\n}\n\nexport interface RecoverFromWrongChainOptions {\n  txid: string;\n  recoveryAddress: string;\n  wallet: string;\n  walletPassphrase?: string;\n  xprv?: string;\n  apiKey?: string;\n  /** @deprecated */\n  coin?: AbstractUtxoCoin;\n  recoveryCoin?: AbstractUtxoCoin;\n  signed?: boolean;\n}\n\nexport interface VerifyKeySignaturesOptions {\n  userKeychain: { pub?: string };\n  keychainToVerify: { pub?: string };\n  keySignature: string;\n}\n\nexport interface VerifyUserPublicKeyOptions {\n  userKeychain?: Keychain;\n  disableNetworking: boolean;\n  txParams: TransactionParams;\n}\n\nexport interface VerifyTransactionOptions<TNumber extends number | bigint = number>\n  extends BaseVerifyTransactionOptions {\n  txPrebuild: TransactionPrebuild<TNumber>;\n  txParams: TransactionParams;\n  wallet: AbstractUtxoCoinWallet;\n}\n\nexport interface SignPsbtRequest {\n  psbt: string;\n}\n\nexport interface SignPsbtResponse {\n  psbt: string;\n}\n\nexport abstract class AbstractUtxoCoin extends BaseCoin {\n  public altScriptHash?: number;\n  public supportAltScriptDestination?: boolean;\n  public readonly amountType: 'number' | 'bigint';\n  private readonly _network: utxolib.Network;\n\n  protected constructor(bitgo: BitGoBase, network: utxolib.Network, amountType: 'number' | 'bigint' = 'number') {\n    super(bitgo);\n    if (!utxolib.isValidNetwork(network)) {\n      throw new Error(\n        'invalid network: please make sure to use the same version of ' +\n          '@bitgo-beta/utxo-lib as this library when initializing an instance of this class'\n      );\n    }\n    this.amountType = amountType;\n    this._network = network;\n  }\n\n  /**\n   * Key Value: Unsigned tx id => PSBT\n   * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.\n   * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.\n   * For more info, check SignTransactionOptions.signingStep\n   *\n   * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.\n   */\n  private static readonly PSBT_CACHE = new Map<string, utxolib.bitgo.UtxoPsbt>();\n\n  get network() {\n    return this._network;\n  }\n\n  sweepWithSendMany(): boolean {\n    return true;\n  }\n\n  /** @deprecated */\n  static get validAddressTypes(): ScriptType2Of3[] {\n    return [...outputScripts.scriptTypes2Of3];\n  }\n\n  /**\n   * Returns the factor between the base unit and its smallest subdivison\n   * @return {number}\n   */\n  getBaseFactor() {\n    return 1e8;\n  }\n\n  /**\n   * @deprecated\n   */\n  getCoinLibrary() {\n    return utxolib;\n  }\n\n  /**\n   * Check if an address is valid\n   * @param address\n   * @param param\n   */\n  isValidAddress(address: string, param?: { anyFormat: boolean } | /* legacy parameter */ boolean): boolean {\n    if (typeof param === 'boolean' && param) {\n      throw new Error('deprecated');\n    }\n\n    const formats = param && param.anyFormat ? undefined : ['default' as const];\n    try {\n      const script = utxolib.addressFormat.toOutputScriptTryFormats(address, this.network, formats);\n      return address === utxolib.address.fromOutputScript(script, this.network);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  isValidPub(pub: string) {\n    try {\n      return bip32.fromBase58(pub).isNeutered();\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Get the latest block height\n   * @param reqId\n   */\n  async getLatestBlockHeight(reqId?: RequestTracer): Promise<number> {\n    if (reqId) {\n      this.bitgo.setRequestTracer(reqId);\n    }\n    const chainhead = await this.bitgo.get(this.url('/public/block/latest')).result();\n    return (chainhead as any).height;\n  }\n\n  /**\n   * Run custom coin logic after a transaction prebuild has been received from BitGo\n   * @param prebuild\n   */\n  async postProcessPrebuild<TNumber extends number | bigint>(\n    prebuild: TransactionPrebuild<TNumber>\n  ): Promise<TransactionPrebuild<TNumber>> {\n    if (_.isUndefined(prebuild.txHex)) {\n      throw new Error('missing required txPrebuild property txHex');\n    }\n    const tx = bitgo.isPsbt(prebuild.txHex)\n      ? bitgo.createPsbtFromHex(prebuild.txHex, this.network)\n      : this.createTransactionFromHex<TNumber>(prebuild.txHex);\n    if (_.isUndefined(prebuild.blockHeight)) {\n      prebuild.blockHeight = (await this.getLatestBlockHeight()) as number;\n    }\n    return _.extend({}, prebuild, { txHex: tx.toHex() });\n  }\n\n  /**\n   * @param first\n   * @param second\n   * @returns {Array} All outputs that are in the first array but not in the second\n   */\n  protected static outputDifference(first: Output[], second: Output[]): Output[] {\n    const keyFunc = ({ address, amount }: Output): string => `${address}:${amount}`;\n    const groupedOutputs = _.groupBy(first, keyFunc);\n\n    second.forEach((output) => {\n      const group = groupedOutputs[keyFunc(output)];\n      if (group) {\n        group.pop();\n      }\n    });\n\n    return _.flatten(_.values(groupedOutputs));\n  }\n\n  /**\n   * Determine an address' type based on its witness and redeem script presence\n   * @param addressDetails\n   */\n  static inferAddressType(addressDetails: { chain: number }): ScriptType2Of3 | null {\n    return isChainCode(addressDetails.chain) ? scriptTypeForChain(addressDetails.chain) : null;\n  }\n\n  createTransactionFromHex<TNumber extends number | bigint = number>(\n    hex: string\n  ): utxolib.bitgo.UtxoTransaction<TNumber> {\n    return utxolib.bitgo.createTransactionFromHex<TNumber>(hex, this.network, this.amountType);\n  }\n\n  /**\n   * Extract and fill transaction details such as internal/change spend, external spend (explicit vs. implicit), etc.\n   * @param params\n   * @returns {*}\n   */\n  async parseTransaction<TNumber extends number | bigint = number>(\n    params: ParseTransactionOptions<TNumber>\n  ): Promise<ParsedTransaction<TNumber>> {\n    const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;\n\n    if (!_.isUndefined(verification.disableNetworking) && !_.isBoolean(verification.disableNetworking)) {\n      throw new Error('verification.disableNetworking must be a boolean');\n    }\n    const disableNetworking = verification.disableNetworking;\n\n    const fetchKeychains = async (wallet: IWallet): Promise<VerificationOptions['keychains']> => {\n      return promiseProps({\n        user: this.keychains().get({ id: wallet.keyIds()[KeyIndices.USER], reqId }),\n        backup: this.keychains().get({ id: wallet.keyIds()[KeyIndices.BACKUP], reqId }),\n        bitgo: this.keychains().get({ id: wallet.keyIds()[KeyIndices.BITGO], reqId }),\n      });\n    };\n\n    // obtain the keychains and key signatures\n    let keychains: VerificationOptions['keychains'] | undefined = verification.keychains;\n    if (!keychains) {\n      if (disableNetworking) {\n        throw new Error('cannot fetch keychains without networking');\n      }\n      keychains = await fetchKeychains(wallet);\n    }\n\n    if (!keychains || !keychains.user || !keychains.backup || !keychains.bitgo) {\n      throw new Error('keychains are required, but could not be fetched');\n    }\n\n    const keychainArray: Triple<Keychain> = [keychains.user, keychains.backup, keychains.bitgo];\n\n    const keySignatures = _.get(wallet, '_wallet.keySignatures', {});\n\n    if (_.isUndefined(txPrebuild.txHex)) {\n      throw new Error('missing required txPrebuild property txHex');\n    }\n    // obtain all outputs\n    const explanation: TransactionExplanation = await this.explainTransaction<TNumber>({\n      txHex: txPrebuild.txHex,\n      txInfo: txPrebuild.txInfo,\n      pubs: keychainArray.map((k) => k.pub) as Triple<string>,\n    });\n    const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];\n\n    let expectedOutputs;\n    if (txParams.rbfTxIds) {\n      assert(txParams.rbfTxIds.length === 1);\n\n      const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });\n      expectedOutputs = txToBeReplaced.outputs\n        .filter((output) => output.wallet !== wallet.id()) // For self-sends, the walletId will be the same as the wallet's id\n        .map((output) => {\n          return { amount: BigInt(output.valueString), address: this.canonicalAddress(output.address) };\n        });\n    } else {\n      // verify that each recipient from txParams has their own output\n      expectedOutputs = _.get(txParams, 'recipients', [] as TransactionRecipient[]).map((output) => {\n        return { ...output, address: this.canonicalAddress(output.address) };\n      });\n      if (params.txParams.allowExternalChangeAddress && params.txParams.changeAddress) {\n        // when an external change address is explicitly specified, count all outputs going towards that\n        // address in the expected outputs (regardless of the output amount)\n        expectedOutputs.push(\n          ...allOutputs\n            .map((output) => {\n              return { ...output, address: this.canonicalAddress(output.address) };\n            })\n            .filter((output) => output.address === this.canonicalAddress(params.txParams.changeAddress as string))\n        );\n      }\n    }\n\n    const missingOutputs = AbstractUtxoCoin.outputDifference(expectedOutputs, allOutputs);\n\n    // get the keychains from the custom change wallet if needed\n    let customChange: CustomChangeOptions | undefined;\n    const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};\n    if (customChangeWalletId) {\n      // fetch keychains from custom change wallet for deriving addresses.\n      // These keychains should be signed and this should be verified in verifyTransaction\n      const customChangeKeySignatures = wallet._wallet.customChangeKeySignatures;\n      const customChangeWallet: Wallet = await this.wallets().get({ id: customChangeWalletId });\n      const customChangeKeys = await fetchKeychains(customChangeWallet);\n\n      if (!customChangeKeys) {\n        throw new Error('failed to fetch keychains for custom change wallet');\n      }\n\n      if (customChangeKeys.user && customChangeKeys.backup && customChangeKeys.bitgo && customChangeWallet) {\n        const customChangeKeychains: [Keychain, Keychain, Keychain] = [\n          customChangeKeys.user,\n          customChangeKeys.backup,\n          customChangeKeys.bitgo,\n        ];\n\n        customChange = {\n          keys: customChangeKeychains,\n          signatures: [\n            customChangeKeySignatures.user,\n            customChangeKeySignatures.backup,\n            customChangeKeySignatures.bitgo,\n          ],\n        };\n      }\n    }\n\n    /**\n     * Loop through all the outputs and classify each of them as either internal spends\n     * or external spends by setting the \"external\" property to true or false on the output object.\n     */\n    const allOutputDetails: Output[] = await Promise.all(\n      allOutputs.map((currentOutput) => {\n        return parseOutput({\n          currentOutput,\n          coin: this,\n          txPrebuild,\n          verification,\n          keychainArray,\n          wallet,\n          txParams,\n          customChange,\n          reqId,\n        });\n      })\n    );\n\n    const needsCustomChangeKeySignatureVerification = allOutputDetails.some(\n      (output) => (output as WalletOutput)?.needsCustomChangeKeySignatureVerification\n    );\n\n    const changeOutputs = _.filter(allOutputDetails, { external: false });\n\n    // these are all the outputs that were not originally explicitly specified in recipients\n    // ideally change outputs or a paygo output that might have been added\n    const implicitOutputs = AbstractUtxoCoin.outputDifference(allOutputDetails, expectedOutputs);\n\n    const explicitOutputs = AbstractUtxoCoin.outputDifference(allOutputDetails, implicitOutputs);\n\n    // these are all the non-wallet outputs that had been originally explicitly specified in recipients\n    const explicitExternalOutputs = _.filter(explicitOutputs, { external: true });\n\n    // this is the sum of all the originally explicitly specified non-wallet output values\n    const explicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n      explicitExternalOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0)) as bigint,\n      this.amountType\n    );\n\n    /**\n     * The calculation of the implicit external spend amount pertains to verifying the pay-as-you-go-fee BitGo\n     * automatically applies to transactions sending money out of the wallet. The logic is fairly straightforward\n     * in that we compare the external spend amount that was specified explicitly by the user to the portion\n     * that was specified implicitly. To protect customers from people tampering with the transaction outputs, we\n     * define a threshold for the maximum percentage of the implicit external spend in relation to the explicit\n     * external spend.\n     */\n\n    // make sure that all the extra addresses are change addresses\n    // get all the additional external outputs the server added and calculate their values\n    const implicitExternalOutputs = _.filter(implicitOutputs, { external: true });\n    const implicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n      implicitExternalOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0)) as bigint,\n      this.amountType\n    );\n\n    return {\n      keychains,\n      keySignatures,\n      outputs: allOutputDetails,\n      missingOutputs,\n      explicitExternalOutputs,\n      implicitExternalOutputs,\n      changeOutputs,\n      explicitExternalSpendAmount,\n      implicitExternalSpendAmount,\n      needsCustomChangeKeySignatureVerification,\n      customChange,\n    };\n  }\n\n  /**\n   * Decrypt the wallet's user private key and verify that the claimed public key matches\n   * @param {VerifyUserPublicKeyOptions} params\n   * @return {boolean}\n   * @protected\n   */\n  protected verifyUserPublicKey(params: VerifyUserPublicKeyOptions): boolean {\n    const { userKeychain, txParams, disableNetworking } = params;\n    if (!userKeychain) {\n      throw new Error('user keychain is required');\n    }\n\n    const userPub = userKeychain.pub;\n\n    // decrypt the user private key, so we can verify that the claimed public key is a match\n    let userPrv = userKeychain.prv;\n    if (!userPrv && txParams.walletPassphrase) {\n      userPrv = decryptKeychainPrivateKey(this.bitgo, userKeychain, txParams.walletPassphrase);\n    }\n\n    if (!userPrv) {\n      const errorMessage = 'user private key unavailable for verification';\n      if (disableNetworking) {\n        console.log(errorMessage);\n        return false;\n      } else {\n        throw new Error(errorMessage);\n      }\n    } else {\n      const userPrivateKey = bip32.fromBase58(userPrv);\n      if (userPrivateKey.toBase58() === userPrivateKey.neutered().toBase58()) {\n        throw new Error('user private key is only public');\n      }\n      if (userPrivateKey.neutered().toBase58() !== userPub) {\n        throw new Error('user private key does not match public key');\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Verify signatures produced by the user key over the backup and bitgo keys.\n   *\n   * If set, these signatures ensure that the wallet keys cannot be changed after the wallet has been created.\n   * @param {VerifyKeySignaturesOptions} params\n   * @return {{backup: boolean, bitgo: boolean}}\n   */\n  public verifyKeySignature(params: VerifyKeySignaturesOptions): boolean {\n    // first, let's verify the integrity of the user key, whose public key is used for subsequent verifications\n    const { userKeychain, keychainToVerify, keySignature } = params;\n    if (!userKeychain) {\n      throw new Error('user keychain is required');\n    }\n\n    if (!keychainToVerify) {\n      throw new Error('keychain to verify is required');\n    }\n\n    if (!keySignature) {\n      throw new Error('key signature is required');\n    }\n\n    // verify the signature against the user public key\n    assert(userKeychain.pub);\n    const publicKey = bip32.fromBase58(userKeychain.pub).publicKey;\n    // Due to interface of `bitcoinMessage`, we need to convert the public key to an address.\n    // Note that this address has no relationship to on-chain transactions. We are\n    // only interested in the address as a representation of the public key.\n    const signingAddress = utxolib.address.toBase58Check(\n      utxolib.crypto.hash160(publicKey),\n      utxolib.networks.bitcoin.pubKeyHash,\n      // we do not pass `this.network` here because it would fail for zcash\n      // the bitcoinMessage library decodes the address and throws away the first byte\n      // because zcash has a two-byte prefix, verify() decodes zcash addresses to an invalid pubkey hash\n      utxolib.networks.bitcoin\n    );\n\n    // BG-5703: use BTC mainnet prefix for all key signature operations\n    // (this means do not pass a prefix parameter, and let it use the default prefix instead)\n    assert(keychainToVerify.pub);\n    try {\n      return bitcoinMessage.verify(keychainToVerify.pub, signingAddress, Buffer.from(keySignature, 'hex'));\n    } catch (e) {\n      debug('error thrown from bitcoinmessage while verifying key signature', e);\n      return false;\n    }\n  }\n\n  /**\n   * Verify signatures against the user private key over the change wallet extended keys\n   * @param {ParsedTransaction} tx\n   * @param {Keychain} userKeychain\n   * @return {boolean}\n   * @protected\n   */\n  protected verifyCustomChangeKeySignatures<TNumber extends number | bigint>(\n    tx: ParsedTransaction<TNumber>,\n    userKeychain: Keychain\n  ): boolean {\n    if (!tx.customChange) {\n      throw new Error('parsed transaction is missing required custom change verification data');\n    }\n\n    if (!Array.isArray(tx.customChange.keys) || !Array.isArray(tx.customChange.signatures)) {\n      throw new Error('customChange property is missing keys or signatures');\n    }\n\n    for (const keyIndex of [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO]) {\n      const keychainToVerify = tx.customChange.keys[keyIndex];\n      const keySignature = tx.customChange.signatures[keyIndex];\n      if (!keychainToVerify) {\n        throw new Error(`missing required custom change ${KeyIndices[keyIndex].toLowerCase()} keychain public key`);\n      }\n      if (!keySignature) {\n        throw new Error(`missing required custom change ${KeyIndices[keyIndex].toLowerCase()} keychain signature`);\n      }\n      if (\n        !this.verifyKeySignature({\n          userKeychain: userKeychain as { pub: string },\n          keychainToVerify: keychainToVerify as { pub: string },\n          keySignature,\n        })\n      ) {\n        debug('failed to verify custom change %s key signature!', KeyIndices[keyIndex].toLowerCase());\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  /**\n   * Get the maximum percentage limit for pay-as-you-go outputs\n   *\n   * @protected\n   */\n  protected getPayGoLimit(allowPaygoOutput?: boolean): number {\n    // allowing paygo outputs needs to be the default behavior, so only disallow paygo outputs if the\n    // relevant verification option is both set and false\n    if (!_.isNil(allowPaygoOutput) && !allowPaygoOutput) {\n      return 0;\n    }\n    // 150 basis points is the absolute permitted maximum if paygo outputs are allowed\n    return 0.015;\n  }\n\n  /**\n   * Verify that a transaction prebuild complies with the original intention\n   *\n   * @param params\n   * @param params.txParams params object passed to send\n   * @param params.txPrebuild prebuild object returned by server\n   * @param params.txPrebuild.txHex prebuilt transaction's txHex form\n   * @param params.wallet Wallet object to obtain keys to verify against\n   * @param params.verification Object specifying some verification parameters\n   * @param params.verification.disableNetworking Disallow fetching any data from the internet for verification purposes\n   * @param params.verification.keychains Pass keychains manually rather than fetching them by id\n   * @param params.verification.addresses Address details to pass in for out-of-band verification\n   * @returns {boolean}\n   */\n  async verifyTransaction<TNumber extends number | bigint = number>(\n    params: VerifyTransactionOptions<TNumber>\n  ): Promise<boolean> {\n    const { txParams, txPrebuild, wallet, verification = { allowPaygoOutput: true }, reqId } = params;\n    const isPsbt = txPrebuild.txHex && bitgo.isPsbt(txPrebuild.txHex);\n    if (isPsbt && txPrebuild.txInfo?.unspents) {\n      throw new Error('should not have unspents in txInfo for psbt');\n    }\n\n    const disableNetworking = !!verification.disableNetworking;\n    const parsedTransaction: ParsedTransaction<TNumber> = await this.parseTransaction<TNumber>({\n      txParams,\n      txPrebuild,\n      wallet,\n      verification,\n      reqId,\n    });\n\n    const keychains = parsedTransaction.keychains;\n\n    // verify that the claimed user public key corresponds to the wallet's user private key\n    let userPublicKeyVerified = false;\n    try {\n      // verify the user public key matches the private key - this will throw if there is no match\n      userPublicKeyVerified = this.verifyUserPublicKey({ userKeychain: keychains.user, disableNetworking, txParams });\n    } catch (e) {\n      debug('failed to verify user public key!', e);\n    }\n\n    // let's verify these keychains\n    const keySignatures = parsedTransaction.keySignatures;\n    if (!_.isEmpty(keySignatures)) {\n      const verify = (key, pub) => {\n        if (!keychains.user || !keychains.user.pub) {\n          throw new Error('missing user keychain');\n        }\n        return this.verifyKeySignature({\n          userKeychain: keychains.user as { pub: string },\n          keychainToVerify: key,\n          keySignature: pub,\n        });\n      };\n      const isBackupKeySignatureValid = verify(keychains.backup, keySignatures.backupPub);\n      const isBitgoKeySignatureValid = verify(keychains.bitgo, keySignatures.bitgoPub);\n      if (!isBackupKeySignatureValid || !isBitgoKeySignatureValid) {\n        throw new Error('secondary public key signatures invalid');\n      }\n      debug('successfully verified backup and bitgo key signatures');\n    } else if (!disableNetworking) {\n      // these keys were obtained online and their signatures were not verified\n      // this could be dangerous\n      console.log('unsigned keys obtained online are being used for address verification');\n    }\n\n    if (parsedTransaction.needsCustomChangeKeySignatureVerification) {\n      if (!keychains.user || !userPublicKeyVerified) {\n        throw new Error('transaction requires verification of user public key, but it was unable to be verified');\n      }\n      const customChangeKeySignaturesVerified = this.verifyCustomChangeKeySignatures(parsedTransaction, keychains.user);\n      if (!customChangeKeySignaturesVerified) {\n        throw new Error(\n          'transaction requires verification of custom change key signatures, but they were unable to be verified'\n        );\n      }\n      debug('successfully verified user public key and custom change key signatures');\n    }\n\n    const missingOutputs = parsedTransaction.missingOutputs;\n    if (missingOutputs.length !== 0) {\n      // there are some outputs in the recipients list that have not made it into the actual transaction\n      throw new Error('expected outputs missing in transaction prebuild');\n    }\n\n    const intendedExternalSpend = parsedTransaction.explicitExternalSpendAmount;\n\n    // this is a limit we impose for the total value that is amended to the transaction beyond what was originally intended\n    const payAsYouGoLimit = new BigNumber(this.getPayGoLimit(verification.allowPaygoOutput)).multipliedBy(\n      intendedExternalSpend.toString()\n    );\n\n    /*\n    Some explanation for why we're doing what we're doing:\n    Some customers will have an output to BitGo's PAYGo wallet added to their transaction, and we need to account for\n    it here. To protect someone tampering with the output to make it send more than it should to BitGo, we define a\n    threshold for the output's value above which we'll throw an error, because the paygo output should never be that\n    high.\n     */\n\n    // make sure that all the extra addresses are change addresses\n    // get all the additional external outputs the server added and calculate their values\n    const nonChangeAmount = new BigNumber(parsedTransaction.implicitExternalSpendAmount.toString());\n\n    debug(\n      'Intended spend is %s, Non-change amount is %s, paygo limit is %s',\n      intendedExternalSpend.toString(),\n      nonChangeAmount.toString(),\n      payAsYouGoLimit.toString()\n    );\n\n    // There are two instances where we will get into this point here\n    if (nonChangeAmount.gt(payAsYouGoLimit)) {\n      if (isPsbt && parsedTransaction.customChange) {\n        // In the case that we have a custom change address on a wallet and we are building the transaction\n        // with a PSBT, we do not have the metadata to verify the address from the custom change wallet, nor\n        // can we fetch that information from the other wallet because we may not have the credentials. Therefore,\n        // we will not throw an error here, but we will log a warning.\n        debug(`cannot verify some of the addresses because it belongs to a separate wallet`);\n      } else {\n        // the additional external outputs can only be BitGo's pay-as-you-go fee, but we cannot verify the wallet address\n        // there are some addresses that are outside the scope of intended recipients that are not change addresses\n        throw new Error('prebuild attempts to spend to unintended external recipients');\n      }\n    }\n\n    const allOutputs = parsedTransaction.outputs;\n    if (!txPrebuild.txHex) {\n      throw new Error(`txPrebuild.txHex not set`);\n    }\n    const inputs = isPsbt\n      ? getPsbtTxInputs(txPrebuild.txHex, this.network).map((v) => ({\n          ...v,\n          value: bitgo.toTNumber(v.value, this.amountType),\n        }))\n      : await getTxInputs({ txPrebuild, bitgo: this.bitgo, coin: this, disableNetworking, reqId });\n    // coins (doge) that can exceed number limits (and thus will use bigint) will have the `valueString` field\n    const inputAmount = inputs.reduce(\n      (sum: bigint, i) => sum + BigInt(this.amountType === 'bigint' ? i.valueString : i.value),\n      BigInt(0)\n    );\n    const outputAmount = allOutputs.reduce((sum: bigint, o: Output) => sum + BigInt(o.amount), BigInt(0));\n    const fee = inputAmount - outputAmount;\n\n    if (fee < 0) {\n      throw new Error(\n        `attempting to spend ${outputAmount} satoshis, which exceeds the input amount (${inputAmount} satoshis) by ${-fee}`\n      );\n    }\n\n    return true;\n  }\n\n  /**\n   * Make sure an address is valid and throw an error if it's not.\n   * @param params.address The address string on the network\n   * @param params.addressType\n   * @param params.keychains Keychain objects with xpubs\n   * @param params.coinSpecific Coin-specific details for the address such as a witness script\n   * @param params.chain Derivation chain\n   * @param params.index Derivation index\n   * @throws {InvalidAddressError}\n   * @throws {InvalidAddressDerivationPropertyError}\n   * @throws {UnexpectedAddressError}\n   */\n  async isWalletAddress(params: VerifyAddressOptions<UtxoCoinSpecific>, wallet?: IWallet): Promise<boolean> {\n    const { address, addressType, keychains, chain, index } = params;\n\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n\n    if (wallet) {\n      const walletCoinSpecific = wallet.coinSpecific();\n      if (walletCoinSpecific && 'descriptors' in walletCoinSpecific) {\n        assertDescriptorWalletAddress(this.network, params, walletCoinSpecific.descriptors);\n        return true;\n      }\n    }\n\n    if ((_.isUndefined(chain) && _.isUndefined(index)) || !(_.isFinite(chain) && _.isFinite(index))) {\n      throw new InvalidAddressDerivationPropertyError(\n        `address validation failure: invalid chain (${chain}) or index (${index})`\n      );\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    const expectedAddress = this.generateAddress({\n      addressType: addressType as ScriptType2Of3,\n      keychains,\n      threshold: 2,\n      chain,\n      index,\n    });\n\n    if (expectedAddress.address !== address) {\n      throw new UnexpectedAddressError(\n        `address validation failure: expected ${expectedAddress.address} but got ${address}`\n      );\n    }\n\n    return true;\n  }\n\n  /**\n   * Indicates whether coin supports a block target\n   * @returns {boolean}\n   */\n  supportsBlockTarget() {\n    return true;\n  }\n\n  /**\n   * @param addressType\n   * @returns true iff coin supports spending from unspentType\n   */\n  supportsAddressType(addressType: ScriptType2Of3): boolean {\n    return utxolib.bitgo.outputScripts.isSupportedScriptType(this.network, addressType);\n  }\n\n  /**\n   * @param chain\n   * @return true iff coin supports spending from chain\n   */\n  supportsAddressChain(chain: number): boolean {\n    return isChainCode(chain) && this.supportsAddressType(utxolib.bitgo.scriptTypeForChain(chain));\n  }\n\n  keyIdsForSigning(): number[] {\n    return [KeyIndices.USER, KeyIndices.BACKUP, KeyIndices.BITGO];\n  }\n\n  /**\n   * TODO(BG-11487): Remove addressType, segwit, and bech32 params in SDKv6\n   * Generate an address for a wallet based on a set of configurations\n   * @param params.addressType {string}   Deprecated\n   * @param params.keychains   {[object]} Array of objects with xpubs\n   * @param params.threshold   {number}   Minimum number of signatures\n   * @param params.chain       {number}   Derivation chain (see https://github.com/BitGo/unspents/blob/master/src/codes.ts for\n   *                                                 the corresponding address type of a given chain code)\n   * @param params.index       {number}   Derivation index\n   * @param params.segwit      {boolean}  Deprecated\n   * @param params.bech32      {boolean}  Deprecated\n   * @returns {{chain: number, index: number, coin: number, coinSpecific: {outputScript, redeemScript}}}\n   */\n  generateAddress(params: GenerateFixedScriptAddressOptions): AddressDetails {\n    let derivationIndex = 0;\n    if (_.isInteger(params.index) && (params.index as number) > 0) {\n      derivationIndex = params.index as number;\n    }\n\n    const { keychains, threshold, chain, segwit = false, bech32 = false } = params as GenerateFixedScriptAddressOptions;\n\n    let derivationChain = getExternalChainCode('p2sh');\n    if (_.isNumber(chain) && _.isInteger(chain) && isChainCode(chain)) {\n      derivationChain = chain;\n    }\n\n    function convertFlagsToAddressType(): ScriptType2Of3 {\n      if (isChainCode(chain)) {\n        return utxolib.bitgo.scriptTypeForChain(chain);\n      }\n      if (_.isBoolean(segwit) && segwit) {\n        return 'p2shP2wsh';\n      } else if (_.isBoolean(bech32) && bech32) {\n        return 'p2wsh';\n      } else {\n        return 'p2sh';\n      }\n    }\n\n    const addressType = params.addressType || convertFlagsToAddressType();\n\n    if (addressType !== utxolib.bitgo.scriptTypeForChain(derivationChain)) {\n      throw new AddressTypeChainMismatchError(addressType, derivationChain);\n    }\n\n    if (!this.supportsAddressType(addressType)) {\n      switch (addressType) {\n        case 'p2sh':\n          throw new Error(`internal error: p2sh should always be supported`);\n        case 'p2shP2wsh':\n          throw new P2shP2wshUnsupportedError();\n        case 'p2wsh':\n          throw new P2wshUnsupportedError();\n        case 'p2tr':\n          throw new P2trUnsupportedError();\n        case 'p2trMusig2':\n          throw new P2trMusig2UnsupportedError();\n        default:\n          throw new UnsupportedAddressTypeError();\n      }\n    }\n\n    let signatureThreshold = 2;\n    if (_.isInteger(threshold)) {\n      signatureThreshold = threshold as number;\n      if (signatureThreshold <= 0) {\n        throw new Error('threshold has to be positive');\n      }\n      if (signatureThreshold > keychains.length) {\n        throw new Error('threshold cannot exceed number of keys');\n      }\n    }\n\n    const path = '0/0/' + derivationChain + '/' + derivationIndex;\n    const hdNodes = keychains.map(({ pub }) => bip32.fromBase58(pub));\n    const derivedKeys = hdNodes.map((hdNode) => hdNode.derivePath(sanitizeLegacyPath(path)).publicKey);\n\n    const { outputScript, redeemScript, witnessScript, address } = this.createMultiSigAddress(\n      addressType,\n      signatureThreshold,\n      derivedKeys\n    );\n\n    return {\n      address,\n      chain: derivationChain,\n      index: derivationIndex,\n      coin: this.getChain(),\n      coinSpecific: {\n        outputScript: outputScript.toString('hex'),\n        redeemScript: redeemScript && redeemScript.toString('hex'),\n        witnessScript: witnessScript && witnessScript.toString('hex'),\n      },\n      addressType,\n    };\n  }\n\n  /**\n   * @returns input psbt added with deterministic MuSig2 nonce for bitgo key for each MuSig2 inputs.\n   * @param psbtHex all MuSig2 inputs should contain user MuSig2 nonce\n   * @param walletId\n   */\n  async signPsbt(psbtHex: string, walletId: string): Promise<SignPsbtResponse> {\n    const params: SignPsbtRequest = { psbt: psbtHex };\n    return await this.bitgo\n      .post(this.url('/wallet/' + walletId + '/tx/signpsbt'))\n      .send(params)\n      .result();\n  }\n\n  /**\n   * @returns input psbt added with deterministic MuSig2 nonce for bitgo key for each MuSig2 inputs from OVC.\n   * @param ovcJson JSON object provided by OVC with fields psbtHex and walletId\n   */\n  async signPsbtFromOVC(ovcJson: Record<string, unknown>): Promise<Record<string, unknown>> {\n    assert(ovcJson['psbtHex'], 'ovcJson must contain psbtHex');\n    assert(ovcJson['walletId'], 'ovcJson must contain walletId');\n    const psbt = (await this.signPsbt(ovcJson['psbtHex'] as string, ovcJson['walletId'] as string)).psbt;\n    assert(psbt, 'psbt not found');\n    return _.extend(ovcJson, { txHex: psbt });\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   * @param params - {@see SignTransactionOptions}\n   * @returns {Promise<SignedTransaction | HalfSignedUtxoTransaction>}\n   */\n  async signTransaction<TNumber extends number | bigint = number>(\n    params: SignTransactionOptions<TNumber>\n  ): Promise<SignedTransaction | HalfSignedUtxoTransaction> {\n    const txPrebuild = params.txPrebuild;\n\n    if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {\n      if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {\n        throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);\n      }\n      throw new Error('missing txPrebuild parameter');\n    }\n\n    let tx = bitgo.isPsbt(txPrebuild.txHex)\n      ? bitgo.createPsbtFromHex(txPrebuild.txHex, this.network)\n      : this.createTransactionFromHex<TNumber>(txPrebuild.txHex);\n\n    const isTxWithKeyPathSpendInput = tx instanceof bitgo.UtxoPsbt && bitgo.isTransactionWithKeyPathSpendInput(tx);\n\n    let isLastSignature = false;\n    if (_.isBoolean(params.isLastSignature)) {\n      // We can only be the first signature on a transaction with taproot key path spend inputs because\n      // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if\n      // deserialized from a hex.\n      if (params.isLastSignature && isTxWithKeyPathSpendInput) {\n        throw new Error('Cannot be last signature on a transaction with key path spend inputs');\n      }\n\n      // if build is called instead of buildIncomplete, no signature placeholders are left in the sig script\n      isLastSignature = params.isLastSignature;\n    }\n\n    const getSignerKeychain = (): utxolib.BIP32Interface => {\n      const userPrv = params.prv;\n      if (_.isUndefined(userPrv) || !_.isString(userPrv)) {\n        if (!_.isUndefined(userPrv)) {\n          throw new Error(`prv must be a string, got type ${typeof userPrv}`);\n        }\n        throw new Error('missing prv parameter to sign transaction');\n      }\n      const signerKeychain = bip32.fromBase58(userPrv, utxolib.networks.bitcoin);\n      if (signerKeychain.isNeutered()) {\n        throw new Error('expected user private key but received public key');\n      }\n      debug(`Here is the public key of the xprv you used to sign: ${signerKeychain.neutered().toBase58()}`);\n      return signerKeychain;\n    };\n\n    const setSignerMusigNonceWithOverride = (\n      psbt: utxolib.bitgo.UtxoPsbt,\n      signerKeychain: utxolib.BIP32Interface,\n      nonSegwitOverride: boolean\n    ) => {\n      utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.setAllInputsMusig2NonceHD(signerKeychain), nonSegwitOverride);\n    };\n\n    let signerKeychain: utxolib.BIP32Interface | undefined;\n\n    if (tx instanceof bitgo.UtxoPsbt && isTxWithKeyPathSpendInput) {\n      switch (params.signingStep) {\n        case 'signerNonce':\n          signerKeychain = getSignerKeychain();\n          setSignerMusigNonceWithOverride(tx, signerKeychain, !!params.allowNonSegwitSigningWithoutPrevTx);\n          AbstractUtxoCoin.PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);\n          return { txHex: tx.toHex() };\n        case 'cosignerNonce':\n          assert(txPrebuild.walletId, 'walletId is required for MuSig2 bitgo nonce');\n          return { txHex: (await this.signPsbt(tx.toHex(), txPrebuild.walletId)).psbt };\n        case 'signerSignature':\n          const txId = tx.getUnsignedTx().getId();\n          const psbt = AbstractUtxoCoin.PSBT_CACHE.get(txId);\n          assert(\n            psbt,\n            `Psbt is missing from txCache (cache size ${AbstractUtxoCoin.PSBT_CACHE.size}).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`\n          );\n          AbstractUtxoCoin.PSBT_CACHE.delete(txId);\n          tx = psbt.combine(tx);\n          break;\n        default:\n          // this instance is not an external signer\n          assert(txPrebuild.walletId, 'walletId is required for MuSig2 bitgo nonce');\n          signerKeychain = getSignerKeychain();\n          setSignerMusigNonceWithOverride(tx, signerKeychain, !!params.allowNonSegwitSigningWithoutPrevTx);\n          const response = await this.signPsbt(tx.toHex(), txPrebuild.walletId);\n          tx.combine(bitgo.createPsbtFromHex(response.psbt, this.network));\n          break;\n      }\n    } else {\n      switch (params.signingStep) {\n        case 'signerNonce':\n        case 'cosignerNonce':\n          /**\n           * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n           * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n           */\n          return { txHex: tx.toHex() };\n      }\n    }\n\n    if (signerKeychain === undefined) {\n      signerKeychain = getSignerKeychain();\n    }\n\n    let signedTransaction: bitgo.UtxoTransaction<bigint> | bitgo.UtxoPsbt;\n    if (tx instanceof bitgo.UtxoPsbt) {\n      signedTransaction = signAndVerifyPsbt(tx, signerKeychain, {\n        isLastSignature,\n        allowNonSegwitSigningWithoutPrevTx: params.allowNonSegwitSigningWithoutPrevTx,\n      });\n    } else {\n      if (tx.ins.length !== txPrebuild.txInfo?.unspents?.length) {\n        throw new Error('length of unspents array should equal to the number of transaction inputs');\n      }\n\n      if (!params.pubs || !isTriple(params.pubs)) {\n        throw new Error(`must provide xpub array`);\n      }\n\n      const keychains = params.pubs.map((pub) => bip32.fromBase58(pub)) as Triple<BIP32Interface>;\n      const cosignerPub = params.cosignerPub ?? params.pubs[2];\n      const cosignerKeychain = bip32.fromBase58(cosignerPub);\n\n      const walletSigner = new bitgo.WalletUnspentSigner<RootWalletKeys>(keychains, signerKeychain, cosignerKeychain);\n      signedTransaction = signAndVerifyWalletTransaction(tx, txPrebuild.txInfo.unspents, walletSigner, {\n        isLastSignature,\n      }) as bitgo.UtxoTransaction<bigint>;\n    }\n\n    return {\n      txHex: signedTransaction.toBuffer().toString('hex'),\n    };\n  }\n\n  /**\n   * Sign a transaction with a custom signing function. Example use case is express external signer\n   * @param customSigningFunction custom signing function that returns a single signed transaction\n   * @param signTransactionParams parameters for custom signing function. Includes txPrebuild and pubs (for legacy tx only).\n   *\n   * @returns signed transaction as hex string\n   */\n  async signWithCustomSigningFunction<TNumber extends number | bigint>(\n    customSigningFunction: UtxoCustomSigningFunction<TNumber>,\n    signTransactionParams: { txPrebuild: TransactionPrebuild<TNumber>; pubs?: string[] }\n  ): Promise<SignedTransaction> {\n    const txHex = signTransactionParams.txPrebuild.txHex;\n    assert(txHex, 'missing txHex parameter');\n\n    const tx = bitgo.isPsbt(txHex)\n      ? bitgo.createPsbtFromHex(txHex, this.network)\n      : this.createTransactionFromHex<TNumber>(txHex);\n\n    const isTxWithKeyPathSpendInput = tx instanceof bitgo.UtxoPsbt && bitgo.isTransactionWithKeyPathSpendInput(tx);\n\n    if (!isTxWithKeyPathSpendInput) {\n      return await customSigningFunction({ ...signTransactionParams, coin: this });\n    }\n\n    const getTxHex = (v: SignedTransaction): string => {\n      if ('txHex' in v) {\n        return v.txHex;\n      }\n      throw new Error('txHex not found in signTransaction result');\n    };\n\n    const signerNonceTx = await customSigningFunction({\n      ...signTransactionParams,\n      signingStep: 'signerNonce',\n      coin: this,\n    });\n\n    const { pubs } = signTransactionParams;\n    assert(pubs === undefined || isTriple(pubs));\n\n    const cosignerNonceTx = await this.signTransaction<TNumber>({\n      ...signTransactionParams,\n      pubs,\n      txPrebuild: { ...signTransactionParams.txPrebuild, txHex: getTxHex(signerNonceTx) },\n      signingStep: 'cosignerNonce',\n    });\n\n    return await customSigningFunction({\n      ...signTransactionParams,\n      txPrebuild: { ...signTransactionParams.txPrebuild, txHex: getTxHex(cosignerNonceTx) },\n      signingStep: 'signerSignature',\n      coin: this,\n    });\n  }\n\n  /**\n   * @param unspent\n   * @returns {boolean}\n   */\n  isBitGoTaintedUnspent<TNumber extends number | bigint>(unspent: Unspent<TNumber>): boolean {\n    return isReplayProtectionUnspent<TNumber>(unspent, this.network);\n  }\n\n  /**\n   * @deprecated - use utxolib.bitgo.getDefaultSigHash(network) instead\n   * @returns {number}\n   */\n  get defaultSigHashType(): number {\n    return utxolib.bitgo.getDefaultSigHash(this.network);\n  }\n\n  /**\n   * @deprecated - use utxolib.bitcoin.verifySignature() instead\n   */\n  verifySignature(\n    transaction: any,\n    inputIndex: number,\n    amount: number,\n    verificationSettings: {\n      signatureIndex?: number;\n      publicKey?: string;\n    } = {}\n  ): boolean {\n    if (transaction.network !== this.network) {\n      throw new Error(`network mismatch`);\n    }\n    return utxolib.bitgo.verifySignature(transaction, inputIndex, amount, {\n      signatureIndex: verificationSettings.signatureIndex,\n      publicKey: verificationSettings.publicKey ? Buffer.from(verificationSettings.publicKey, 'hex') : undefined,\n    });\n  }\n\n  /**\n   * Decompose a raw psbt/transaction into useful information, such as the total amounts,\n   * change amounts, and transaction outputs.\n   * @param params\n   */\n  async explainTransaction<TNumber extends number | bigint = number>(\n    params: ExplainTransactionOptions<TNumber>\n  ): Promise<TransactionExplanation> {\n    const { txHex } = params;\n    if (typeof txHex !== 'string' || !txHex.match(/^([a-f0-9]{2})+$/i)) {\n      throw new Error('invalid transaction hex, must be a valid hex string');\n    }\n    return utxolib.bitgo.isPsbt(txHex) ? explainPsbt(params, this.network) : explainTx(params, this);\n  }\n\n  /**\n   * Create a multisig address of a given type from a list of keychains and a signing threshold\n   * @param addressType\n   * @param signatureThreshold\n   * @param keys\n   */\n  createMultiSigAddress(addressType: ScriptType2Of3, signatureThreshold: number, keys: Buffer[]): MultiSigAddress {\n    const {\n      scriptPubKey: outputScript,\n      redeemScript,\n      witnessScript,\n    } = utxolib.bitgo.outputScripts.createOutputScript2of3(keys, addressType);\n\n    return {\n      outputScript,\n      redeemScript,\n      witnessScript,\n      address: utxolib.address.fromOutputScript(outputScript, this.network),\n    };\n  }\n\n  /**\n   * @deprecated - use {@see backupKeyRecovery}\n   * Builds a funds recovery transaction without BitGo\n   * @param params - {@see backupKeyRecovery}\n   */\n  async recover(params: RecoverParams): ReturnType<typeof backupKeyRecovery> {\n    return backupKeyRecovery(this, this.bitgo, params);\n  }\n\n  async recoverV1(params: V1RecoverParams): ReturnType<typeof v1BackupKeyRecovery> {\n    return v1BackupKeyRecovery(this, this.bitgo, params);\n  }\n\n  async sweepV1(params: V1SweepParams): ReturnType<typeof v1Sweep> {\n    return v1Sweep(this, this.bitgo, params);\n  }\n\n  /**\n   * Recover coin that was sent to wrong chain\n   * @param params\n   * @param params.txid The txid of the faulty transaction\n   * @param params.recoveryAddress address to send recovered funds to\n   * @param params.wallet the wallet that received the funds\n   * @param params.recoveryCoin the coin type of the wallet that received the funds\n   * @param params.signed return a half-signed transaction (default=true)\n   * @param params.walletPassphrase the wallet passphrase\n   * @param params.xprv the unencrypted xprv (used instead of wallet passphrase)\n   * @param params.apiKey for utxo coins other than [BTC,TBTC] this is a Block Chair api key\n   * @returns {*}\n   */\n  async recoverFromWrongChain<TNumber extends number | bigint = number>(\n    params: RecoverFromWrongChainOptions\n  ): Promise<CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>> {\n    const { txid, recoveryAddress, wallet, walletPassphrase, xprv, apiKey } = params;\n\n    // params.recoveryCoin used to be params.coin, backwards compatibility\n    const recoveryCoin = params.coin || params.recoveryCoin;\n    if (!recoveryCoin) {\n      throw new Error('missing required object recoveryCoin');\n    }\n    // signed should default to true, and only be disabled if explicitly set to false (not undefined)\n    const signed = params.signed !== false;\n\n    const sourceCoinFamily = this.getFamily();\n    const recoveryCoinFamily = recoveryCoin.getFamily();\n    const supportedRecoveryCoins = supportedCrossChainRecoveries[sourceCoinFamily];\n\n    if (_.isUndefined(supportedRecoveryCoins) || !supportedRecoveryCoins.includes(recoveryCoinFamily)) {\n      throw new Error(`Recovery of ${sourceCoinFamily} balances from ${recoveryCoinFamily} wallets is not supported.`);\n    }\n\n    return await recoverCrossChain<TNumber>(this.bitgo, {\n      sourceCoin: this,\n      recoveryCoin,\n      walletId: wallet,\n      txid,\n      recoveryAddress,\n      walletPassphrase: signed ? walletPassphrase : undefined,\n      xprv: signed ? xprv : undefined,\n      apiKey,\n    });\n  }\n\n  /**\n   * Generate bip32 key pair\n   *\n   * @param seed\n   * @returns {Object} object with generated pub and prv\n   */\n  generateKeyPair(seed: Buffer): { pub: string; prv: string } {\n    if (!seed) {\n      // An extended private key has both a normal 256 bit private key and a 256\n      // bit chain code, both of which must be random. 512 bits is therefore the\n      // maximum entropy and gives us maximum security against cracking.\n      seed = randomBytes(512 / 8);\n    }\n    const extendedKey = bip32.fromSeed(seed);\n    return {\n      pub: extendedKey.neutered().toBase58(),\n      prv: extendedKey.toBase58(),\n    };\n  }\n\n  private shouldDefaultToPsbtTxFormat(buildParams: ExtraPrebuildParamsOptions & { wallet: Wallet }) {\n    const walletFlagMusigKp = buildParams.wallet.flag('musigKp') === 'true';\n    const isHotWallet = buildParams.wallet.type() === 'hot';\n\n    // if not txFormat is already specified figure out if we should default to psbt format\n    return (\n      buildParams.txFormat === undefined &&\n      (buildParams.wallet.subType() === 'distributedCustody' ||\n        // default to testnet for all utxo coins except zcash\n        (isTestnet(this.network) &&\n          // FIXME(BTC-1322): fix zcash PSBT support\n          getMainnet(this.network) !== utxolib.networks.zcash &&\n          isHotWallet) ||\n        // if mainnet, only default to psbt for btc hot wallets\n        (isMainnet(this.network) && getMainnet(this.network) === utxolib.networks.bitcoin && isHotWallet) ||\n        // default to psbt if it has the wallet flag\n        walletFlagMusigKp)\n    );\n  }\n\n  async getExtraPrebuildParams(buildParams: ExtraPrebuildParamsOptions & { wallet: Wallet }): Promise<{\n    txFormat?: 'legacy' | 'psbt';\n    changeAddressType?: ScriptType2Of3[] | ScriptType2Of3;\n  }> {\n    let txFormat = buildParams.txFormat as 'legacy' | 'psbt' | undefined;\n    let changeAddressType = buildParams.changeAddressType as ScriptType2Of3[] | ScriptType2Of3 | undefined;\n\n    if (this.shouldDefaultToPsbtTxFormat(buildParams)) {\n      txFormat = 'psbt';\n    }\n\n    // if the addressType is not specified, we need to default to p2trMusig2 for testnet hot wallets for staged rollout of p2trMusig2\n    if (\n      buildParams.addressType === undefined && // addressType is deprecated and replaced by `changeAddress`\n      buildParams.changeAddressType === undefined &&\n      buildParams.changeAddress === undefined &&\n      buildParams.wallet.type() === 'hot'\n    ) {\n      changeAddressType = ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'];\n    }\n\n    return {\n      txFormat,\n      changeAddressType,\n    };\n  }\n\n  preCreateBitGo(params: PrecreateBitGoOptions): void {\n    return;\n  }\n\n  async presignTransaction(params: PresignTransactionOptions): Promise<any> {\n    // In the case that we have a 'psbt-lite' transaction format, we want to indicate in signing to not fail\n    const txHex = (params.txHex ?? params.txPrebuild?.txHex) as string;\n    if (\n      txHex &&\n      utxolib.bitgo.isPsbt(txHex as string) &&\n      utxolib.bitgo.isPsbtLite(utxolib.bitgo.createPsbtFromHex(txHex, this.network)) &&\n      params.allowNonSegwitSigningWithoutPrevTx === undefined\n    ) {\n      return { ...params, allowNonSegwitSigningWithoutPrevTx: true };\n    }\n    return params;\n  }\n\n  async supplementGenerateWallet(\n    walletParams: SupplementGenerateWalletOptions,\n    keychains: KeychainsTriplet\n  ): Promise<any> {\n    return walletParams;\n  }\n\n  transactionDataAllowed(): boolean {\n    return false;\n  }\n\n  valuelessTransferAllowed(): boolean {\n    return false;\n  }\n\n  getRecoveryProvider(apiToken?: string): RecoveryProvider {\n    return forCoin(this.getChain(), apiToken);\n  }\n}\n"]}
|