utxo-lib 1.0.9 → 1.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +19 -16
- package/dist/src/address.d.ts.map +1 -1
- package/dist/src/address.js +11 -1
- package/dist/src/addressFormat.d.ts +1 -1
- package/dist/src/addressFormat.d.ts.map +1 -1
- package/dist/src/addressFormat.js +1 -1
- package/dist/src/bitgo/Musig2.d.ts +115 -17
- package/dist/src/bitgo/Musig2.d.ts.map +1 -1
- package/dist/src/bitgo/Musig2.js +283 -100
- package/dist/src/bitgo/PsbtUtil.d.ts +59 -0
- package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -0
- package/dist/src/bitgo/PsbtUtil.js +91 -0
- package/dist/src/bitgo/UtxoPsbt.d.ts +180 -47
- package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoPsbt.js +651 -107
- package/dist/src/bitgo/bitcoincash/address.js +2 -2
- package/dist/src/bitgo/index.d.ts +11 -0
- package/dist/src/bitgo/index.d.ts.map +1 -1
- package/dist/src/bitgo/index.js +6 -2
- package/dist/src/bitgo/legacysafe/index.d.ts +15 -0
- package/dist/src/bitgo/legacysafe/index.d.ts.map +1 -0
- package/dist/src/bitgo/legacysafe/index.js +61 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.js +17 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts +16 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.js +46 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.js +15 -0
- package/dist/src/bitgo/litecoin/index.d.ts +4 -0
- package/dist/src/bitgo/litecoin/index.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/index.js +16 -0
- package/dist/src/bitgo/outputScripts.d.ts +3 -1
- package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
- package/dist/src/bitgo/outputScripts.js +18 -10
- package/dist/src/bitgo/parseInput.d.ts +49 -20
- package/dist/src/bitgo/parseInput.d.ts.map +1 -1
- package/dist/src/bitgo/parseInput.js +109 -25
- package/dist/src/bitgo/psbt/fromHalfSigned.d.ts.map +1 -1
- package/dist/src/bitgo/psbt/fromHalfSigned.js +9 -6
- package/dist/src/bitgo/signature.d.ts +3 -3
- package/dist/src/bitgo/signature.d.ts.map +1 -1
- package/dist/src/bitgo/signature.js +48 -16
- package/dist/src/bitgo/transaction.d.ts +18 -3
- package/dist/src/bitgo/transaction.d.ts.map +1 -1
- package/dist/src/bitgo/transaction.js +28 -15
- package/dist/src/bitgo/types.d.ts +2 -0
- package/dist/src/bitgo/types.d.ts.map +1 -1
- package/dist/src/bitgo/types.js +1 -1
- package/dist/src/bitgo/wallet/Psbt.d.ts +104 -12
- package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Psbt.js +285 -70
- package/dist/src/bitgo/wallet/Unspent.d.ts +28 -0
- package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Unspent.js +172 -68
- package/dist/src/bitgo/wallet/WalletOutput.d.ts +17 -1
- package/dist/src/bitgo/wallet/WalletOutput.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/WalletOutput.js +64 -23
- package/dist/src/bitgo/wallet/chains.d.ts +2 -2
- package/dist/src/bitgo/wallet/chains.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/chains.js +1 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts +0 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.js +6 -14
- package/dist/src/musig.d.ts +0 -1
- package/dist/src/musig.d.ts.map +1 -1
- package/dist/src/musig.js +15 -29
- package/dist/src/networks.d.ts +1 -2
- package/dist/src/networks.d.ts.map +1 -1
- package/dist/src/networks.js +22 -29
- package/dist/src/noble_ecc.d.ts.map +1 -1
- package/dist/src/noble_ecc.js +7 -3
- package/dist/src/payments/p2tr.d.ts.map +1 -1
- package/dist/src/payments/p2tr.js +15 -9
- package/dist/src/taproot.d.ts +16 -0
- package/dist/src/taproot.d.ts.map +1 -1
- package/dist/src/taproot.js +44 -2
- package/dist/src/testutil/index.d.ts +2 -0
- package/dist/src/testutil/index.d.ts.map +1 -1
- package/dist/src/testutil/index.js +3 -1
- package/dist/src/testutil/keys.d.ts +3 -0
- package/dist/src/testutil/keys.d.ts.map +1 -1
- package/dist/src/testutil/keys.js +17 -2
- package/dist/src/testutil/mock.d.ts +1 -1
- package/dist/src/testutil/mock.d.ts.map +1 -1
- package/dist/src/testutil/mock.js +12 -4
- package/dist/src/testutil/psbt.d.ts +89 -0
- package/dist/src/testutil/psbt.d.ts.map +1 -0
- package/dist/src/testutil/psbt.js +150 -0
- package/dist/src/testutil/transaction.d.ts +70 -0
- package/dist/src/testutil/transaction.d.ts.map +1 -0
- package/dist/src/testutil/transaction.js +107 -0
- package/package.json +6 -6
@@ -1,9 +1,13 @@
|
|
1
1
|
"use strict";
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
-
exports.UtxoPsbt =
|
3
|
+
exports.UtxoPsbt = void 0;
|
4
|
+
const assert = require("assert");
|
4
5
|
const bip174_1 = require("bip174");
|
5
6
|
const utils_1 = require("bip174/src/lib/utils");
|
6
7
|
const bufferutils_1 = require("bitcoinjs-lib/src/bufferutils");
|
8
|
+
const bip32_1 = require("bip32");
|
9
|
+
const bs58check = require("bs58check");
|
10
|
+
const proprietaryKeyVal_1 = require("bip174/src/lib/proprietaryKeyVal");
|
7
11
|
const __1 = require("..");
|
8
12
|
const UtxoTransaction_1 = require("./UtxoTransaction");
|
9
13
|
const Unspent_1 = require("./Unspent");
|
@@ -11,20 +15,43 @@ const scriptTypes_1 = require("./psbt/scriptTypes");
|
|
11
15
|
const fromHalfSigned_1 = require("./psbt/fromHalfSigned");
|
12
16
|
const outputScripts_1 = require("./outputScripts");
|
13
17
|
const parseInput_1 = require("./parseInput");
|
14
|
-
const
|
15
|
-
const
|
16
|
-
const
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
18
|
+
const Musig2_1 = require("./Musig2");
|
19
|
+
const types_1 = require("./types");
|
20
|
+
const taproot_1 = require("../taproot");
|
21
|
+
const PsbtUtil_1 = require("./PsbtUtil");
|
22
|
+
function defaultSighashTypes(network) {
|
23
|
+
const sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL];
|
24
|
+
switch (__1.getMainnet(network)) {
|
25
|
+
case __1.networks.bitcoincash:
|
26
|
+
case __1.networks.bitcoinsv:
|
27
|
+
case __1.networks.bitcoingold:
|
28
|
+
case __1.networks.ecash:
|
29
|
+
return [...sighashTypes, ...sighashTypes.map((s) => s | UtxoTransaction_1.UtxoTransaction.SIGHASH_FORKID)];
|
30
|
+
default:
|
31
|
+
return sighashTypes;
|
32
|
+
}
|
33
|
+
}
|
34
|
+
function toSignatureParams(network, v) {
|
35
|
+
if (Array.isArray(v))
|
36
|
+
return toSignatureParams(network, { sighashTypes: v });
|
37
|
+
return { deterministic: false, sighashTypes: defaultSighashTypes(network), ...v };
|
38
|
+
}
|
39
|
+
/**
|
40
|
+
* @param a
|
41
|
+
* @param b
|
42
|
+
* @returns true if the two public keys are equal ignoring the y coordinate.
|
43
|
+
*/
|
44
|
+
function equalPublicKeyIgnoreY(a, b) {
|
45
|
+
return outputScripts_1.toXOnlyPublicKey(a).equals(outputScripts_1.toXOnlyPublicKey(b));
|
46
|
+
}
|
24
47
|
// TODO: upstream does `checkInputsForPartialSigs` before doing things like
|
25
48
|
// `setVersion`. Our inputs could have tapscriptsigs (or in future tapkeysigs)
|
26
49
|
// and not fail that check. Do we want to do anything about that?
|
27
50
|
class UtxoPsbt extends __1.Psbt {
|
51
|
+
constructor() {
|
52
|
+
super(...arguments);
|
53
|
+
this.nonceStore = new Musig2_1.Musig2NonceStore();
|
54
|
+
}
|
28
55
|
static transactionFromBuffer(buffer, network) {
|
29
56
|
return UtxoTransaction_1.UtxoTransaction.fromBuffer(buffer, false, 'bigint', network);
|
30
57
|
}
|
@@ -46,6 +73,40 @@ class UtxoPsbt extends __1.Psbt {
|
|
46
73
|
static fromHex(data, opts) {
|
47
74
|
return this.fromBuffer(Buffer.from(data, 'hex'), opts);
|
48
75
|
}
|
76
|
+
/**
|
77
|
+
* @param parent - Parent key. Matched with `bip32Derivations` using `fingerprint` property.
|
78
|
+
* @param bip32Derivations - possible derivations for input or output
|
79
|
+
* @param ignoreY - when true, ignore the y coordinate when matching public keys
|
80
|
+
* @return derived bip32 node if matching derivation is found, undefined if none is found
|
81
|
+
* @throws Error if more than one match is found
|
82
|
+
*/
|
83
|
+
static deriveKeyPair(parent, bip32Derivations, { ignoreY }) {
|
84
|
+
const matchingDerivations = bip32Derivations.filter((bipDv) => {
|
85
|
+
return bipDv.masterFingerprint.equals(parent.fingerprint);
|
86
|
+
});
|
87
|
+
if (!matchingDerivations.length) {
|
88
|
+
// No fingerprint match
|
89
|
+
return undefined;
|
90
|
+
}
|
91
|
+
if (matchingDerivations.length !== 1) {
|
92
|
+
throw new Error(`more than one matching derivation for fingerprint ${parent.fingerprint.toString('hex')}: ${matchingDerivations.length}`);
|
93
|
+
}
|
94
|
+
const [derivation] = matchingDerivations;
|
95
|
+
const node = parent.derivePath(derivation.path);
|
96
|
+
if (!node.publicKey.equals(derivation.pubkey)) {
|
97
|
+
if (!ignoreY || !equalPublicKeyIgnoreY(node.publicKey, derivation.pubkey)) {
|
98
|
+
throw new Error('pubkey did not match bip32Derivation');
|
99
|
+
}
|
100
|
+
}
|
101
|
+
return node;
|
102
|
+
}
|
103
|
+
static deriveKeyPairForInput(bip32, input) {
|
104
|
+
return input.tapBip32Derivation?.length
|
105
|
+
? UtxoPsbt.deriveKeyPair(bip32, input.tapBip32Derivation, { ignoreY: true })?.publicKey
|
106
|
+
: input.bip32Derivation?.length
|
107
|
+
? UtxoPsbt.deriveKeyPair(bip32, input.bip32Derivation, { ignoreY: false })?.publicKey
|
108
|
+
: bip32?.publicKey;
|
109
|
+
}
|
49
110
|
get network() {
|
50
111
|
return this.tx.network;
|
51
112
|
}
|
@@ -53,21 +114,14 @@ class UtxoPsbt extends __1.Psbt {
|
|
53
114
|
return this.toBuffer().toString('hex');
|
54
115
|
}
|
55
116
|
/**
|
56
|
-
*
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
}
|
62
|
-
/**
|
63
|
-
* @return partialSig/tapScriptSig count iff input is not finalized
|
117
|
+
* It is expensive to attempt to compute every output address using psbt.txOutputs[outputIndex]
|
118
|
+
* to then just get the script. Here, we are doing the same thing as what txOutputs() does in
|
119
|
+
* bitcoinjs-lib, but without iterating over each output.
|
120
|
+
* @param outputIndex
|
121
|
+
* @returns output script at the given index
|
64
122
|
*/
|
65
|
-
|
66
|
-
|
67
|
-
throw new Error('Input is already finalized');
|
68
|
-
}
|
69
|
-
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
70
|
-
return Math.max(Array.isArray(input.partialSig) ? input.partialSig.length : 0, Array.isArray(input.tapScriptSig) ? input.tapScriptSig.length : 0);
|
123
|
+
getOutputScript(outputIndex) {
|
124
|
+
return this.__CACHE.__TX.outs[outputIndex].script;
|
71
125
|
}
|
72
126
|
getNonWitnessPreviousTxids() {
|
73
127
|
const txInputs = this.txInputs; // These are somewhat costly to extract
|
@@ -130,17 +184,104 @@ class UtxoPsbt extends __1.Psbt {
|
|
130
184
|
}
|
131
185
|
});
|
132
186
|
}
|
187
|
+
/**
|
188
|
+
* @returns true if the input at inputIndex is a taproot key path.
|
189
|
+
* Checks for presence of minimum required key path input fields and absence of any script path only input fields.
|
190
|
+
*/
|
191
|
+
isTaprootKeyPathInput(inputIndex) {
|
192
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
193
|
+
return (!!input.tapInternalKey &&
|
194
|
+
!!input.tapMerkleRoot &&
|
195
|
+
!(input.tapLeafScript?.length ||
|
196
|
+
input.tapScriptSig?.length ||
|
197
|
+
input.tapBip32Derivation?.some((v) => v.leafHashes.length)));
|
198
|
+
}
|
199
|
+
/**
|
200
|
+
* @returns true if the input at inputIndex is a taproot script path.
|
201
|
+
* Checks for presence of minimum required script path input fields and absence of any key path only input fields.
|
202
|
+
*/
|
203
|
+
isTaprootScriptPathInput(inputIndex) {
|
204
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
205
|
+
return (!!input.tapLeafScript?.length &&
|
206
|
+
!(this.getProprietaryKeyVals(inputIndex, {
|
207
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
208
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
|
209
|
+
}).length ||
|
210
|
+
this.getProprietaryKeyVals(inputIndex, {
|
211
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
212
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
|
213
|
+
}).length ||
|
214
|
+
this.getProprietaryKeyVals(inputIndex, {
|
215
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
216
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
|
217
|
+
}).length));
|
218
|
+
}
|
219
|
+
/**
|
220
|
+
* @returns true if the input at inputIndex is a taproot
|
221
|
+
*/
|
222
|
+
isTaprootInput(inputIndex) {
|
223
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
224
|
+
const isP2TR = (script) => {
|
225
|
+
try {
|
226
|
+
taproot_1.getTaprootOutputKey(script);
|
227
|
+
return true;
|
228
|
+
}
|
229
|
+
catch (e) {
|
230
|
+
return false;
|
231
|
+
}
|
232
|
+
};
|
233
|
+
return !!(input.tapInternalKey ||
|
234
|
+
input.tapMerkleRoot ||
|
235
|
+
input.tapLeafScript?.length ||
|
236
|
+
input.tapBip32Derivation?.length ||
|
237
|
+
input.tapScriptSig?.length ||
|
238
|
+
this.getProprietaryKeyVals(inputIndex, {
|
239
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
240
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
|
241
|
+
}).length ||
|
242
|
+
this.getProprietaryKeyVals(inputIndex, {
|
243
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
244
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
|
245
|
+
}).length ||
|
246
|
+
this.getProprietaryKeyVals(inputIndex, {
|
247
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
248
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
|
249
|
+
}).length ||
|
250
|
+
(input.witnessUtxo && isP2TR(input.witnessUtxo.script)));
|
251
|
+
}
|
252
|
+
isMultisigTaprootScript(script) {
|
253
|
+
try {
|
254
|
+
parseInput_1.parsePubScript2Of3(script, 'taprootScriptPathSpend');
|
255
|
+
return true;
|
256
|
+
}
|
257
|
+
catch (e) {
|
258
|
+
return false;
|
259
|
+
}
|
260
|
+
}
|
133
261
|
/**
|
134
262
|
* Mostly copied from bitcoinjs-lib/ts_src/psbt.ts
|
135
263
|
*/
|
136
264
|
finalizeAllInputs() {
|
137
265
|
utils_1.checkForInput(this.data.inputs, 0); // making sure we have at least one
|
138
266
|
this.data.inputs.map((input, idx) => {
|
139
|
-
|
267
|
+
if (input.tapLeafScript?.length) {
|
268
|
+
return this.isMultisigTaprootScript(input.tapLeafScript[0].script)
|
269
|
+
? this.finalizeTaprootInput(idx)
|
270
|
+
: this.finalizeTapInputWithSingleLeafScriptAndSignature(idx);
|
271
|
+
}
|
272
|
+
else if (this.isTaprootKeyPathInput(idx)) {
|
273
|
+
return this.finalizeTaprootMusig2Input(idx);
|
274
|
+
}
|
275
|
+
return this.finalizeInput(idx);
|
140
276
|
});
|
141
277
|
return this;
|
142
278
|
}
|
143
279
|
finalizeTaprootInput(inputIndex) {
|
280
|
+
const checkPartialSigSighashes = (sig) => {
|
281
|
+
const sighashType = sig.length === 64 ? __1.Transaction.SIGHASH_DEFAULT : sig.readUInt8(sig.length - 1);
|
282
|
+
const inputSighashType = input.sighashType === undefined ? __1.Transaction.SIGHASH_DEFAULT : input.sighashType;
|
283
|
+
assert(sighashType === inputSighashType, 'signature sighash does not match input sighash type');
|
284
|
+
};
|
144
285
|
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
145
286
|
// witness = control-block script first-sig second-sig
|
146
287
|
if (input.tapLeafScript?.length !== 1) {
|
@@ -148,12 +289,13 @@ class UtxoPsbt extends __1.Psbt {
|
|
148
289
|
}
|
149
290
|
const { controlBlock, script } = input.tapLeafScript[0];
|
150
291
|
const witness = [script, controlBlock];
|
151
|
-
const [pubkey1, pubkey2] = parseInput_1.
|
292
|
+
const [pubkey1, pubkey2] = parseInput_1.parsePubScript2Of3(script, 'taprootScriptPathSpend').publicKeys;
|
152
293
|
for (const pk of [pubkey1, pubkey2]) {
|
153
|
-
const sig = input.tapScriptSig?.find(({ pubkey }) =>
|
294
|
+
const sig = input.tapScriptSig?.find(({ pubkey }) => equalPublicKeyIgnoreY(pk, pubkey));
|
154
295
|
if (!sig) {
|
155
296
|
throw new Error('Could not find signatures in Script Sig.');
|
156
297
|
}
|
298
|
+
checkPartialSigSighashes(sig.signature);
|
157
299
|
witness.unshift(sig.signature);
|
158
300
|
}
|
159
301
|
const witnessLength = witness.reduce((s, b) => s + b.length + bufferutils_1.varuint.encodingLength(b.length), 1);
|
@@ -164,6 +306,30 @@ class UtxoPsbt extends __1.Psbt {
|
|
164
306
|
this.data.clearFinalizedInput(inputIndex);
|
165
307
|
return this;
|
166
308
|
}
|
309
|
+
/**
|
310
|
+
* Finalizes a taproot musig2 input by aggregating all partial sigs.
|
311
|
+
* IMPORTANT: Always call validate* function before finalizing.
|
312
|
+
*/
|
313
|
+
finalizeTaprootMusig2Input(inputIndex) {
|
314
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
315
|
+
const partialSigs = Musig2_1.parsePsbtMusig2PartialSigs(input);
|
316
|
+
if (partialSigs?.length !== 2) {
|
317
|
+
throw new Error(`invalid number of partial signatures ${partialSigs ? partialSigs.length : 0} to finalize`);
|
318
|
+
}
|
319
|
+
const { partialSigs: pSigs, sigHashType } = Musig2_1.getSigHashTypeFromSigs(partialSigs);
|
320
|
+
const { sessionKey } = this.getMusig2SessionKey(inputIndex, sigHashType);
|
321
|
+
const aggSig = Musig2_1.musig2AggregateSigs(pSigs.map((pSig) => pSig.partialSig), sessionKey);
|
322
|
+
const sig = sigHashType === __1.Transaction.SIGHASH_DEFAULT ? aggSig : Buffer.concat([aggSig, Buffer.of(sigHashType)]);
|
323
|
+
// single signature with 64/65 bytes size is script witness for key path spend
|
324
|
+
const bufferWriter = bufferutils_1.BufferWriter.withCapacity(1 + bufferutils_1.varuint.encodingLength(sig.length) + sig.length);
|
325
|
+
bufferWriter.writeVector([sig]);
|
326
|
+
const finalScriptWitness = bufferWriter.end();
|
327
|
+
this.data.updateInput(inputIndex, { finalScriptWitness });
|
328
|
+
this.data.clearFinalizedInput(inputIndex);
|
329
|
+
// deleting only BitGo proprietary key values.
|
330
|
+
this.deleteProprietaryKeyVals(inputIndex, { identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER });
|
331
|
+
return this;
|
332
|
+
}
|
167
333
|
finalizeTapInputWithSingleLeafScriptAndSignature(inputIndex) {
|
168
334
|
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
169
335
|
if (input.tapLeafScript?.length !== 1) {
|
@@ -192,12 +358,97 @@ class UtxoPsbt extends __1.Psbt {
|
|
192
358
|
validateSignaturesOfAllInputs() {
|
193
359
|
utils_1.checkForInput(this.data.inputs, 0); // making sure we have at least one
|
194
360
|
const results = this.data.inputs.map((input, idx) => {
|
195
|
-
return
|
196
|
-
? this.validateTaprootSignaturesOfInput(idx)
|
197
|
-
: this.validateSignaturesOfInput(idx, (p, m, s) => __1.ecc.verify(m, p, s));
|
361
|
+
return this.validateSignaturesOfInputCommon(idx);
|
198
362
|
});
|
199
363
|
return results.reduce((final, res) => res && final, true);
|
200
364
|
}
|
365
|
+
/**
|
366
|
+
* @returns true iff any matching valid signature is found for a derived pub key from given HD key pair.
|
367
|
+
*/
|
368
|
+
validateSignaturesOfInputHD(inputIndex, hdKeyPair) {
|
369
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
370
|
+
const pubKey = UtxoPsbt.deriveKeyPairForInput(hdKeyPair, input);
|
371
|
+
if (!pubKey) {
|
372
|
+
throw new Error('can not derive from HD key pair');
|
373
|
+
}
|
374
|
+
return this.validateSignaturesOfInputCommon(inputIndex, pubKey);
|
375
|
+
}
|
376
|
+
/**
|
377
|
+
* @returns true iff any valid signature(s) are found from bip32 data of PSBT or for given pub key.
|
378
|
+
*/
|
379
|
+
validateSignaturesOfInputCommon(inputIndex, pubkey) {
|
380
|
+
try {
|
381
|
+
if (this.isTaprootScriptPathInput(inputIndex)) {
|
382
|
+
return this.validateTaprootSignaturesOfInput(inputIndex, pubkey);
|
383
|
+
}
|
384
|
+
else if (this.isTaprootKeyPathInput(inputIndex)) {
|
385
|
+
return this.validateTaprootMusig2SignaturesOfInput(inputIndex, pubkey);
|
386
|
+
}
|
387
|
+
return this.validateSignaturesOfInput(inputIndex, (p, m, s) => __1.ecc.verify(m, p, s, true), pubkey);
|
388
|
+
}
|
389
|
+
catch (err) {
|
390
|
+
// Not an elegant solution. Might need upstream changes like custom error types.
|
391
|
+
if (err.message === 'No signatures for this pubkey') {
|
392
|
+
return false;
|
393
|
+
}
|
394
|
+
throw err;
|
395
|
+
}
|
396
|
+
}
|
397
|
+
getMusig2SessionKey(inputIndex, sigHashType) {
|
398
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
399
|
+
if (!input.tapInternalKey || !input.tapMerkleRoot) {
|
400
|
+
throw new Error('both tapInternalKey and tapMerkleRoot are required');
|
401
|
+
}
|
402
|
+
const participants = this.getMusig2Participants(inputIndex, input.tapInternalKey, input.tapMerkleRoot);
|
403
|
+
const nonces = this.getMusig2Nonces(inputIndex, participants);
|
404
|
+
const { hash } = this.getTaprootHashForSig(inputIndex, [sigHashType]);
|
405
|
+
const sessionKey = Musig2_1.createMusig2SigningSession({
|
406
|
+
pubNonces: [nonces[0].pubNonce, nonces[1].pubNonce],
|
407
|
+
pubKeys: participants.participantPubKeys,
|
408
|
+
txHash: hash,
|
409
|
+
internalPubKey: input.tapInternalKey,
|
410
|
+
tapTreeRoot: input.tapMerkleRoot,
|
411
|
+
});
|
412
|
+
return { participants, nonces, hash, sessionKey };
|
413
|
+
}
|
414
|
+
/**
|
415
|
+
* @returns true for following cases.
|
416
|
+
* If valid musig2 partial signatures exists for both 2 keys, it will also verify aggregated sig
|
417
|
+
* for aggregated tweaked key (output key), otherwise only verifies partial sig.
|
418
|
+
* If pubkey is passed in input, it will check sig only for that pubkey,
|
419
|
+
* if no sig exits for such key, throws error.
|
420
|
+
* For invalid state of input data, it will throw errors.
|
421
|
+
*/
|
422
|
+
validateTaprootMusig2SignaturesOfInput(inputIndex, pubkey) {
|
423
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
424
|
+
const partialSigs = Musig2_1.parsePsbtMusig2PartialSigs(input);
|
425
|
+
if (!partialSigs) {
|
426
|
+
throw new Error(`No signatures to validate`);
|
427
|
+
}
|
428
|
+
let myPartialSigs = partialSigs;
|
429
|
+
if (pubkey) {
|
430
|
+
myPartialSigs = partialSigs.filter((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, pubkey));
|
431
|
+
if (myPartialSigs?.length < 1) {
|
432
|
+
throw new Error('No signatures for this pubkey');
|
433
|
+
}
|
434
|
+
}
|
435
|
+
const { partialSigs: mySigs, sigHashType } = Musig2_1.getSigHashTypeFromSigs(myPartialSigs);
|
436
|
+
const { participants, nonces, hash, sessionKey } = this.getMusig2SessionKey(inputIndex, sigHashType);
|
437
|
+
const results = mySigs.map((mySig) => {
|
438
|
+
const myNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, mySig.participantPubKey));
|
439
|
+
if (!myNonce) {
|
440
|
+
throw new Error('Found no pub nonce for pubkey');
|
441
|
+
}
|
442
|
+
return Musig2_1.musig2PartialSigVerify(mySig.partialSig, mySig.participantPubKey, myNonce.pubNonce, sessionKey);
|
443
|
+
});
|
444
|
+
// For valid single sig or 1 or 2 failure sigs, no need to validate aggregated sig. So skip.
|
445
|
+
const result = results.every((res) => res);
|
446
|
+
if (!result || mySigs.length < 2) {
|
447
|
+
return result;
|
448
|
+
}
|
449
|
+
const aggSig = Musig2_1.musig2AggregateSigs(mySigs.map((mySig) => mySig.partialSig), sessionKey);
|
450
|
+
return __1.ecc.verifySchnorr(hash, participants.tapOutputKey, aggSig);
|
451
|
+
}
|
201
452
|
validateTaprootSignaturesOfInput(inputIndex, pubkey) {
|
202
453
|
const input = this.data.inputs[inputIndex];
|
203
454
|
const tapSigs = (input || {}).tapScriptSig;
|
@@ -206,8 +457,7 @@ class UtxoPsbt extends __1.Psbt {
|
|
206
457
|
}
|
207
458
|
let mySigs;
|
208
459
|
if (pubkey) {
|
209
|
-
|
210
|
-
mySigs = tapSigs.filter((sig) => sig.pubkey.equals(xOnlyPubkey));
|
460
|
+
mySigs = tapSigs.filter((sig) => equalPublicKeyIgnoreY(sig.pubkey, pubkey));
|
211
461
|
if (mySigs.length < 1) {
|
212
462
|
throw new Error('No signatures for this pubkey');
|
213
463
|
}
|
@@ -216,8 +466,16 @@ class UtxoPsbt extends __1.Psbt {
|
|
216
466
|
mySigs = tapSigs;
|
217
467
|
}
|
218
468
|
const results = [];
|
469
|
+
assert(input.tapLeafScript?.length === 1, `single tapLeafScript is expected. Got ${input.tapLeafScript?.length}`);
|
470
|
+
const [tapLeafScript] = input.tapLeafScript;
|
471
|
+
const pubKeys = this.isMultisigTaprootScript(tapLeafScript.script)
|
472
|
+
? parseInput_1.parsePubScript2Of3(tapLeafScript.script, 'taprootScriptPathSpend').publicKeys
|
473
|
+
: undefined;
|
219
474
|
for (const pSig of mySigs) {
|
220
475
|
const { signature, leafHash, pubkey } = pSig;
|
476
|
+
if (pubKeys) {
|
477
|
+
assert(pubKeys.find((pk) => pubkey.equals(pk)), 'public key not found in tap leaf script');
|
478
|
+
}
|
221
479
|
let sigHashType;
|
222
480
|
let sig;
|
223
481
|
if (signature.length === 65) {
|
@@ -231,33 +489,39 @@ class UtxoPsbt extends __1.Psbt {
|
|
231
489
|
const { hash } = this.getTaprootHashForSig(inputIndex, [sigHashType], leafHash);
|
232
490
|
results.push(__1.ecc.verifySchnorr(hash, pubkey, sig));
|
233
491
|
}
|
234
|
-
return results.every((res) => res
|
492
|
+
return results.every((res) => res);
|
235
493
|
}
|
236
494
|
/**
|
495
|
+
* @param inputIndex
|
496
|
+
* @param rootNodes optional input root bip32 nodes to verify with. If it is not provided, globalXpub will be used.
|
237
497
|
* @return array of boolean values. True when corresponding index in `publicKeys` has signed the transaction.
|
238
498
|
* If no signature in the tx or no public key matching signature, the validation is considered as false.
|
239
499
|
*/
|
240
|
-
getSignatureValidationArray(inputIndex) {
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
500
|
+
getSignatureValidationArray(inputIndex, { rootNodes } = {}) {
|
501
|
+
if (!rootNodes && (!this.data.globalMap.globalXpub?.length || !types_1.isTriple(this.data.globalMap.globalXpub))) {
|
502
|
+
throw new Error('Cannot get signature validation array without 3 global xpubs');
|
503
|
+
}
|
504
|
+
const bip32s = rootNodes
|
505
|
+
? rootNodes
|
506
|
+
: this.data.globalMap.globalXpub?.map((xpub) => bip32_1.BIP32Factory(__1.ecc).fromBase58(bs58check.encode(xpub.extendedPubkey)));
|
507
|
+
if (!bip32s) {
|
508
|
+
throw new Error('either globalMap or rootNodes is required');
|
246
509
|
}
|
247
|
-
|
248
|
-
|
510
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
511
|
+
if (!PsbtUtil_1.getPsbtInputSignatureCount(input)) {
|
512
|
+
return [false, false, false];
|
249
513
|
}
|
250
|
-
return
|
251
|
-
|
252
|
-
|
514
|
+
return bip32s.map((bip32) => {
|
515
|
+
const pubKey = UtxoPsbt.deriveKeyPairForInput(bip32, input);
|
516
|
+
if (!pubKey) {
|
517
|
+
return false;
|
518
|
+
}
|
253
519
|
try {
|
254
|
-
return
|
255
|
-
? this.validateTaprootSignaturesOfInput(inputIndex, bip32.publicKey)
|
256
|
-
: this.validateSignaturesOfInput(inputIndex, (p, m, s) => __1.ecc.verify(m, p, s), bip32.publicKey);
|
520
|
+
return this.validateSignaturesOfInputCommon(inputIndex, pubKey);
|
257
521
|
}
|
258
522
|
catch (err) {
|
259
523
|
// Not an elegant solution. Might need upstream changes like custom error types.
|
260
|
-
if (
|
524
|
+
if (err.message === 'No signatures for this pubkey') {
|
261
525
|
return false;
|
262
526
|
}
|
263
527
|
throw err;
|
@@ -267,26 +531,22 @@ class UtxoPsbt extends __1.Psbt {
|
|
267
531
|
/**
|
268
532
|
* Mostly copied from bitcoinjs-lib/ts_src/psbt.ts
|
269
533
|
*/
|
270
|
-
signAllInputsHD(hdKeyPair,
|
534
|
+
signAllInputsHD(hdKeyPair, params) {
|
271
535
|
if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {
|
272
536
|
throw new Error('Need HDSigner to sign input');
|
273
537
|
}
|
538
|
+
const { sighashTypes, deterministic } = toSignatureParams(this.network, params);
|
274
539
|
const results = [];
|
275
540
|
for (let i = 0; i < this.data.inputs.length; i++) {
|
276
541
|
try {
|
277
|
-
|
278
|
-
this.signTaprootInputHD(i, hdKeyPair, sighashTypes);
|
279
|
-
}
|
280
|
-
else {
|
281
|
-
this.signInputHD(i, hdKeyPair, sighashTypes);
|
282
|
-
}
|
542
|
+
this.signInputHD(i, hdKeyPair, { sighashTypes, deterministic });
|
283
543
|
results.push(true);
|
284
544
|
}
|
285
545
|
catch (err) {
|
286
546
|
results.push(false);
|
287
547
|
}
|
288
548
|
}
|
289
|
-
if (results.every((v) => v
|
549
|
+
if (results.every((v) => !v)) {
|
290
550
|
throw new Error('No inputs were signed');
|
291
551
|
}
|
292
552
|
return this;
|
@@ -294,7 +554,10 @@ class UtxoPsbt extends __1.Psbt {
|
|
294
554
|
/**
|
295
555
|
* Mostly copied from bitcoinjs-lib/ts_src/psbt.ts:signInputHD
|
296
556
|
*/
|
297
|
-
signTaprootInputHD(inputIndex, hdKeyPair, sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL]) {
|
557
|
+
signTaprootInputHD(inputIndex, hdKeyPair, { sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL], deterministic = false } = {}) {
|
558
|
+
if (!this.isTaprootInput(inputIndex)) {
|
559
|
+
throw new Error('not a taproot input');
|
560
|
+
}
|
298
561
|
if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {
|
299
562
|
throw new Error('Need HDSigner to sign input');
|
300
563
|
}
|
@@ -312,28 +575,144 @@ class UtxoPsbt extends __1.Psbt {
|
|
312
575
|
if (myDerivations.length === 0) {
|
313
576
|
throw new Error('Need one tapBip32Derivation masterFingerprint to match the HDSigner fingerprint');
|
314
577
|
}
|
315
|
-
|
578
|
+
function getDerivedNode(bipDv) {
|
316
579
|
const node = hdKeyPair.derivePath(bipDv.path);
|
317
|
-
if (!bipDv.pubkey
|
580
|
+
if (!equalPublicKeyIgnoreY(bipDv.pubkey, node.publicKey)) {
|
318
581
|
throw new Error('pubkey did not match tapBip32Derivation');
|
319
582
|
}
|
320
|
-
return
|
583
|
+
return node;
|
584
|
+
}
|
585
|
+
if (input.tapLeafScript?.length) {
|
586
|
+
const signers = myDerivations.map((bipDv) => {
|
587
|
+
const signer = getDerivedNode(bipDv);
|
588
|
+
if (!('signSchnorr' in signer)) {
|
589
|
+
throw new Error('signSchnorr function is required to sign p2tr');
|
590
|
+
}
|
591
|
+
return { signer, leafHashes: bipDv.leafHashes };
|
592
|
+
});
|
593
|
+
signers.forEach(({ signer, leafHashes }) => this.signTaprootInput(inputIndex, signer, leafHashes, sighashTypes));
|
594
|
+
}
|
595
|
+
else if (input.tapInternalKey?.length) {
|
596
|
+
const signers = myDerivations.map((bipDv) => {
|
597
|
+
const signer = getDerivedNode(bipDv);
|
598
|
+
if (!('privateKey' in signer) || !signer.privateKey) {
|
599
|
+
throw new Error('privateKey is required to sign p2tr musig2');
|
600
|
+
}
|
601
|
+
return signer;
|
602
|
+
});
|
603
|
+
signers.forEach((signer) => this.signTaprootMusig2Input(inputIndex, signer, { sighashTypes, deterministic }));
|
604
|
+
}
|
605
|
+
return this;
|
606
|
+
}
|
607
|
+
signInputHD(inputIndex, hdKeyPair, params) {
|
608
|
+
const { sighashTypes, deterministic } = toSignatureParams(this.network, params);
|
609
|
+
if (this.isTaprootInput(inputIndex)) {
|
610
|
+
return this.signTaprootInputHD(inputIndex, hdKeyPair, { sighashTypes, deterministic });
|
611
|
+
}
|
612
|
+
else {
|
613
|
+
return super.signInputHD(inputIndex, hdKeyPair, sighashTypes);
|
614
|
+
}
|
615
|
+
}
|
616
|
+
getMusig2Participants(inputIndex, tapInternalKey, tapMerkleRoot) {
|
617
|
+
const participantsKeyValData = Musig2_1.parsePsbtMusig2Participants(this.data.inputs[inputIndex]);
|
618
|
+
if (!participantsKeyValData) {
|
619
|
+
throw new Error(`Found 0 matching participant key value instead of 1`);
|
620
|
+
}
|
621
|
+
Musig2_1.assertPsbtMusig2Participants(participantsKeyValData, tapInternalKey, tapMerkleRoot);
|
622
|
+
return participantsKeyValData;
|
623
|
+
}
|
624
|
+
getMusig2Nonces(inputIndex, participantsKeyValData) {
|
625
|
+
const noncesKeyValsData = Musig2_1.parsePsbtMusig2Nonces(this.data.inputs[inputIndex]);
|
626
|
+
if (!noncesKeyValsData || !types_1.isTuple(noncesKeyValsData)) {
|
627
|
+
throw new Error(`Found ${noncesKeyValsData?.length ? noncesKeyValsData.length : 0} matching nonce key value instead of 2`);
|
628
|
+
}
|
629
|
+
Musig2_1.assertPsbtMusig2Nonces(noncesKeyValsData, participantsKeyValData);
|
630
|
+
return noncesKeyValsData;
|
631
|
+
}
|
632
|
+
/**
|
633
|
+
* Signs p2tr musig2 key path input with 2 aggregated keys.
|
634
|
+
*
|
635
|
+
* Note: Only can sign deterministically as the cosigner
|
636
|
+
* @param inputIndex
|
637
|
+
* @param signer - XY public key and private key are required
|
638
|
+
* @param sighashTypes
|
639
|
+
* @param deterministic If true, sign the musig input deterministically
|
640
|
+
*/
|
641
|
+
signTaprootMusig2Input(inputIndex, signer, { sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL], deterministic = false } = {}) {
|
642
|
+
if (!this.isTaprootKeyPathInput(inputIndex)) {
|
643
|
+
throw new Error('not a taproot musig2 input');
|
644
|
+
}
|
645
|
+
const input = this.data.inputs[inputIndex];
|
646
|
+
if (!input.tapInternalKey || !input.tapMerkleRoot) {
|
647
|
+
throw new Error('missing required input data');
|
648
|
+
}
|
649
|
+
// Retrieve and check that we have two participant nonces
|
650
|
+
const participants = this.getMusig2Participants(inputIndex, input.tapInternalKey, input.tapMerkleRoot);
|
651
|
+
const { tapOutputKey, participantPubKeys } = participants;
|
652
|
+
const signerPubKey = participantPubKeys.find((pubKey) => equalPublicKeyIgnoreY(pubKey, signer.publicKey));
|
653
|
+
if (!signerPubKey) {
|
654
|
+
throw new Error('signer pub key should match one of participant pub keys');
|
655
|
+
}
|
656
|
+
const nonces = this.getMusig2Nonces(inputIndex, participants);
|
657
|
+
const { hash, sighashType } = this.getTaprootHashForSig(inputIndex, sighashTypes);
|
658
|
+
let partialSig;
|
659
|
+
if (deterministic) {
|
660
|
+
if (!equalPublicKeyIgnoreY(signerPubKey, participantPubKeys[1])) {
|
661
|
+
throw new Error('can only add a deterministic signature on the cosigner');
|
662
|
+
}
|
663
|
+
const firstSignerNonce = nonces.find((n) => equalPublicKeyIgnoreY(n.participantPubKey, participantPubKeys[0]));
|
664
|
+
if (!firstSignerNonce) {
|
665
|
+
throw new Error('could not find the user nonce');
|
666
|
+
}
|
667
|
+
partialSig = Musig2_1.musig2DeterministicSign({
|
668
|
+
privateKey: signer.privateKey,
|
669
|
+
otherNonce: firstSignerNonce.pubNonce,
|
670
|
+
publicKeys: participantPubKeys,
|
671
|
+
internalPubKey: input.tapInternalKey,
|
672
|
+
tapTreeRoot: input.tapMerkleRoot,
|
673
|
+
hash,
|
674
|
+
}).sig;
|
675
|
+
}
|
676
|
+
else {
|
677
|
+
const sessionKey = Musig2_1.createMusig2SigningSession({
|
678
|
+
pubNonces: [nonces[0].pubNonce, nonces[1].pubNonce],
|
679
|
+
pubKeys: participantPubKeys,
|
680
|
+
txHash: hash,
|
681
|
+
internalPubKey: input.tapInternalKey,
|
682
|
+
tapTreeRoot: input.tapMerkleRoot,
|
683
|
+
});
|
684
|
+
const signerNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, signerPubKey));
|
685
|
+
if (!signerNonce) {
|
686
|
+
throw new Error('pubNonce is missing. retry signing process');
|
687
|
+
}
|
688
|
+
partialSig = Musig2_1.musig2PartialSign(signer.privateKey, signerNonce.pubNonce, sessionKey, this.nonceStore);
|
689
|
+
}
|
690
|
+
if (sighashType !== __1.Transaction.SIGHASH_DEFAULT) {
|
691
|
+
partialSig = Buffer.concat([partialSig, Buffer.of(sighashType)]);
|
692
|
+
}
|
693
|
+
const sig = Musig2_1.encodePsbtMusig2PartialSig({
|
694
|
+
participantPubKey: signerPubKey,
|
695
|
+
tapOutputKey,
|
696
|
+
partialSig: partialSig,
|
321
697
|
});
|
322
|
-
|
698
|
+
this.addProprietaryKeyValToInput(inputIndex, sig);
|
323
699
|
return this;
|
324
700
|
}
|
325
701
|
signTaprootInput(inputIndex, signer, leafHashes, sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL]) {
|
326
|
-
const pubkey = outputScripts_1.toXOnlyPublicKey(signer.publicKey);
|
327
702
|
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
328
703
|
// Figure out if this is script path or not, if not, tweak the private key
|
329
704
|
if (!input.tapLeafScript?.length) {
|
330
|
-
|
331
|
-
throw new Error('Taproot key path signing is not supported.');
|
705
|
+
throw new Error('tapLeafScript is required for p2tr script path');
|
332
706
|
}
|
707
|
+
const pubkey = outputScripts_1.toXOnlyPublicKey(signer.publicKey);
|
333
708
|
if (input.tapLeafScript.length !== 1) {
|
334
709
|
throw new Error('Only one leaf script supported for signing');
|
335
710
|
}
|
336
|
-
const tapLeafScript = input.tapLeafScript
|
711
|
+
const [tapLeafScript] = input.tapLeafScript;
|
712
|
+
if (this.isMultisigTaprootScript(tapLeafScript.script)) {
|
713
|
+
const pubKeys = parseInput_1.parsePubScript2Of3(tapLeafScript.script, 'taprootScriptPathSpend').publicKeys;
|
714
|
+
assert(pubKeys.find((pk) => pubkey.equals(pk)), 'public key not found in tap leaf script');
|
715
|
+
}
|
337
716
|
const parsedControlBlock = __1.taproot.parseControlBlock(__1.ecc, tapLeafScript.controlBlock);
|
338
717
|
const { leafVersion } = parsedControlBlock;
|
339
718
|
if (leafVersion !== tapLeafScript.leafVersion) {
|
@@ -359,7 +738,26 @@ class UtxoPsbt extends __1.Psbt {
|
|
359
738
|
});
|
360
739
|
return this;
|
361
740
|
}
|
741
|
+
getTaprootOutputScript(inputIndex) {
|
742
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
743
|
+
if (input.tapLeafScript?.length) {
|
744
|
+
return __1.taproot.createTaprootOutputScript({
|
745
|
+
controlBlock: input.tapLeafScript[0].controlBlock,
|
746
|
+
leafScript: input.tapLeafScript[0].script,
|
747
|
+
});
|
748
|
+
}
|
749
|
+
else if (input.tapInternalKey && input.tapMerkleRoot) {
|
750
|
+
return __1.taproot.createTaprootOutputScript({
|
751
|
+
internalPubKey: input.tapInternalKey,
|
752
|
+
taptreeRoot: input.tapMerkleRoot,
|
753
|
+
});
|
754
|
+
}
|
755
|
+
throw new Error('not a taproot input');
|
756
|
+
}
|
362
757
|
getTaprootHashForSig(inputIndex, sighashTypes, leafHash) {
|
758
|
+
if (!this.isTaprootInput(inputIndex)) {
|
759
|
+
throw new Error('not a taproot input');
|
760
|
+
}
|
363
761
|
const sighashType = this.data.inputs[inputIndex].sighashType || __1.Transaction.SIGHASH_DEFAULT;
|
364
762
|
if (sighashTypes && sighashTypes.indexOf(sighashType) < 0) {
|
365
763
|
throw new Error(`Sighash type is not allowed. Retry the sign method passing the ` +
|
@@ -391,39 +789,13 @@ class UtxoPsbt extends __1.Psbt {
|
|
391
789
|
prevoutScripts.push(prevout.script);
|
392
790
|
prevoutValues.push(prevout.value);
|
393
791
|
});
|
792
|
+
const outputScript = this.getTaprootOutputScript(inputIndex);
|
793
|
+
if (!outputScript.equals(prevoutScripts[inputIndex])) {
|
794
|
+
throw new Error(`Witness script for input #${inputIndex} doesn't match the scriptPubKey in the prevout`);
|
795
|
+
}
|
394
796
|
const hash = this.tx.hashForWitnessV1(inputIndex, prevoutScripts, prevoutValues, sighashType, leafHash);
|
395
797
|
return { hash, sighashType };
|
396
798
|
}
|
397
|
-
/**
|
398
|
-
* @retuns true iff the input is taproot.
|
399
|
-
*/
|
400
|
-
isTaprootInput(inputIndex) {
|
401
|
-
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
402
|
-
function isP2tr(output) {
|
403
|
-
try {
|
404
|
-
__1.p2trPayments.p2tr({ output }, { eccLib: __1.ecc });
|
405
|
-
return true;
|
406
|
-
}
|
407
|
-
catch (err) {
|
408
|
-
return false;
|
409
|
-
}
|
410
|
-
}
|
411
|
-
return !!(input.tapInternalKey ||
|
412
|
-
input.tapMerkleRoot ||
|
413
|
-
(input.tapLeafScript && input.tapLeafScript.length) ||
|
414
|
-
(input.tapBip32Derivation && input.tapBip32Derivation.length) ||
|
415
|
-
(input.witnessUtxo && isP2tr(input.witnessUtxo.script)));
|
416
|
-
}
|
417
|
-
/**
|
418
|
-
* @returns hash and hashType for taproot input at inputIndex
|
419
|
-
* @throws error if input at inputIndex is not a taproot input
|
420
|
-
*/
|
421
|
-
getTaprootHashForSigChecked(inputIndex, sighashTypes = [__1.Transaction.SIGHASH_DEFAULT, __1.Transaction.SIGHASH_ALL], leafHash) {
|
422
|
-
if (!this.isTaprootInput(inputIndex)) {
|
423
|
-
throw new Error(`${inputIndex} input is not a taproot type to take taproot tx hash`);
|
424
|
-
}
|
425
|
-
return this.getTaprootHashForSig(inputIndex, sighashTypes, leafHash);
|
426
|
-
}
|
427
799
|
/**
|
428
800
|
* Adds proprietary key value pair to PSBT input.
|
429
801
|
* Default identifierEncoding is utf-8 for identifier.
|
@@ -435,27 +807,199 @@ class UtxoPsbt extends __1.Psbt {
|
|
435
807
|
});
|
436
808
|
}
|
437
809
|
/**
|
438
|
-
*
|
810
|
+
* Adds or updates (if exists) proprietary key value pair to PSBT input.
|
811
|
+
* Default identifierEncoding is utf-8 for identifier.
|
812
|
+
*/
|
813
|
+
addOrUpdateProprietaryKeyValToInput(inputIndex, keyValueData) {
|
814
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
815
|
+
const key = proprietaryKeyVal_1.encodeProprietaryKey(keyValueData.key);
|
816
|
+
const { value } = keyValueData;
|
817
|
+
if (input.unknownKeyVals?.length) {
|
818
|
+
const ukvIndex = input.unknownKeyVals.findIndex((ukv) => ukv.key.equals(key));
|
819
|
+
if (ukvIndex > -1) {
|
820
|
+
input.unknownKeyVals[ukvIndex] = { key, value };
|
821
|
+
return this;
|
822
|
+
}
|
823
|
+
}
|
824
|
+
this.addUnknownKeyValToInput(inputIndex, {
|
825
|
+
key,
|
826
|
+
value,
|
827
|
+
});
|
828
|
+
return this;
|
829
|
+
}
|
830
|
+
/**
|
831
|
+
* To search any data from proprietary key value against keydata.
|
439
832
|
* Default identifierEncoding is utf-8 for identifier.
|
440
833
|
*/
|
441
834
|
getProprietaryKeyVals(inputIndex, keySearch) {
|
442
835
|
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
443
|
-
|
444
|
-
|
836
|
+
return PsbtUtil_1.getPsbtInputProprietaryKeyVals(input, keySearch);
|
837
|
+
}
|
838
|
+
/**
|
839
|
+
* To delete any data from proprietary key value.
|
840
|
+
* Default identifierEncoding is utf-8 for identifier.
|
841
|
+
*/
|
842
|
+
deleteProprietaryKeyVals(inputIndex, keysToDelete) {
|
843
|
+
const input = utils_1.checkForInput(this.data.inputs, inputIndex);
|
844
|
+
if (!input.unknownKeyVals?.length) {
|
845
|
+
return this;
|
445
846
|
}
|
446
|
-
|
447
|
-
|
847
|
+
if (keysToDelete && keysToDelete.subtype === undefined && Buffer.isBuffer(keysToDelete.keydata)) {
|
848
|
+
throw new Error('invalid proprietary key search filter combination. subtype is required');
|
849
|
+
}
|
850
|
+
input.unknownKeyVals = input.unknownKeyVals.filter((keyValue, i) => {
|
851
|
+
const key = proprietaryKeyVal_1.decodeProprietaryKey(keyValue.key);
|
852
|
+
return !(keysToDelete === undefined ||
|
853
|
+
(keysToDelete.identifier === key.identifier &&
|
854
|
+
(keysToDelete.subtype === undefined ||
|
855
|
+
(keysToDelete.subtype === key.subtype &&
|
856
|
+
(!Buffer.isBuffer(keysToDelete.keydata) || keysToDelete.keydata.equals(key.keydata))))));
|
448
857
|
});
|
449
|
-
return
|
450
|
-
|
451
|
-
|
452
|
-
|
453
|
-
|
858
|
+
return this;
|
859
|
+
}
|
860
|
+
createMusig2NonceForInput(inputIndex, keyPair, keyType, params = { deterministic: false }) {
|
861
|
+
const input = this.data.inputs[inputIndex];
|
862
|
+
if (!input.tapInternalKey) {
|
863
|
+
throw new Error('tapInternalKey is required to create nonce');
|
864
|
+
}
|
865
|
+
if (!input.tapMerkleRoot) {
|
866
|
+
throw new Error('tapMerkleRoot is required to create nonce');
|
867
|
+
}
|
868
|
+
const getDerivedKeyPair = () => {
|
869
|
+
if (!input.tapBip32Derivation?.length) {
|
870
|
+
throw new Error('tapBip32Derivation is required to create nonce');
|
871
|
+
}
|
872
|
+
const derived = UtxoPsbt.deriveKeyPair(keyPair, input.tapBip32Derivation, { ignoreY: true });
|
873
|
+
if (!derived) {
|
874
|
+
throw new Error('No bip32Derivation masterFingerprint matched the HD keyPair fingerprint');
|
875
|
+
}
|
876
|
+
return derived;
|
877
|
+
};
|
878
|
+
const derivedKeyPair = keyType === 'root' ? getDerivedKeyPair() : keyPair;
|
879
|
+
if (!derivedKeyPair.privateKey) {
|
880
|
+
throw new Error('privateKey is required to create nonce');
|
881
|
+
}
|
882
|
+
const participants = Musig2_1.parsePsbtMusig2Participants(input);
|
883
|
+
if (!participants) {
|
884
|
+
throw new Error(`Found 0 matching participant key value instead of 1`);
|
885
|
+
}
|
886
|
+
Musig2_1.assertPsbtMusig2Participants(participants, input.tapInternalKey, input.tapMerkleRoot);
|
887
|
+
const { tapOutputKey, participantPubKeys } = participants;
|
888
|
+
const participantPubKey = participantPubKeys.find((pubKey) => equalPublicKeyIgnoreY(pubKey, derivedKeyPair.publicKey));
|
889
|
+
if (!Buffer.isBuffer(participantPubKey)) {
|
890
|
+
throw new Error('participant plain pub key should match one bip32Derivation plain pub key');
|
891
|
+
}
|
892
|
+
const { hash } = this.getTaprootHashForSig(inputIndex);
|
893
|
+
let pubNonce;
|
894
|
+
if (params.deterministic) {
|
895
|
+
if (params.sessionId) {
|
896
|
+
throw new Error('Cannot add extra entropy when generating a deterministic nonce');
|
897
|
+
}
|
898
|
+
// There must be only 2 participant pubKeys if it got to this point
|
899
|
+
if (!equalPublicKeyIgnoreY(participantPubKey, participantPubKeys[1])) {
|
900
|
+
throw new Error(`Only the cosigner's nonce can be set deterministically`);
|
901
|
+
}
|
902
|
+
const nonces = Musig2_1.parsePsbtMusig2Nonces(input);
|
903
|
+
if (!nonces) {
|
904
|
+
throw new Error(`No nonces found on input #${inputIndex}`);
|
905
|
+
}
|
906
|
+
if (nonces.length > 2) {
|
907
|
+
throw new Error(`Cannot have more than 2 nonces`);
|
908
|
+
}
|
909
|
+
const firstSignerNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, participantPubKeys[0]));
|
910
|
+
if (!firstSignerNonce) {
|
911
|
+
throw new Error('signer nonce must be set if cosigner nonce is to be derived deterministically');
|
912
|
+
}
|
913
|
+
pubNonce = Musig2_1.createMusig2DeterministicNonce({
|
914
|
+
privateKey: derivedKeyPair.privateKey,
|
915
|
+
otherNonce: firstSignerNonce.pubNonce,
|
916
|
+
publicKeys: participantPubKeys,
|
917
|
+
internalPubKey: input.tapInternalKey,
|
918
|
+
tapTreeRoot: input.tapMerkleRoot,
|
919
|
+
hash,
|
920
|
+
});
|
921
|
+
}
|
922
|
+
else {
|
923
|
+
pubNonce = Buffer.from(this.nonceStore.createMusig2Nonce(derivedKeyPair.privateKey, participantPubKey, tapOutputKey, hash, params.sessionId));
|
924
|
+
}
|
925
|
+
return { tapOutputKey, participantPubKey, pubNonce };
|
926
|
+
}
|
927
|
+
setMusig2NoncesInner(keyPair, keyType, inputIndex, params = { deterministic: false }) {
|
928
|
+
if (keyPair.isNeutered()) {
|
929
|
+
throw new Error('private key is required to generate nonce');
|
930
|
+
}
|
931
|
+
if (Buffer.isBuffer(params.sessionId) && params.sessionId.length !== 32) {
|
932
|
+
throw new Error(`Invalid sessionId size ${params.sessionId.length}`);
|
933
|
+
}
|
934
|
+
const inputIndexes = inputIndex === undefined ? [...Array(this.inputCount).keys()] : [inputIndex];
|
935
|
+
inputIndexes.forEach((index) => {
|
936
|
+
if (!this.isTaprootKeyPathInput(index)) {
|
937
|
+
return;
|
938
|
+
}
|
939
|
+
const nonce = this.createMusig2NonceForInput(index, keyPair, keyType, params);
|
940
|
+
this.addOrUpdateProprietaryKeyValToInput(index, Musig2_1.encodePsbtMusig2PubNonce(nonce));
|
454
941
|
});
|
942
|
+
return this;
|
943
|
+
}
|
944
|
+
/**
|
945
|
+
* Generates and sets MuSig2 nonce to taproot key path input at inputIndex.
|
946
|
+
* If input is not a taproot key path, no action.
|
947
|
+
*
|
948
|
+
* @param inputIndex input index
|
949
|
+
* @param keyPair derived key pair
|
950
|
+
* @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,
|
951
|
+
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
952
|
+
* @param deterministic If true, set the cosigner nonce deterministically
|
953
|
+
*/
|
954
|
+
setInputMusig2Nonce(inputIndex, derivedKeyPair, params = { deterministic: false }) {
|
955
|
+
return this.setMusig2NoncesInner(derivedKeyPair, 'derived', inputIndex, params);
|
956
|
+
}
|
957
|
+
/**
|
958
|
+
* Generates and sets MuSig2 nonce to taproot key path input at inputIndex.
|
959
|
+
* If input is not a taproot key path, no action.
|
960
|
+
*
|
961
|
+
* @param inputIndex input index
|
962
|
+
* @param keyPair HD root key pair
|
963
|
+
* @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,
|
964
|
+
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
965
|
+
* @param deterministic If true, set the cosigner nonce deterministically
|
966
|
+
*/
|
967
|
+
setInputMusig2NonceHD(inputIndex, keyPair, params = { deterministic: false }) {
|
968
|
+
utils_1.checkForInput(this.data.inputs, inputIndex);
|
969
|
+
return this.setMusig2NoncesInner(keyPair, 'root', inputIndex, params);
|
970
|
+
}
|
971
|
+
/**
|
972
|
+
* Generates and sets MuSig2 nonce to all taproot key path inputs. Other inputs will be skipped.
|
973
|
+
*
|
974
|
+
* @param inputIndex input index
|
975
|
+
* @param keyPair derived key pair
|
976
|
+
* @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,
|
977
|
+
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
978
|
+
*/
|
979
|
+
setAllInputsMusig2Nonce(keyPair, params = { deterministic: false }) {
|
980
|
+
return this.setMusig2NoncesInner(keyPair, 'derived', undefined, params);
|
981
|
+
}
|
982
|
+
/**
|
983
|
+
* Generates and sets MuSig2 nonce to all taproot key path inputs. Other inputs will be skipped.
|
984
|
+
*
|
985
|
+
* @param inputIndex input index
|
986
|
+
* @param keyPair HD root key pair
|
987
|
+
* @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,
|
988
|
+
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
989
|
+
*/
|
990
|
+
setAllInputsMusig2NonceHD(keyPair, params = { deterministic: false }) {
|
991
|
+
return this.setMusig2NoncesInner(keyPair, 'root', undefined, params);
|
455
992
|
}
|
456
993
|
clone() {
|
457
994
|
return super.clone();
|
458
995
|
}
|
996
|
+
extractTransaction(disableFeeCheck) {
|
997
|
+
const tx = super.extractTransaction(disableFeeCheck);
|
998
|
+
if (tx instanceof UtxoTransaction_1.UtxoTransaction) {
|
999
|
+
return tx;
|
1000
|
+
}
|
1001
|
+
throw new Error('extractTransaction did not return instace of UtxoTransaction');
|
1002
|
+
}
|
459
1003
|
}
|
460
1004
|
exports.UtxoPsbt = UtxoPsbt;
|
461
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UtxoPsbt.js","sourceRoot":"","sources":["../../../src/bitgo/UtxoPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import { Psbt as PsbtBase } from 'bip174';\nimport { TapBip32Derivation, Transaction as ITransaction, TransactionFromBuffer } from 'bip174/src/lib/interfaces';\nimport { checkForInput } from 'bip174/src/lib/utils';\nimport { BufferWriter, varuint } from 'bitcoinjs-lib/src/bufferutils';\n\nimport {\n  taproot,\n  HDSigner,\n  Psbt,\n  PsbtTransaction,\n  Transaction,\n  TxOutput,\n  Network,\n  ecc as eccLib,\n  p2trPayments,\n} from '..';\nimport { UtxoTransaction } from './UtxoTransaction';\nimport { getOutputIdForInput } from './Unspent';\nimport { isSegwit } from './psbt/scriptTypes';\nimport { unsign } from './psbt/fromHalfSigned';\nimport { toXOnlyPublicKey } from './outputScripts';\nimport { parsePubScript } from './parseInput';\nimport { BIP32Factory } from 'bip32';\nimport * as bs58check from 'bs58check';\nimport { decodeProprietaryKey, encodeProprietaryKey, ProprietaryKey } from 'bip174/src/lib/proprietaryKeyVal';\n\nexport const PSBT_PROPRIETARY_IDENTIFIER = 'BITGO';\n\nexport enum ProprietaryKeySubtype {\n  ZEC_CONSENSUS_BRANCH_ID = 0x00,\n  MUSIG2_PARTICIPANT_PUB_KEYS = 0x01,\n  MUSIG2_PUB_NONCE = 0x02,\n}\n\nexport interface HDTaprootSigner extends HDSigner {\n  /**\n   * The path string must match /^m(\\/\\d+'?)+$/\n   * ex. m/44'/0'/0'/1/23 levels with ' must be hard derivations\n   */\n  derivePath(path: string): HDTaprootSigner;\n  /**\n   * Input hash (the \"message digest\") for the signature algorithm\n   * Return a 64 byte signature (32 byte r and 32 byte s in that order)\n   */\n  signSchnorr(hash: Buffer): Buffer;\n}\n\nexport interface SchnorrSigner {\n  publicKey: Buffer;\n  signSchnorr(hash: Buffer): Buffer;\n}\n\nexport interface TaprootSigner {\n  leafHashes: Buffer[];\n  signer: SchnorrSigner;\n}\n\nexport interface PsbtOpts {\n  network: Network;\n  maximumFeeRate?: number; // [sat/byte]\n  bip32PathsAbsolute?: boolean;\n}\n\n/**\n * Psbt proprietary keydata object.\n * <compact size uint identifier length> <bytes identifier> <compact size uint subtype> <bytes subkeydata>\n * => <bytes valuedata>\n */\nexport interface ProprietaryKeyValueData {\n  key: ProprietaryKey;\n  value: Buffer;\n}\n\n/**\n * Psbt proprietary keydata object search fields.\n * <compact size uint identifier length> <bytes identifier> <compact size uint subtype> <bytes subkeydata>\n */\nexport interface ProprietaryKeySearch {\n  identifier: string;\n  subtype: number;\n  keydata?: Buffer;\n  identifierEncoding?: BufferEncoding;\n}\n\n// TODO: upstream does `checkInputsForPartialSigs` before doing things like\n// `setVersion`. Our inputs could have tapscriptsigs (or in future tapkeysigs)\n// and not fail that check. Do we want to do anything about that?\nexport class UtxoPsbt<Tx extends UtxoTransaction<bigint> = UtxoTransaction<bigint>> extends Psbt {\n  protected static transactionFromBuffer(buffer: Buffer, network: Network): UtxoTransaction<bigint> {\n    return UtxoTransaction.fromBuffer<bigint>(buffer, false, 'bigint', network);\n  }\n\n  static createPsbt(opts: PsbtOpts, data?: PsbtBase): UtxoPsbt {\n    return new UtxoPsbt(\n      opts,\n      data || new PsbtBase(new PsbtTransaction({ tx: new UtxoTransaction<bigint>(opts.network) }))\n    );\n  }\n\n  static fromBuffer(buffer: Buffer, opts: PsbtOpts): UtxoPsbt {\n    const transactionFromBuffer: TransactionFromBuffer = (buffer: Buffer): ITransaction => {\n      const tx = this.transactionFromBuffer(buffer, opts.network);\n      return new PsbtTransaction({ tx });\n    };\n    const psbtBase = PsbtBase.fromBuffer(buffer, transactionFromBuffer, {\n      bip32PathsAbsolute: opts.bip32PathsAbsolute,\n    });\n    const psbt = this.createPsbt(opts, psbtBase);\n    // Upstream checks for duplicate inputs here, but it seems to be of dubious value.\n    return psbt;\n  }\n\n  static fromHex(data: string, opts: PsbtOpts): UtxoPsbt {\n    return this.fromBuffer(Buffer.from(data, 'hex'), opts);\n  }\n\n  get network(): Network {\n    return this.tx.network;\n  }\n\n  toHex(): string {\n    return this.toBuffer().toString('hex');\n  }\n\n  /**\n   * @return true iff PSBT input is finalized\n   */\n  isInputFinalized(inputIndex: number): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    return Buffer.isBuffer(input.finalScriptSig) || Buffer.isBuffer(input.finalScriptWitness);\n  }\n\n  /**\n   * @return partialSig/tapScriptSig count iff input is not finalized\n   */\n  getSignatureCount(inputIndex: number): number {\n    if (this.isInputFinalized(inputIndex)) {\n      throw new Error('Input is already finalized');\n    }\n    const input = checkForInput(this.data.inputs, inputIndex);\n    return Math.max(\n      Array.isArray(input.partialSig) ? input.partialSig.length : 0,\n      Array.isArray(input.tapScriptSig) ? input.tapScriptSig.length : 0\n    );\n  }\n\n  getNonWitnessPreviousTxids(): string[] {\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    const txidSet = new Set<string>();\n    this.data.inputs.forEach((input, index) => {\n      if (!input.witnessUtxo) {\n        throw new Error('Must have witness UTXO for all inputs');\n      }\n      if (!isSegwit(input.witnessUtxo.script, input.redeemScript)) {\n        txidSet.add(getOutputIdForInput(txInputs[index]).txid);\n      }\n    });\n    return [...txidSet];\n  }\n\n  addNonWitnessUtxos(txBufs: Record<string, Buffer>): this {\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    this.data.inputs.forEach((input, index) => {\n      if (!input.witnessUtxo) {\n        throw new Error('Must have witness UTXO for all inputs');\n      }\n      if (!isSegwit(input.witnessUtxo.script, input.redeemScript)) {\n        const { txid } = getOutputIdForInput(txInputs[index]);\n        if (txBufs[txid] === undefined) {\n          throw new Error('Not all required previous transactions provided');\n        }\n        this.updateInput(index, { nonWitnessUtxo: txBufs[txid] });\n      }\n    });\n    return this;\n  }\n\n  static fromTransaction(transaction: UtxoTransaction<bigint>, prevOutputs: TxOutput<bigint>[]): UtxoPsbt {\n    if (prevOutputs.length !== transaction.ins.length) {\n      throw new Error(\n        `Transaction has ${transaction.ins.length} inputs, but ${prevOutputs.length} previous outputs provided`\n      );\n    }\n    const clonedTransaction = transaction.clone();\n    const updates = unsign(clonedTransaction, prevOutputs);\n\n    const psbtBase = new PsbtBase(new PsbtTransaction({ tx: clonedTransaction }));\n    clonedTransaction.ins.forEach(() => psbtBase.inputs.push({ unknownKeyVals: [] }));\n    clonedTransaction.outs.forEach(() => psbtBase.outputs.push({ unknownKeyVals: [] }));\n    const psbt = this.createPsbt({ network: transaction.network }, psbtBase);\n\n    updates.forEach((update, index) => {\n      psbt.updateInput(index, update);\n      psbt.updateInput(index, { witnessUtxo: { script: prevOutputs[index].script, value: prevOutputs[index].value } });\n    });\n\n    return psbt;\n  }\n\n  getUnsignedTx(): UtxoTransaction<bigint> {\n    return this.tx.clone();\n  }\n\n  protected static newTransaction(network: Network): UtxoTransaction<bigint> {\n    return new UtxoTransaction<bigint>(network);\n  }\n\n  protected get tx(): Tx {\n    return (this.data.globalMap.unsignedTx as PsbtTransaction).tx as Tx;\n  }\n\n  protected checkForSignatures(propName?: string): void {\n    this.data.inputs.forEach((input) => {\n      if (input.tapScriptSig?.length || input.tapKeySig || input.partialSig?.length) {\n        throw new Error(`Cannot modify ${propName ?? 'transaction'} - signatures exist.`);\n      }\n    });\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   */\n  finalizeAllInputs(): this {\n    checkForInput(this.data.inputs, 0); // making sure we have at least one\n    this.data.inputs.map((input, idx) => {\n      return input.tapScriptSig?.length ? this.finalizeTaprootInput(idx) : this.finalizeInput(idx);\n    });\n    return this;\n  }\n\n  finalizeTaprootInput(inputIndex: number): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    // witness = control-block script first-sig second-sig\n    if (input.tapLeafScript?.length !== 1) {\n      throw new Error('Only one leaf script supported for finalizing');\n    }\n    const { controlBlock, script } = input.tapLeafScript[0];\n    const witness: Buffer[] = [script, controlBlock];\n    const [pubkey1, pubkey2] = parsePubScript(script, 'p2tr').publicKeys;\n    for (const pk of [pubkey1, pubkey2]) {\n      const sig = input.tapScriptSig?.find(({ pubkey }) => pubkey.equals(pk));\n      if (!sig) {\n        throw new Error('Could not find signatures in Script Sig.');\n      }\n      witness.unshift(sig.signature);\n    }\n\n    const witnessLength = witness.reduce((s, b) => s + b.length + varuint.encodingLength(b.length), 1);\n\n    const bufferWriter = BufferWriter.withCapacity(witnessLength);\n    bufferWriter.writeVector(witness);\n    const finalScriptWitness = bufferWriter.end();\n\n    this.data.updateInput(inputIndex, { finalScriptWitness });\n    this.data.clearFinalizedInput(inputIndex);\n\n    return this;\n  }\n\n  finalizeTapInputWithSingleLeafScriptAndSignature(inputIndex: number): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (input.tapLeafScript?.length !== 1) {\n      throw new Error('Only one leaf script supported for finalizing');\n    }\n    if (input.tapScriptSig?.length !== 1) {\n      throw new Error('Could not find signatures in Script Sig.');\n    }\n\n    const { controlBlock, script } = input.tapLeafScript[0];\n    const witness: Buffer[] = [input.tapScriptSig[0].signature, script, controlBlock];\n    const witnessLength = witness.reduce((s, b) => s + b.length + varuint.encodingLength(b.length), 1);\n\n    const bufferWriter = BufferWriter.withCapacity(witnessLength);\n    bufferWriter.writeVector(witness);\n    const finalScriptWitness = bufferWriter.end();\n\n    this.data.updateInput(inputIndex, { finalScriptWitness });\n    this.data.clearFinalizedInput(inputIndex);\n\n    return this;\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   *\n   * Unlike the function it overrides, this does not take a validator. In BitGo\n   * context, we know how we want to validate so we just hard code the right\n   * validator.\n   */\n  validateSignaturesOfAllInputs(): boolean {\n    checkForInput(this.data.inputs, 0); // making sure we have at least one\n    const results = this.data.inputs.map((input, idx) => {\n      return input.tapScriptSig?.length\n        ? this.validateTaprootSignaturesOfInput(idx)\n        : this.validateSignaturesOfInput(idx, (p, m, s) => eccLib.verify(m, p, s));\n    });\n    return results.reduce((final, res) => res && final, true);\n  }\n\n  validateTaprootSignaturesOfInput(inputIndex: number, pubkey?: Buffer): boolean {\n    const input = this.data.inputs[inputIndex];\n    const tapSigs = (input || {}).tapScriptSig;\n    if (!input || !tapSigs || tapSigs.length < 1) {\n      throw new Error('No signatures to validate');\n    }\n    let mySigs;\n    if (pubkey) {\n      const xOnlyPubkey = toXOnlyPublicKey(pubkey);\n      mySigs = tapSigs.filter((sig) => sig.pubkey.equals(xOnlyPubkey));\n      if (mySigs.length < 1) {\n        throw new Error('No signatures for this pubkey');\n      }\n    } else {\n      mySigs = tapSigs;\n    }\n    const results: boolean[] = [];\n\n    for (const pSig of mySigs) {\n      const { signature, leafHash, pubkey } = pSig;\n      let sigHashType: number;\n      let sig: Buffer;\n      if (signature.length === 65) {\n        sigHashType = signature[64];\n        sig = signature.slice(0, 64);\n      } else {\n        sigHashType = Transaction.SIGHASH_DEFAULT;\n        sig = signature;\n      }\n      const { hash } = this.getTaprootHashForSig(inputIndex, [sigHashType], leafHash);\n      results.push(eccLib.verifySchnorr(hash, pubkey, sig));\n    }\n    return results.every((res) => res === true);\n  }\n\n  /**\n   * @return array of boolean values. True when corresponding index in `publicKeys` has signed the transaction.\n   * If no signature in the tx or no public key matching signature, the validation is considered as false.\n   */\n  getSignatureValidationArray(inputIndex: number): boolean[] {\n    const noSigErrorMessages = ['No signatures to validate', 'No signatures for this pubkey'];\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const isP2tr = input.tapScriptSig?.length;\n    if (!this.data.globalMap.globalXpub) {\n      throw new Error('Cannot get signature validation array without global xpubs');\n    }\n    if (this.data.globalMap.globalXpub.length !== 3) {\n      throw new Error(`There must be 3 global xpubs and there are ${this.data.globalMap.globalXpub.length}`);\n    }\n    return this.data.globalMap.globalXpub.map((xpub) => {\n      // const bip32 = ECPair.fromPublicKey(xpub.extendedPubkey, { network: (this as any).opts.network });\n      const bip32 = BIP32Factory(eccLib).fromBase58(bs58check.encode(xpub.extendedPubkey));\n      try {\n        return isP2tr\n          ? this.validateTaprootSignaturesOfInput(inputIndex, bip32.publicKey)\n          : this.validateSignaturesOfInput(inputIndex, (p, m, s) => eccLib.verify(m, p, s), bip32.publicKey);\n      } catch (err) {\n        // Not an elegant solution. Might need upstream changes like custom error types.\n        if (noSigErrorMessages.includes(err.message)) {\n          return false;\n        }\n        throw err;\n      }\n    });\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   */\n  signAllInputsHD(\n    hdKeyPair: HDTaprootSigner,\n    sighashTypes: number[] = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL]\n  ): this {\n    if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {\n      throw new Error('Need HDSigner to sign input');\n    }\n\n    const results: boolean[] = [];\n    for (let i = 0; i < this.data.inputs.length; i++) {\n      try {\n        if (this.data.inputs[i].tapBip32Derivation?.length) {\n          this.signTaprootInputHD(i, hdKeyPair, sighashTypes);\n        } else {\n          this.signInputHD(i, hdKeyPair, sighashTypes);\n        }\n        results.push(true);\n      } catch (err) {\n        results.push(false);\n      }\n    }\n    if (results.every((v) => v === false)) {\n      throw new Error('No inputs were signed');\n    }\n    return this;\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts:signInputHD\n   */\n  signTaprootInputHD(\n    inputIndex: number,\n    hdKeyPair: HDTaprootSigner,\n    sighashTypes: number[] = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL]\n  ): this {\n    if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {\n      throw new Error('Need HDSigner to sign input');\n    }\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!input.tapBip32Derivation || input.tapBip32Derivation.length === 0) {\n      throw new Error('Need tapBip32Derivation to sign Taproot with HD');\n    }\n    const myDerivations = input.tapBip32Derivation\n      .map((bipDv) => {\n        if (bipDv.masterFingerprint.equals(hdKeyPair.fingerprint)) {\n          return bipDv;\n        }\n      })\n      .filter((v) => !!v) as TapBip32Derivation[];\n    if (myDerivations.length === 0) {\n      throw new Error('Need one tapBip32Derivation masterFingerprint to match the HDSigner fingerprint');\n    }\n    const signers: TaprootSigner[] = myDerivations.map((bipDv) => {\n      const node = hdKeyPair.derivePath(bipDv.path);\n      if (!bipDv.pubkey.equals(node.publicKey.slice(1))) {\n        throw new Error('pubkey did not match tapBip32Derivation');\n      }\n      return { signer: node, leafHashes: bipDv.leafHashes };\n    });\n    signers.forEach(({ signer, leafHashes }) => this.signTaprootInput(inputIndex, signer, leafHashes, sighashTypes));\n    return this;\n  }\n\n  signTaprootInput(\n    inputIndex: number,\n    signer: SchnorrSigner,\n    leafHashes: Buffer[],\n    sighashTypes: number[] = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL]\n  ): this {\n    const pubkey = toXOnlyPublicKey(signer.publicKey);\n    const input = checkForInput(this.data.inputs, inputIndex);\n    // Figure out if this is script path or not, if not, tweak the private key\n    if (!input.tapLeafScript?.length) {\n      // See BitGo/BitGoJS/modules/utxo_lib/src/transaction_builder.ts:trySign for how to support it.\n      throw new Error('Taproot key path signing is not supported.');\n    }\n    if (input.tapLeafScript.length !== 1) {\n      throw new Error('Only one leaf script supported for signing');\n    }\n    const tapLeafScript = input.tapLeafScript[0];\n    const parsedControlBlock = taproot.parseControlBlock(eccLib, tapLeafScript.controlBlock);\n    const { leafVersion } = parsedControlBlock;\n    if (leafVersion !== tapLeafScript.leafVersion) {\n      throw new Error('Tap script leaf version mismatch with control block');\n    }\n    const leafHash = taproot.getTapleafHash(eccLib, parsedControlBlock, tapLeafScript.script);\n    if (!leafHashes.find((l) => l.equals(leafHash))) {\n      throw new Error(`Signer cannot sign for leaf hash ${leafHash.toString('hex')}`);\n    }\n    const { hash, sighashType } = this.getTaprootHashForSig(inputIndex, sighashTypes, leafHash);\n    let signature = signer.signSchnorr(hash);\n    if (sighashType !== Transaction.SIGHASH_DEFAULT) {\n      signature = Buffer.concat([signature, Buffer.of(sighashType)]);\n    }\n    this.data.updateInput(inputIndex, {\n      tapScriptSig: [\n        {\n          pubkey,\n          signature,\n          leafHash,\n        },\n      ],\n    });\n    return this;\n  }\n\n  private getTaprootHashForSig(\n    inputIndex: number,\n    sighashTypes?: number[],\n    leafHash?: Buffer\n  ): {\n    hash: Buffer;\n    sighashType: number;\n  } {\n    const sighashType = this.data.inputs[inputIndex].sighashType || Transaction.SIGHASH_DEFAULT;\n    if (sighashTypes && sighashTypes.indexOf(sighashType) < 0) {\n      throw new Error(\n        `Sighash type is not allowed. Retry the sign method passing the ` +\n          `sighashTypes array of whitelisted types. Sighash type: ${sighashType}`\n      );\n    }\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    const prevoutScripts: Buffer[] = [];\n    const prevoutValues: bigint[] = [];\n\n    this.data.inputs.forEach((input, i) => {\n      let prevout;\n      if (input.nonWitnessUtxo) {\n        // TODO: This could be costly, either cache it here, or find a way to share with super\n        const nonWitnessUtxoTx = (this.constructor as typeof UtxoPsbt).transactionFromBuffer(\n          input.nonWitnessUtxo,\n          this.tx.network\n        );\n\n        const prevoutHash = txInputs[i].hash;\n        const utxoHash = nonWitnessUtxoTx.getHash();\n\n        // If a non-witness UTXO is provided, its hash must match the hash specified in the prevout\n        if (!prevoutHash.equals(utxoHash)) {\n          throw new Error(`Non-witness UTXO hash for input #${i} doesn't match the hash specified in the prevout`);\n        }\n\n        const prevoutIndex = txInputs[i].index;\n        prevout = nonWitnessUtxoTx.outs[prevoutIndex];\n      } else if (input.witnessUtxo) {\n        prevout = input.witnessUtxo;\n      } else {\n        throw new Error('Need a Utxo input item for signing');\n      }\n      prevoutScripts.push(prevout.script);\n      prevoutValues.push(prevout.value);\n    });\n    const hash = this.tx.hashForWitnessV1(inputIndex, prevoutScripts, prevoutValues, sighashType, leafHash);\n    return { hash, sighashType };\n  }\n\n  /**\n   * @retuns true iff the input is taproot.\n   */\n  isTaprootInput(inputIndex: number): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    function isP2tr(output: Buffer): boolean {\n      try {\n        p2trPayments.p2tr({ output }, { eccLib });\n        return true;\n      } catch (err) {\n        return false;\n      }\n    }\n    return !!(\n      input.tapInternalKey ||\n      input.tapMerkleRoot ||\n      (input.tapLeafScript && input.tapLeafScript.length) ||\n      (input.tapBip32Derivation && input.tapBip32Derivation.length) ||\n      (input.witnessUtxo && isP2tr(input.witnessUtxo.script))\n    );\n  }\n\n  /**\n   * @returns hash and hashType for taproot input at inputIndex\n   * @throws error if input at inputIndex is not a taproot input\n   */\n  getTaprootHashForSigChecked(\n    inputIndex: number,\n    sighashTypes: number[] = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL],\n    leafHash?: Buffer\n  ): {\n    hash: Buffer;\n    sighashType: number;\n  } {\n    if (!this.isTaprootInput(inputIndex)) {\n      throw new Error(`${inputIndex} input is not a taproot type to take taproot tx hash`);\n    }\n    return this.getTaprootHashForSig(inputIndex, sighashTypes, leafHash);\n  }\n\n  /**\n   * Adds proprietary key value pair to PSBT input.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  addProprietaryKeyValToInput(inputIndex: number, keyValueData: ProprietaryKeyValueData): this {\n    return this.addUnknownKeyValToInput(inputIndex, {\n      key: encodeProprietaryKey(keyValueData.key),\n      value: keyValueData.value,\n    });\n  }\n\n  /**\n   * To search any data from proprietary key value againts keydata.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  getProprietaryKeyVals(inputIndex: number, keySearch?: ProprietaryKeySearch): ProprietaryKeyValueData[] {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!input.unknownKeyVals || input.unknownKeyVals.length === 0) {\n      return [];\n    }\n    const keyVals = input.unknownKeyVals.map(({ key, value }, i) => {\n      return { key: decodeProprietaryKey(key), value };\n    });\n    return keyVals.filter((keyVal) => {\n      return (\n        keySearch === undefined ||\n        (keySearch.identifier === keyVal.key.identifier &&\n          keySearch.subtype === keyVal.key.subtype &&\n          (!Buffer.isBuffer(keySearch.keydata) || keySearch.keydata.equals(keyVal.key.keydata)))\n      );\n    });\n  }\n\n  clone(): this {\n    return super.clone() as this;\n  }\n}\n"]}
|
1005
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UtxoPsbt.js","sourceRoot":"","sources":["../../../src/bitgo/UtxoPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport { Psbt as PsbtBase } from 'bip174';\nimport {\n  Bip32Derivation,\n  PsbtInput,\n  TapBip32Derivation,\n  Transaction as ITransaction,\n  TransactionFromBuffer,\n} from 'bip174/src/lib/interfaces';\nimport { checkForInput } from 'bip174/src/lib/utils';\nimport { BufferWriter, varuint } from 'bitcoinjs-lib/src/bufferutils';\nimport { SessionKey } from '@brandonblack/musig';\nimport { BIP32Factory, BIP32Interface } from 'bip32';\nimport * as bs58check from 'bs58check';\nimport { decodeProprietaryKey, encodeProprietaryKey } from 'bip174/src/lib/proprietaryKeyVal';\n\nimport {\n  taproot,\n  HDSigner,\n  Psbt,\n  PsbtTransaction,\n  Transaction,\n  TxOutput,\n  Network,\n  ecc as eccLib,\n  getMainnet,\n  networks,\n} from '..';\nimport { UtxoTransaction } from './UtxoTransaction';\nimport { getOutputIdForInput } from './Unspent';\nimport { isSegwit } from './psbt/scriptTypes';\nimport { unsign } from './psbt/fromHalfSigned';\nimport { toXOnlyPublicKey } from './outputScripts';\nimport { parsePubScript2Of3 } from './parseInput';\nimport {\n  createMusig2SigningSession,\n  encodePsbtMusig2PartialSig,\n  encodePsbtMusig2PubNonce,\n  musig2PartialSign,\n  parsePsbtMusig2Nonces,\n  parsePsbtMusig2Participants,\n  PsbtMusig2Participants,\n  assertPsbtMusig2Nonces,\n  assertPsbtMusig2Participants,\n  Musig2NonceStore,\n  PsbtMusig2PubNonce,\n  parsePsbtMusig2PartialSigs,\n  musig2PartialSigVerify,\n  musig2AggregateSigs,\n  getSigHashTypeFromSigs,\n  musig2DeterministicSign,\n  createMusig2DeterministicNonce,\n} from './Musig2';\nimport { isTriple, isTuple, Triple, Tuple } from './types';\nimport { getTaprootOutputKey } from '../taproot';\nimport {\n  getPsbtInputProprietaryKeyVals,\n  getPsbtInputSignatureCount,\n  ProprietaryKeySearch,\n  ProprietaryKeySubtype,\n  ProprietaryKeyValue,\n  PSBT_PROPRIETARY_IDENTIFIER,\n} from './PsbtUtil';\n\ntype SignatureParams = {\n  /** When true, and add the second (last) nonce and signature for a taproot key\n   * path spend deterministically. Throws an error if done for the first nonce/signature\n   * of a taproot keypath spend. Ignore for all other input types.\n   */\n  deterministic: boolean;\n  /** Allowed sighash types */\n  sighashTypes: number[];\n};\n\nfunction defaultSighashTypes(network: Network): number[] {\n  const sighashTypes = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL];\n  switch (getMainnet(network)) {\n    case networks.bitcoincash:\n    case networks.bitcoinsv:\n    case networks.bitcoingold:\n    case networks.ecash:\n      return [...sighashTypes, ...sighashTypes.map((s) => s | UtxoTransaction.SIGHASH_FORKID)];\n    default:\n      return sighashTypes;\n  }\n}\n\nfunction toSignatureParams(network: Network, v?: Partial<SignatureParams> | number[]): SignatureParams {\n  if (Array.isArray(v)) return toSignatureParams(network, { sighashTypes: v });\n  return { deterministic: false, sighashTypes: defaultSighashTypes(network), ...v };\n}\n\n/**\n * @param a\n * @param b\n * @returns true if the two public keys are equal ignoring the y coordinate.\n */\nfunction equalPublicKeyIgnoreY(a: Buffer, b: Buffer): boolean {\n  return toXOnlyPublicKey(a).equals(toXOnlyPublicKey(b));\n}\n\nexport interface HDTaprootSigner extends HDSigner {\n  /**\n   * The path string must match /^m(\\/\\d+'?)+$/\n   * ex. m/44'/0'/0'/1/23 levels with ' must be hard derivations\n   */\n  derivePath(path: string): HDTaprootSigner;\n  /**\n   * Input hash (the \"message digest\") for the signature algorithm\n   * Return a 64 byte signature (32 byte r and 32 byte s in that order)\n   */\n  signSchnorr(hash: Buffer): Buffer;\n}\n\n/**\n * HD signer object for taproot p2tr musig2 key path sign\n */\nexport interface HDTaprootMusig2Signer extends HDSigner {\n  /**\n   * Musig2 requires signer's 32-bytes private key to be passed to it.\n   */\n  privateKey: Buffer;\n\n  /**\n   * The path string must match /^m(\\/\\d+'?)+$/\n   * ex. m/44'/0'/0'/1/23 levels with ' must be hard derivations\n   */\n  derivePath(path: string): HDTaprootMusig2Signer;\n}\n\nexport interface SchnorrSigner {\n  publicKey: Buffer;\n  signSchnorr(hash: Buffer): Buffer;\n}\n\nexport interface Musig2Signer {\n  publicKey: Buffer;\n  privateKey: Buffer;\n}\n\nexport interface TaprootSigner {\n  leafHashes: Buffer[];\n  signer: SchnorrSigner;\n}\n\nexport interface PsbtOpts {\n  network: Network;\n  maximumFeeRate?: number; // [sat/byte]\n  bip32PathsAbsolute?: boolean;\n}\n\n// TODO: upstream does `checkInputsForPartialSigs` before doing things like\n// `setVersion`. Our inputs could have tapscriptsigs (or in future tapkeysigs)\n// and not fail that check. Do we want to do anything about that?\nexport class UtxoPsbt<Tx extends UtxoTransaction<bigint> = UtxoTransaction<bigint>> extends Psbt {\n  private nonceStore = new Musig2NonceStore();\n\n  protected static transactionFromBuffer(buffer: Buffer, network: Network): UtxoTransaction<bigint> {\n    return UtxoTransaction.fromBuffer<bigint>(buffer, false, 'bigint', network);\n  }\n\n  static createPsbt(opts: PsbtOpts, data?: PsbtBase): UtxoPsbt {\n    return new UtxoPsbt(\n      opts,\n      data || new PsbtBase(new PsbtTransaction({ tx: new UtxoTransaction<bigint>(opts.network) }))\n    );\n  }\n\n  static fromBuffer(buffer: Buffer, opts: PsbtOpts): UtxoPsbt {\n    const transactionFromBuffer: TransactionFromBuffer = (buffer: Buffer): ITransaction => {\n      const tx = this.transactionFromBuffer(buffer, opts.network);\n      return new PsbtTransaction({ tx });\n    };\n    const psbtBase = PsbtBase.fromBuffer(buffer, transactionFromBuffer, {\n      bip32PathsAbsolute: opts.bip32PathsAbsolute,\n    });\n    const psbt = this.createPsbt(opts, psbtBase);\n    // Upstream checks for duplicate inputs here, but it seems to be of dubious value.\n    return psbt;\n  }\n\n  static fromHex(data: string, opts: PsbtOpts): UtxoPsbt {\n    return this.fromBuffer(Buffer.from(data, 'hex'), opts);\n  }\n\n  /**\n   * @param parent - Parent key. Matched with `bip32Derivations` using `fingerprint` property.\n   * @param bip32Derivations - possible derivations for input or output\n   * @param ignoreY - when true, ignore the y coordinate when matching public keys\n   * @return derived bip32 node if matching derivation is found, undefined if none is found\n   * @throws Error if more than one match is found\n   */\n  static deriveKeyPair(\n    parent: BIP32Interface,\n    bip32Derivations: Bip32Derivation[],\n    { ignoreY }: { ignoreY: boolean }\n  ): BIP32Interface | undefined {\n    const matchingDerivations = bip32Derivations.filter((bipDv) => {\n      return bipDv.masterFingerprint.equals(parent.fingerprint);\n    });\n\n    if (!matchingDerivations.length) {\n      // No fingerprint match\n      return undefined;\n    }\n\n    if (matchingDerivations.length !== 1) {\n      throw new Error(\n        `more than one matching derivation for fingerprint ${parent.fingerprint.toString('hex')}: ${\n          matchingDerivations.length\n        }`\n      );\n    }\n\n    const [derivation] = matchingDerivations;\n    const node = parent.derivePath(derivation.path);\n\n    if (!node.publicKey.equals(derivation.pubkey)) {\n      if (!ignoreY || !equalPublicKeyIgnoreY(node.publicKey, derivation.pubkey)) {\n        throw new Error('pubkey did not match bip32Derivation');\n      }\n    }\n\n    return node;\n  }\n\n  static deriveKeyPairForInput(bip32: BIP32Interface, input: PsbtInput): Buffer | undefined {\n    return input.tapBip32Derivation?.length\n      ? UtxoPsbt.deriveKeyPair(bip32, input.tapBip32Derivation, { ignoreY: true })?.publicKey\n      : input.bip32Derivation?.length\n      ? UtxoPsbt.deriveKeyPair(bip32, input.bip32Derivation, { ignoreY: false })?.publicKey\n      : bip32?.publicKey;\n  }\n\n  get network(): Network {\n    return this.tx.network;\n  }\n\n  toHex(): string {\n    return this.toBuffer().toString('hex');\n  }\n\n  /**\n   * It is expensive to attempt to compute every output address using psbt.txOutputs[outputIndex]\n   * to then just get the script. Here, we are doing the same thing as what txOutputs() does in\n   * bitcoinjs-lib, but without iterating over each output.\n   * @param outputIndex\n   * @returns output script at the given index\n   */\n  getOutputScript(outputIndex: number): Buffer {\n    return (this as any).__CACHE.__TX.outs[outputIndex].script as Buffer;\n  }\n\n  getNonWitnessPreviousTxids(): string[] {\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    const txidSet = new Set<string>();\n    this.data.inputs.forEach((input, index) => {\n      if (!input.witnessUtxo) {\n        throw new Error('Must have witness UTXO for all inputs');\n      }\n      if (!isSegwit(input.witnessUtxo.script, input.redeemScript)) {\n        txidSet.add(getOutputIdForInput(txInputs[index]).txid);\n      }\n    });\n    return [...txidSet];\n  }\n\n  addNonWitnessUtxos(txBufs: Record<string, Buffer>): this {\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    this.data.inputs.forEach((input, index) => {\n      if (!input.witnessUtxo) {\n        throw new Error('Must have witness UTXO for all inputs');\n      }\n      if (!isSegwit(input.witnessUtxo.script, input.redeemScript)) {\n        const { txid } = getOutputIdForInput(txInputs[index]);\n        if (txBufs[txid] === undefined) {\n          throw new Error('Not all required previous transactions provided');\n        }\n        this.updateInput(index, { nonWitnessUtxo: txBufs[txid] });\n      }\n    });\n    return this;\n  }\n\n  static fromTransaction(transaction: UtxoTransaction<bigint>, prevOutputs: TxOutput<bigint>[]): UtxoPsbt {\n    if (prevOutputs.length !== transaction.ins.length) {\n      throw new Error(\n        `Transaction has ${transaction.ins.length} inputs, but ${prevOutputs.length} previous outputs provided`\n      );\n    }\n    const clonedTransaction = transaction.clone();\n    const updates = unsign(clonedTransaction, prevOutputs);\n\n    const psbtBase = new PsbtBase(new PsbtTransaction({ tx: clonedTransaction }));\n    clonedTransaction.ins.forEach(() => psbtBase.inputs.push({ unknownKeyVals: [] }));\n    clonedTransaction.outs.forEach(() => psbtBase.outputs.push({ unknownKeyVals: [] }));\n    const psbt = this.createPsbt({ network: transaction.network }, psbtBase);\n\n    updates.forEach((update, index) => {\n      psbt.updateInput(index, update);\n      psbt.updateInput(index, { witnessUtxo: { script: prevOutputs[index].script, value: prevOutputs[index].value } });\n    });\n\n    return psbt;\n  }\n\n  getUnsignedTx(): UtxoTransaction<bigint> {\n    return this.tx.clone();\n  }\n\n  protected static newTransaction(network: Network): UtxoTransaction<bigint> {\n    return new UtxoTransaction<bigint>(network);\n  }\n\n  protected get tx(): Tx {\n    return (this.data.globalMap.unsignedTx as PsbtTransaction).tx as Tx;\n  }\n\n  protected checkForSignatures(propName?: string): void {\n    this.data.inputs.forEach((input) => {\n      if (input.tapScriptSig?.length || input.tapKeySig || input.partialSig?.length) {\n        throw new Error(`Cannot modify ${propName ?? 'transaction'} - signatures exist.`);\n      }\n    });\n  }\n\n  /**\n   * @returns true if the input at inputIndex is a taproot key path.\n   * Checks for presence of minimum required key path input fields and absence of any script path only input fields.\n   */\n  isTaprootKeyPathInput(inputIndex: number): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    return (\n      !!input.tapInternalKey &&\n      !!input.tapMerkleRoot &&\n      !(\n        input.tapLeafScript?.length ||\n        input.tapScriptSig?.length ||\n        input.tapBip32Derivation?.some((v) => v.leafHashes.length)\n      )\n    );\n  }\n\n  /**\n   * @returns true if the input at inputIndex is a taproot script path.\n   * Checks for presence of minimum required script path input fields and absence of any key path only input fields.\n   */\n  isTaprootScriptPathInput(inputIndex: number): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    return (\n      !!input.tapLeafScript?.length &&\n      !(\n        this.getProprietaryKeyVals(inputIndex, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n        }).length ||\n        this.getProprietaryKeyVals(inputIndex, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        }).length ||\n        this.getProprietaryKeyVals(inputIndex, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n        }).length\n      )\n    );\n  }\n\n  /**\n   * @returns true if the input at inputIndex is a taproot\n   */\n  isTaprootInput(inputIndex: number): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const isP2TR = (script: Buffer): boolean => {\n      try {\n        getTaprootOutputKey(script);\n        return true;\n      } catch (e) {\n        return false;\n      }\n    };\n    return !!(\n      input.tapInternalKey ||\n      input.tapMerkleRoot ||\n      input.tapLeafScript?.length ||\n      input.tapBip32Derivation?.length ||\n      input.tapScriptSig?.length ||\n      this.getProprietaryKeyVals(inputIndex, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n      }).length ||\n      this.getProprietaryKeyVals(inputIndex, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      }).length ||\n      this.getProprietaryKeyVals(inputIndex, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n      }).length ||\n      (input.witnessUtxo && isP2TR(input.witnessUtxo.script))\n    );\n  }\n\n  private isMultisigTaprootScript(script: Buffer): boolean {\n    try {\n      parsePubScript2Of3(script, 'taprootScriptPathSpend');\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   */\n  finalizeAllInputs(): this {\n    checkForInput(this.data.inputs, 0); // making sure we have at least one\n    this.data.inputs.map((input, idx) => {\n      if (input.tapLeafScript?.length) {\n        return this.isMultisigTaprootScript(input.tapLeafScript[0].script)\n          ? this.finalizeTaprootInput(idx)\n          : this.finalizeTapInputWithSingleLeafScriptAndSignature(idx);\n      } else if (this.isTaprootKeyPathInput(idx)) {\n        return this.finalizeTaprootMusig2Input(idx);\n      }\n      return this.finalizeInput(idx);\n    });\n    return this;\n  }\n\n  finalizeTaprootInput(inputIndex: number): this {\n    const checkPartialSigSighashes = (sig: Buffer) => {\n      const sighashType = sig.length === 64 ? Transaction.SIGHASH_DEFAULT : sig.readUInt8(sig.length - 1);\n      const inputSighashType = input.sighashType === undefined ? Transaction.SIGHASH_DEFAULT : input.sighashType;\n      assert(sighashType === inputSighashType, 'signature sighash does not match input sighash type');\n    };\n    const input = checkForInput(this.data.inputs, inputIndex);\n    // witness = control-block script first-sig second-sig\n    if (input.tapLeafScript?.length !== 1) {\n      throw new Error('Only one leaf script supported for finalizing');\n    }\n    const { controlBlock, script } = input.tapLeafScript[0];\n    const witness: Buffer[] = [script, controlBlock];\n    const [pubkey1, pubkey2] = parsePubScript2Of3(script, 'taprootScriptPathSpend').publicKeys;\n    for (const pk of [pubkey1, pubkey2]) {\n      const sig = input.tapScriptSig?.find(({ pubkey }) => equalPublicKeyIgnoreY(pk, pubkey));\n      if (!sig) {\n        throw new Error('Could not find signatures in Script Sig.');\n      }\n      checkPartialSigSighashes(sig.signature);\n      witness.unshift(sig.signature);\n    }\n\n    const witnessLength = witness.reduce((s, b) => s + b.length + varuint.encodingLength(b.length), 1);\n\n    const bufferWriter = BufferWriter.withCapacity(witnessLength);\n    bufferWriter.writeVector(witness);\n    const finalScriptWitness = bufferWriter.end();\n\n    this.data.updateInput(inputIndex, { finalScriptWitness });\n    this.data.clearFinalizedInput(inputIndex);\n\n    return this;\n  }\n\n  /**\n   * Finalizes a taproot musig2 input by aggregating all partial sigs.\n   * IMPORTANT: Always call validate* function before finalizing.\n   */\n  finalizeTaprootMusig2Input(inputIndex: number): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const partialSigs = parsePsbtMusig2PartialSigs(input);\n    if (partialSigs?.length !== 2) {\n      throw new Error(`invalid number of partial signatures ${partialSigs ? partialSigs.length : 0} to finalize`);\n    }\n    const { partialSigs: pSigs, sigHashType } = getSigHashTypeFromSigs(partialSigs);\n    const { sessionKey } = this.getMusig2SessionKey(inputIndex, sigHashType);\n\n    const aggSig = musig2AggregateSigs(\n      pSigs.map((pSig) => pSig.partialSig),\n      sessionKey\n    );\n\n    const sig = sigHashType === Transaction.SIGHASH_DEFAULT ? aggSig : Buffer.concat([aggSig, Buffer.of(sigHashType)]);\n\n    // single signature with 64/65 bytes size is script witness for key path spend\n    const bufferWriter = BufferWriter.withCapacity(1 + varuint.encodingLength(sig.length) + sig.length);\n    bufferWriter.writeVector([sig]);\n    const finalScriptWitness = bufferWriter.end();\n\n    this.data.updateInput(inputIndex, { finalScriptWitness });\n    this.data.clearFinalizedInput(inputIndex);\n    // deleting only BitGo proprietary key values.\n    this.deleteProprietaryKeyVals(inputIndex, { identifier: PSBT_PROPRIETARY_IDENTIFIER });\n    return this;\n  }\n\n  finalizeTapInputWithSingleLeafScriptAndSignature(inputIndex: number): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (input.tapLeafScript?.length !== 1) {\n      throw new Error('Only one leaf script supported for finalizing');\n    }\n    if (input.tapScriptSig?.length !== 1) {\n      throw new Error('Could not find signatures in Script Sig.');\n    }\n\n    const { controlBlock, script } = input.tapLeafScript[0];\n    const witness: Buffer[] = [input.tapScriptSig[0].signature, script, controlBlock];\n    const witnessLength = witness.reduce((s, b) => s + b.length + varuint.encodingLength(b.length), 1);\n\n    const bufferWriter = BufferWriter.withCapacity(witnessLength);\n    bufferWriter.writeVector(witness);\n    const finalScriptWitness = bufferWriter.end();\n\n    this.data.updateInput(inputIndex, { finalScriptWitness });\n    this.data.clearFinalizedInput(inputIndex);\n\n    return this;\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   *\n   * Unlike the function it overrides, this does not take a validator. In BitGo\n   * context, we know how we want to validate so we just hard code the right\n   * validator.\n   */\n  validateSignaturesOfAllInputs(): boolean {\n    checkForInput(this.data.inputs, 0); // making sure we have at least one\n    const results = this.data.inputs.map((input, idx) => {\n      return this.validateSignaturesOfInputCommon(idx);\n    });\n    return results.reduce((final, res) => res && final, true);\n  }\n\n  /**\n   * @returns true iff any matching valid signature is found for a derived pub key from given HD key pair.\n   */\n  validateSignaturesOfInputHD(inputIndex: number, hdKeyPair: BIP32Interface): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const pubKey = UtxoPsbt.deriveKeyPairForInput(hdKeyPair, input);\n    if (!pubKey) {\n      throw new Error('can not derive from HD key pair');\n    }\n    return this.validateSignaturesOfInputCommon(inputIndex, pubKey);\n  }\n\n  /**\n   * @returns true iff any valid signature(s) are found from bip32 data of PSBT or for given pub key.\n   */\n  validateSignaturesOfInputCommon(inputIndex: number, pubkey?: Buffer): boolean {\n    try {\n      if (this.isTaprootScriptPathInput(inputIndex)) {\n        return this.validateTaprootSignaturesOfInput(inputIndex, pubkey);\n      } else if (this.isTaprootKeyPathInput(inputIndex)) {\n        return this.validateTaprootMusig2SignaturesOfInput(inputIndex, pubkey);\n      }\n      return this.validateSignaturesOfInput(inputIndex, (p, m, s) => eccLib.verify(m, p, s, true), pubkey);\n    } catch (err) {\n      // Not an elegant solution. Might need upstream changes like custom error types.\n      if (err.message === 'No signatures for this pubkey') {\n        return false;\n      }\n      throw err;\n    }\n  }\n\n  private getMusig2SessionKey(\n    inputIndex: number,\n    sigHashType: number\n  ): {\n    participants: PsbtMusig2Participants;\n    nonces: Tuple<PsbtMusig2PubNonce>;\n    hash: Buffer;\n    sessionKey: SessionKey;\n  } {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!input.tapInternalKey || !input.tapMerkleRoot) {\n      throw new Error('both tapInternalKey and tapMerkleRoot are required');\n    }\n\n    const participants = this.getMusig2Participants(inputIndex, input.tapInternalKey, input.tapMerkleRoot);\n    const nonces = this.getMusig2Nonces(inputIndex, participants);\n\n    const { hash } = this.getTaprootHashForSig(inputIndex, [sigHashType]);\n\n    const sessionKey = createMusig2SigningSession({\n      pubNonces: [nonces[0].pubNonce, nonces[1].pubNonce],\n      pubKeys: participants.participantPubKeys,\n      txHash: hash,\n      internalPubKey: input.tapInternalKey,\n      tapTreeRoot: input.tapMerkleRoot,\n    });\n    return { participants, nonces, hash, sessionKey };\n  }\n\n  /**\n   * @returns true for following cases.\n   * If valid musig2 partial signatures exists for both 2 keys, it will also verify aggregated sig\n   * for aggregated tweaked key (output key), otherwise only verifies partial sig.\n   * If pubkey is passed in input, it will check sig only for that pubkey,\n   * if no sig exits for such key, throws error.\n   * For invalid state of input data, it will throw errors.\n   */\n  validateTaprootMusig2SignaturesOfInput(inputIndex: number, pubkey?: Buffer): boolean {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const partialSigs = parsePsbtMusig2PartialSigs(input);\n    if (!partialSigs) {\n      throw new Error(`No signatures to validate`);\n    }\n\n    let myPartialSigs = partialSigs;\n    if (pubkey) {\n      myPartialSigs = partialSigs.filter((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, pubkey));\n      if (myPartialSigs?.length < 1) {\n        throw new Error('No signatures for this pubkey');\n      }\n    }\n\n    const { partialSigs: mySigs, sigHashType } = getSigHashTypeFromSigs(myPartialSigs);\n    const { participants, nonces, hash, sessionKey } = this.getMusig2SessionKey(inputIndex, sigHashType);\n\n    const results = mySigs.map((mySig) => {\n      const myNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, mySig.participantPubKey));\n      if (!myNonce) {\n        throw new Error('Found no pub nonce for pubkey');\n      }\n      return musig2PartialSigVerify(mySig.partialSig, mySig.participantPubKey, myNonce.pubNonce, sessionKey);\n    });\n\n    // For valid single sig or 1 or 2 failure sigs, no need to validate aggregated sig. So skip.\n    const result = results.every((res) => res);\n    if (!result || mySigs.length < 2) {\n      return result;\n    }\n\n    const aggSig = musig2AggregateSigs(\n      mySigs.map((mySig) => mySig.partialSig),\n      sessionKey\n    );\n\n    return eccLib.verifySchnorr(hash, participants.tapOutputKey, aggSig);\n  }\n\n  validateTaprootSignaturesOfInput(inputIndex: number, pubkey?: Buffer): boolean {\n    const input = this.data.inputs[inputIndex];\n    const tapSigs = (input || {}).tapScriptSig;\n    if (!input || !tapSigs || tapSigs.length < 1) {\n      throw new Error('No signatures to validate');\n    }\n    let mySigs;\n    if (pubkey) {\n      mySigs = tapSigs.filter((sig) => equalPublicKeyIgnoreY(sig.pubkey, pubkey));\n      if (mySigs.length < 1) {\n        throw new Error('No signatures for this pubkey');\n      }\n    } else {\n      mySigs = tapSigs;\n    }\n    const results: boolean[] = [];\n\n    assert(input.tapLeafScript?.length === 1, `single tapLeafScript is expected. Got ${input.tapLeafScript?.length}`);\n    const [tapLeafScript] = input.tapLeafScript;\n    const pubKeys = this.isMultisigTaprootScript(tapLeafScript.script)\n      ? parsePubScript2Of3(tapLeafScript.script, 'taprootScriptPathSpend').publicKeys\n      : undefined;\n\n    for (const pSig of mySigs) {\n      const { signature, leafHash, pubkey } = pSig;\n      if (pubKeys) {\n        assert(\n          pubKeys.find((pk) => pubkey.equals(pk)),\n          'public key not found in tap leaf script'\n        );\n      }\n      let sigHashType: number;\n      let sig: Buffer;\n      if (signature.length === 65) {\n        sigHashType = signature[64];\n        sig = signature.slice(0, 64);\n      } else {\n        sigHashType = Transaction.SIGHASH_DEFAULT;\n        sig = signature;\n      }\n      const { hash } = this.getTaprootHashForSig(inputIndex, [sigHashType], leafHash);\n      results.push(eccLib.verifySchnorr(hash, pubkey, sig));\n    }\n    return results.every((res) => res);\n  }\n\n  /**\n   * @param inputIndex\n   * @param rootNodes optional input root bip32 nodes to verify with. If it is not provided, globalXpub will be used.\n   * @return array of boolean values. True when corresponding index in `publicKeys` has signed the transaction.\n   * If no signature in the tx or no public key matching signature, the validation is considered as false.\n   */\n  getSignatureValidationArray(\n    inputIndex: number,\n    { rootNodes }: { rootNodes?: Triple<BIP32Interface> } = {}\n  ): Triple<boolean> {\n    if (!rootNodes && (!this.data.globalMap.globalXpub?.length || !isTriple(this.data.globalMap.globalXpub))) {\n      throw new Error('Cannot get signature validation array without 3 global xpubs');\n    }\n\n    const bip32s = rootNodes\n      ? rootNodes\n      : this.data.globalMap.globalXpub?.map((xpub) =>\n          BIP32Factory(eccLib).fromBase58(bs58check.encode(xpub.extendedPubkey))\n        );\n\n    if (!bip32s) {\n      throw new Error('either globalMap or rootNodes is required');\n    }\n\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!getPsbtInputSignatureCount(input)) {\n      return [false, false, false];\n    }\n\n    return bip32s.map((bip32) => {\n      const pubKey = UtxoPsbt.deriveKeyPairForInput(bip32, input);\n      if (!pubKey) {\n        return false;\n      }\n      try {\n        return this.validateSignaturesOfInputCommon(inputIndex, pubKey);\n      } catch (err) {\n        // Not an elegant solution. Might need upstream changes like custom error types.\n        if (err.message === 'No signatures for this pubkey') {\n          return false;\n        }\n        throw err;\n      }\n    }) as Triple<boolean>;\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts\n   */\n  signAllInputsHD(\n    hdKeyPair: HDTaprootSigner | HDTaprootMusig2Signer,\n    params?: number[] | Partial<SignatureParams>\n  ): this {\n    if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {\n      throw new Error('Need HDSigner to sign input');\n    }\n    const { sighashTypes, deterministic } = toSignatureParams(this.network, params);\n\n    const results: boolean[] = [];\n    for (let i = 0; i < this.data.inputs.length; i++) {\n      try {\n        this.signInputHD(i, hdKeyPair, { sighashTypes, deterministic });\n        results.push(true);\n      } catch (err) {\n        results.push(false);\n      }\n    }\n    if (results.every((v) => !v)) {\n      throw new Error('No inputs were signed');\n    }\n    return this;\n  }\n\n  /**\n   * Mostly copied from bitcoinjs-lib/ts_src/psbt.ts:signInputHD\n   */\n  signTaprootInputHD(\n    inputIndex: number,\n    hdKeyPair: HDTaprootSigner | HDTaprootMusig2Signer,\n    { sighashTypes = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL], deterministic = false } = {}\n  ): this {\n    if (!this.isTaprootInput(inputIndex)) {\n      throw new Error('not a taproot input');\n    }\n    if (!hdKeyPair || !hdKeyPair.publicKey || !hdKeyPair.fingerprint) {\n      throw new Error('Need HDSigner to sign input');\n    }\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!input.tapBip32Derivation || input.tapBip32Derivation.length === 0) {\n      throw new Error('Need tapBip32Derivation to sign Taproot with HD');\n    }\n    const myDerivations = input.tapBip32Derivation\n      .map((bipDv) => {\n        if (bipDv.masterFingerprint.equals(hdKeyPair.fingerprint)) {\n          return bipDv;\n        }\n      })\n      .filter((v) => !!v) as TapBip32Derivation[];\n    if (myDerivations.length === 0) {\n      throw new Error('Need one tapBip32Derivation masterFingerprint to match the HDSigner fingerprint');\n    }\n\n    function getDerivedNode(bipDv: TapBip32Derivation): HDTaprootMusig2Signer | HDTaprootSigner {\n      const node = hdKeyPair.derivePath(bipDv.path);\n      if (!equalPublicKeyIgnoreY(bipDv.pubkey, node.publicKey)) {\n        throw new Error('pubkey did not match tapBip32Derivation');\n      }\n      return node;\n    }\n\n    if (input.tapLeafScript?.length) {\n      const signers: TaprootSigner[] = myDerivations.map((bipDv) => {\n        const signer = getDerivedNode(bipDv);\n        if (!('signSchnorr' in signer)) {\n          throw new Error('signSchnorr function is required to sign p2tr');\n        }\n        return { signer, leafHashes: bipDv.leafHashes };\n      });\n      signers.forEach(({ signer, leafHashes }) => this.signTaprootInput(inputIndex, signer, leafHashes, sighashTypes));\n    } else if (input.tapInternalKey?.length) {\n      const signers: Musig2Signer[] = myDerivations.map((bipDv) => {\n        const signer = getDerivedNode(bipDv);\n        if (!('privateKey' in signer) || !signer.privateKey) {\n          throw new Error('privateKey is required to sign p2tr musig2');\n        }\n        return signer;\n      });\n      signers.forEach((signer) => this.signTaprootMusig2Input(inputIndex, signer, { sighashTypes, deterministic }));\n    }\n    return this;\n  }\n\n  signInputHD(\n    inputIndex: number,\n    hdKeyPair: HDTaprootSigner | HDTaprootMusig2Signer,\n    params?: number[] | Partial<SignatureParams>\n  ): this {\n    const { sighashTypes, deterministic } = toSignatureParams(this.network, params);\n    if (this.isTaprootInput(inputIndex)) {\n      return this.signTaprootInputHD(inputIndex, hdKeyPair, { sighashTypes, deterministic });\n    } else {\n      return super.signInputHD(inputIndex, hdKeyPair, sighashTypes);\n    }\n  }\n\n  private getMusig2Participants(inputIndex: number, tapInternalKey: Buffer, tapMerkleRoot: Buffer) {\n    const participantsKeyValData = parsePsbtMusig2Participants(this.data.inputs[inputIndex]);\n    if (!participantsKeyValData) {\n      throw new Error(`Found 0 matching participant key value instead of 1`);\n    }\n    assertPsbtMusig2Participants(participantsKeyValData, tapInternalKey, tapMerkleRoot);\n    return participantsKeyValData;\n  }\n\n  private getMusig2Nonces(inputIndex: number, participantsKeyValData: PsbtMusig2Participants) {\n    const noncesKeyValsData = parsePsbtMusig2Nonces(this.data.inputs[inputIndex]);\n    if (!noncesKeyValsData || !isTuple(noncesKeyValsData)) {\n      throw new Error(\n        `Found ${noncesKeyValsData?.length ? noncesKeyValsData.length : 0} matching nonce key value instead of 2`\n      );\n    }\n    assertPsbtMusig2Nonces(noncesKeyValsData, participantsKeyValData);\n    return noncesKeyValsData;\n  }\n\n  /**\n   * Signs p2tr musig2 key path input with 2 aggregated keys.\n   *\n   * Note: Only can sign deterministically as the cosigner\n   * @param inputIndex\n   * @param signer - XY public key and private key are required\n   * @param sighashTypes\n   * @param deterministic If true, sign the musig input deterministically\n   */\n  signTaprootMusig2Input(\n    inputIndex: number,\n    signer: Musig2Signer,\n    { sighashTypes = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL], deterministic = false } = {}\n  ): this {\n    if (!this.isTaprootKeyPathInput(inputIndex)) {\n      throw new Error('not a taproot musig2 input');\n    }\n\n    const input = this.data.inputs[inputIndex];\n\n    if (!input.tapInternalKey || !input.tapMerkleRoot) {\n      throw new Error('missing required input data');\n    }\n\n    // Retrieve and check that we have two participant nonces\n    const participants = this.getMusig2Participants(inputIndex, input.tapInternalKey, input.tapMerkleRoot);\n    const { tapOutputKey, participantPubKeys } = participants;\n    const signerPubKey = participantPubKeys.find((pubKey) => equalPublicKeyIgnoreY(pubKey, signer.publicKey));\n    if (!signerPubKey) {\n      throw new Error('signer pub key should match one of participant pub keys');\n    }\n\n    const nonces = this.getMusig2Nonces(inputIndex, participants);\n    const { hash, sighashType } = this.getTaprootHashForSig(inputIndex, sighashTypes);\n\n    let partialSig: Buffer;\n    if (deterministic) {\n      if (!equalPublicKeyIgnoreY(signerPubKey, participantPubKeys[1])) {\n        throw new Error('can only add a deterministic signature on the cosigner');\n      }\n\n      const firstSignerNonce = nonces.find((n) => equalPublicKeyIgnoreY(n.participantPubKey, participantPubKeys[0]));\n      if (!firstSignerNonce) {\n        throw new Error('could not find the user nonce');\n      }\n\n      partialSig = musig2DeterministicSign({\n        privateKey: signer.privateKey,\n        otherNonce: firstSignerNonce.pubNonce,\n        publicKeys: participantPubKeys,\n        internalPubKey: input.tapInternalKey,\n        tapTreeRoot: input.tapMerkleRoot,\n        hash,\n      }).sig;\n    } else {\n      const sessionKey = createMusig2SigningSession({\n        pubNonces: [nonces[0].pubNonce, nonces[1].pubNonce],\n        pubKeys: participantPubKeys,\n        txHash: hash,\n        internalPubKey: input.tapInternalKey,\n        tapTreeRoot: input.tapMerkleRoot,\n      });\n\n      const signerNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, signerPubKey));\n      if (!signerNonce) {\n        throw new Error('pubNonce is missing. retry signing process');\n      }\n      partialSig = musig2PartialSign(signer.privateKey, signerNonce.pubNonce, sessionKey, this.nonceStore);\n    }\n\n    if (sighashType !== Transaction.SIGHASH_DEFAULT) {\n      partialSig = Buffer.concat([partialSig, Buffer.of(sighashType)]);\n    }\n\n    const sig = encodePsbtMusig2PartialSig({\n      participantPubKey: signerPubKey,\n      tapOutputKey,\n      partialSig: partialSig,\n    });\n    this.addProprietaryKeyValToInput(inputIndex, sig);\n    return this;\n  }\n\n  signTaprootInput(\n    inputIndex: number,\n    signer: SchnorrSigner,\n    leafHashes: Buffer[],\n    sighashTypes: number[] = [Transaction.SIGHASH_DEFAULT, Transaction.SIGHASH_ALL]\n  ): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    // Figure out if this is script path or not, if not, tweak the private key\n    if (!input.tapLeafScript?.length) {\n      throw new Error('tapLeafScript is required for p2tr script path');\n    }\n    const pubkey = toXOnlyPublicKey(signer.publicKey);\n    if (input.tapLeafScript.length !== 1) {\n      throw new Error('Only one leaf script supported for signing');\n    }\n    const [tapLeafScript] = input.tapLeafScript;\n\n    if (this.isMultisigTaprootScript(tapLeafScript.script)) {\n      const pubKeys = parsePubScript2Of3(tapLeafScript.script, 'taprootScriptPathSpend').publicKeys;\n      assert(\n        pubKeys.find((pk) => pubkey.equals(pk)),\n        'public key not found in tap leaf script'\n      );\n    }\n\n    const parsedControlBlock = taproot.parseControlBlock(eccLib, tapLeafScript.controlBlock);\n    const { leafVersion } = parsedControlBlock;\n    if (leafVersion !== tapLeafScript.leafVersion) {\n      throw new Error('Tap script leaf version mismatch with control block');\n    }\n    const leafHash = taproot.getTapleafHash(eccLib, parsedControlBlock, tapLeafScript.script);\n    if (!leafHashes.find((l) => l.equals(leafHash))) {\n      throw new Error(`Signer cannot sign for leaf hash ${leafHash.toString('hex')}`);\n    }\n    const { hash, sighashType } = this.getTaprootHashForSig(inputIndex, sighashTypes, leafHash);\n    let signature = signer.signSchnorr(hash);\n    if (sighashType !== Transaction.SIGHASH_DEFAULT) {\n      signature = Buffer.concat([signature, Buffer.of(sighashType)]);\n    }\n    this.data.updateInput(inputIndex, {\n      tapScriptSig: [\n        {\n          pubkey,\n          signature,\n          leafHash,\n        },\n      ],\n    });\n    return this;\n  }\n\n  private getTaprootOutputScript(inputIndex: number) {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (input.tapLeafScript?.length) {\n      return taproot.createTaprootOutputScript({\n        controlBlock: input.tapLeafScript[0].controlBlock,\n        leafScript: input.tapLeafScript[0].script,\n      });\n    } else if (input.tapInternalKey && input.tapMerkleRoot) {\n      return taproot.createTaprootOutputScript({\n        internalPubKey: input.tapInternalKey,\n        taptreeRoot: input.tapMerkleRoot,\n      });\n    }\n    throw new Error('not a taproot input');\n  }\n\n  private getTaprootHashForSig(\n    inputIndex: number,\n    sighashTypes?: number[],\n    leafHash?: Buffer\n  ): {\n    hash: Buffer;\n    sighashType: number;\n  } {\n    if (!this.isTaprootInput(inputIndex)) {\n      throw new Error('not a taproot input');\n    }\n    const sighashType = this.data.inputs[inputIndex].sighashType || Transaction.SIGHASH_DEFAULT;\n    if (sighashTypes && sighashTypes.indexOf(sighashType) < 0) {\n      throw new Error(\n        `Sighash type is not allowed. Retry the sign method passing the ` +\n          `sighashTypes array of whitelisted types. Sighash type: ${sighashType}`\n      );\n    }\n    const txInputs = this.txInputs; // These are somewhat costly to extract\n    const prevoutScripts: Buffer[] = [];\n    const prevoutValues: bigint[] = [];\n\n    this.data.inputs.forEach((input, i) => {\n      let prevout;\n      if (input.nonWitnessUtxo) {\n        // TODO: This could be costly, either cache it here, or find a way to share with super\n        const nonWitnessUtxoTx = (this.constructor as typeof UtxoPsbt).transactionFromBuffer(\n          input.nonWitnessUtxo,\n          this.tx.network\n        );\n\n        const prevoutHash = txInputs[i].hash;\n        const utxoHash = nonWitnessUtxoTx.getHash();\n\n        // If a non-witness UTXO is provided, its hash must match the hash specified in the prevout\n        if (!prevoutHash.equals(utxoHash)) {\n          throw new Error(`Non-witness UTXO hash for input #${i} doesn't match the hash specified in the prevout`);\n        }\n\n        const prevoutIndex = txInputs[i].index;\n        prevout = nonWitnessUtxoTx.outs[prevoutIndex];\n      } else if (input.witnessUtxo) {\n        prevout = input.witnessUtxo;\n      } else {\n        throw new Error('Need a Utxo input item for signing');\n      }\n      prevoutScripts.push(prevout.script);\n      prevoutValues.push(prevout.value);\n    });\n    const outputScript = this.getTaprootOutputScript(inputIndex);\n    if (!outputScript.equals(prevoutScripts[inputIndex])) {\n      throw new Error(`Witness script for input #${inputIndex} doesn't match the scriptPubKey in the prevout`);\n    }\n    const hash = this.tx.hashForWitnessV1(inputIndex, prevoutScripts, prevoutValues, sighashType, leafHash);\n    return { hash, sighashType };\n  }\n\n  /**\n   * Adds proprietary key value pair to PSBT input.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  addProprietaryKeyValToInput(inputIndex: number, keyValueData: ProprietaryKeyValue): this {\n    return this.addUnknownKeyValToInput(inputIndex, {\n      key: encodeProprietaryKey(keyValueData.key),\n      value: keyValueData.value,\n    });\n  }\n\n  /**\n   * Adds or updates (if exists) proprietary key value pair to PSBT input.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  addOrUpdateProprietaryKeyValToInput(inputIndex: number, keyValueData: ProprietaryKeyValue): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    const key = encodeProprietaryKey(keyValueData.key);\n    const { value } = keyValueData;\n    if (input.unknownKeyVals?.length) {\n      const ukvIndex = input.unknownKeyVals.findIndex((ukv) => ukv.key.equals(key));\n      if (ukvIndex > -1) {\n        input.unknownKeyVals[ukvIndex] = { key, value };\n        return this;\n      }\n    }\n    this.addUnknownKeyValToInput(inputIndex, {\n      key,\n      value,\n    });\n    return this;\n  }\n\n  /**\n   * To search any data from proprietary key value against keydata.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  getProprietaryKeyVals(inputIndex: number, keySearch?: ProprietaryKeySearch): ProprietaryKeyValue[] {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    return getPsbtInputProprietaryKeyVals(input, keySearch);\n  }\n\n  /**\n   * To delete any data from proprietary key value.\n   * Default identifierEncoding is utf-8 for identifier.\n   */\n  deleteProprietaryKeyVals(inputIndex: number, keysToDelete?: ProprietaryKeySearch): this {\n    const input = checkForInput(this.data.inputs, inputIndex);\n    if (!input.unknownKeyVals?.length) {\n      return this;\n    }\n    if (keysToDelete && keysToDelete.subtype === undefined && Buffer.isBuffer(keysToDelete.keydata)) {\n      throw new Error('invalid proprietary key search filter combination. subtype is required');\n    }\n    input.unknownKeyVals = input.unknownKeyVals.filter((keyValue, i) => {\n      const key = decodeProprietaryKey(keyValue.key);\n      return !(\n        keysToDelete === undefined ||\n        (keysToDelete.identifier === key.identifier &&\n          (keysToDelete.subtype === undefined ||\n            (keysToDelete.subtype === key.subtype &&\n              (!Buffer.isBuffer(keysToDelete.keydata) || keysToDelete.keydata.equals(key.keydata)))))\n      );\n    });\n    return this;\n  }\n\n  private createMusig2NonceForInput(\n    inputIndex: number,\n    keyPair: BIP32Interface,\n    keyType: 'root' | 'derived',\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): PsbtMusig2PubNonce {\n    const input = this.data.inputs[inputIndex];\n    if (!input.tapInternalKey) {\n      throw new Error('tapInternalKey is required to create nonce');\n    }\n    if (!input.tapMerkleRoot) {\n      throw new Error('tapMerkleRoot is required to create nonce');\n    }\n    const getDerivedKeyPair = (): BIP32Interface => {\n      if (!input.tapBip32Derivation?.length) {\n        throw new Error('tapBip32Derivation is required to create nonce');\n      }\n      const derived = UtxoPsbt.deriveKeyPair(keyPair, input.tapBip32Derivation, { ignoreY: true });\n      if (!derived) {\n        throw new Error('No bip32Derivation masterFingerprint matched the HD keyPair fingerprint');\n      }\n      return derived;\n    };\n    const derivedKeyPair = keyType === 'root' ? getDerivedKeyPair() : keyPair;\n    if (!derivedKeyPair.privateKey) {\n      throw new Error('privateKey is required to create nonce');\n    }\n    const participants = parsePsbtMusig2Participants(input);\n    if (!participants) {\n      throw new Error(`Found 0 matching participant key value instead of 1`);\n    }\n    assertPsbtMusig2Participants(participants, input.tapInternalKey, input.tapMerkleRoot);\n    const { tapOutputKey, participantPubKeys } = participants;\n\n    const participantPubKey = participantPubKeys.find((pubKey) =>\n      equalPublicKeyIgnoreY(pubKey, derivedKeyPair.publicKey)\n    );\n    if (!Buffer.isBuffer(participantPubKey)) {\n      throw new Error('participant plain pub key should match one bip32Derivation plain pub key');\n    }\n\n    const { hash } = this.getTaprootHashForSig(inputIndex);\n\n    let pubNonce: Buffer;\n    if (params.deterministic) {\n      if (params.sessionId) {\n        throw new Error('Cannot add extra entropy when generating a deterministic nonce');\n      }\n      // There must be only 2 participant pubKeys if it got to this point\n      if (!equalPublicKeyIgnoreY(participantPubKey, participantPubKeys[1])) {\n        throw new Error(`Only the cosigner's nonce can be set deterministically`);\n      }\n      const nonces = parsePsbtMusig2Nonces(input);\n      if (!nonces) {\n        throw new Error(`No nonces found on input #${inputIndex}`);\n      }\n      if (nonces.length > 2) {\n        throw new Error(`Cannot have more than 2 nonces`);\n      }\n      const firstSignerNonce = nonces.find((kv) => equalPublicKeyIgnoreY(kv.participantPubKey, participantPubKeys[0]));\n      if (!firstSignerNonce) {\n        throw new Error('signer nonce must be set if cosigner nonce is to be derived deterministically');\n      }\n\n      pubNonce = createMusig2DeterministicNonce({\n        privateKey: derivedKeyPair.privateKey,\n        otherNonce: firstSignerNonce.pubNonce,\n        publicKeys: participantPubKeys,\n        internalPubKey: input.tapInternalKey,\n        tapTreeRoot: input.tapMerkleRoot,\n        hash,\n      });\n    } else {\n      pubNonce = Buffer.from(\n        this.nonceStore.createMusig2Nonce(\n          derivedKeyPair.privateKey,\n          participantPubKey,\n          tapOutputKey,\n          hash,\n          params.sessionId\n        )\n      );\n    }\n\n    return { tapOutputKey, participantPubKey, pubNonce };\n  }\n\n  private setMusig2NoncesInner(\n    keyPair: BIP32Interface,\n    keyType: 'root' | 'derived',\n    inputIndex?: number,\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): this {\n    if (keyPair.isNeutered()) {\n      throw new Error('private key is required to generate nonce');\n    }\n    if (Buffer.isBuffer(params.sessionId) && params.sessionId.length !== 32) {\n      throw new Error(`Invalid sessionId size ${params.sessionId.length}`);\n    }\n\n    const inputIndexes = inputIndex === undefined ? [...Array(this.inputCount).keys()] : [inputIndex];\n    inputIndexes.forEach((index) => {\n      if (!this.isTaprootKeyPathInput(index)) {\n        return;\n      }\n      const nonce = this.createMusig2NonceForInput(index, keyPair, keyType, params);\n      this.addOrUpdateProprietaryKeyValToInput(index, encodePsbtMusig2PubNonce(nonce));\n    });\n    return this;\n  }\n\n  /**\n   * Generates and sets MuSig2 nonce to taproot key path input at inputIndex.\n   * If input is not a taproot key path, no action.\n   *\n   * @param inputIndex input index\n   * @param keyPair derived key pair\n   * @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,\n   * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n   * @param deterministic If true, set the cosigner nonce deterministically\n   */\n  setInputMusig2Nonce(\n    inputIndex: number,\n    derivedKeyPair: BIP32Interface,\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): this {\n    return this.setMusig2NoncesInner(derivedKeyPair, 'derived', inputIndex, params);\n  }\n\n  /**\n   * Generates and sets MuSig2 nonce to taproot key path input at inputIndex.\n   * If input is not a taproot key path, no action.\n   *\n   * @param inputIndex input index\n   * @param keyPair HD root key pair\n   * @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,\n   * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n   * @param deterministic If true, set the cosigner nonce deterministically\n   */\n  setInputMusig2NonceHD(\n    inputIndex: number,\n    keyPair: BIP32Interface,\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): this {\n    checkForInput(this.data.inputs, inputIndex);\n    return this.setMusig2NoncesInner(keyPair, 'root', inputIndex, params);\n  }\n\n  /**\n   * Generates and sets MuSig2 nonce to all taproot key path inputs. Other inputs will be skipped.\n   *\n   * @param inputIndex input index\n   * @param keyPair derived key pair\n   * @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,\n   * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n   */\n  setAllInputsMusig2Nonce(\n    keyPair: BIP32Interface,\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): this {\n    return this.setMusig2NoncesInner(keyPair, 'derived', undefined, params);\n  }\n\n  /**\n   * Generates and sets MuSig2 nonce to all taproot key path inputs. Other inputs will be skipped.\n   *\n   * @param inputIndex input index\n   * @param keyPair HD root key pair\n   * @param sessionId Optional extra entropy. If provided it must either be a counter unique to this secret key,\n   * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n   */\n  setAllInputsMusig2NonceHD(\n    keyPair: BIP32Interface,\n    params: { sessionId?: Buffer; deterministic?: boolean } = { deterministic: false }\n  ): this {\n    return this.setMusig2NoncesInner(keyPair, 'root', undefined, params);\n  }\n\n  clone(): this {\n    return super.clone() as this;\n  }\n\n  extractTransaction(disableFeeCheck?: boolean): UtxoTransaction<bigint> {\n    const tx = super.extractTransaction(disableFeeCheck);\n    if (tx instanceof UtxoTransaction) {\n      return tx;\n    }\n    throw new Error('extractTransaction did not return instace of UtxoTransaction');\n  }\n}\n"]}
|