@bitgo-beta/utxo-lib 8.0.3-beta.43 → 8.0.3-beta.430
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +19 -16
- package/dist/src/address.js +5 -5
- package/dist/src/addressFormat.js +9 -9
- package/dist/src/bitgo/Musig2.js +19 -19
- package/dist/src/bitgo/PsbtUtil.d.ts +5 -0
- package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -1
- package/dist/src/bitgo/PsbtUtil.js +18 -3
- package/dist/src/bitgo/Unspent.js +3 -3
- package/dist/src/bitgo/UtxoPsbt.d.ts +15 -3
- package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoPsbt.js +118 -76
- package/dist/src/bitgo/UtxoTransaction.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoTransaction.js +9 -9
- package/dist/src/bitgo/bitcoincash/address.js +6 -6
- package/dist/src/bitgo/bitcoincash/index.js +6 -2
- package/dist/src/bitgo/dash/DashTransaction.js +3 -3
- package/dist/src/bitgo/dash/index.js +6 -2
- package/dist/src/bitgo/index.d.ts +2 -0
- package/dist/src/bitgo/index.d.ts.map +1 -1
- package/dist/src/bitgo/index.js +9 -3
- 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/LitecoinTransaction.js +2 -2
- package/dist/src/bitgo/litecoin/index.js +6 -2
- package/dist/src/bitgo/outputScripts.d.ts +1 -0
- package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
- package/dist/src/bitgo/outputScripts.js +29 -14
- package/dist/src/bitgo/parseInput.d.ts +4 -0
- package/dist/src/bitgo/parseInput.d.ts.map +1 -1
- package/dist/src/bitgo/parseInput.js +23 -7
- package/dist/src/bitgo/psbt/fromHalfSigned.js +5 -5
- package/dist/src/bitgo/signature.d.ts.map +1 -1
- package/dist/src/bitgo/signature.js +17 -8
- package/dist/src/bitgo/transaction.js +10 -10
- package/dist/src/bitgo/transactionAmounts.d.ts +9 -0
- package/dist/src/bitgo/transactionAmounts.d.ts.map +1 -0
- package/dist/src/bitgo/transactionAmounts.js +33 -0
- package/dist/src/bitgo/types.d.ts +7 -0
- package/dist/src/bitgo/types.d.ts.map +1 -1
- package/dist/src/bitgo/types.js +11 -2
- package/dist/src/bitgo/wallet/Psbt.d.ts +31 -1
- package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Psbt.js +129 -29
- package/dist/src/bitgo/wallet/Unspent.d.ts +23 -13
- package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Unspent.js +59 -54
- 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 +68 -27
- package/dist/src/bitgo/wallet/WalletScripts.js +2 -2
- package/dist/src/bitgo/wallet/WalletUnspentSigner.js +4 -4
- package/dist/src/bitgo/wallet/index.d.ts +2 -0
- package/dist/src/bitgo/wallet/index.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/index.js +8 -2
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.d.ts +50 -0
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.d.ts.map +1 -0
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.js +87 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.d.ts +32 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.d.ts.map +1 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.js +125 -0
- 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 +5 -14
- package/dist/src/bitgo/zcash/ZcashTransaction.js +15 -15
- package/dist/src/bitgo/zcash/ZcashTransactionBuilder.js +4 -4
- package/dist/src/bitgo/zcash/address.js +3 -3
- package/dist/src/bitgo/zcash/hashZip0244.js +3 -3
- package/dist/src/bitgo/zcash/index.js +6 -2
- package/dist/src/index.js +6 -2
- package/dist/src/networks.d.ts +1 -1
- package/dist/src/networks.d.ts.map +1 -1
- package/dist/src/networks.js +19 -1
- package/dist/src/noble_ecc.d.ts +7 -7
- package/dist/src/noble_ecc.d.ts.map +1 -1
- package/dist/src/noble_ecc.js +4 -4
- package/dist/src/testutil/index.js +6 -2
- 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 +18 -3
- package/dist/src/testutil/mock.js +15 -15
- package/dist/src/testutil/psbt.d.ts +26 -5
- package/dist/src/testutil/psbt.d.ts.map +1 -1
- package/dist/src/testutil/psbt.js +40 -29
- package/dist/src/testutil/transaction.d.ts +14 -5
- package/dist/src/testutil/transaction.d.ts.map +1 -1
- package/dist/src/testutil/transaction.js +20 -20
- package/dist/src/transaction_builder.d.ts.map +1 -1
- package/dist/src/transaction_builder.js +1 -6
- package/package.json +7 -7
|
@@ -3,9 +3,10 @@ import { PsbtInput } from 'bip174/src/lib/interfaces';
|
|
|
3
3
|
import { BIP32Interface } from 'bip32';
|
|
4
4
|
import { UtxoPsbt } from '../UtxoPsbt';
|
|
5
5
|
import { UtxoTransaction } from '../UtxoTransaction';
|
|
6
|
+
import { ScriptType2Of3 } from '../outputScripts';
|
|
6
7
|
import { RootWalletKeys } from './WalletKeys';
|
|
7
8
|
import { WalletUnspent } from './Unspent';
|
|
8
|
-
import { ParsedPubScriptP2ms, ParsedPubScriptTaprootScriptPath, ParsedPubScriptTaprootKeyPath, ParsedPubScriptP2shP2pk, ParsedScriptType } from '../parseInput';
|
|
9
|
+
import { ParsedPubScriptP2ms, ParsedPubScriptTaprootScriptPath, ParsedPubScriptTaprootKeyPath, ParsedPubScriptP2shP2pk, ParsedScriptType, ParsedScriptType2Of3 } from '../parseInput';
|
|
9
10
|
import { Triple } from '../types';
|
|
10
11
|
import { TxInput } from 'bitcoinjs-lib';
|
|
11
12
|
declare type BaseSignatureContainer<T> = {
|
|
@@ -86,6 +87,12 @@ export declare function getPsbtInputScriptType(input: PsbtInput): ParsedScriptTy
|
|
|
86
87
|
* public key (tapOutputkey), signatures (partial signer sigs).
|
|
87
88
|
*/
|
|
88
89
|
export declare function parsePsbtInput(input: PsbtInput): ParsedPsbtP2ms | ParsedPsbtTaproot | ParsedPsbtP2shP2pk;
|
|
90
|
+
/**
|
|
91
|
+
* Converts a parsed script type into an array of script types.
|
|
92
|
+
* @param parsedScriptType - The parsed script type.
|
|
93
|
+
* @returns An array of ScriptType2Of3 values corresponding to the parsed script type.
|
|
94
|
+
*/
|
|
95
|
+
export declare function toScriptType2Of3s(parsedScriptType: ParsedScriptType2Of3): ScriptType2Of3[];
|
|
89
96
|
/**
|
|
90
97
|
* @returns strictly parse the input and get signature count.
|
|
91
98
|
* unsigned(0), half-signed(1) or fully-signed(2)
|
|
@@ -121,5 +128,28 @@ export declare function addXpubsToPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalle
|
|
|
121
128
|
* For p2shP2pk input, [false, false, false] is returned since it is not a 2 of 3 sig input.
|
|
122
129
|
*/
|
|
123
130
|
export declare function getSignatureValidationArrayPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalletKeys): SignatureValidation[];
|
|
131
|
+
/**
|
|
132
|
+
* Extracts the half signed transaction from the psbt for p2ms based script types - p2sh, p2wsh, and p2shP2wsh.
|
|
133
|
+
* The purpose is to provide backward compatibility to keyternal (KRS) that only supports network transaction and p2ms script types.
|
|
134
|
+
*/
|
|
135
|
+
export declare function extractP2msOnlyHalfSignedTx(psbt: UtxoPsbt): UtxoTransaction<bigint>;
|
|
136
|
+
/**
|
|
137
|
+
* Clones the psbt without nonWitnessUtxo for non-segwit inputs and witnessUtxo is added instead.
|
|
138
|
+
* It is not BIP-174 compliant, so use it carefully.
|
|
139
|
+
*/
|
|
140
|
+
export declare function clonePsbtWithoutNonWitnessUtxo(psbt: UtxoPsbt): UtxoPsbt;
|
|
141
|
+
/**
|
|
142
|
+
* Returns true if there are non-segwit inputs in the PSBT that do not contain the
|
|
143
|
+
* nonWitnessUtxo.
|
|
144
|
+
*
|
|
145
|
+
* isPsbtLite(clonePsbtWithoutNonWitnessUtxo(psbt)) === true
|
|
146
|
+
*
|
|
147
|
+
* @param psbt
|
|
148
|
+
*/
|
|
149
|
+
export declare function isPsbtLite(psbt: UtxoPsbt): boolean;
|
|
150
|
+
/**
|
|
151
|
+
* Deletes witnessUtxo for non-segwit inputs to make the PSBT BIP-174 compliant.
|
|
152
|
+
*/
|
|
153
|
+
export declare function deleteWitnessUtxoForNonSegwitInputs(psbt: UtxoPsbt): void;
|
|
124
154
|
export {};
|
|
125
155
|
//# sourceMappingURL=Psbt.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Psbt.d.ts","sourceRoot":"","sources":["../../../../src/bitgo/wallet/Psbt.ts"],"names":[],"mappings":";AAEA,OAAO,EAA0B,SAAS,EAAgB,MAAM,2BAA2B,CAAC;AAE5F,OAAO,EAAE,cAAc,EAAE,MAAM,OAAO,CAAC;AAEvC,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AACvC,OAAO,EAAE,eAAe,EAAE,MAAM,oBAAoB,CAAC;
|
|
1
|
+
{"version":3,"file":"Psbt.d.ts","sourceRoot":"","sources":["../../../../src/bitgo/wallet/Psbt.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,12 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.getSignatureValidationArrayPsbt = exports.addXpubsToPsbt = exports.isTransactionWithKeyPathSpendInput = exports.isTxInputArray = exports.isPsbtInputArray = exports.getStrictSignatureCounts = exports.getStrictSignatureCount = exports.parsePsbtInput = exports.getPsbtInputScriptType = exports.signWalletPsbt = exports.toWalletPsbt = void 0;
|
|
3
|
+
exports.deleteWitnessUtxoForNonSegwitInputs = exports.isPsbtLite = exports.clonePsbtWithoutNonWitnessUtxo = exports.extractP2msOnlyHalfSignedTx = exports.getSignatureValidationArrayPsbt = exports.addXpubsToPsbt = exports.isTransactionWithKeyPathSpendInput = exports.isTxInputArray = exports.isPsbtInputArray = exports.getStrictSignatureCounts = exports.getStrictSignatureCount = exports.toScriptType2Of3s = exports.parsePsbtInput = exports.getPsbtInputScriptType = exports.signWalletPsbt = exports.toWalletPsbt = void 0;
|
|
4
4
|
const assert = require("assert");
|
|
5
5
|
const utils_1 = require("bip174/src/lib/utils");
|
|
6
6
|
const bs58check = require("bs58check");
|
|
7
7
|
const UtxoPsbt_1 = require("../UtxoPsbt");
|
|
8
8
|
const UtxoTransaction_1 = require("../UtxoTransaction");
|
|
9
9
|
const outputScripts_1 = require("../outputScripts");
|
|
10
|
+
const WalletKeys_1 = require("./WalletKeys");
|
|
10
11
|
const Unspent_1 = require("../Unspent");
|
|
11
12
|
const transaction_1 = require("../transaction");
|
|
12
13
|
const Unspent_2 = require("./Unspent");
|
|
@@ -18,9 +19,9 @@ const bitcoinjs_lib_1 = require("bitcoinjs-lib");
|
|
|
18
19
|
const index_1 = require("../../index");
|
|
19
20
|
const PsbtUtil_1 = require("../PsbtUtil");
|
|
20
21
|
function getTaprootSigners(script, walletKeys) {
|
|
21
|
-
const parsedPublicKeys = parseInput_1.parsePubScript2Of3(script, 'taprootScriptPathSpend').publicKeys;
|
|
22
|
+
const parsedPublicKeys = (0, parseInput_1.parsePubScript2Of3)(script, 'taprootScriptPathSpend').publicKeys;
|
|
22
23
|
const walletSigners = parsedPublicKeys.map((publicKey) => {
|
|
23
|
-
const index = walletKeys.publicKeys.findIndex((walletPublicKey) => outputScripts_1.toXOnlyPublicKey(walletPublicKey).equals(publicKey));
|
|
24
|
+
const index = walletKeys.publicKeys.findIndex((walletPublicKey) => (0, outputScripts_1.toXOnlyPublicKey)(walletPublicKey).equals(publicKey));
|
|
24
25
|
if (index >= 0) {
|
|
25
26
|
return { walletKey: walletKeys.triple[index], rootKey: walletKeys.parent.triple[index] };
|
|
26
27
|
}
|
|
@@ -29,9 +30,9 @@ function getTaprootSigners(script, walletKeys) {
|
|
|
29
30
|
return [walletSigners[0], walletSigners[1]];
|
|
30
31
|
}
|
|
31
32
|
function updatePsbtInput(psbt, inputIndex, unspent, rootWalletKeys) {
|
|
32
|
-
const input = utils_1.checkForInput(psbt.data.inputs, inputIndex);
|
|
33
|
-
const signatureCount = PsbtUtil_1.getPsbtInputSignatureCount(input);
|
|
34
|
-
const scriptType = outputScripts_1.scriptTypeForChain(unspent.chain);
|
|
33
|
+
const input = (0, utils_1.checkForInput)(psbt.data.inputs, inputIndex);
|
|
34
|
+
const signatureCount = (0, PsbtUtil_1.getPsbtInputSignatureCount)(input);
|
|
35
|
+
const scriptType = (0, outputScripts_1.scriptTypeForChain)(unspent.chain);
|
|
35
36
|
if (signatureCount === 0 && scriptType === 'p2tr') {
|
|
36
37
|
return;
|
|
37
38
|
}
|
|
@@ -44,7 +45,7 @@ function updatePsbtInput(psbt, inputIndex, unspent, rootWalletKeys) {
|
|
|
44
45
|
throw new Error('Bitgo only supports a single tap leaf script per input');
|
|
45
46
|
}
|
|
46
47
|
const [signer, cosigner] = getTaprootSigners(input.tapLeafScript[0].script, walletKeys);
|
|
47
|
-
const leafHash = outputScripts_1.getLeafHash({
|
|
48
|
+
const leafHash = (0, outputScripts_1.getLeafHash)({
|
|
48
49
|
publicKeys: walletKeys.publicKeys,
|
|
49
50
|
signer: signer.walletKey.publicKey,
|
|
50
51
|
cosigner: cosigner.walletKey.publicKey,
|
|
@@ -52,7 +53,7 @@ function updatePsbtInput(psbt, inputIndex, unspent, rootWalletKeys) {
|
|
|
52
53
|
psbt.updateInput(inputIndex, {
|
|
53
54
|
tapBip32Derivation: [signer, cosigner].map((walletSigner) => ({
|
|
54
55
|
leafHashes: [leafHash],
|
|
55
|
-
pubkey: outputScripts_1.toXOnlyPublicKey(walletSigner.walletKey.publicKey),
|
|
56
|
+
pubkey: (0, outputScripts_1.toXOnlyPublicKey)(walletSigner.walletKey.publicKey),
|
|
56
57
|
path: rootWalletKeys.getDerivationPath(walletSigner.rootKey, unspent.chain, unspent.index),
|
|
57
58
|
masterFingerprint: walletSigner.rootKey.fingerprint,
|
|
58
59
|
})),
|
|
@@ -60,7 +61,7 @@ function updatePsbtInput(psbt, inputIndex, unspent, rootWalletKeys) {
|
|
|
60
61
|
}
|
|
61
62
|
else {
|
|
62
63
|
if (signatureCount === 0) {
|
|
63
|
-
const { witnessScript, redeemScript } = outputScripts_1.createOutputScript2of3(walletKeys.publicKeys, scriptType);
|
|
64
|
+
const { witnessScript, redeemScript } = (0, outputScripts_1.createOutputScript2of3)(walletKeys.publicKeys, scriptType);
|
|
64
65
|
if (witnessScript && psbt.data.inputs[inputIndex].witnessScript === undefined) {
|
|
65
66
|
psbt.updateInput(inputIndex, { witnessScript });
|
|
66
67
|
}
|
|
@@ -86,12 +87,12 @@ function updatePsbtInput(psbt, inputIndex, unspent, rootWalletKeys) {
|
|
|
86
87
|
*/
|
|
87
88
|
function toWalletPsbt(tx, unspents, rootWalletKeys) {
|
|
88
89
|
const prevOutputs = unspents.map((u) => {
|
|
89
|
-
assert.notStrictEqual(outputScripts_1.scriptTypeForChain(u.chain), 'p2trMusig2');
|
|
90
|
-
return Unspent_1.toPrevOutputWithPrevTx(u, tx.network);
|
|
90
|
+
assert.notStrictEqual((0, outputScripts_1.scriptTypeForChain)(u.chain), 'p2trMusig2');
|
|
91
|
+
return (0, Unspent_1.toPrevOutputWithPrevTx)(u, tx.network);
|
|
91
92
|
});
|
|
92
|
-
const psbt = transaction_1.createPsbtFromTransaction(tx, prevOutputs);
|
|
93
|
+
const psbt = (0, transaction_1.createPsbtFromTransaction)(tx, prevOutputs);
|
|
93
94
|
unspents.forEach((u, i) => {
|
|
94
|
-
if (Unspent_2.isWalletUnspent(u) && u.index !== undefined) {
|
|
95
|
+
if ((0, Unspent_2.isWalletUnspent)(u) && u.index !== undefined) {
|
|
95
96
|
updatePsbtInput(psbt, i, u, rootWalletKeys);
|
|
96
97
|
}
|
|
97
98
|
});
|
|
@@ -106,7 +107,7 @@ exports.toWalletPsbt = toWalletPsbt;
|
|
|
106
107
|
* @return signed PSBT with signer's key for unspent
|
|
107
108
|
*/
|
|
108
109
|
function signWalletPsbt(psbt, inputIndex, signer, unspent) {
|
|
109
|
-
const scriptType = outputScripts_1.scriptTypeForChain(unspent.chain);
|
|
110
|
+
const scriptType = (0, outputScripts_1.scriptTypeForChain)(unspent.chain);
|
|
110
111
|
if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {
|
|
111
112
|
psbt.signTaprootInputHD(inputIndex, signer);
|
|
112
113
|
}
|
|
@@ -163,13 +164,13 @@ function getPsbtInputScriptType(input) {
|
|
|
163
164
|
}
|
|
164
165
|
exports.getPsbtInputScriptType = getPsbtInputScriptType;
|
|
165
166
|
function parseTaprootKeyPathSignatures(input) {
|
|
166
|
-
const partialSigs = Musig2_1.parsePsbtMusig2PartialSigs(input);
|
|
167
|
+
const partialSigs = (0, Musig2_1.parsePsbtMusig2PartialSigs)(input);
|
|
167
168
|
if (!partialSigs) {
|
|
168
169
|
return { signatures: undefined, participantPublicKeys: undefined };
|
|
169
170
|
}
|
|
170
171
|
const signatures = partialSigs.map((pSig) => pSig.partialSig);
|
|
171
172
|
const participantPublicKeys = partialSigs.map((pSig) => pSig.participantPubKey);
|
|
172
|
-
return types_1.isTuple(signatures) && types_1.isTuple(participantPublicKeys)
|
|
173
|
+
return (0, types_1.isTuple)(signatures) && (0, types_1.isTuple)(participantPublicKeys)
|
|
173
174
|
? { signatures, participantPublicKeys }
|
|
174
175
|
: { signatures: [signatures[0]], participantPublicKeys: [participantPublicKeys[0]] };
|
|
175
176
|
}
|
|
@@ -181,7 +182,7 @@ function parsePartialOrTapScriptSignatures(sig) {
|
|
|
181
182
|
throw new Error('unexpected signature count');
|
|
182
183
|
}
|
|
183
184
|
const signatures = sig.map((tSig) => tSig.signature);
|
|
184
|
-
return types_1.isTuple(signatures) ? { signatures } : { signatures: [signatures[0]] };
|
|
185
|
+
return (0, types_1.isTuple)(signatures) ? { signatures } : { signatures: [signatures[0]] };
|
|
185
186
|
}
|
|
186
187
|
function parseSignatures(input, scriptType) {
|
|
187
188
|
return scriptType === 'taprootKeyPathSpend'
|
|
@@ -207,13 +208,13 @@ function parseScript(input, scriptType) {
|
|
|
207
208
|
pubScript = input.witnessUtxo.script;
|
|
208
209
|
}
|
|
209
210
|
else if (input.tapInternalKey && input.tapMerkleRoot) {
|
|
210
|
-
pubScript = taproot_1.createTaprootOutputScript({ internalPubKey: input.tapInternalKey, taptreeRoot: input.tapMerkleRoot });
|
|
211
|
+
pubScript = (0, taproot_1.createTaprootOutputScript)({ internalPubKey: input.tapInternalKey, taptreeRoot: input.tapMerkleRoot });
|
|
211
212
|
}
|
|
212
213
|
}
|
|
213
214
|
if (!pubScript) {
|
|
214
215
|
throw new Error(`Invalid PSBT state for ${scriptType}. Missing required fields.`);
|
|
215
216
|
}
|
|
216
|
-
return parseInput_1.parsePubScript(pubScript, scriptType);
|
|
217
|
+
return (0, parseInput_1.parsePubScript)(pubScript, scriptType);
|
|
217
218
|
}
|
|
218
219
|
/**
|
|
219
220
|
* @return psbt metadata are parsed as per below conditions.
|
|
@@ -229,7 +230,7 @@ function parseScript(input, scriptType) {
|
|
|
229
230
|
* public key (tapOutputkey), signatures (partial signer sigs).
|
|
230
231
|
*/
|
|
231
232
|
function parsePsbtInput(input) {
|
|
232
|
-
if (PsbtUtil_1.isPsbtInputFinalized(input)) {
|
|
233
|
+
if ((0, PsbtUtil_1.isPsbtInputFinalized)(input)) {
|
|
233
234
|
throw new Error('Finalized PSBT parsing is not supported');
|
|
234
235
|
}
|
|
235
236
|
const scriptType = getPsbtInputScriptType(input);
|
|
@@ -246,11 +247,11 @@ function parsePsbtInput(input) {
|
|
|
246
247
|
throw new Error('Invalid PSBT state for taprootScriptPathSpend. Missing required fields.');
|
|
247
248
|
}
|
|
248
249
|
const controlBlock = input.tapLeafScript[0].controlBlock;
|
|
249
|
-
if (!parseInput_1.isValidControlBock(controlBlock)) {
|
|
250
|
+
if (!(0, parseInput_1.isValidControlBock)(controlBlock)) {
|
|
250
251
|
throw new Error('Invalid PSBT taprootScriptPathSpend controlBlock.');
|
|
251
252
|
}
|
|
252
|
-
const scriptPathLevel = parseInput_1.calculateScriptPathLevel(controlBlock);
|
|
253
|
-
const leafVersion = parseInput_1.getLeafVersion(controlBlock);
|
|
253
|
+
const scriptPathLevel = (0, parseInput_1.calculateScriptPathLevel)(controlBlock);
|
|
254
|
+
const leafVersion = (0, parseInput_1.getLeafVersion)(controlBlock);
|
|
254
255
|
return {
|
|
255
256
|
...parsedPubScript,
|
|
256
257
|
...signatures,
|
|
@@ -270,7 +271,7 @@ function parsePsbtInput(input) {
|
|
|
270
271
|
...signatures,
|
|
271
272
|
};
|
|
272
273
|
}
|
|
273
|
-
if (parsedPubScript.scriptType === 'p2shP2pk' && (!signatures.signatures || !types_1.isTuple(signatures.signatures))) {
|
|
274
|
+
if (parsedPubScript.scriptType === 'p2shP2pk' && (!signatures.signatures || !(0, types_1.isTuple)(signatures.signatures))) {
|
|
274
275
|
return {
|
|
275
276
|
...parsedPubScript,
|
|
276
277
|
signatures: signatures.signatures,
|
|
@@ -279,6 +280,19 @@ function parsePsbtInput(input) {
|
|
|
279
280
|
throw new Error('invalid pub script');
|
|
280
281
|
}
|
|
281
282
|
exports.parsePsbtInput = parsePsbtInput;
|
|
283
|
+
/**
|
|
284
|
+
* Converts a parsed script type into an array of script types.
|
|
285
|
+
* @param parsedScriptType - The parsed script type.
|
|
286
|
+
* @returns An array of ScriptType2Of3 values corresponding to the parsed script type.
|
|
287
|
+
*/
|
|
288
|
+
function toScriptType2Of3s(parsedScriptType) {
|
|
289
|
+
return parsedScriptType === 'taprootScriptPathSpend'
|
|
290
|
+
? ['p2trMusig2', 'p2tr']
|
|
291
|
+
: parsedScriptType === 'taprootKeyPathSpend'
|
|
292
|
+
? ['p2trMusig2']
|
|
293
|
+
: [parsedScriptType];
|
|
294
|
+
}
|
|
295
|
+
exports.toScriptType2Of3s = toScriptType2Of3s;
|
|
282
296
|
/**
|
|
283
297
|
* @returns strictly parse the input and get signature count.
|
|
284
298
|
* unsigned(0), half-signed(1) or fully-signed(2)
|
|
@@ -286,7 +300,7 @@ exports.parsePsbtInput = parsePsbtInput;
|
|
|
286
300
|
function getStrictSignatureCount(input) {
|
|
287
301
|
var _a, _b;
|
|
288
302
|
const calculateSignatureCount = (signatures) => {
|
|
289
|
-
const count = signatures ? signatures.filter((s) => !parseInput_1.isPlaceholderSignature(s)).length : 0;
|
|
303
|
+
const count = signatures ? signatures.filter((s) => !(0, parseInput_1.isPlaceholderSignature)(s)).length : 0;
|
|
290
304
|
if (count === 0 || count === 1 || count === 2) {
|
|
291
305
|
return count;
|
|
292
306
|
}
|
|
@@ -294,7 +308,7 @@ function getStrictSignatureCount(input) {
|
|
|
294
308
|
};
|
|
295
309
|
if ('hash' in input) {
|
|
296
310
|
if (((_a = input.script) === null || _a === void 0 ? void 0 : _a.length) || ((_b = input.witness) === null || _b === void 0 ? void 0 : _b.length)) {
|
|
297
|
-
const parsedInput = parseInput_1.parseSignatureScript(input);
|
|
311
|
+
const parsedInput = (0, parseInput_1.parseSignatureScript)(input);
|
|
298
312
|
return parsedInput.scriptType === 'taprootKeyPathSpend' ? 2 : calculateSignatureCount(parsedInput.signatures);
|
|
299
313
|
}
|
|
300
314
|
return 0;
|
|
@@ -345,7 +359,7 @@ function isTransactionWithKeyPathSpendInput(data) {
|
|
|
345
359
|
if (getStrictSignatureCount(input) === 0) {
|
|
346
360
|
return false;
|
|
347
361
|
}
|
|
348
|
-
return parseInput_1.parseSignatureScript(input).scriptType === 'taprootKeyPathSpend';
|
|
362
|
+
return (0, parseInput_1.parseSignatureScript)(input).scriptType === 'taprootKeyPathSpend';
|
|
349
363
|
});
|
|
350
364
|
}
|
|
351
365
|
exports.isTransactionWithKeyPathSpendInput = isTransactionWithKeyPathSpendInput;
|
|
@@ -356,7 +370,8 @@ exports.isTransactionWithKeyPathSpendInput = isTransactionWithKeyPathSpendInput;
|
|
|
356
370
|
* extendedPubkey.
|
|
357
371
|
*/
|
|
358
372
|
function addXpubsToPsbt(psbt, rootWalletKeys) {
|
|
359
|
-
const
|
|
373
|
+
const safeRootWalletKeys = new WalletKeys_1.RootWalletKeys(rootWalletKeys.triple.map((bip32) => bip32.neutered()), rootWalletKeys.derivationPrefixes);
|
|
374
|
+
const xPubs = safeRootWalletKeys.triple.map((bip32) => ({
|
|
360
375
|
extendedPubkey: bs58check.decode(bip32.toBase58()),
|
|
361
376
|
masterFingerprint: bip32.fingerprint,
|
|
362
377
|
// TODO: BG-73797 - bip174 currently requires m prefix for this to be a valid globalXpub
|
|
@@ -379,4 +394,89 @@ function getSignatureValidationArrayPsbt(psbt, rootWalletKeys) {
|
|
|
379
394
|
});
|
|
380
395
|
}
|
|
381
396
|
exports.getSignatureValidationArrayPsbt = getSignatureValidationArrayPsbt;
|
|
382
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Psbt.js","sourceRoot":"","sources":["../../../../src/bitgo/wallet/Psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { GlobalXpub, PartialSig, PsbtInput, TapScriptSig } from 'bip174/src/lib/interfaces';\nimport { checkForInput } from 'bip174/src/lib/utils';\nimport { BIP32Interface } from 'bip32';\nimport * as bs58check from 'bs58check';\nimport { UtxoPsbt } from '../UtxoPsbt';\nimport { UtxoTransaction } from '../UtxoTransaction';\nimport { createOutputScript2of3, getLeafHash, scriptTypeForChain, toXOnlyPublicKey } from '../outputScripts';\nimport { DerivedWalletKeys, RootWalletKeys } from './WalletKeys';\nimport { toPrevOutputWithPrevTx } from '../Unspent';\nimport { createPsbtFromTransaction } from '../transaction';\nimport { isWalletUnspent, WalletUnspent } from './Unspent';\n\nimport {\n  getLeafVersion,\n  calculateScriptPathLevel,\n  isValidControlBock,\n  ParsedPubScriptP2ms,\n  ParsedPubScriptTaprootScriptPath,\n  parsePubScript2Of3,\n  ParsedPubScriptTaproot,\n  ParsedPubScriptTaprootKeyPath,\n  parsePubScript,\n  ParsedPubScriptP2shP2pk,\n  ParsedScriptType,\n  isPlaceholderSignature,\n  parseSignatureScript,\n} from '../parseInput';\nimport { parsePsbtMusig2PartialSigs } from '../Musig2';\nimport { isTuple, Triple } from '../types';\nimport { createTaprootOutputScript } from '../../taproot';\nimport { script as bscript, TxInput } from 'bitcoinjs-lib';\nimport { opcodes } from '../../index';\nimport { getPsbtInputSignatureCount, isPsbtInputFinalized } from '../PsbtUtil';\n\n// only used for building `SignatureContainer`\ntype BaseSignatureContainer<T> = {\n  signatures: T;\n};\n\ntype UnsignedSignatureContainer = BaseSignatureContainer<undefined>;\ntype HalfSignedSignatureContainer = BaseSignatureContainer<[Buffer]>;\ntype FullSignedSignatureContainer = BaseSignatureContainer<[Buffer, Buffer]>;\n\ntype SignatureContainer = UnsignedSignatureContainer | HalfSignedSignatureContainer | FullSignedSignatureContainer;\n\n/**\n * Contents of a pre-finalized PSBT Input for p2trMusig2 key path in the non-finalized state.\n * T is [Buffer] for first signature, [Buffer, Buffer] for both signatures and `undefined` for no signatures.\n */\ntype BaseTaprootKeyPathSignatureContainer<T> = {\n  signatures: T;\n  /** Only contains participants that have added a signature */\n  participantPublicKeys: T;\n};\n\ntype UnsignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<undefined>;\ntype HalfSignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<[Buffer]>;\ntype FullSignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<[Buffer, Buffer]>;\n\ntype TaprootKeyPathSignatureContainer =\n  | UnsignedTaprootKeyPathSignatureContainer\n  | HalfSignedTaprootKeyPathSignatureContainer\n  | FullSignedTaprootKeyPathSignatureContainer;\n\n/**\n * To hold parsed psbt data for p2ms based script types - p2sh, p2wsh, and p2shP2wsh\n */\nexport type ParsedPsbtP2ms = ParsedPubScriptP2ms & SignatureContainer;\n\n/**\n * To hold parsed psbt data for TaprootKeyPathSpend script type.\n */\nexport type ParsedPsbtTaprootKeyPath = ParsedPubScriptTaprootKeyPath & TaprootKeyPathSignatureContainer;\n\n/**\n * To hold parsed psbt data for TaprootScriptPathSpend script path script type.\n */\nexport type ParsedPsbtTaprootScriptPath = ParsedPubScriptTaprootScriptPath &\n  SignatureContainer & {\n    controlBlock: Buffer;\n    leafVersion: number;\n    /** Indicates the level inside the taptree. */\n    scriptPathLevel: number;\n  };\n\nexport type ParsedPsbtTaproot = ParsedPsbtTaprootKeyPath | ParsedPsbtTaprootScriptPath;\n\ntype P2shP2pkSignatureContainer = UnsignedSignatureContainer | HalfSignedSignatureContainer;\n\nexport type ParsedPsbtP2shP2pk = ParsedPubScriptP2shP2pk & P2shP2pkSignatureContainer;\n\ninterface WalletSigner {\n  walletKey: BIP32Interface;\n  rootKey: BIP32Interface;\n}\n\n/**\n * psbt input index and its user, backup, bitgo signatures status\n */\nexport type SignatureValidation = [index: number, sigTriple: Triple<boolean>];\n\nfunction getTaprootSigners(script: Buffer, walletKeys: DerivedWalletKeys): [WalletSigner, WalletSigner] {\n  const parsedPublicKeys = parsePubScript2Of3(script, 'taprootScriptPathSpend').publicKeys;\n  const walletSigners = parsedPublicKeys.map((publicKey) => {\n    const index = walletKeys.publicKeys.findIndex((walletPublicKey) =>\n      toXOnlyPublicKey(walletPublicKey).equals(publicKey)\n    );\n    if (index >= 0) {\n      return { walletKey: walletKeys.triple[index], rootKey: walletKeys.parent.triple[index] };\n    }\n    throw new Error('Taproot public key is not a wallet public key');\n  });\n  return [walletSigners[0], walletSigners[1]];\n}\n\nfunction updatePsbtInput(\n  psbt: UtxoPsbt,\n  inputIndex: number,\n  unspent: WalletUnspent<bigint>,\n  rootWalletKeys: RootWalletKeys\n): void {\n  const input = checkForInput(psbt.data.inputs, inputIndex);\n  const signatureCount = getPsbtInputSignatureCount(input);\n  const scriptType = scriptTypeForChain(unspent.chain);\n  if (signatureCount === 0 && scriptType === 'p2tr') {\n    return;\n  }\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n\n  if (scriptType === 'p2tr') {\n    if (!Array.isArray(input.tapLeafScript) || input.tapLeafScript.length === 0) {\n      throw new Error('Invalid PSBT state. Missing required fields.');\n    }\n\n    if (input.tapLeafScript.length > 1) {\n      throw new Error('Bitgo only supports a single tap leaf script per input');\n    }\n\n    const [signer, cosigner] = getTaprootSigners(input.tapLeafScript[0].script, walletKeys);\n\n    const leafHash = getLeafHash({\n      publicKeys: walletKeys.publicKeys,\n      signer: signer.walletKey.publicKey,\n      cosigner: cosigner.walletKey.publicKey,\n    });\n\n    psbt.updateInput(inputIndex, {\n      tapBip32Derivation: [signer, cosigner].map((walletSigner) => ({\n        leafHashes: [leafHash],\n        pubkey: toXOnlyPublicKey(walletSigner.walletKey.publicKey),\n        path: rootWalletKeys.getDerivationPath(walletSigner.rootKey, unspent.chain, unspent.index),\n        masterFingerprint: walletSigner.rootKey.fingerprint,\n      })),\n    });\n  } else {\n    if (signatureCount === 0) {\n      const { witnessScript, redeemScript } = createOutputScript2of3(walletKeys.publicKeys, scriptType);\n      if (witnessScript && psbt.data.inputs[inputIndex].witnessScript === undefined) {\n        psbt.updateInput(inputIndex, { witnessScript });\n      }\n      if (redeemScript && psbt.data.inputs[inputIndex].redeemScript === undefined) {\n        psbt.updateInput(inputIndex, { redeemScript });\n      }\n    }\n\n    psbt.updateInput(inputIndex, {\n      bip32Derivation: [0, 1, 2].map((idx) => ({\n        pubkey: walletKeys.triple[idx].publicKey,\n        path: walletKeys.paths[idx],\n        masterFingerprint: rootWalletKeys.triple[idx].fingerprint,\n      })),\n    });\n  }\n}\n\n/**\n * @return PSBT filled with metatdata as per input params tx, unspents and rootWalletKeys.\n * Unsigned PSBT for taproot input with witnessUtxo\n * Unsigned PSBT for other input with witnessUtxo/nonWitnessUtxo, redeemScript/witnessScript, bip32Derivation\n * Signed PSBT for taproot input with witnessUtxo, tapLeafScript, tapBip32Derivation, tapScriptSig\n * Signed PSBT for other input with witnessUtxo/nonWitnessUtxo, redeemScript/witnessScript, bip32Derivation, partialSig\n */\nexport function toWalletPsbt(\n  tx: UtxoTransaction<bigint>,\n  unspents: WalletUnspent<bigint>[],\n  rootWalletKeys: RootWalletKeys\n): UtxoPsbt {\n  const prevOutputs = unspents.map((u) => {\n    assert.notStrictEqual(scriptTypeForChain(u.chain), 'p2trMusig2');\n    return toPrevOutputWithPrevTx(u, tx.network);\n  });\n  const psbt = createPsbtFromTransaction(tx, prevOutputs);\n  unspents.forEach((u, i) => {\n    if (isWalletUnspent(u) && u.index !== undefined) {\n      updatePsbtInput(psbt, i, u, rootWalletKeys);\n    }\n  });\n  return psbt;\n}\n\n/**\n * @param psbt\n * @param inputIndex\n * @param signer\n * @param unspent\n * @return signed PSBT with signer's key for unspent\n */\nexport function signWalletPsbt(\n  psbt: UtxoPsbt,\n  inputIndex: number,\n  signer: BIP32Interface,\n  unspent: WalletUnspent<bigint>\n): void {\n  const scriptType = scriptTypeForChain(unspent.chain);\n  if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {\n    psbt.signTaprootInputHD(inputIndex, signer);\n  } else {\n    psbt.signInputHD(inputIndex, signer);\n  }\n}\n\n/**\n * @returns script type of the input\n */\nexport function getPsbtInputScriptType(input: PsbtInput): ParsedScriptType {\n  const isP2pk = (script: Buffer) => {\n    try {\n      const chunks = bscript.decompile(script);\n      return (\n        chunks?.length === 2 &&\n        Buffer.isBuffer(chunks[0]) &&\n        bscript.isCanonicalPubKey(chunks[0]) &&\n        chunks[1] === opcodes.OP_CHECKSIG\n      );\n    } catch (e) {\n      return false;\n    }\n  };\n  let scriptType: ParsedScriptType | undefined;\n  if (Buffer.isBuffer(input.redeemScript) && Buffer.isBuffer(input.witnessScript)) {\n    scriptType = 'p2shP2wsh';\n  } else if (Buffer.isBuffer(input.redeemScript)) {\n    scriptType = isP2pk(input.redeemScript) ? 'p2shP2pk' : 'p2sh';\n  } else if (Buffer.isBuffer(input.witnessScript)) {\n    scriptType = 'p2wsh';\n  }\n  if (Array.isArray(input.tapLeafScript) && input.tapLeafScript.length > 0) {\n    if (scriptType) {\n      throw new Error(`Found both ${scriptType} and taprootScriptPath PSBT metadata.`);\n    }\n    if (input.tapLeafScript.length > 1) {\n      throw new Error('Bitgo only supports a single tap leaf script per input.');\n    }\n    scriptType = 'taprootScriptPathSpend';\n  }\n  if (input.tapInternalKey) {\n    if (scriptType) {\n      throw new Error(`Found both ${scriptType} and taprootKeyPath PSBT metadata.`);\n    }\n    scriptType = 'taprootKeyPathSpend';\n  }\n  if (scriptType) {\n    return scriptType;\n  }\n  throw new Error('could not parse input');\n}\n\nfunction parseTaprootKeyPathSignatures(input: PsbtInput): TaprootKeyPathSignatureContainer {\n  const partialSigs = parsePsbtMusig2PartialSigs(input);\n  if (!partialSigs) {\n    return { signatures: undefined, participantPublicKeys: undefined };\n  }\n  const signatures = partialSigs.map((pSig) => pSig.partialSig);\n  const participantPublicKeys = partialSigs.map((pSig) => pSig.participantPubKey);\n  return isTuple<Buffer>(signatures) && isTuple<Buffer>(participantPublicKeys)\n    ? { signatures, participantPublicKeys }\n    : { signatures: [signatures[0]], participantPublicKeys: [participantPublicKeys[0]] };\n}\n\nfunction parsePartialOrTapScriptSignatures(sig: PartialSig[] | TapScriptSig[] | undefined): SignatureContainer {\n  if (!sig?.length) {\n    return { signatures: undefined };\n  }\n  if (sig.length > 2) {\n    throw new Error('unexpected signature count');\n  }\n  const signatures = sig.map((tSig) => tSig.signature);\n  return isTuple<Buffer>(signatures) ? { signatures } : { signatures: [signatures[0]] };\n}\n\nfunction parseSignatures(\n  input: PsbtInput,\n  scriptType: ParsedScriptType\n): SignatureContainer | TaprootKeyPathSignatureContainer {\n  return scriptType === 'taprootKeyPathSpend'\n    ? parseTaprootKeyPathSignatures(input)\n    : scriptType === 'taprootScriptPathSpend'\n    ? parsePartialOrTapScriptSignatures(input.tapScriptSig)\n    : parsePartialOrTapScriptSignatures(input.partialSig);\n}\n\nfunction parseScript(\n  input: PsbtInput,\n  scriptType: ParsedScriptType\n): ParsedPubScriptP2ms | ParsedPubScriptTaproot | ParsedPubScriptP2shP2pk {\n  let pubScript: Buffer | undefined;\n  if (scriptType === 'p2sh' || scriptType === 'p2shP2pk') {\n    pubScript = input.redeemScript;\n  } else if (scriptType === 'p2wsh' || scriptType === 'p2shP2wsh') {\n    pubScript = input.witnessScript;\n  } else if (scriptType === 'taprootScriptPathSpend') {\n    pubScript = input.tapLeafScript ? input.tapLeafScript[0].script : undefined;\n  } else if (scriptType === 'taprootKeyPathSpend') {\n    if (input.witnessUtxo?.script) {\n      pubScript = input.witnessUtxo.script;\n    } else if (input.tapInternalKey && input.tapMerkleRoot) {\n      pubScript = createTaprootOutputScript({ internalPubKey: input.tapInternalKey, taptreeRoot: input.tapMerkleRoot });\n    }\n  }\n  if (!pubScript) {\n    throw new Error(`Invalid PSBT state for ${scriptType}. Missing required fields.`);\n  }\n  return parsePubScript(pubScript, scriptType);\n}\n\n/**\n * @return psbt metadata are parsed as per below conditions.\n * redeemScript/witnessScript/tapLeafScript matches BitGo.\n * signature and public key count matches BitGo.\n * P2SH-P2PK => scriptType, redeemScript, public key, signature.\n * P2SH => scriptType, redeemScript, public keys, signatures.\n * PW2SH => scriptType, witnessScript, public keys, signatures.\n * P2SH-PW2SH => scriptType, redeemScript, witnessScript, public keys, signatures.\n * P2TR and P2TR MUSIG2 script path => scriptType (taprootScriptPathSpend), pubScript (leaf script), controlBlock,\n * scriptPathLevel, leafVersion, public keys, signatures.\n * P2TR MUSIG2 kep path => scriptType (taprootKeyPathSpend), pubScript (scriptPubKey), participant pub keys (signer),\n * public key (tapOutputkey), signatures (partial signer sigs).\n */\nexport function parsePsbtInput(input: PsbtInput): ParsedPsbtP2ms | ParsedPsbtTaproot | ParsedPsbtP2shP2pk {\n  if (isPsbtInputFinalized(input)) {\n    throw new Error('Finalized PSBT parsing is not supported');\n  }\n  const scriptType = getPsbtInputScriptType(input);\n  const parsedPubScript = parseScript(input, scriptType);\n  const signatures = parseSignatures(input, scriptType);\n\n  if (parsedPubScript.scriptType === 'taprootKeyPathSpend' && 'participantPublicKeys' in signatures) {\n    return {\n      ...parsedPubScript,\n      ...signatures,\n    };\n  }\n  if (parsedPubScript.scriptType === 'taprootScriptPathSpend') {\n    if (!input.tapLeafScript) {\n      throw new Error('Invalid PSBT state for taprootScriptPathSpend. Missing required fields.');\n    }\n    const controlBlock = input.tapLeafScript[0].controlBlock;\n    if (!isValidControlBock(controlBlock)) {\n      throw new Error('Invalid PSBT taprootScriptPathSpend controlBlock.');\n    }\n    const scriptPathLevel = calculateScriptPathLevel(controlBlock);\n    const leafVersion = getLeafVersion(controlBlock);\n    return {\n      ...parsedPubScript,\n      ...signatures,\n      controlBlock,\n      scriptPathLevel,\n      leafVersion,\n    };\n  }\n  if (\n    parsedPubScript.scriptType === 'p2sh' ||\n    parsedPubScript.scriptType === 'p2wsh' ||\n    parsedPubScript.scriptType === 'p2shP2wsh'\n  ) {\n    if (parsedPubScript.scriptType === 'p2shP2wsh') {\n      parsedPubScript.redeemScript = input.redeemScript;\n    }\n    return {\n      ...parsedPubScript,\n      ...signatures,\n    };\n  }\n  if (parsedPubScript.scriptType === 'p2shP2pk' && (!signatures.signatures || !isTuple(signatures.signatures))) {\n    return {\n      ...parsedPubScript,\n      signatures: signatures.signatures,\n    };\n  }\n  throw new Error('invalid pub script');\n}\n\n/**\n * @returns strictly parse the input and get signature count.\n * unsigned(0), half-signed(1) or fully-signed(2)\n */\nexport function getStrictSignatureCount(input: TxInput | PsbtInput): 0 | 1 | 2 {\n  const calculateSignatureCount = (\n    signatures: [Buffer | 0, Buffer | 0, Buffer | 0] | [Buffer, Buffer] | [Buffer] | undefined\n  ): 0 | 1 | 2 => {\n    const count = signatures ? signatures.filter((s) => !isPlaceholderSignature(s)).length : 0;\n    if (count === 0 || count === 1 || count === 2) {\n      return count;\n    }\n    throw new Error('invalid signature count');\n  };\n\n  if ('hash' in input) {\n    if (input.script?.length || input.witness?.length) {\n      const parsedInput = parseSignatureScript(input);\n      return parsedInput.scriptType === 'taprootKeyPathSpend' ? 2 : calculateSignatureCount(parsedInput.signatures);\n    }\n    return 0;\n  } else {\n    return calculateSignatureCount(parsePsbtInput(input).signatures);\n  }\n}\n\n/**\n * @returns strictly parse input and get signature count for all inputs.\n * 0=unsigned, 1=half-signed or 2=fully-signed\n */\nexport function getStrictSignatureCounts(\n  tx: UtxoPsbt | UtxoTransaction<number | bigint> | PsbtInput[] | TxInput[]\n): (0 | 1 | 2)[] {\n  const inputs = tx instanceof UtxoPsbt ? tx.data.inputs : tx instanceof UtxoTransaction ? tx.ins : tx;\n  return inputs.map((input, _) => getStrictSignatureCount(input));\n}\n\n/**\n * @return true iff inputs array is of PsbtInputType type\n * */\nexport function isPsbtInputArray(inputs: PsbtInput[] | TxInput[]): inputs is PsbtInput[] {\n  return !isTxInputArray(inputs);\n}\n\n/**\n * @return true iff inputs array is of TxInput type\n * */\nexport function isTxInputArray(inputs: PsbtInput[] | TxInput[]): inputs is TxInput[] {\n  assert(!!inputs.length, 'empty inputs array');\n  return 'hash' in inputs[0];\n}\n\n/**\n * @returns true iff given psbt/transaction/tx-input-array/psbt-input-array contains at least one taproot key path spend input\n */\nexport function isTransactionWithKeyPathSpendInput(\n  data: UtxoPsbt | UtxoTransaction<bigint | number> | PsbtInput[] | TxInput[]\n): boolean {\n  const inputs = data instanceof UtxoPsbt ? data.data.inputs : data instanceof UtxoTransaction ? data.ins : data;\n  if (!inputs.length) {\n    return false;\n  }\n  if (isPsbtInputArray(inputs)) {\n    return inputs.some((input, _) => getPsbtInputScriptType(input) === 'taprootKeyPathSpend');\n  }\n  return inputs.some((input, _) => {\n    // If the input is not signed, it cannot be a taprootKeyPathSpend input because you can only\n    // extract a fully signed psbt into a transaction with taprootKeyPathSpend inputs.\n    if (getStrictSignatureCount(input) === 0) {\n      return false;\n    }\n    return parseSignatureScript(input).scriptType === 'taprootKeyPathSpend';\n  });\n}\n\n/**\n * Set the RootWalletKeys as the globalXpubs on the psbt\n *\n * We do all the matching of the (tap)bip32Derivations masterFingerprint to the fingerprint of the\n * extendedPubkey.\n */\nexport function addXpubsToPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalletKeys): void {\n  const xPubs = rootWalletKeys.triple.map(\n    (bip32): GlobalXpub => ({\n      extendedPubkey: bs58check.decode(bip32.toBase58()),\n      masterFingerprint: bip32.fingerprint,\n      // TODO: BG-73797 - bip174 currently requires m prefix for this to be a valid globalXpub\n      path: 'm',\n    })\n  );\n  psbt.updateGlobal({ globalXpub: xPubs });\n}\n\n/**\n * validates signatures for each 2 of 3 input against user, backup, bitgo keys derived from rootWalletKeys.\n * @returns array of input index and its [is valid user sig exist, is valid backup sig exist, is valid user bitgo exist]\n * For p2shP2pk input, [false, false, false] is returned since it is not a 2 of 3 sig input.\n */\nexport function getSignatureValidationArrayPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalletKeys): SignatureValidation[] {\n  return psbt.data.inputs.map((input, i) => {\n    const sigValArrayForInput: Triple<boolean> =\n      getPsbtInputScriptType(input) === 'p2shP2pk'\n        ? [false, false, false]\n        : psbt.getSignatureValidationArray(i, { rootNodes: rootWalletKeys.triple });\n    return [i, sigValArrayForInput];\n  });\n}\n"]}
|
|
397
|
+
/**
|
|
398
|
+
* Extracts the half signed transaction from the psbt for p2ms based script types - p2sh, p2wsh, and p2shP2wsh.
|
|
399
|
+
* The purpose is to provide backward compatibility to keyternal (KRS) that only supports network transaction and p2ms script types.
|
|
400
|
+
*/
|
|
401
|
+
function extractP2msOnlyHalfSignedTx(psbt) {
|
|
402
|
+
assert(!!(psbt.data.inputs.length && psbt.data.outputs.length), 'empty inputs or outputs');
|
|
403
|
+
const tx = psbt.getUnsignedTx();
|
|
404
|
+
function isP2msParsedPsbtInput(parsed) {
|
|
405
|
+
return ['p2sh', 'p2shP2wsh', 'p2wsh'].includes(parsed.scriptType);
|
|
406
|
+
}
|
|
407
|
+
psbt.data.inputs.forEach((input, i) => {
|
|
408
|
+
var _a, _b;
|
|
409
|
+
const parsed = parsePsbtInput(input);
|
|
410
|
+
assert(isP2msParsedPsbtInput(parsed), `unsupported script type ${parsed.scriptType}`);
|
|
411
|
+
assert(((_a = input.partialSig) === null || _a === void 0 ? void 0 : _a.length) === 1, `unexpected signature count ${(_b = input.partialSig) === null || _b === void 0 ? void 0 : _b.length}`);
|
|
412
|
+
const [partialSig] = input.partialSig;
|
|
413
|
+
assert(input.sighashType !== undefined && input.sighashType === bitcoinjs_lib_1.script.signature.decode(partialSig.signature).hashType, 'signature sighash does not match input sighash type');
|
|
414
|
+
// type casting is to address the invalid type checking in payments.p2ms
|
|
415
|
+
const signatures = parsed.publicKeys.map((pk) => partialSig.pubkey.equals(pk) ? partialSig.signature : bitcoinjs_lib_1.opcodes.OP_0);
|
|
416
|
+
const isP2SH = !!parsed.redeemScript;
|
|
417
|
+
const isP2WSH = !!parsed.witnessScript;
|
|
418
|
+
const payment = index_1.payments.p2ms({ output: parsed.pubScript, signatures }, { validate: false, allowIncomplete: true });
|
|
419
|
+
const p2wsh = isP2WSH ? index_1.payments.p2wsh({ redeem: payment }) : undefined;
|
|
420
|
+
const p2sh = isP2SH ? index_1.payments.p2sh({ redeem: p2wsh || payment }) : undefined;
|
|
421
|
+
if (p2sh === null || p2sh === void 0 ? void 0 : p2sh.input) {
|
|
422
|
+
tx.setInputScript(i, p2sh.input);
|
|
423
|
+
}
|
|
424
|
+
if (p2wsh === null || p2wsh === void 0 ? void 0 : p2wsh.witness) {
|
|
425
|
+
tx.setWitness(i, p2wsh.witness);
|
|
426
|
+
}
|
|
427
|
+
});
|
|
428
|
+
return tx;
|
|
429
|
+
}
|
|
430
|
+
exports.extractP2msOnlyHalfSignedTx = extractP2msOnlyHalfSignedTx;
|
|
431
|
+
/**
|
|
432
|
+
* Clones the psbt without nonWitnessUtxo for non-segwit inputs and witnessUtxo is added instead.
|
|
433
|
+
* It is not BIP-174 compliant, so use it carefully.
|
|
434
|
+
*/
|
|
435
|
+
function clonePsbtWithoutNonWitnessUtxo(psbt) {
|
|
436
|
+
const newPsbt = (0, transaction_1.createPsbtFromHex)(psbt.toHex(), psbt.network);
|
|
437
|
+
const txInputs = psbt.txInputs;
|
|
438
|
+
psbt.data.inputs.forEach((input, i) => {
|
|
439
|
+
if (input.nonWitnessUtxo && !input.witnessUtxo) {
|
|
440
|
+
const tx = (0, transaction_1.createTransactionFromBuffer)(input.nonWitnessUtxo, psbt.network, { amountType: 'bigint' });
|
|
441
|
+
if (!txInputs[i].hash.equals(tx.getHash())) {
|
|
442
|
+
throw new Error(`Non-witness UTXO hash for input #${i} doesn't match the hash specified in the prevout`);
|
|
443
|
+
}
|
|
444
|
+
newPsbt.data.inputs[i].witnessUtxo = tx.outs[txInputs[i].index];
|
|
445
|
+
}
|
|
446
|
+
delete newPsbt.data.inputs[i].nonWitnessUtxo;
|
|
447
|
+
});
|
|
448
|
+
return newPsbt;
|
|
449
|
+
}
|
|
450
|
+
exports.clonePsbtWithoutNonWitnessUtxo = clonePsbtWithoutNonWitnessUtxo;
|
|
451
|
+
/**
|
|
452
|
+
* Returns true if there are non-segwit inputs in the PSBT that do not contain the
|
|
453
|
+
* nonWitnessUtxo.
|
|
454
|
+
*
|
|
455
|
+
* isPsbtLite(clonePsbtWithoutNonWitnessUtxo(psbt)) === true
|
|
456
|
+
*
|
|
457
|
+
* @param psbt
|
|
458
|
+
*/
|
|
459
|
+
function isPsbtLite(psbt) {
|
|
460
|
+
let isFull = true;
|
|
461
|
+
const nonSegwitInputTypes = ['p2shP2pk', 'p2sh'];
|
|
462
|
+
psbt.data.inputs.forEach((input) => {
|
|
463
|
+
if (isFull && nonSegwitInputTypes.includes(getPsbtInputScriptType(input))) {
|
|
464
|
+
isFull = !!input.nonWitnessUtxo;
|
|
465
|
+
}
|
|
466
|
+
});
|
|
467
|
+
return !isFull;
|
|
468
|
+
}
|
|
469
|
+
exports.isPsbtLite = isPsbtLite;
|
|
470
|
+
/**
|
|
471
|
+
* Deletes witnessUtxo for non-segwit inputs to make the PSBT BIP-174 compliant.
|
|
472
|
+
*/
|
|
473
|
+
function deleteWitnessUtxoForNonSegwitInputs(psbt) {
|
|
474
|
+
psbt.data.inputs.forEach((input, i) => {
|
|
475
|
+
const scriptType = getPsbtInputScriptType(input);
|
|
476
|
+
if (scriptType === 'p2sh' || scriptType === 'p2shP2pk') {
|
|
477
|
+
delete input.witnessUtxo;
|
|
478
|
+
}
|
|
479
|
+
});
|
|
480
|
+
}
|
|
481
|
+
exports.deleteWitnessUtxoForNonSegwitInputs = deleteWitnessUtxoForNonSegwitInputs;
|
|
482
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Psbt.js","sourceRoot":"","sources":["../../../../src/bitgo/wallet/Psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { GlobalXpub, PartialSig, PsbtInput, TapScriptSig } from 'bip174/src/lib/interfaces';\nimport { checkForInput } from 'bip174/src/lib/utils';\nimport { BIP32Interface } from 'bip32';\nimport * as bs58check from 'bs58check';\nimport { UtxoPsbt } from '../UtxoPsbt';\nimport { UtxoTransaction } from '../UtxoTransaction';\nimport {\n  createOutputScript2of3,\n  getLeafHash,\n  ScriptType2Of3,\n  scriptTypeForChain,\n  toXOnlyPublicKey,\n} from '../outputScripts';\nimport { DerivedWalletKeys, RootWalletKeys } from './WalletKeys';\nimport { toPrevOutputWithPrevTx } from '../Unspent';\nimport { createPsbtFromHex, createPsbtFromTransaction, createTransactionFromBuffer } from '../transaction';\nimport { isWalletUnspent, WalletUnspent } from './Unspent';\n\nimport {\n  getLeafVersion,\n  calculateScriptPathLevel,\n  isValidControlBock,\n  ParsedPubScriptP2ms,\n  ParsedPubScriptTaprootScriptPath,\n  parsePubScript2Of3,\n  ParsedPubScriptTaproot,\n  ParsedPubScriptTaprootKeyPath,\n  parsePubScript,\n  ParsedPubScriptP2shP2pk,\n  ParsedScriptType,\n  isPlaceholderSignature,\n  parseSignatureScript,\n  ParsedScriptType2Of3,\n} from '../parseInput';\nimport { parsePsbtMusig2PartialSigs } from '../Musig2';\nimport { isTuple, Triple } from '../types';\nimport { createTaprootOutputScript } from '../../taproot';\nimport { opcodes as ops, script as bscript, TxInput } from 'bitcoinjs-lib';\nimport { opcodes, payments } from '../../index';\nimport { getPsbtInputSignatureCount, isPsbtInputFinalized } from '../PsbtUtil';\n\n// only used for building `SignatureContainer`\ntype BaseSignatureContainer<T> = {\n  signatures: T;\n};\n\ntype UnsignedSignatureContainer = BaseSignatureContainer<undefined>;\ntype HalfSignedSignatureContainer = BaseSignatureContainer<[Buffer]>;\ntype FullSignedSignatureContainer = BaseSignatureContainer<[Buffer, Buffer]>;\n\ntype SignatureContainer = UnsignedSignatureContainer | HalfSignedSignatureContainer | FullSignedSignatureContainer;\n\n/**\n * Contents of a pre-finalized PSBT Input for p2trMusig2 key path in the non-finalized state.\n * T is [Buffer] for first signature, [Buffer, Buffer] for both signatures and `undefined` for no signatures.\n */\ntype BaseTaprootKeyPathSignatureContainer<T> = {\n  signatures: T;\n  /** Only contains participants that have added a signature */\n  participantPublicKeys: T;\n};\n\ntype UnsignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<undefined>;\ntype HalfSignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<[Buffer]>;\ntype FullSignedTaprootKeyPathSignatureContainer = BaseTaprootKeyPathSignatureContainer<[Buffer, Buffer]>;\n\ntype TaprootKeyPathSignatureContainer =\n  | UnsignedTaprootKeyPathSignatureContainer\n  | HalfSignedTaprootKeyPathSignatureContainer\n  | FullSignedTaprootKeyPathSignatureContainer;\n\n/**\n * To hold parsed psbt data for p2ms based script types - p2sh, p2wsh, and p2shP2wsh\n */\nexport type ParsedPsbtP2ms = ParsedPubScriptP2ms & SignatureContainer;\n\n/**\n * To hold parsed psbt data for TaprootKeyPathSpend script type.\n */\nexport type ParsedPsbtTaprootKeyPath = ParsedPubScriptTaprootKeyPath & TaprootKeyPathSignatureContainer;\n\n/**\n * To hold parsed psbt data for TaprootScriptPathSpend script path script type.\n */\nexport type ParsedPsbtTaprootScriptPath = ParsedPubScriptTaprootScriptPath &\n  SignatureContainer & {\n    controlBlock: Buffer;\n    leafVersion: number;\n    /** Indicates the level inside the taptree. */\n    scriptPathLevel: number;\n  };\n\nexport type ParsedPsbtTaproot = ParsedPsbtTaprootKeyPath | ParsedPsbtTaprootScriptPath;\n\ntype P2shP2pkSignatureContainer = UnsignedSignatureContainer | HalfSignedSignatureContainer;\n\nexport type ParsedPsbtP2shP2pk = ParsedPubScriptP2shP2pk & P2shP2pkSignatureContainer;\n\ninterface WalletSigner {\n  walletKey: BIP32Interface;\n  rootKey: BIP32Interface;\n}\n\n/**\n * psbt input index and its user, backup, bitgo signatures status\n */\nexport type SignatureValidation = [index: number, sigTriple: Triple<boolean>];\n\nfunction getTaprootSigners(script: Buffer, walletKeys: DerivedWalletKeys): [WalletSigner, WalletSigner] {\n  const parsedPublicKeys = parsePubScript2Of3(script, 'taprootScriptPathSpend').publicKeys;\n  const walletSigners = parsedPublicKeys.map((publicKey) => {\n    const index = walletKeys.publicKeys.findIndex((walletPublicKey) =>\n      toXOnlyPublicKey(walletPublicKey).equals(publicKey)\n    );\n    if (index >= 0) {\n      return { walletKey: walletKeys.triple[index], rootKey: walletKeys.parent.triple[index] };\n    }\n    throw new Error('Taproot public key is not a wallet public key');\n  });\n  return [walletSigners[0], walletSigners[1]];\n}\n\nfunction updatePsbtInput(\n  psbt: UtxoPsbt,\n  inputIndex: number,\n  unspent: WalletUnspent<bigint>,\n  rootWalletKeys: RootWalletKeys\n): void {\n  const input = checkForInput(psbt.data.inputs, inputIndex);\n  const signatureCount = getPsbtInputSignatureCount(input);\n  const scriptType = scriptTypeForChain(unspent.chain);\n  if (signatureCount === 0 && scriptType === 'p2tr') {\n    return;\n  }\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n\n  if (scriptType === 'p2tr') {\n    if (!Array.isArray(input.tapLeafScript) || input.tapLeafScript.length === 0) {\n      throw new Error('Invalid PSBT state. Missing required fields.');\n    }\n\n    if (input.tapLeafScript.length > 1) {\n      throw new Error('Bitgo only supports a single tap leaf script per input');\n    }\n\n    const [signer, cosigner] = getTaprootSigners(input.tapLeafScript[0].script, walletKeys);\n\n    const leafHash = getLeafHash({\n      publicKeys: walletKeys.publicKeys,\n      signer: signer.walletKey.publicKey,\n      cosigner: cosigner.walletKey.publicKey,\n    });\n\n    psbt.updateInput(inputIndex, {\n      tapBip32Derivation: [signer, cosigner].map((walletSigner) => ({\n        leafHashes: [leafHash],\n        pubkey: toXOnlyPublicKey(walletSigner.walletKey.publicKey),\n        path: rootWalletKeys.getDerivationPath(walletSigner.rootKey, unspent.chain, unspent.index),\n        masterFingerprint: walletSigner.rootKey.fingerprint,\n      })),\n    });\n  } else {\n    if (signatureCount === 0) {\n      const { witnessScript, redeemScript } = createOutputScript2of3(walletKeys.publicKeys, scriptType);\n      if (witnessScript && psbt.data.inputs[inputIndex].witnessScript === undefined) {\n        psbt.updateInput(inputIndex, { witnessScript });\n      }\n      if (redeemScript && psbt.data.inputs[inputIndex].redeemScript === undefined) {\n        psbt.updateInput(inputIndex, { redeemScript });\n      }\n    }\n\n    psbt.updateInput(inputIndex, {\n      bip32Derivation: [0, 1, 2].map((idx) => ({\n        pubkey: walletKeys.triple[idx].publicKey,\n        path: walletKeys.paths[idx],\n        masterFingerprint: rootWalletKeys.triple[idx].fingerprint,\n      })),\n    });\n  }\n}\n\n/**\n * @return PSBT filled with metatdata as per input params tx, unspents and rootWalletKeys.\n * Unsigned PSBT for taproot input with witnessUtxo\n * Unsigned PSBT for other input with witnessUtxo/nonWitnessUtxo, redeemScript/witnessScript, bip32Derivation\n * Signed PSBT for taproot input with witnessUtxo, tapLeafScript, tapBip32Derivation, tapScriptSig\n * Signed PSBT for other input with witnessUtxo/nonWitnessUtxo, redeemScript/witnessScript, bip32Derivation, partialSig\n */\nexport function toWalletPsbt(\n  tx: UtxoTransaction<bigint>,\n  unspents: WalletUnspent<bigint>[],\n  rootWalletKeys: RootWalletKeys\n): UtxoPsbt {\n  const prevOutputs = unspents.map((u) => {\n    assert.notStrictEqual(scriptTypeForChain(u.chain), 'p2trMusig2');\n    return toPrevOutputWithPrevTx(u, tx.network);\n  });\n  const psbt = createPsbtFromTransaction(tx, prevOutputs);\n  unspents.forEach((u, i) => {\n    if (isWalletUnspent(u) && u.index !== undefined) {\n      updatePsbtInput(psbt, i, u, rootWalletKeys);\n    }\n  });\n  return psbt;\n}\n\n/**\n * @param psbt\n * @param inputIndex\n * @param signer\n * @param unspent\n * @return signed PSBT with signer's key for unspent\n */\nexport function signWalletPsbt(\n  psbt: UtxoPsbt,\n  inputIndex: number,\n  signer: BIP32Interface,\n  unspent: WalletUnspent<bigint>\n): void {\n  const scriptType = scriptTypeForChain(unspent.chain);\n  if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {\n    psbt.signTaprootInputHD(inputIndex, signer);\n  } else {\n    psbt.signInputHD(inputIndex, signer);\n  }\n}\n\n/**\n * @returns script type of the input\n */\nexport function getPsbtInputScriptType(input: PsbtInput): ParsedScriptType {\n  const isP2pk = (script: Buffer) => {\n    try {\n      const chunks = bscript.decompile(script);\n      return (\n        chunks?.length === 2 &&\n        Buffer.isBuffer(chunks[0]) &&\n        bscript.isCanonicalPubKey(chunks[0]) &&\n        chunks[1] === opcodes.OP_CHECKSIG\n      );\n    } catch (e) {\n      return false;\n    }\n  };\n  let scriptType: ParsedScriptType | undefined;\n  if (Buffer.isBuffer(input.redeemScript) && Buffer.isBuffer(input.witnessScript)) {\n    scriptType = 'p2shP2wsh';\n  } else if (Buffer.isBuffer(input.redeemScript)) {\n    scriptType = isP2pk(input.redeemScript) ? 'p2shP2pk' : 'p2sh';\n  } else if (Buffer.isBuffer(input.witnessScript)) {\n    scriptType = 'p2wsh';\n  }\n  if (Array.isArray(input.tapLeafScript) && input.tapLeafScript.length > 0) {\n    if (scriptType) {\n      throw new Error(`Found both ${scriptType} and taprootScriptPath PSBT metadata.`);\n    }\n    if (input.tapLeafScript.length > 1) {\n      throw new Error('Bitgo only supports a single tap leaf script per input.');\n    }\n    scriptType = 'taprootScriptPathSpend';\n  }\n  if (input.tapInternalKey) {\n    if (scriptType) {\n      throw new Error(`Found both ${scriptType} and taprootKeyPath PSBT metadata.`);\n    }\n    scriptType = 'taprootKeyPathSpend';\n  }\n  if (scriptType) {\n    return scriptType;\n  }\n  throw new Error('could not parse input');\n}\n\nfunction parseTaprootKeyPathSignatures(input: PsbtInput): TaprootKeyPathSignatureContainer {\n  const partialSigs = parsePsbtMusig2PartialSigs(input);\n  if (!partialSigs) {\n    return { signatures: undefined, participantPublicKeys: undefined };\n  }\n  const signatures = partialSigs.map((pSig) => pSig.partialSig);\n  const participantPublicKeys = partialSigs.map((pSig) => pSig.participantPubKey);\n  return isTuple<Buffer>(signatures) && isTuple<Buffer>(participantPublicKeys)\n    ? { signatures, participantPublicKeys }\n    : { signatures: [signatures[0]], participantPublicKeys: [participantPublicKeys[0]] };\n}\n\nfunction parsePartialOrTapScriptSignatures(sig: PartialSig[] | TapScriptSig[] | undefined): SignatureContainer {\n  if (!sig?.length) {\n    return { signatures: undefined };\n  }\n  if (sig.length > 2) {\n    throw new Error('unexpected signature count');\n  }\n  const signatures = sig.map((tSig) => tSig.signature);\n  return isTuple<Buffer>(signatures) ? { signatures } : { signatures: [signatures[0]] };\n}\n\nfunction parseSignatures(\n  input: PsbtInput,\n  scriptType: ParsedScriptType\n): SignatureContainer | TaprootKeyPathSignatureContainer {\n  return scriptType === 'taprootKeyPathSpend'\n    ? parseTaprootKeyPathSignatures(input)\n    : scriptType === 'taprootScriptPathSpend'\n    ? parsePartialOrTapScriptSignatures(input.tapScriptSig)\n    : parsePartialOrTapScriptSignatures(input.partialSig);\n}\n\nfunction parseScript(\n  input: PsbtInput,\n  scriptType: ParsedScriptType\n): ParsedPubScriptP2ms | ParsedPubScriptTaproot | ParsedPubScriptP2shP2pk {\n  let pubScript: Buffer | undefined;\n  if (scriptType === 'p2sh' || scriptType === 'p2shP2pk') {\n    pubScript = input.redeemScript;\n  } else if (scriptType === 'p2wsh' || scriptType === 'p2shP2wsh') {\n    pubScript = input.witnessScript;\n  } else if (scriptType === 'taprootScriptPathSpend') {\n    pubScript = input.tapLeafScript ? input.tapLeafScript[0].script : undefined;\n  } else if (scriptType === 'taprootKeyPathSpend') {\n    if (input.witnessUtxo?.script) {\n      pubScript = input.witnessUtxo.script;\n    } else if (input.tapInternalKey && input.tapMerkleRoot) {\n      pubScript = createTaprootOutputScript({ internalPubKey: input.tapInternalKey, taptreeRoot: input.tapMerkleRoot });\n    }\n  }\n  if (!pubScript) {\n    throw new Error(`Invalid PSBT state for ${scriptType}. Missing required fields.`);\n  }\n  return parsePubScript(pubScript, scriptType);\n}\n\n/**\n * @return psbt metadata are parsed as per below conditions.\n * redeemScript/witnessScript/tapLeafScript matches BitGo.\n * signature and public key count matches BitGo.\n * P2SH-P2PK => scriptType, redeemScript, public key, signature.\n * P2SH => scriptType, redeemScript, public keys, signatures.\n * PW2SH => scriptType, witnessScript, public keys, signatures.\n * P2SH-PW2SH => scriptType, redeemScript, witnessScript, public keys, signatures.\n * P2TR and P2TR MUSIG2 script path => scriptType (taprootScriptPathSpend), pubScript (leaf script), controlBlock,\n * scriptPathLevel, leafVersion, public keys, signatures.\n * P2TR MUSIG2 kep path => scriptType (taprootKeyPathSpend), pubScript (scriptPubKey), participant pub keys (signer),\n * public key (tapOutputkey), signatures (partial signer sigs).\n */\nexport function parsePsbtInput(input: PsbtInput): ParsedPsbtP2ms | ParsedPsbtTaproot | ParsedPsbtP2shP2pk {\n  if (isPsbtInputFinalized(input)) {\n    throw new Error('Finalized PSBT parsing is not supported');\n  }\n  const scriptType = getPsbtInputScriptType(input);\n  const parsedPubScript = parseScript(input, scriptType);\n  const signatures = parseSignatures(input, scriptType);\n\n  if (parsedPubScript.scriptType === 'taprootKeyPathSpend' && 'participantPublicKeys' in signatures) {\n    return {\n      ...parsedPubScript,\n      ...signatures,\n    };\n  }\n  if (parsedPubScript.scriptType === 'taprootScriptPathSpend') {\n    if (!input.tapLeafScript) {\n      throw new Error('Invalid PSBT state for taprootScriptPathSpend. Missing required fields.');\n    }\n    const controlBlock = input.tapLeafScript[0].controlBlock;\n    if (!isValidControlBock(controlBlock)) {\n      throw new Error('Invalid PSBT taprootScriptPathSpend controlBlock.');\n    }\n    const scriptPathLevel = calculateScriptPathLevel(controlBlock);\n    const leafVersion = getLeafVersion(controlBlock);\n    return {\n      ...parsedPubScript,\n      ...signatures,\n      controlBlock,\n      scriptPathLevel,\n      leafVersion,\n    };\n  }\n  if (\n    parsedPubScript.scriptType === 'p2sh' ||\n    parsedPubScript.scriptType === 'p2wsh' ||\n    parsedPubScript.scriptType === 'p2shP2wsh'\n  ) {\n    if (parsedPubScript.scriptType === 'p2shP2wsh') {\n      parsedPubScript.redeemScript = input.redeemScript;\n    }\n    return {\n      ...parsedPubScript,\n      ...signatures,\n    };\n  }\n  if (parsedPubScript.scriptType === 'p2shP2pk' && (!signatures.signatures || !isTuple(signatures.signatures))) {\n    return {\n      ...parsedPubScript,\n      signatures: signatures.signatures,\n    };\n  }\n  throw new Error('invalid pub script');\n}\n\n/**\n * Converts a parsed script type into an array of script types.\n * @param parsedScriptType - The parsed script type.\n * @returns An array of ScriptType2Of3 values corresponding to the parsed script type.\n */\nexport function toScriptType2Of3s(parsedScriptType: ParsedScriptType2Of3): ScriptType2Of3[] {\n  return parsedScriptType === 'taprootScriptPathSpend'\n    ? ['p2trMusig2', 'p2tr']\n    : parsedScriptType === 'taprootKeyPathSpend'\n    ? ['p2trMusig2']\n    : [parsedScriptType];\n}\n\n/**\n * @returns strictly parse the input and get signature count.\n * unsigned(0), half-signed(1) or fully-signed(2)\n */\nexport function getStrictSignatureCount(input: TxInput | PsbtInput): 0 | 1 | 2 {\n  const calculateSignatureCount = (\n    signatures: [Buffer | 0, Buffer | 0, Buffer | 0] | [Buffer, Buffer] | [Buffer] | undefined\n  ): 0 | 1 | 2 => {\n    const count = signatures ? signatures.filter((s) => !isPlaceholderSignature(s)).length : 0;\n    if (count === 0 || count === 1 || count === 2) {\n      return count;\n    }\n    throw new Error('invalid signature count');\n  };\n\n  if ('hash' in input) {\n    if (input.script?.length || input.witness?.length) {\n      const parsedInput = parseSignatureScript(input);\n      return parsedInput.scriptType === 'taprootKeyPathSpend' ? 2 : calculateSignatureCount(parsedInput.signatures);\n    }\n    return 0;\n  } else {\n    return calculateSignatureCount(parsePsbtInput(input).signatures);\n  }\n}\n\n/**\n * @returns strictly parse input and get signature count for all inputs.\n * 0=unsigned, 1=half-signed or 2=fully-signed\n */\nexport function getStrictSignatureCounts(\n  tx: UtxoPsbt | UtxoTransaction<number | bigint> | PsbtInput[] | TxInput[]\n): (0 | 1 | 2)[] {\n  const inputs = tx instanceof UtxoPsbt ? tx.data.inputs : tx instanceof UtxoTransaction ? tx.ins : tx;\n  return inputs.map((input, _) => getStrictSignatureCount(input));\n}\n\n/**\n * @return true iff inputs array is of PsbtInputType type\n * */\nexport function isPsbtInputArray(inputs: PsbtInput[] | TxInput[]): inputs is PsbtInput[] {\n  return !isTxInputArray(inputs);\n}\n\n/**\n * @return true iff inputs array is of TxInput type\n * */\nexport function isTxInputArray(inputs: PsbtInput[] | TxInput[]): inputs is TxInput[] {\n  assert(!!inputs.length, 'empty inputs array');\n  return 'hash' in inputs[0];\n}\n\n/**\n * @returns true iff given psbt/transaction/tx-input-array/psbt-input-array contains at least one taproot key path spend input\n */\nexport function isTransactionWithKeyPathSpendInput(\n  data: UtxoPsbt | UtxoTransaction<bigint | number> | PsbtInput[] | TxInput[]\n): boolean {\n  const inputs = data instanceof UtxoPsbt ? data.data.inputs : data instanceof UtxoTransaction ? data.ins : data;\n  if (!inputs.length) {\n    return false;\n  }\n  if (isPsbtInputArray(inputs)) {\n    return inputs.some((input, _) => getPsbtInputScriptType(input) === 'taprootKeyPathSpend');\n  }\n  return inputs.some((input, _) => {\n    // If the input is not signed, it cannot be a taprootKeyPathSpend input because you can only\n    // extract a fully signed psbt into a transaction with taprootKeyPathSpend inputs.\n    if (getStrictSignatureCount(input) === 0) {\n      return false;\n    }\n    return parseSignatureScript(input).scriptType === 'taprootKeyPathSpend';\n  });\n}\n\n/**\n * Set the RootWalletKeys as the globalXpubs on the psbt\n *\n * We do all the matching of the (tap)bip32Derivations masterFingerprint to the fingerprint of the\n * extendedPubkey.\n */\nexport function addXpubsToPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalletKeys): void {\n  const safeRootWalletKeys = new RootWalletKeys(\n    rootWalletKeys.triple.map((bip32) => bip32.neutered()) as Triple<BIP32Interface>,\n    rootWalletKeys.derivationPrefixes\n  );\n  const xPubs = safeRootWalletKeys.triple.map(\n    (bip32): GlobalXpub => ({\n      extendedPubkey: bs58check.decode(bip32.toBase58()),\n      masterFingerprint: bip32.fingerprint,\n      // TODO: BG-73797 - bip174 currently requires m prefix for this to be a valid globalXpub\n      path: 'm',\n    })\n  );\n  psbt.updateGlobal({ globalXpub: xPubs });\n}\n\n/**\n * validates signatures for each 2 of 3 input against user, backup, bitgo keys derived from rootWalletKeys.\n * @returns array of input index and its [is valid user sig exist, is valid backup sig exist, is valid user bitgo exist]\n * For p2shP2pk input, [false, false, false] is returned since it is not a 2 of 3 sig input.\n */\nexport function getSignatureValidationArrayPsbt(psbt: UtxoPsbt, rootWalletKeys: RootWalletKeys): SignatureValidation[] {\n  return psbt.data.inputs.map((input, i) => {\n    const sigValArrayForInput: Triple<boolean> =\n      getPsbtInputScriptType(input) === 'p2shP2pk'\n        ? [false, false, false]\n        : psbt.getSignatureValidationArray(i, { rootNodes: rootWalletKeys.triple });\n    return [i, sigValArrayForInput];\n  });\n}\n\n/**\n * Extracts the half signed transaction from the psbt for p2ms based script types - p2sh, p2wsh, and p2shP2wsh.\n * The purpose is to provide backward compatibility to keyternal (KRS) that only supports network transaction and p2ms script types.\n */\nexport function extractP2msOnlyHalfSignedTx(psbt: UtxoPsbt): UtxoTransaction<bigint> {\n  assert(!!(psbt.data.inputs.length && psbt.data.outputs.length), 'empty inputs or outputs');\n  const tx = psbt.getUnsignedTx();\n\n  function isP2msParsedPsbtInput(\n    parsed: ParsedPsbtP2ms | ParsedPsbtTaproot | ParsedPsbtP2shP2pk\n  ): parsed is ParsedPsbtP2ms {\n    return ['p2sh', 'p2shP2wsh', 'p2wsh'].includes(parsed.scriptType);\n  }\n\n  psbt.data.inputs.forEach((input, i) => {\n    const parsed = parsePsbtInput(input);\n    assert(isP2msParsedPsbtInput(parsed), `unsupported script type ${parsed.scriptType}`);\n    assert(input.partialSig?.length === 1, `unexpected signature count ${input.partialSig?.length}`);\n    const [partialSig] = input.partialSig;\n    assert(\n      input.sighashType !== undefined && input.sighashType === bscript.signature.decode(partialSig.signature).hashType,\n      'signature sighash does not match input sighash type'\n    );\n\n    // type casting is to address the invalid type checking in payments.p2ms\n    const signatures = parsed.publicKeys.map((pk) =>\n      partialSig.pubkey.equals(pk) ? partialSig.signature : (ops.OP_0 as unknown as Buffer)\n    );\n\n    const isP2SH = !!parsed.redeemScript;\n    const isP2WSH = !!parsed.witnessScript;\n\n    const payment = payments.p2ms({ output: parsed.pubScript, signatures }, { validate: false, allowIncomplete: true });\n    const p2wsh = isP2WSH ? payments.p2wsh({ redeem: payment }) : undefined;\n    const p2sh = isP2SH ? payments.p2sh({ redeem: p2wsh || payment }) : undefined;\n\n    if (p2sh?.input) {\n      tx.setInputScript(i, p2sh.input);\n    }\n    if (p2wsh?.witness) {\n      tx.setWitness(i, p2wsh.witness);\n    }\n  });\n\n  return tx;\n}\n\n/**\n * Clones the psbt without nonWitnessUtxo for non-segwit inputs and witnessUtxo is added instead.\n * It is not BIP-174 compliant, so use it carefully.\n */\nexport function clonePsbtWithoutNonWitnessUtxo(psbt: UtxoPsbt): UtxoPsbt {\n  const newPsbt = createPsbtFromHex(psbt.toHex(), psbt.network);\n  const txInputs = psbt.txInputs;\n\n  psbt.data.inputs.forEach((input, i) => {\n    if (input.nonWitnessUtxo && !input.witnessUtxo) {\n      const tx = createTransactionFromBuffer(input.nonWitnessUtxo, psbt.network, { amountType: 'bigint' });\n      if (!txInputs[i].hash.equals(tx.getHash())) {\n        throw new Error(`Non-witness UTXO hash for input #${i} doesn't match the hash specified in the prevout`);\n      }\n      newPsbt.data.inputs[i].witnessUtxo = tx.outs[txInputs[i].index];\n    }\n    delete newPsbt.data.inputs[i].nonWitnessUtxo;\n  });\n\n  return newPsbt;\n}\n\n/**\n * Returns true if there are non-segwit inputs in the PSBT that do not contain the\n * nonWitnessUtxo.\n *\n * isPsbtLite(clonePsbtWithoutNonWitnessUtxo(psbt)) === true\n *\n * @param psbt\n */\nexport function isPsbtLite(psbt: UtxoPsbt): boolean {\n  let isFull = true;\n  const nonSegwitInputTypes = ['p2shP2pk', 'p2sh'];\n  psbt.data.inputs.forEach((input) => {\n    if (isFull && nonSegwitInputTypes.includes(getPsbtInputScriptType(input))) {\n      isFull = !!input.nonWitnessUtxo;\n    }\n  });\n  return !isFull;\n}\n\n/**\n * Deletes witnessUtxo for non-segwit inputs to make the PSBT BIP-174 compliant.\n */\nexport function deleteWitnessUtxoForNonSegwitInputs(psbt: UtxoPsbt): void {\n  psbt.data.inputs.forEach((input, i) => {\n    const scriptType = getPsbtInputScriptType(input);\n    if (scriptType === 'p2sh' || scriptType === 'p2shP2pk') {\n      delete input.witnessUtxo;\n    }\n  });\n}\n"]}
|