PHP WebShell
Текущая директория: /opt/BitGoJS/modules/utxo-lib/dist/test/bitgo/psbt
Просмотр файла: Musig2Util.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.invalidPartialSig = exports.invalidParticipantPubKeys = exports.invalidTxHash = exports.invalidTapInputKey = exports.invalidTapOutputKey = exports.dummyPartialSig = exports.dummyPrivateKey = exports.dummyAggNonce = exports.dummyPubNonce = exports.dummyParticipantPubKeys = exports.dummyTapInternalKey = exports.dummyTapOutputKey = exports.rootWalletKeys = exports.network = void 0;
exports.constructPsbt = constructPsbt;
exports.getUnspents = getUnspents;
exports.validatePsbtP2trMusig2Input = validatePsbtP2trMusig2Input;
exports.validatePsbtP2trMusig2Output = validatePsbtP2trMusig2Output;
exports.validateNoncesKeyVals = validateNoncesKeyVals;
exports.validatePartialSigKeyVals = validatePartialSigKeyVals;
exports.validateParticipantsKeyVals = validateParticipantsKeyVals;
exports.validateFinalizedInput = validateFinalizedInput;
exports.validateParsedTaprootKeyPathPsbt = validateParsedTaprootKeyPathPsbt;
exports.validateParsedTaprootScriptPathPsbt = validateParsedTaprootScriptPathPsbt;
exports.validateParsedTaprootKeyPathTxInput = validateParsedTaprootKeyPathTxInput;
exports.validateParsedTaprootScriptPathTxInput = validateParsedTaprootScriptPathTxInput;
const assert = require("assert");
const bitgo_1 = require("../../../src/bitgo");
const outputScripts_1 = require("../../../src/bitgo/outputScripts");
const testutil_1 = require("../../../src/testutil");
const src_1 = require("../../../src");
const PsbtUtil_1 = require("../../../src/bitgo/PsbtUtil");
exports.network = src_1.networks.bitcoin;
const outputType = 'p2trMusig2';
const CHANGE_INDEX = 100;
const FEE = BigInt(100);
const keys = [1, 2, 3].map((v) => src_1.bip32.fromSeed(Buffer.alloc(16, `test/2/${v}`), exports.network));
exports.rootWalletKeys = new bitgo_1.RootWalletKeys([keys[0], keys[1], keys[2]]);
const dummyKey1 = exports.rootWalletKeys.deriveForChainAndIndex(50, 200);
const dummyKey2 = exports.rootWalletKeys.deriveForChainAndIndex(60, 201);
exports.dummyTapOutputKey = dummyKey1.user.publicKey.subarray(1, 33);
exports.dummyTapInternalKey = dummyKey1.bitgo.publicKey.subarray(1, 33);
exports.dummyParticipantPubKeys = [dummyKey1.user.publicKey, dummyKey1.backup.publicKey];
exports.dummyPubNonce = Buffer.concat([dummyKey2.user.publicKey, dummyKey2.bitgo.publicKey]);
exports.dummyAggNonce = Buffer.concat([dummyKey2.backup.publicKey, dummyKey2.bitgo.publicKey]);
exports.dummyPrivateKey = dummyKey2.user.privateKey;
exports.dummyPartialSig = dummyKey2.backup.privateKey;
exports.invalidTapOutputKey = Buffer.allocUnsafe(1);
exports.invalidTapInputKey = Buffer.allocUnsafe(1);
exports.invalidTxHash = Buffer.allocUnsafe(1);
exports.invalidParticipantPubKeys = [Buffer.allocUnsafe(1), Buffer.allocUnsafe(1)];
exports.invalidPartialSig = Buffer.allocUnsafe(1);
function constructPsbt(unspents, rootWalletKeys, signer, cosigner, outputs) {
const psbt = (0, bitgo_1.createPsbtForNetwork)({ network: exports.network });
if (Array.isArray(outputs)) {
outputs.forEach((output) => (0, bitgo_1.addWalletOutputToPsbt)(psbt, rootWalletKeys, output.chain, output.index, output.value));
}
else {
const total = BigInt((0, bitgo_1.unspentSum)(unspents, 'bigint'));
(0, bitgo_1.addWalletOutputToPsbt)(psbt, rootWalletKeys, (0, bitgo_1.getInternalChainCode)(outputs), CHANGE_INDEX, total - FEE);
}
unspents.forEach((u) => {
if ((0, bitgo_1.isWalletUnspent)(u)) {
(0, bitgo_1.addWalletUnspentToPsbt)(psbt, u, rootWalletKeys, signer, cosigner);
}
else {
const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(testutil_1.replayProtectionKeyPair.publicKey);
assert.ok(redeemScript);
(0, bitgo_1.addReplayProtectionUnspentToPsbt)(psbt, u, redeemScript);
}
});
return psbt;
}
function getUnspents(inputScriptTypes, rootWalletKeys) {
return inputScriptTypes.map((t, i) => {
if (!bitgo_1.outputScripts.isScriptType2Of3(t)) {
throw new Error(`invalid input type ${t}`);
}
const unspent = (0, testutil_1.mockWalletUnspent)(exports.network, BigInt('10000000000000000'), {
keys: rootWalletKeys,
chain: (0, bitgo_1.getExternalChainCode)(t),
vout: i,
});
if ((0, bitgo_1.isWalletUnspent)(unspent)) {
return unspent;
}
throw new Error('Invalid unspent');
});
}
function validatePsbtP2trMusig2Input(psbt, index, unspent, spendType) {
const input = psbt.data.inputs[index];
assert.strictEqual(input.tapBip32Derivation?.length, 2);
let leafHashesCount = 0;
if (spendType === 'keyPath') {
const inputWalletKeys = exports.rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);
const { internalPubkey, taptreeRoot } = (0, outputScripts_1.createKeyPathP2trMusig2)(inputWalletKeys.publicKeys);
assert.ok(!input.tapLeafScript);
assert.ok(input.tapInternalKey);
assert.ok(input.tapMerkleRoot);
assert.ok(input.tapInternalKey.equals(internalPubkey));
assert.ok(input.tapMerkleRoot.equals(taptreeRoot));
}
else {
assert.ok(input.tapLeafScript);
assert.ok(!input.tapInternalKey);
assert.ok(!input.tapMerkleRoot);
leafHashesCount = 1;
}
input.tapBip32Derivation?.forEach((bv) => {
assert.strictEqual(bv.leafHashes.length, leafHashesCount);
});
}
function validatePsbtP2trMusig2Output(psbt, index) {
const outputWalletKeys = exports.rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getInternalChainCode)(outputType), CHANGE_INDEX);
const payment = (0, outputScripts_1.createPaymentP2trMusig2)(outputWalletKeys.publicKeys);
const output = psbt.data.outputs[index];
assert.ok(!!payment.internalPubkey);
assert.ok(!!output.tapInternalKey);
assert.ok(output.tapInternalKey.equals(payment.internalPubkey));
assert.strictEqual(output.tapBip32Derivation?.length, 3);
output.tapBip32Derivation?.forEach((bv) => {
const leafHashesCount = bv.pubkey.equals((0, outputScripts_1.toXOnlyPublicKey)(outputWalletKeys.backup.publicKey)) ? 2 : 1;
assert.strictEqual(bv.leafHashes.length, leafHashesCount);
});
}
function validateNoncesKeyVals(psbt, index, unspent) {
const keyVals = psbt.getProprietaryKeyVals(index);
const walletKeys = exports.rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);
const { outputPubkey } = (0, outputScripts_1.createKeyPathP2trMusig2)(walletKeys.publicKeys);
const participantPubKeys = [walletKeys.user.publicKey, walletKeys.bitgo.publicKey];
const nonces = keyVals.filter((kv) => kv.key.subtype === bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE);
assert.strictEqual(nonces.length, 2);
const nonceKeydata = participantPubKeys.map((p) => {
const keydata = Buffer.alloc(65);
p.copy(keydata);
outputPubkey.copy(keydata, 33);
return keydata;
});
nonces.forEach((kv) => {
assert.strictEqual(kv.key.identifier, bitgo_1.PSBT_PROPRIETARY_IDENTIFIER);
assert.strictEqual(kv.value.length, 66);
assert.strictEqual(nonceKeydata.filter((kd) => kd.equals(kv.key.keydata)).length, 1);
});
}
function validatePartialSigKeyVals(psbt, index, unspent) {
const keyVals = psbt.getProprietaryKeyVals(index);
const inputWalletKeys = exports.rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);
const { outputPubkey } = (0, outputScripts_1.createKeyPathP2trMusig2)(inputWalletKeys.publicKeys);
const participantPubKeys = [inputWalletKeys.user.publicKey, inputWalletKeys.bitgo.publicKey];
const partialSigs = keyVals.filter((kv) => kv.key.subtype === bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG);
assert.strictEqual(partialSigs.length, 2);
const partialSigKeydata = participantPubKeys.map((p) => {
const keydata = Buffer.alloc(65);
p.copy(keydata);
outputPubkey.copy(keydata, 33);
return keydata;
});
partialSigs.forEach((kv) => {
assert.strictEqual(kv.key.identifier, bitgo_1.PSBT_PROPRIETARY_IDENTIFIER);
assert.strictEqual(kv.value.length, 32);
assert.strictEqual(partialSigKeydata.filter((kd) => kd.equals(kv.key.keydata)).length, 1);
});
}
function validateParticipantsKeyVals(psbt, index, unspent) {
const keyVals = psbt.getProprietaryKeyVals(index);
const walletKeys = exports.rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);
const { internalPubkey, outputPubkey } = (0, outputScripts_1.createKeyPathP2trMusig2)(walletKeys.publicKeys);
const participantPubKeys = [walletKeys.user.publicKey, walletKeys.bitgo.publicKey];
const participantsKeyVals = keyVals.filter((kv) => kv.key.subtype === bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS);
assert.strictEqual(participantsKeyVals.length, 1);
const kv = participantsKeyVals[0];
assert.strictEqual(kv.key.identifier, bitgo_1.PSBT_PROPRIETARY_IDENTIFIER);
assert.ok(Buffer.concat([outputPubkey, internalPubkey]).equals(kv.key.keydata));
const valueMatch = [Buffer.concat(participantPubKeys), Buffer.concat(participantPubKeys.reverse())].some((pks) => {
return pks.equals(kv.value);
});
assert.ok(valueMatch);
}
function validateFinalizedInput(psbt, index, unspent, spendType) {
const input = psbt.data.inputs[index];
assert.ok((0, PsbtUtil_1.isPsbtInputFinalized)(input));
if ((0, bitgo_1.scriptTypeForChain)(unspent.chain) === 'p2trMusig2' && spendType === 'keyPath') {
assert.strictEqual(input.finalScriptWitness?.length, 66);
}
assert.ok(!input.unknownKeyVals?.length);
}
function validateParsedTaprootKeyPathPsbt(psbt, index, signature) {
const parsed = (0, bitgo_1.parsePsbtInput)(psbt.data.inputs[0]);
assert.ok(parsed.scriptType === 'taprootKeyPathSpend');
assert.strictEqual(parsed.pubScript.length, 34);
assert.strictEqual(parsed.publicKeys.length, 1);
assert.strictEqual(parsed.publicKeys[0].length, 32);
if (signature === 'unsigned') {
assert.strictEqual(parsed.signatures, undefined);
assert.strictEqual(parsed.participantPublicKeys, undefined);
}
else {
const expected = signature === 'halfsigned' ? 1 : 2;
assert.strictEqual(parsed.signatures?.length, expected);
parsed.signatures.forEach((sig) => {
assert.strictEqual(sig.length, 32);
});
assert.strictEqual(parsed.participantPublicKeys?.length, expected);
parsed.participantPublicKeys.forEach((pk) => {
assert.strictEqual(pk.length, 33);
});
}
}
function validateParsedTaprootScriptPathPsbt(psbt, index, signature) {
const input = psbt.data.inputs[index];
const parsed = (0, bitgo_1.parsePsbtInput)(psbt.data.inputs[0]);
assert.ok(parsed.scriptType === 'taprootScriptPathSpend');
assert.ok(input.tapLeafScript);
assert.ok(parsed.pubScript.equals(input.tapLeafScript[0].script));
assert.ok(parsed.controlBlock.equals(input.tapLeafScript[0].controlBlock));
assert.strictEqual(parsed.scriptPathLevel, 1);
assert.strictEqual(parsed.leafVersion, input.tapLeafScript[0].leafVersion);
parsed.publicKeys.forEach((pk) => {
assert.strictEqual(pk.length, 32);
});
if (signature === 'unsigned') {
assert.strictEqual(parsed.signatures, undefined);
}
else {
const expected = signature === 'halfsigned' ? 1 : 2;
assert.strictEqual(parsed.signatures?.length, expected);
parsed.signatures.forEach((sig) => {
assert.strictEqual(sig.length, 64);
});
}
}
function validateParsedTaprootKeyPathTxInput(psbt, tx) {
const parsedTxInput = (0, bitgo_1.parseSignatureScript2Of3)(tx.ins[0]);
assert.ok(parsedTxInput.scriptType === 'taprootKeyPathSpend');
assert.strictEqual(parsedTxInput.signatures.length, 1);
assert.strictEqual(parsedTxInput.signatures[0].length, 64);
}
function validateParsedTaprootScriptPathTxInput(psbt, tx, index) {
const input = psbt.data.inputs[index];
const parsedTxInput = (0, bitgo_1.parseSignatureScript2Of3)(tx.ins[0]);
assert.ok(parsedTxInput);
assert.ok(parsedTxInput.scriptType === 'taprootScriptPathSpend');
assert.ok(input.tapLeafScript);
assert.ok(parsedTxInput.pubScript.equals(input.tapLeafScript[0].script));
assert.ok(parsedTxInput.controlBlock.equals(input.tapLeafScript[0].controlBlock));
assert.strictEqual(parsedTxInput.scriptPathLevel, 1);
assert.strictEqual(parsedTxInput.leafVersion, input.tapLeafScript[0].leafVersion);
parsedTxInput.publicKeys.forEach((pk) => {
assert.strictEqual(pk.length, 32);
});
assert.strictEqual(parsedTxInput.signatures?.length, 2);
parsedTxInput.signatures.forEach((sig) => {
assert.strictEqual(sig.length, 64);
});
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Musig2Util.js","sourceRoot":"","sources":["../../../../test/bitgo/psbt/Musig2Util.ts"],"names":[],"mappings":";;;AA+DA,sCAyBC;AAED,kCAmBC;AAED,kEA4BC;AAED,oEAYC;AAED,sDAyBC;AAED,8DAyBC;AAED,kEAsBC;AAED,wDAYC;AAED,4EAyBC;AAED,kFAyBC;AAED,kFAQC;AAED,wFAqBC;AA5UD,iCAAiC;AAEjC,8CAuB4B;AAC5B,oEAM0C;AAC1C,oDAAmF;AACnF,sCAA+C;AAE/C,0DAAmE;AAEtD,QAAA,OAAO,GAAG,cAAQ,CAAC,OAAO,CAAC;AACxC,MAAM,UAAU,GAAG,YAAY,CAAC;AAChC,MAAM,YAAY,GAAG,GAAG,CAAC;AACzB,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;AAExB,MAAM,IAAI,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,WAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,UAAU,CAAC,EAAE,CAAC,EAAE,eAAO,CAAC,CAAqB,CAAC;AACnG,QAAA,cAAc,GAAG,IAAI,sBAAc,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AAE9E,MAAM,SAAS,GAAG,sBAAc,CAAC,sBAAsB,CAAC,EAAE,EAAE,GAAG,CAAC,CAAC;AACjE,MAAM,SAAS,GAAG,sBAAc,CAAC,sBAAsB,CAAC,EAAE,EAAE,GAAG,CAAC,CAAC;AAEpD,QAAA,iBAAiB,GAAG,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;AAC7D,QAAA,mBAAmB,GAAG,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;AAChE,QAAA,uBAAuB,GAAkB,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,EAAE,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;AAChG,QAAA,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,EAAE,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;AACrF,QAAA,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;AACvF,QAAA,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,UAAW,CAAC;AAC7C,QAAA,eAAe,GAAG,SAAS,CAAC,MAAM,CAAC,UAAW,CAAC;AAE/C,QAAA,mBAAmB,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AAC5C,QAAA,kBAAkB,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AAC3C,QAAA,aAAa,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AACtC,QAAA,yBAAyB,GAAkB,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;AAC1F,QAAA,iBAAiB,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AAEvD,SAAgB,aAAa,CAC3B,QAAmD,EACnD,cAA8B,EAC9B,MAAe,EACf,QAAiB,EACjB,OAA4F;IAE5F,MAAM,IAAI,GAAG,IAAA,4BAAoB,EAAC,EAAE,OAAO,EAAP,eAAO,EAAE,CAAC,CAAC;IAE/C,IAAI,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QAC3B,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,IAAA,6BAAqB,EAAC,IAAI,EAAE,cAAc,EAAE,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACrH,CAAC;SAAM,CAAC;QACN,MAAM,KAAK,GAAG,MAAM,CAAC,IAAA,kBAAU,EAAS,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC;QAC7D,IAAA,6BAAqB,EAAC,IAAI,EAAE,cAAc,EAAE,IAAA,4BAAoB,EAAC,OAAO,CAAC,EAAE,YAAY,EAAE,KAAK,GAAG,GAAG,CAAC,CAAC;IACxG,CAAC;IACD,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;QACrB,IAAI,IAAA,uBAAe,EAAC,CAAC,CAAC,EAAE,CAAC;YACvB,IAAA,8BAAsB,EAAC,IAAI,EAAE,CAAC,EAAE,cAAc,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QACpE,CAAC;aAAM,CAAC;YACN,MAAM,EAAE,YAAY,EAAE,GAAG,IAAA,0CAA0B,EAAC,kCAAuB,CAAC,SAAS,CAAC,CAAC;YACvF,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;YACxB,IAAA,wCAAgC,EAAC,IAAI,EAAE,CAAC,EAAE,YAAY,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC,CAAC,CAAC;IACH,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAgB,WAAW,CACzB,gBAAkC,EAClC,cAA8B;IAE9B,OAAO,gBAAgB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAyB,EAAE;QAC1D,IAAI,CAAC,qBAAa,CAAC,gBAAgB,CAAC,CAAC,CAAC,EAAE,CAAC;YACvC,MAAM,IAAI,KAAK,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC;QAC7C,CAAC;QACD,MAAM,OAAO,GAAG,IAAA,4BAAiB,EAAC,eAAO,EAAE,MAAM,CAAC,mBAAmB,CAAC,EAAE;YACtE,IAAI,EAAE,cAAc;YACpB,KAAK,EAAE,IAAA,4BAAoB,EAAC,CAAC,CAAC;YAC9B,IAAI,EAAE,CAAC;SACR,CAAC,CAAC;QAEH,IAAI,IAAA,uBAAe,EAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,OAAO,OAAO,CAAC;QACjB,CAAC;QACD,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,2BAA2B,CACzC,IAAuC,EACvC,KAAa,EACb,OAA8B,EAC9B,SAAmC;IAEnC,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACtC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,kBAAkB,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;IACxD,IAAI,eAAe,GAAG,CAAC,CAAC;IAExB,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;QAC5B,MAAM,eAAe,GAAG,sBAAc,CAAC,sBAAsB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;QAC5F,MAAM,EAAE,cAAc,EAAE,WAAW,EAAE,GAAG,IAAA,uCAAuB,EAAC,eAAe,CAAC,UAAU,CAAC,CAAC;QAE5F,MAAM,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAChC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QAChC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAC/B,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC;QACvD,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;IACrD,CAAC;SAAM,CAAC;QACN,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAC/B,MAAM,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACjC,MAAM,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAChC,eAAe,GAAG,CAAC,CAAC;IACtB,CAAC;IACD,KAAK,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QACvC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC;IAC5D,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,4BAA4B,CAAC,IAAuC,EAAE,KAAa;IACjG,MAAM,gBAAgB,GAAG,sBAAc,CAAC,sBAAsB,CAAC,IAAA,4BAAoB,EAAC,UAAU,CAAC,EAAE,YAAY,CAAC,CAAC;IAC/G,MAAM,OAAO,GAAG,IAAA,uCAAuB,EAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC;IACrE,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACxC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;IACpC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;IACnC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,CAAC;IAChE,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,kBAAkB,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;IACzD,MAAM,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QACxC,MAAM,eAAe,GAAG,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAA,gCAAgB,EAAC,gBAAgB,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACtG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC;IAC5D,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,qBAAqB,CACnC,IAAuC,EACvC,KAAa,EACb,OAA8B;IAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;IAClD,MAAM,UAAU,GAAG,sBAAc,CAAC,sBAAsB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;IACvF,MAAM,EAAE,YAAY,EAAE,GAAG,IAAA,uCAAuB,EAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IACxE,MAAM,kBAAkB,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAEnF,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,6BAAqB,CAAC,gBAAgB,CAAC,CAAC;IACjG,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAErC,MAAM,YAAY,GAAG,kBAAkB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;QAChD,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChB,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAC/B,OAAO,OAAO,CAAC;IACjB,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QACpB,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,EAAE,mCAA2B,CAAC,CAAC;QACnE,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACxC,MAAM,CAAC,WAAW,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IACvF,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,yBAAyB,CACvC,IAAuC,EACvC,KAAa,EACb,OAA8B;IAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;IAClD,MAAM,eAAe,GAAG,sBAAc,CAAC,sBAAsB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;IAC5F,MAAM,EAAE,YAAY,EAAE,GAAG,IAAA,uCAAuB,EAAC,eAAe,CAAC,UAAU,CAAC,CAAC;IAC7E,MAAM,kBAAkB,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAE7F,MAAM,WAAW,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,6BAAqB,CAAC,kBAAkB,CAAC,CAAC;IACxG,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAE1C,MAAM,iBAAiB,GAAG,kBAAkB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;QACrD,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChB,YAAY,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAC/B,OAAO,OAAO,CAAC;IACjB,CAAC,CAAC,CAAC;IAEH,WAAW,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QACzB,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,EAAE,mCAA2B,CAAC,CAAC;QACnE,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACxC,MAAM,CAAC,WAAW,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAC5F,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,2BAA2B,CACzC,IAAuC,EACvC,KAAa,EACb,OAA8B;IAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;IAClD,MAAM,UAAU,GAAG,sBAAc,CAAC,sBAAsB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;IACvF,MAAM,EAAE,cAAc,EAAE,YAAY,EAAE,GAAG,IAAA,uCAAuB,EAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IACxF,MAAM,kBAAkB,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAEnF,MAAM,mBAAmB,GAAG,OAAO,CAAC,MAAM,CACxC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,6BAAqB,CAAC,2BAA2B,CAC7E,CAAC;IACF,MAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAElD,MAAM,EAAE,GAAG,mBAAmB,CAAC,CAAC,CAAC,CAAC;IAClC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,EAAE,mCAA2B,CAAC,CAAC;IACnE,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,EAAE,cAAc,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC;IAChF,MAAM,UAAU,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,kBAAkB,CAAC,EAAE,MAAM,CAAC,MAAM,CAAC,kBAAkB,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;QAC/G,OAAO,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC,CAAC,CAAC;IACH,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC;AACxB,CAAC;AAED,SAAgB,sBAAsB,CACpC,IAAuC,EACvC,KAAa,EACb,OAA8B,EAC9B,SAAoC;IAEpC,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACtC,MAAM,CAAC,EAAE,CAAC,IAAA,+BAAoB,EAAC,KAAK,CAAC,CAAC,CAAC;IACvC,IAAI,IAAA,0BAAkB,EAAC,OAAO,CAAC,KAAK,CAAC,KAAK,YAAY,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;QAClF,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,kBAAkB,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IACD,MAAM,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC;AAC3C,CAAC;AAED,SAAgB,gCAAgC,CAC9C,IAAuC,EACvC,KAAa,EACb,SAAoD;IAEpD,MAAM,MAAM,GAAG,IAAA,sBAAc,EAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;IACnD,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,UAAU,KAAK,qBAAqB,CAAC,CAAC;IACvD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;IAChD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAChD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;IAEpD,IAAI,SAAS,KAAK,UAAU,EAAE,CAAC;QAC7B,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QACjD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,qBAAqB,EAAE,SAAS,CAAC,CAAC;IAC9D,CAAC;SAAM,CAAC;QACN,MAAM,QAAQ,GAAG,SAAS,KAAK,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QACxD,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAChC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,qBAAqB,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QACnE,MAAM,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;YAC1C,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,SAAgB,mCAAmC,CACjD,IAAuC,EACvC,KAAa,EACb,SAAoD;IAEpD,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACtC,MAAM,MAAM,GAAG,IAAA,sBAAc,EAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;IACnD,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,UAAU,KAAK,wBAAwB,CAAC,CAAC;IAC1D,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IAC/B,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IAClE,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;IAC3E,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;IAC9C,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,WAAW,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;IAC3E,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QAC/B,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IACH,IAAI,SAAS,KAAK,UAAU,EAAE,CAAC;QAC7B,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;IACnD,CAAC;SAAM,CAAC;QACN,MAAM,QAAQ,GAAG,SAAS,KAAK,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpD,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QACxD,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAChC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,SAAgB,mCAAmC,CACjD,IAAuC,EACvC,EAA2B;IAE3B,MAAM,aAAa,GAAG,IAAA,gCAAwB,EAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1D,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,UAAU,KAAK,qBAAqB,CAAC,CAAC;IAC9D,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IACvD,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;AAC7D,CAAC;AAED,SAAgB,sCAAsC,CACpD,IAAuC,EACvC,EAA2B,EAC3B,KAAa;IAEb,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACtC,MAAM,aAAa,GAAG,IAAA,gCAAwB,EAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1D,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC;IACzB,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,UAAU,KAAK,wBAAwB,CAAC,CAAC;IACjE,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IAC/B,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IACzE,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;IAClF,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC;IACrD,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,WAAW,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;IAClF,aAAa,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;QACtC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,UAAU,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;IACxD,aAAa,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;QACvC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import * as assert from 'assert';\n\nimport {\n  addReplayProtectionUnspentToPsbt,\n  addWalletOutputToPsbt,\n  addWalletUnspentToPsbt,\n  createPsbtForNetwork,\n  getExternalChainCode,\n  getInternalChainCode,\n  isWalletUnspent,\n  KeyName,\n  outputScripts,\n  parsePsbtInput,\n  parseSignatureScript2Of3,\n  ProprietaryKeySubtype,\n  PSBT_PROPRIETARY_IDENTIFIER,\n  RootWalletKeys,\n  scriptTypeForChain,\n  Tuple,\n  Unspent,\n  unspentSum,\n  UtxoPsbt,\n  UtxoTransaction,\n  WalletUnspent,\n  ChainCode,\n} from '../../../src/bitgo';\nimport {\n  createKeyPathP2trMusig2,\n  createOutputScriptP2shP2pk,\n  createPaymentP2trMusig2,\n  ScriptType2Of3,\n  toXOnlyPublicKey,\n} from '../../../src/bitgo/outputScripts';\nimport { mockWalletUnspent, replayProtectionKeyPair } from '../../../src/testutil';\nimport { bip32, networks } from '../../../src';\nimport { BIP32Interface } from 'bip32';\nimport { isPsbtInputFinalized } from '../../../src/bitgo/PsbtUtil';\n\nexport const network = networks.bitcoin;\nconst outputType = 'p2trMusig2';\nconst CHANGE_INDEX = 100;\nconst FEE = BigInt(100);\n\nconst keys = [1, 2, 3].map((v) => bip32.fromSeed(Buffer.alloc(16, `test/2/${v}`), network)) as BIP32Interface[];\nexport const rootWalletKeys = new RootWalletKeys([keys[0], keys[1], keys[2]]);\n\nconst dummyKey1 = rootWalletKeys.deriveForChainAndIndex(50, 200);\nconst dummyKey2 = rootWalletKeys.deriveForChainAndIndex(60, 201);\n\nexport const dummyTapOutputKey = dummyKey1.user.publicKey.subarray(1, 33);\nexport const dummyTapInternalKey = dummyKey1.bitgo.publicKey.subarray(1, 33);\nexport const dummyParticipantPubKeys: Tuple<Buffer> = [dummyKey1.user.publicKey, dummyKey1.backup.publicKey];\nexport const dummyPubNonce = Buffer.concat([dummyKey2.user.publicKey, dummyKey2.bitgo.publicKey]);\nexport const dummyAggNonce = Buffer.concat([dummyKey2.backup.publicKey, dummyKey2.bitgo.publicKey]);\nexport const dummyPrivateKey = dummyKey2.user.privateKey!;\nexport const dummyPartialSig = dummyKey2.backup.privateKey!;\n\nexport const invalidTapOutputKey = Buffer.allocUnsafe(1);\nexport const invalidTapInputKey = Buffer.allocUnsafe(1);\nexport const invalidTxHash = Buffer.allocUnsafe(1);\nexport const invalidParticipantPubKeys: Tuple<Buffer> = [Buffer.allocUnsafe(1), Buffer.allocUnsafe(1)];\nexport const invalidPartialSig = Buffer.allocUnsafe(1);\n\nexport function constructPsbt(\n  unspents: (Unspent<bigint> & { prevTx?: Buffer })[],\n  rootWalletKeys: RootWalletKeys,\n  signer: KeyName,\n  cosigner: KeyName,\n  outputs: { chain: ChainCode; index: number; value: bigint }[] | outputScripts.ScriptType2Of3\n): UtxoPsbt {\n  const psbt = createPsbtForNetwork({ network });\n\n  if (Array.isArray(outputs)) {\n    outputs.forEach((output) => addWalletOutputToPsbt(psbt, rootWalletKeys, output.chain, output.index, output.value));\n  } else {\n    const total = BigInt(unspentSum<bigint>(unspents, 'bigint'));\n    addWalletOutputToPsbt(psbt, rootWalletKeys, getInternalChainCode(outputs), CHANGE_INDEX, total - FEE);\n  }\n  unspents.forEach((u) => {\n    if (isWalletUnspent(u)) {\n      addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signer, cosigner);\n    } else {\n      const { redeemScript } = createOutputScriptP2shP2pk(replayProtectionKeyPair.publicKey);\n      assert.ok(redeemScript);\n      addReplayProtectionUnspentToPsbt(psbt, u, redeemScript);\n    }\n  });\n  return psbt;\n}\n\nexport function getUnspents(\n  inputScriptTypes: ScriptType2Of3[],\n  rootWalletKeys: RootWalletKeys\n): WalletUnspent<bigint>[] {\n  return inputScriptTypes.map((t, i): WalletUnspent<bigint> => {\n    if (!outputScripts.isScriptType2Of3(t)) {\n      throw new Error(`invalid input type ${t}`);\n    }\n    const unspent = mockWalletUnspent(network, BigInt('10000000000000000'), {\n      keys: rootWalletKeys,\n      chain: getExternalChainCode(t),\n      vout: i,\n    });\n\n    if (isWalletUnspent(unspent)) {\n      return unspent;\n    }\n    throw new Error('Invalid unspent');\n  });\n}\n\nexport function validatePsbtP2trMusig2Input(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  unspent: WalletUnspent<bigint>,\n  spendType: 'keyPath' | 'scriptPath'\n): void {\n  const input = psbt.data.inputs[index];\n  assert.strictEqual(input.tapBip32Derivation?.length, 2);\n  let leafHashesCount = 0;\n\n  if (spendType === 'keyPath') {\n    const inputWalletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n    const { internalPubkey, taptreeRoot } = createKeyPathP2trMusig2(inputWalletKeys.publicKeys);\n\n    assert.ok(!input.tapLeafScript);\n    assert.ok(input.tapInternalKey);\n    assert.ok(input.tapMerkleRoot);\n    assert.ok(input.tapInternalKey.equals(internalPubkey));\n    assert.ok(input.tapMerkleRoot.equals(taptreeRoot));\n  } else {\n    assert.ok(input.tapLeafScript);\n    assert.ok(!input.tapInternalKey);\n    assert.ok(!input.tapMerkleRoot);\n    leafHashesCount = 1;\n  }\n  input.tapBip32Derivation?.forEach((bv) => {\n    assert.strictEqual(bv.leafHashes.length, leafHashesCount);\n  });\n}\n\nexport function validatePsbtP2trMusig2Output(psbt: UtxoPsbt<UtxoTransaction<bigint>>, index: number): void {\n  const outputWalletKeys = rootWalletKeys.deriveForChainAndIndex(getInternalChainCode(outputType), CHANGE_INDEX);\n  const payment = createPaymentP2trMusig2(outputWalletKeys.publicKeys);\n  const output = psbt.data.outputs[index];\n  assert.ok(!!payment.internalPubkey);\n  assert.ok(!!output.tapInternalKey);\n  assert.ok(output.tapInternalKey.equals(payment.internalPubkey));\n  assert.strictEqual(output.tapBip32Derivation?.length, 3);\n  output.tapBip32Derivation?.forEach((bv) => {\n    const leafHashesCount = bv.pubkey.equals(toXOnlyPublicKey(outputWalletKeys.backup.publicKey)) ? 2 : 1;\n    assert.strictEqual(bv.leafHashes.length, leafHashesCount);\n  });\n}\n\nexport function validateNoncesKeyVals(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  unspent: WalletUnspent<bigint>\n): void {\n  const keyVals = psbt.getProprietaryKeyVals(index);\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n  const { outputPubkey } = createKeyPathP2trMusig2(walletKeys.publicKeys);\n  const participantPubKeys = [walletKeys.user.publicKey, walletKeys.bitgo.publicKey];\n\n  const nonces = keyVals.filter((kv) => kv.key.subtype === ProprietaryKeySubtype.MUSIG2_PUB_NONCE);\n  assert.strictEqual(nonces.length, 2);\n\n  const nonceKeydata = participantPubKeys.map((p) => {\n    const keydata = Buffer.alloc(65);\n    p.copy(keydata);\n    outputPubkey.copy(keydata, 33);\n    return keydata;\n  });\n\n  nonces.forEach((kv) => {\n    assert.strictEqual(kv.key.identifier, PSBT_PROPRIETARY_IDENTIFIER);\n    assert.strictEqual(kv.value.length, 66);\n    assert.strictEqual(nonceKeydata.filter((kd) => kd.equals(kv.key.keydata)).length, 1);\n  });\n}\n\nexport function validatePartialSigKeyVals(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  unspent: WalletUnspent<bigint>\n): void {\n  const keyVals = psbt.getProprietaryKeyVals(index);\n  const inputWalletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n  const { outputPubkey } = createKeyPathP2trMusig2(inputWalletKeys.publicKeys);\n  const participantPubKeys = [inputWalletKeys.user.publicKey, inputWalletKeys.bitgo.publicKey];\n\n  const partialSigs = keyVals.filter((kv) => kv.key.subtype === ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG);\n  assert.strictEqual(partialSigs.length, 2);\n\n  const partialSigKeydata = participantPubKeys.map((p) => {\n    const keydata = Buffer.alloc(65);\n    p.copy(keydata);\n    outputPubkey.copy(keydata, 33);\n    return keydata;\n  });\n\n  partialSigs.forEach((kv) => {\n    assert.strictEqual(kv.key.identifier, PSBT_PROPRIETARY_IDENTIFIER);\n    assert.strictEqual(kv.value.length, 32);\n    assert.strictEqual(partialSigKeydata.filter((kd) => kd.equals(kv.key.keydata)).length, 1);\n  });\n}\n\nexport function validateParticipantsKeyVals(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  unspent: WalletUnspent<bigint>\n): void {\n  const keyVals = psbt.getProprietaryKeyVals(index);\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n  const { internalPubkey, outputPubkey } = createKeyPathP2trMusig2(walletKeys.publicKeys);\n  const participantPubKeys = [walletKeys.user.publicKey, walletKeys.bitgo.publicKey];\n\n  const participantsKeyVals = keyVals.filter(\n    (kv) => kv.key.subtype === ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS\n  );\n  assert.strictEqual(participantsKeyVals.length, 1);\n\n  const kv = participantsKeyVals[0];\n  assert.strictEqual(kv.key.identifier, PSBT_PROPRIETARY_IDENTIFIER);\n  assert.ok(Buffer.concat([outputPubkey, internalPubkey]).equals(kv.key.keydata));\n  const valueMatch = [Buffer.concat(participantPubKeys), Buffer.concat(participantPubKeys.reverse())].some((pks) => {\n    return pks.equals(kv.value);\n  });\n  assert.ok(valueMatch);\n}\n\nexport function validateFinalizedInput(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  unspent: WalletUnspent<bigint>,\n  spendType?: 'keyPath' | 'scriptPath'\n): void {\n  const input = psbt.data.inputs[index];\n  assert.ok(isPsbtInputFinalized(input));\n  if (scriptTypeForChain(unspent.chain) === 'p2trMusig2' && spendType === 'keyPath') {\n    assert.strictEqual(input.finalScriptWitness?.length, 66);\n  }\n  assert.ok(!input.unknownKeyVals?.length);\n}\n\nexport function validateParsedTaprootKeyPathPsbt(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  signature: 'unsigned' | 'halfsigned' | 'fullysigned'\n): void {\n  const parsed = parsePsbtInput(psbt.data.inputs[0]);\n  assert.ok(parsed.scriptType === 'taprootKeyPathSpend');\n  assert.strictEqual(parsed.pubScript.length, 34);\n  assert.strictEqual(parsed.publicKeys.length, 1);\n  assert.strictEqual(parsed.publicKeys[0].length, 32);\n\n  if (signature === 'unsigned') {\n    assert.strictEqual(parsed.signatures, undefined);\n    assert.strictEqual(parsed.participantPublicKeys, undefined);\n  } else {\n    const expected = signature === 'halfsigned' ? 1 : 2;\n    assert.strictEqual(parsed.signatures?.length, expected);\n    parsed.signatures.forEach((sig) => {\n      assert.strictEqual(sig.length, 32);\n    });\n    assert.strictEqual(parsed.participantPublicKeys?.length, expected);\n    parsed.participantPublicKeys.forEach((pk) => {\n      assert.strictEqual(pk.length, 33);\n    });\n  }\n}\n\nexport function validateParsedTaprootScriptPathPsbt(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  index: number,\n  signature: 'unsigned' | 'halfsigned' | 'fullysigned'\n): void {\n  const input = psbt.data.inputs[index];\n  const parsed = parsePsbtInput(psbt.data.inputs[0]);\n  assert.ok(parsed.scriptType === 'taprootScriptPathSpend');\n  assert.ok(input.tapLeafScript);\n  assert.ok(parsed.pubScript.equals(input.tapLeafScript[0].script));\n  assert.ok(parsed.controlBlock.equals(input.tapLeafScript[0].controlBlock));\n  assert.strictEqual(parsed.scriptPathLevel, 1);\n  assert.strictEqual(parsed.leafVersion, input.tapLeafScript[0].leafVersion);\n  parsed.publicKeys.forEach((pk) => {\n    assert.strictEqual(pk.length, 32);\n  });\n  if (signature === 'unsigned') {\n    assert.strictEqual(parsed.signatures, undefined);\n  } else {\n    const expected = signature === 'halfsigned' ? 1 : 2;\n    assert.strictEqual(parsed.signatures?.length, expected);\n    parsed.signatures.forEach((sig) => {\n      assert.strictEqual(sig.length, 64);\n    });\n  }\n}\n\nexport function validateParsedTaprootKeyPathTxInput(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  tx: UtxoTransaction<bigint>\n): void {\n  const parsedTxInput = parseSignatureScript2Of3(tx.ins[0]);\n  assert.ok(parsedTxInput.scriptType === 'taprootKeyPathSpend');\n  assert.strictEqual(parsedTxInput.signatures.length, 1);\n  assert.strictEqual(parsedTxInput.signatures[0].length, 64);\n}\n\nexport function validateParsedTaprootScriptPathTxInput(\n  psbt: UtxoPsbt<UtxoTransaction<bigint>>,\n  tx: UtxoTransaction<bigint>,\n  index: number\n): void {\n  const input = psbt.data.inputs[index];\n  const parsedTxInput = parseSignatureScript2Of3(tx.ins[0]);\n  assert.ok(parsedTxInput);\n  assert.ok(parsedTxInput.scriptType === 'taprootScriptPathSpend');\n  assert.ok(input.tapLeafScript);\n  assert.ok(parsedTxInput.pubScript.equals(input.tapLeafScript[0].script));\n  assert.ok(parsedTxInput.controlBlock.equals(input.tapLeafScript[0].controlBlock));\n  assert.strictEqual(parsedTxInput.scriptPathLevel, 1);\n  assert.strictEqual(parsedTxInput.leafVersion, input.tapLeafScript[0].leafVersion);\n  parsedTxInput.publicKeys.forEach((pk) => {\n    assert.strictEqual(pk.length, 32);\n  });\n  assert.strictEqual(parsedTxInput.signatures?.length, 2);\n  parsedTxInput.signatures.forEach((sig) => {\n    assert.strictEqual(sig.length, 64);\n  });\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!