PHP WebShell
Текущая директория: /opt/BitGoJS/modules/utxo-lib/dist/test/bitgo/psbt
Просмотр файла: Musig2.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const bitgo_1 = require("../../../src/bitgo");
const testutil_1 = require("../../../src/testutil");
const Musig2_1 = require("../../../src/bitgo/Musig2");
const outputScripts_1 = require("../../../src/bitgo/outputScripts");
const Musig2Util_1 = require("./Musig2Util");
const p2trMusig2Unspent = (0, Musig2Util_1.getUnspents)(['p2trMusig2'], Musig2Util_1.rootWalletKeys);
const outputType = 'p2trMusig2';
const CHANGE_INDEX = 100;
describe('p2trMusig2', function () {
describe('p2trMusig2 key path', function () {
it(`create psbt, nonces, sign (internal verify) - success`, function () {
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getExternalChainCode)('p2trMusig2'), 0);
const unspents = (0, Musig2Util_1.getUnspents)(outputScripts_1.scriptTypes2Of3.map((t) => t), Musig2Util_1.rootWalletKeys);
// WP creates PSBT during build API, serializes it, and sends the psbt to user
const buildPsbt = (0, Musig2Util_1.constructPsbt)(unspents, Musig2Util_1.rootWalletKeys, 'bitgo', 'user', outputType);
const buildPsbtSer = buildPsbt.toHex();
// User de-serialises the psbt, ands the user nonce, and sends it to the hsm so that it can add the bitgo nonce
const userPsbt = (0, bitgo_1.createPsbtFromHex)(buildPsbtSer, Musig2Util_1.network);
userPsbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
const userPsbtSer = userPsbt.toHex();
// HSM deserializes the user psbt, adds the deterministic bitgo nonce, and sends that back to the user
const bitgoPsbt = (0, bitgo_1.createPsbtFromHex)(userPsbtSer, Musig2Util_1.network);
bitgoPsbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
const bitgoPsbtSer = bitgoPsbt.toHex();
// User combines the psbt with the bitgo nonce, adds user signature, and sends half-signed to hsm
const bitgoPsbtDeser = (0, bitgo_1.createPsbtFromHex)(bitgoPsbtSer, Musig2Util_1.network);
userPsbt.combine(bitgoPsbtDeser);
userPsbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
const userPsbtHalfSignedHex = userPsbt.toHex();
// WP de-serialises the psbt and validates user sig
const userPsbtDes = (0, bitgo_1.createPsbtFromHex)(userPsbtHalfSignedHex, Musig2Util_1.network);
assert.ok(userPsbtDes.validateTaprootMusig2SignaturesOfInput(4, walletKeys.user.publicKey));
// WP sends to hsm for signature and returns a fully signed psbt
const psbt = (0, bitgo_1.createPsbtFromHex)(userPsbtHalfSignedHex, Musig2Util_1.network);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
unspents.forEach((unspent, index) => {
if ((0, bitgo_1.scriptTypeForChain)(unspent.chain) !== 'p2trMusig2') {
assert.strictEqual(psbt.getProprietaryKeyVals(index).length, 0);
return;
}
(0, Musig2Util_1.validatePsbtP2trMusig2Input)(psbt, index, unspent, 'keyPath');
(0, Musig2Util_1.validatePsbtP2trMusig2Output)(psbt, 0);
(0, Musig2Util_1.validateParticipantsKeyVals)(psbt, index, unspent);
(0, Musig2Util_1.validateNoncesKeyVals)(psbt, index, unspent);
(0, Musig2Util_1.validatePartialSigKeyVals)(psbt, index, unspent);
});
assert.ok(psbt.validateSignaturesOfAllInputs());
psbt.finalizeAllInputs();
unspents.forEach((unspent, index) => {
(0, Musig2Util_1.validateFinalizedInput)(psbt, index, unspent);
});
const tx = psbt.extractTransaction();
assert.ok((0, testutil_1.verifyFullySignedSignatures)(tx, unspents, Musig2Util_1.rootWalletKeys, 'bitgo', 'user'));
unspents.map((unspent, inputIndex) => {
assert.deepStrictEqual((0, bitgo_1.verifySignatureWithUnspent)(tx, inputIndex, unspents, Musig2Util_1.rootWalletKeys), [
true,
false,
true,
]);
});
});
it(`parse tx`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'bitgo', 'user', outputType);
(0, Musig2Util_1.validateParsedTaprootKeyPathPsbt)(psbt, 0, 'unsigned');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
(0, Musig2Util_1.validateParsedTaprootKeyPathPsbt)(psbt, 0, 'halfsigned');
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
(0, Musig2Util_1.validateParsedTaprootKeyPathPsbt)(psbt, 0, 'fullysigned');
psbt.finalizeAllInputs();
assert.throws(() => (0, bitgo_1.parsePsbtInput)(psbt.data.inputs[0]), (e) => e.message === 'Finalized PSBT parsing is not supported');
const tx = psbt.extractTransaction();
(0, Musig2Util_1.validateParsedTaprootKeyPathTxInput)(psbt, tx);
});
describe('create nonce', function () {
it(`update with new nonce should be allowed`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
let noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 1);
const userNonceKey = noncesKeyVals[0].key.keydata;
const userNonceValue = noncesKeyVals[0].value;
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 2);
noncesKeyVals = noncesKeyVals.filter((kv) => kv.key.keydata.equals(userNonceKey));
assert.strictEqual(noncesKeyVals.length, 1);
assert.ok(!noncesKeyVals[0].value.equals(userNonceValue));
});
it(`Cosigner nonce creation fail should not enforce the signer to recreate nonce`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
const tapBip32Derivation = psbt.data.inputs[0].tapBip32Derivation;
psbt.data.inputs[0].tapBip32Derivation = undefined;
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo), (e) => e.message === 'tapBip32Derivation is required to create nonce');
psbt.data.inputs[0].tapBip32Derivation = tapBip32Derivation;
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
const noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 2);
const partialSigKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
});
assert.strictEqual(partialSigKeyVals.length, 2);
const participantKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
});
assert.strictEqual(participantKeyVals.length, 1);
});
it('Cosigner can create a deterministic nonce', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
const noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 2);
});
it('Cosigner cannot create a deterministic nonce if there is no signer nonce', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true }), (e) => e.message === 'No nonces found on input #0');
let noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 0);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true }), (e) => e.message === 'signer nonce must be set if cosigner nonce is to be derived deterministically');
noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 1);
});
it('Cosigner cannot add entropy to deterministic nonce creation', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, {
deterministic: true,
sessionId: Buffer.allocUnsafe(32),
}), (e) => e.message === 'Cannot add extra entropy when generating a deterministic nonce');
const noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 1);
});
it('Signer cannot create a deterministic nonce', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user, { deterministic: true }), (e) => e.message === `Only the cosigner's nonce can be set deterministically`);
const noncesKeyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(noncesKeyVals.length, 0);
});
it(`skipped if tapInternalKey doesn't match participant pub keys agg`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.data.inputs[0].tapInternalKey = Musig2Util_1.dummyTapInternalKey;
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'tapInternalKey and aggregated participant pub keys does not match');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if sessionId size is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user, { sessionId: Buffer.allocUnsafe(33) }), (e) => e.message === 'Invalid sessionId size 33');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if private key is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user.neutered()), (e) => e.message === 'private key is required to generate nonce');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if tapBip32Derivation is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.data.inputs[0].tapBip32Derivation = [];
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'tapBip32Derivation is required to create nonce');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if participant pub keys is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.data.inputs[0].unknownKeyVals = [];
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'Found 0 matching participant key value instead of 1');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);
});
it(`fails if participant pub keys keydata size is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].key.keydata = Buffer.concat([keyVals[0].key.keydata, Buffer.from('dummy')]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid keydata size ${keyVals[0].key.keydata.length} for participant pub keys`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if participant keydata tapOutputKey in invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].key.keydata = Buffer.concat([Musig2Util_1.dummyTapOutputKey, keyVals[0].key.keydata.subarray(32)]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid participants keydata tapOutputKey`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if participant keydata tapInternalKey in invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].key.keydata = Buffer.concat([keyVals[0].key.keydata.subarray(0, 32), Musig2Util_1.dummyTapInternalKey]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid participants keydata tapInternalKey`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if tapInternalKey and aggregated participant pub keys don't match`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getInternalChainCode)('p2trMusig2'), 1);
const tapInternalKey = (0, Musig2_1.createTapInternalKey)([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);
const tapOutputKey = (0, Musig2_1.createTapOutputKey)(tapInternalKey, psbt.data.inputs[0].tapMerkleRoot);
keyVals[0].key.keydata = Buffer.concat([tapOutputKey, tapInternalKey]);
keyVals[0].value = Buffer.concat([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `tapInternalKey and aggregated participant pub keys does not match`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if keydata size of participant pub keys is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].key.keydata = Buffer.allocUnsafe(65);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid keydata size 65 for participant pub keys`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if valuedata size of participant pub keys is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].value = Buffer.allocUnsafe(67);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid valuedata size 67 for participant pub keys`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if duplicate participant pub keys found`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
keyVals[0].value = Buffer.concat([keyVals[0].value.subarray(33), keyVals[0].value.subarray(33)]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Duplicate participant pub keys found`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if no fingerprint match`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.data.inputs[0].tapBip32Derivation?.forEach((bv) => (bv.masterFingerprint = Buffer.allocUnsafe(4)));
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'No bip32Derivation masterFingerprint matched the HD keyPair fingerprint');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if pubkey did not match tapBip32Derivation`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getInternalChainCode)('p2trMusig2'), CHANGE_INDEX);
psbt.data.inputs[0].tapBip32Derivation?.forEach((bv) => {
bv.path = walletKeys.paths[2];
});
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'pubkey did not match bip32Derivation');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if root wallet key derive more than one tapBip32Derivation`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
psbt.data.inputs[0].tapBip32Derivation?.forEach((bv, index) => {
bv.path = walletKeys.paths[0];
bv.pubkey = (0, outputScripts_1.toXOnlyPublicKey)(walletKeys.publicKeys[0]);
bv.masterFingerprint = Musig2Util_1.rootWalletKeys.user.fingerprint;
});
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message.startsWith('more than one matching derivation for fingerprint'));
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
it(`fails if derived wallet key does not match any participant key`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const keyVals = psbt.getProprietaryKeyVals(0);
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getInternalChainCode)('p2trMusig2'), 1);
const tapInternalKey = (0, Musig2_1.createTapInternalKey)([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);
psbt.data.inputs[0].tapInternalKey = tapInternalKey;
keyVals[0].value = Buffer.concat([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);
const tapOutputKey = (0, Musig2_1.createTapOutputKey)(tapInternalKey, psbt.data.inputs[0].tapMerkleRoot);
keyVals[0].key.keydata = Buffer.concat([tapOutputKey, tapInternalKey]);
psbt.data.inputs[0].unknownKeyVals = [];
psbt.addProprietaryKeyValToInput(0, keyVals[0]);
assert.throws(() => psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user), (e) => e.message === `participant plain pub key should match one bip32Derivation plain pub key`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);
});
});
describe('sign', function () {
it(`fails if privateKey is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user.neutered()), (e) => e.message === 'privateKey is required to sign p2tr musig2');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if tapInternalKey is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.data.inputs[0].tapInternalKey = undefined;
assert.throws(() => psbt.signTaprootMusig2Input(0, {
publicKey: walletKeys.user.publicKey,
privateKey: walletKeys.user.privateKey,
}), (e) => e.message === 'not a taproot musig2 input');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it('only the cosigner can add a deterministic signature', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
assert.throws(() => psbt.signInputHD(0, Musig2Util_1.rootWalletKeys.user, { deterministic: true }), (e) => e.message === 'can only add a deterministic signature on the cosigner');
});
it('cosigner can sign deterministically', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: true });
assert.ok(psbt.validateSignaturesOfAllInputs());
assert.ok(psbt.finalizeAllInputs());
});
it('cosigner can sign non-deterministically', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: false });
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo, { deterministic: false });
assert.ok(psbt.validateSignaturesOfAllInputs());
assert.ok(psbt.finalizeAllInputs());
});
});
it(`fails if tapMerkleRoot is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.data.inputs[0].tapMerkleRoot = undefined;
assert.throws(() => psbt.signTaprootMusig2Input(0, {
publicKey: walletKeys.user.publicKey,
privateKey: walletKeys.user.privateKey,
}), (e) => e.message === 'not a taproot musig2 input');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if participant pub keys is missing`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.data.inputs[0].unknownKeyVals = [];
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'Found 0 matching participant key value instead of 1');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);
});
it(`fails if signer pub key is not matching any participant pub keys`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
assert.throws(() => psbt.signTaprootMusig2Input(0, {
privateKey: Musig2Util_1.rootWalletKeys.backup.privateKey,
publicKey: Musig2Util_1.rootWalletKeys.backup.publicKey,
}), (e) => e.message === 'signer pub key should match one of participant pub keys');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if more than 2 nonce key value exists`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.data.inputs[0].unknownKeyVals?.push(psbt.data.inputs[0].unknownKeyVals[2]);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'Found 3 matching nonce key value instead of 1 or 2');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 4);
});
it(`fails if 2 nonce key value do not exist`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.data.inputs[0].unknownKeyVals?.splice(2);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === 'Found 1 matching nonce key value instead of 2');
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 2);
});
it(`fails if nonce keydata size is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
const keyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
keyVals[1].key.keydata = Buffer.concat([keyVals[1].key.keydata, Buffer.from('dummy')]);
psbt.data.inputs[0].unknownKeyVals?.splice(2);
psbt.addProprietaryKeyValToInput(0, keyVals[1]);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid keydata size ${keyVals[1].key.keydata.length} for nonce`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if nonce valuedata size is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
const keyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
keyVals[1].value = Buffer.concat([keyVals[1].value, Buffer.from('dummy')]);
psbt.data.inputs[0].unknownKeyVals?.splice(2);
psbt.addProprietaryKeyValToInput(0, keyVals[1]);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid valuedata size ${keyVals[1].value.length} for nonce`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if nonce keydata is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
const dummyRootWalletKeys = new bitgo_1.RootWalletKeys((0, testutil_1.getKeyTriple)('dummy'));
const dummyP2trMusig2Unspent = (0, Musig2Util_1.getUnspents)(['p2trMusig2'], dummyRootWalletKeys);
const dummyPsbt = (0, Musig2Util_1.constructPsbt)(dummyP2trMusig2Unspent, dummyRootWalletKeys, 'user', 'bitgo', 'p2sh');
dummyPsbt.setAllInputsMusig2NonceHD(dummyRootWalletKeys.user);
dummyPsbt.setAllInputsMusig2NonceHD(dummyRootWalletKeys.bitgo);
const dummyKeyVals = dummyPsbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
psbt.data.inputs[0].unknownKeyVals?.splice(1);
dummyKeyVals.forEach((kv, i) => psbt.addProprietaryKeyValToInput(0, dummyKeyVals[i]));
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid nonce keydata participant pub key`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
it(`fails if nonce keydata tapOutputKey is invalid`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', 'p2sh');
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
const keyVals = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
keyVals[1].key.keydata = Buffer.concat([keyVals[1].key.keydata.subarray(0, 33), Musig2Util_1.dummyTapOutputKey]);
psbt.data.inputs[0].unknownKeyVals?.splice(2);
psbt.addProprietaryKeyValToInput(0, keyVals[1]);
assert.throws(() => psbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Invalid nonce keydata tapOutputKey`);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);
});
});
describe('p2trMusig2 script path', function () {
it(`psbt creation success and musig2 skips`, function () {
let psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'backup', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.backup);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.backup);
(0, Musig2Util_1.validatePsbtP2trMusig2Input)(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');
(0, Musig2Util_1.validatePsbtP2trMusig2Output)(psbt, 0);
assert.ok(psbt.validateSignaturesOfAllInputs());
psbt.finalizeAllInputs();
(0, Musig2Util_1.validateFinalizedInput)(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');
let tx = psbt.extractTransaction();
assert.ok((0, testutil_1.verifyFullySignedSignatures)(tx, p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'backup'));
psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'bitgo', 'backup', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.backup);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.backup);
assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);
(0, Musig2Util_1.validatePsbtP2trMusig2Input)(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');
(0, Musig2Util_1.validatePsbtP2trMusig2Output)(psbt, 0);
assert.ok(psbt.validateSignaturesOfAllInputs());
psbt.finalizeAllInputs();
(0, Musig2Util_1.validateFinalizedInput)(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');
tx = psbt.extractTransaction();
assert.ok((0, testutil_1.verifyFullySignedSignatures)(tx, p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'bitgo', 'backup'));
});
it(`parse tx`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'backup', outputType);
(0, Musig2Util_1.validateParsedTaprootScriptPathPsbt)(psbt, 0, 'unsigned');
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
(0, Musig2Util_1.validateParsedTaprootScriptPathPsbt)(psbt, 0, 'halfsigned');
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.backup);
(0, Musig2Util_1.validateParsedTaprootScriptPathPsbt)(psbt, 0, 'fullysigned');
psbt.finalizeAllInputs();
const tx = psbt.extractTransaction();
const psbtDuplicate = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'backup', outputType);
(0, Musig2Util_1.validateParsedTaprootScriptPathTxInput)(psbtDuplicate, tx, 0);
});
});
describe('validate p2tr Musig2 signatures', function () {
it(`validate with pubkey`, function () {
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
let psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2Nonce(walletKeys.user);
psbt.setAllInputsMusig2Nonce(walletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.user.publicKey));
psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setInputMusig2Nonce(0, walletKeys.user);
psbt.setInputMusig2NonceHD(0, Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.user.publicKey));
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey));
});
it(`fails if no sig`, function () {
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
assert.throws(() => psbt.validateSignaturesOfAllInputs(), (e) => e.message === `No signatures to validate`);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
assert.throws(() => psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey), (e) => e.message === `No signatures for this pubkey`);
});
it(`fails if no tapInternalKey and tapMerkleRoot`, function () {
const walletKeys = Musig2Util_1.rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
const tapInternalKey = psbt.data.inputs[0].tapInternalKey;
psbt.data.inputs[0].tapInternalKey = undefined;
assert.throws(() => psbt.validateTaprootMusig2SignaturesOfInput(0), (e) => e.message === `both tapInternalKey and tapMerkleRoot are required`);
psbt.data.inputs[0].tapInternalKey = tapInternalKey;
psbt.data.inputs[0].tapMerkleRoot = undefined;
assert.throws(() => psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey), (e) => e.message === `both tapInternalKey and tapMerkleRoot are required`);
});
it(`fails if no nonce and sig pub key match`, function () {
let psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
const partialSigs = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
});
const myRootWalletKeys = new bitgo_1.RootWalletKeys((0, testutil_1.getKeyTriple)('dummy'));
const myUnspents = (0, Musig2Util_1.getUnspents)(['p2trMusig2'], myRootWalletKeys);
psbt = (0, Musig2Util_1.constructPsbt)(myUnspents, myRootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(myRootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(myRootWalletKeys.bitgo);
psbt.signAllInputsHD(myRootWalletKeys.user);
psbt.signAllInputsHD(myRootWalletKeys.bitgo);
const participants = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
});
const nonces = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
psbt.data.inputs[0].unknownKeyVals = undefined;
psbt.addProprietaryKeyValToInput(0, participants[0]);
psbt.addProprietaryKeyValToInput(0, nonces[0]);
psbt.addProprietaryKeyValToInput(0, nonces[1]);
psbt.addProprietaryKeyValToInput(0, partialSigs[0]);
psbt.addProprietaryKeyValToInput(0, partialSigs[1]);
assert.throws(() => psbt.validateSignaturesOfAllInputs(), (e) => e.message === `Found no pub nonce for pubkey`);
});
it(`fails if no valid sig`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.bitgo);
const partialSigs = psbt.getProprietaryKeyVals(0, {
identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER,
subtype: bitgo_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
});
partialSigs[1].value = Musig2Util_1.dummyPartialSig;
psbt.addOrUpdateProprietaryKeyValToInput(0, partialSigs[1]);
assert.ok(!psbt.validateSignaturesOfAllInputs());
});
});
describe('finalizeTaprootMusig2Input', function () {
it('fails if invalid number for sigs', function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
psbt.signAllInputsHD(Musig2Util_1.rootWalletKeys.user);
assert.throws(() => psbt.finalizeAllInputs(), (e) => e.message === `invalid number of partial signatures 1 to finalize`);
});
});
describe('Psbt musig2 common functions', function () {
it('output script should match the scriptPubKey in the prevout', function () {
const myRootWalletKeys = new bitgo_1.RootWalletKeys((0, testutil_1.getKeyTriple)('dummy'));
const unspents = (0, Musig2Util_1.getUnspents)(outputScripts_1.scriptTypes2Of3.map((t) => t), myRootWalletKeys);
const psbt = (0, Musig2Util_1.constructPsbt)(unspents, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
unspents.forEach((u, index) => {
const scriptType = (0, bitgo_1.scriptTypeForChain)(u.chain);
assert.throws(() => scriptType === 'p2trMusig2'
? psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user)
: scriptType === 'p2tr'
? psbt.signTaprootInputHD(index, Musig2Util_1.rootWalletKeys.user)
: psbt.signInputHD(index, Musig2Util_1.rootWalletKeys.user), (e) => (0, bitgo_1.isSegwit)(u.chain) && scriptType !== 'p2shP2wsh'
? e.message === `Witness script for input #${index} doesn't match the scriptPubKey in the prevout`
: e.message === `Redeem script for input #${index} doesn't match the scriptPubKey in the prevout`);
});
const p2trMusig2ScriptPathPsbt = (0, Musig2Util_1.constructPsbt)([unspents[4]], Musig2Util_1.rootWalletKeys, 'user', 'backup', outputType);
assert.throws(() => p2trMusig2ScriptPathPsbt.signTaprootInputHD(0, Musig2Util_1.rootWalletKeys.user), (e) => e.message === `Witness script for input #0 doesn't match the scriptPubKey in the prevout`);
});
it(`decodePsbtMusig2ParticipantsKeyValData fails if invalid subtype or identifier is passed`, function () {
const kv = {
key: {
identifier: 'dummy',
subtype: 0x05,
keydata: Buffer.allocUnsafe(1),
},
value: Buffer.allocUnsafe(1),
};
assert.throws(() => (0, Musig2_1.decodePsbtMusig2Participants)(kv), (e) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`);
kv.key.identifier = bitgo_1.PSBT_PROPRIETARY_IDENTIFIER;
assert.throws(() => (0, Musig2_1.decodePsbtMusig2Participants)(kv), (e) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`);
});
it(`decodePsbtMusig2NonceKeyValData fails if invalid subtype or identifier is passed`, function () {
const kv = {
key: {
identifier: 'dummy',
subtype: 0x05,
keydata: Buffer.allocUnsafe(1),
},
value: Buffer.allocUnsafe(1),
};
assert.throws(() => (0, Musig2_1.decodePsbtMusig2Nonce)(kv), (e) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`);
kv.key.identifier = bitgo_1.PSBT_PROPRIETARY_IDENTIFIER;
assert.throws(() => (0, Musig2_1.decodePsbtMusig2Nonce)(kv), (e) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`);
});
it(`validatePsbtMusig2NoncesKeyValData fails if participant pub keys are duplicate`, function () {
const nonceKeyValData = [0, 1].map((i) => ({
participantPubKey: Musig2Util_1.dummyParticipantPubKeys[i],
tapOutputKey: Musig2Util_1.dummyTapOutputKey,
pubNonce: Musig2Util_1.dummyPubNonce,
}));
let participantKeyValData = {
participantPubKeys: Musig2Util_1.dummyParticipantPubKeys,
tapInternalKey: Musig2Util_1.dummyTapInternalKey,
tapOutputKey: Musig2Util_1.invalidTapOutputKey,
};
assert.throws(() => (0, Musig2_1.assertPsbtMusig2Nonces)(nonceKeyValData, participantKeyValData), (e) => e.message === `invalid size 1. Must use x-only key.`);
participantKeyValData = {
participantPubKeys: [Musig2Util_1.invalidParticipantPubKeys[0], Musig2Util_1.dummyParticipantPubKeys[0]],
tapInternalKey: Musig2Util_1.dummyTapInternalKey,
tapOutputKey: Musig2Util_1.dummyTapOutputKey,
};
assert.throws(() => (0, Musig2_1.assertPsbtMusig2Nonces)(nonceKeyValData, participantKeyValData), (e) => e.message === `invalid size 1. Must use plain key.`);
participantKeyValData = {
participantPubKeys: [Musig2Util_1.dummyParticipantPubKeys[0], Musig2Util_1.dummyParticipantPubKeys[0]],
tapInternalKey: Musig2Util_1.dummyTapInternalKey,
tapOutputKey: Musig2Util_1.dummyTapOutputKey,
};
assert.throws(() => (0, Musig2_1.assertPsbtMusig2Nonces)(nonceKeyValData, participantKeyValData), (e) => e.message === `Duplicate participant pub keys found`);
});
it(`createTapTweak fails if invalid tapInternalKey or tapMerkleRoot is passed`, function () {
assert.throws(() => (0, Musig2_1.createTapTweak)(Musig2Util_1.invalidTapInputKey, Musig2Util_1.dummyTapOutputKey), (e) => e.message === `invalid size 1. Must use x-only key.`);
assert.throws(() => (0, Musig2_1.createTapTweak)(Musig2Util_1.dummyTapInternalKey, Musig2Util_1.invalidTapOutputKey), (e) => e.message === `invalid size 1. Must use tap merkle root.`);
});
it(`musig2PartialSign fails if invalid txHash is passed`, function () {
assert.throws(() => (0, Musig2_1.musig2PartialSign)(Musig2Util_1.dummyPrivateKey, Musig2Util_1.dummyPubNonce, {
publicKey: Musig2Util_1.dummyParticipantPubKeys[0],
aggNonce: Musig2Util_1.dummyAggNonce,
msg: Musig2Util_1.invalidTxHash,
}, new Musig2_1.Musig2NonceStore()), (e) => e.message === `invalid size 1. Must use tx hash.`);
});
it(`encodePsbtMusig2PartialSigKeyKeyValData fails if invalid txHash is passed`, function () {
assert.throws(() => (0, Musig2_1.encodePsbtMusig2PartialSig)({
partialSig: Musig2Util_1.invalidPartialSig,
participantPubKey: Musig2Util_1.dummyParticipantPubKeys[0],
tapOutputKey: Musig2Util_1.dummyTapOutputKey,
}), (e) => e.message === `Invalid partialSig length 1`);
});
it(`deleteProprietaryKeyVals`, function () {
const psbt = (0, Musig2Util_1.constructPsbt)(p2trMusig2Unspent, Musig2Util_1.rootWalletKeys, 'user', 'bitgo', outputType);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(Musig2Util_1.rootWalletKeys.bitgo);
const key = {
identifier: 'DUMMY',
subtype: 100,
keydata: Musig2Util_1.dummyTapOutputKey,
};
psbt.addProprietaryKeyValToInput(0, { key, value: Musig2Util_1.dummyTapInternalKey });
psbt.deleteProprietaryKeyVals(0, { identifier: bitgo_1.PSBT_PROPRIETARY_IDENTIFIER });
const keyVal = psbt.getProprietaryKeyVals(0);
assert.strictEqual(keyVal.length, 1);
assert.strictEqual(keyVal[0].key.identifier, 'DUMMY');
});
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Musig2.js","sourceRoot":"","sources":["../../../../test/bitgo/psbt/Musig2.ts"],"names":[],"mappings":";;AAAA,iCAAiC;AAEjC,8CAY4B;AAE5B,oDAAkF;AAClF,sDAUmC;AACnC,oEAAqF;AACrF,6CA2BsB;AAEtB,MAAM,iBAAiB,GAAG,IAAA,wBAAW,EAAC,CAAC,YAAY,CAAC,EAAE,2BAAc,CAAC,CAAC;AACtE,MAAM,UAAU,GAAG,YAAY,CAAC;AAChC,MAAM,YAAY,GAAG,GAAG,CAAC;AAEzB,QAAQ,CAAC,YAAY,EAAE;IACrB,QAAQ,CAAC,qBAAqB,EAAE;QAC9B,EAAE,CAAC,uDAAuD,EAAE;YAC1D,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,IAAA,4BAAoB,EAAC,YAAY,CAAC,EAAE,CAAC,CAAC,CAAC;YAChG,MAAM,QAAQ,GAAG,IAAA,wBAAW,EAC1B,+BAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAC7B,2BAAc,CACf,CAAC;YACF,8EAA8E;YAC9E,MAAM,SAAS,GAAG,IAAA,0BAAa,EAAC,QAAQ,EAAE,2BAAc,EAAE,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,CAAC;YACvF,MAAM,YAAY,GAAG,SAAS,CAAC,KAAK,EAAE,CAAC;YAEvC,+GAA+G;YAC/G,MAAM,QAAQ,GAAG,IAAA,yBAAiB,EAAC,YAAY,EAAE,oBAAO,CAAC,CAAC;YAC1D,QAAQ,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACxD,MAAM,WAAW,GAAG,QAAQ,CAAC,KAAK,EAAE,CAAC;YAErC,sGAAsG;YACtG,MAAM,SAAS,GAAG,IAAA,yBAAiB,EAAC,WAAW,EAAE,oBAAO,CAAC,CAAC;YAC1D,SAAS,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;YACnF,MAAM,YAAY,GAAG,SAAS,CAAC,KAAK,EAAE,CAAC;YAEvC,iGAAiG;YACjG,MAAM,cAAc,GAAG,IAAA,yBAAiB,EAAC,YAAY,EAAE,oBAAO,CAAC,CAAC;YAChE,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YACjC,QAAQ,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC9C,MAAM,qBAAqB,GAAG,QAAQ,CAAC,KAAK,EAAE,CAAC;YAE/C,mDAAmD;YACnD,MAAM,WAAW,GAAG,IAAA,yBAAiB,EAAC,qBAAqB,EAAE,oBAAO,CAAC,CAAC;YACtE,MAAM,CAAC,EAAE,CAAC,WAAW,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;YAE5F,gEAAgE;YAChE,MAAM,IAAI,GAAG,IAAA,yBAAiB,EAAC,qBAAqB,EAAE,oBAAO,CAAC,CAAC;YAC/D,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;YAEpE,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE;gBAClC,IAAI,IAAA,0BAAkB,EAAC,OAAO,CAAC,KAAK,CAAC,KAAK,YAAY,EAAE,CAAC;oBACvD,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;oBAChE,OAAO;gBACT,CAAC;gBACD,IAAA,wCAA2B,EAAC,IAAI,EAAE,KAAK,EAAE,OAAO,EAAE,SAAS,CAAC,CAAC;gBAC7D,IAAA,yCAA4B,EAAC,IAAI,EAAE,CAAC,CAAC,CAAC;gBACtC,IAAA,wCAA2B,EAAC,IAAI,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;gBAClD,IAAA,kCAAqB,EAAC,IAAI,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;gBAC5C,IAAA,sCAAyB,EAAC,IAAI,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;YAEH,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;YAChD,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE;gBAClC,IAAA,mCAAsB,EAAC,IAAI,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;YAC/C,CAAC,CAAC,CAAC;YACH,MAAM,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAA6B,CAAC;YAChE,MAAM,CAAC,EAAE,CAAC,IAAA,sCAA2B,EAAC,EAAE,EAAE,QAAQ,EAAE,2BAAc,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC;YACtF,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE;gBACnC,MAAM,CAAC,eAAe,CAAC,IAAA,kCAA0B,EAAC,EAAE,EAAE,UAAU,EAAE,QAAQ,EAAE,2BAAc,CAAC,EAAE;oBAC3F,IAAI;oBACJ,KAAK;oBACL,IAAI;iBACL,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,UAAU,EAAE;YACb,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,OAAO,EAAE,MAAM,EAAE,UAAU,CAAC,CAAC;YAC3F,IAAA,6CAAgC,EAAC,IAAI,EAAE,CAAC,EAAE,UAAU,CAAC,CAAC;YAEtD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAA,6CAAgC,EAAC,IAAI,EAAE,CAAC,EAAE,YAAY,CAAC,CAAC;YAExD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAC3C,IAAA,6CAAgC,EAAC,IAAI,EAAE,CAAC,EAAE,aAAa,CAAC,CAAC;YAEzD,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,sBAAc,EAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EACzC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,yCAAyC,CACpE,CAAC;YAEF,MAAM,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAA6B,CAAC;YAChE,IAAA,gDAAmC,EAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,cAAc,EAAE;YACvB,EAAE,CAAC,yCAAyC,EAAE;gBAC5C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBAEvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAEpD,IAAI,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAChD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAC5C,MAAM,YAAY,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC;gBAClD,MAAM,cAAc,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;gBAE9C,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBACrD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAEpD,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAC5C,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAE5C,aAAa,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;gBAClF,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAC5C,MAAM,CAAC,EAAE,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC;YAC5D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,8EAA8E,EAAE;gBACjF,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBAEvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAEpD,MAAM,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,CAAC;gBAClE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,GAAG,SAAS,CAAC;gBAEnD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,EAC1D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,gDAAgD,CAC3E,CAAC;gBAEF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,GAAG,kBAAkB,CAAC;gBAE5D,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBAErD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBAE3C,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAClD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAE5C,MAAM,iBAAiB,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBACtD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,kBAAkB;iBAClD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAEhD,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBACvD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,2BAA2B;iBAC3D,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,kBAAkB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACnD,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,2CAA2C,EAAE;gBAC9C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBAE9E,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAClD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,0EAA0E,EAAE;gBAC7E,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBAEvF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EACnF,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,6BAA6B,CACxD,CAAC;gBAEF,IAAI,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAChD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAE5C,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBACrD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EACnF,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,+EAA+E,CAC1G,CAAC;gBAEF,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAC5C,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,6DAA6D,EAAE;gBAChE,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE;oBACnD,aAAa,EAAE,IAAI;oBACnB,SAAS,EAAE,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;iBAClC,CAAC,EACJ,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,gEAAgE,CAC3F,CAAC;gBACF,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAClD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4CAA4C,EAAE;gBAC/C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EAClF,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,wDAAwD,CACnF,CAAC;gBACF,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;oBAClD,UAAU,EAAE,mCAA2B;oBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;iBAChD,CAAC,CAAC;gBACH,MAAM,CAAC,WAAW,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9C,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,kEAAkE,EAAE;gBACrE,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,gCAAmB,CAAC;gBACzD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,mEAAmE,CAC9F,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,oCAAoC,EAAE;gBACvC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,EAAE,CAAC,EAChG,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2BAA2B,CACtD,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,iCAAiC,EAAE;gBACpC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EACpE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2CAA2C,CACtE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wCAAwC,EAAE;gBAC3C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,GAAG,EAAE,CAAC;gBAC5C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,gDAAgD,CAC3E,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,0CAA0C,EAAE;gBAC7C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,qDAAqD,CAChF,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,uDAAuD,EAAE;gBAC1D,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,wBAAwB,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,2BAA2B,CAC3G,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,sDAAsD,EAAE;gBACzD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,8BAAiB,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;gBACjG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2CAA2C,CACtE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wDAAwD,EAAE;gBAC3D,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,gCAAmB,CAAC,CAAC,CAAC;gBACtG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,6CAA6C,CACxE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,yEAAyE,EAAE;gBAC5E,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAE9C,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,IAAA,4BAAoB,EAAC,YAAY,CAAC,EAAE,CAAC,CAAC,CAAC;gBAChG,MAAM,cAAc,GAAG,IAAA,6BAAoB,EAAC,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;gBACrG,MAAM,YAAY,GAAG,IAAA,2BAAkB,EAAC,cAAc,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,aAAc,CAAC,CAAC;gBAE5F,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,EAAE,cAAc,CAAC,CAAC,CAAC;gBACvE,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;gBAE1F,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,mEAAmE,CAC9F,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,0DAA0D,EAAE;gBAC7D,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;gBAChD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,kDAAkD,CAC7E,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,4DAA4D,EAAE;gBAC/D,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;gBAC1C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oDAAoD,CAC/E,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,+CAA+C,EAAE;gBAClD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;gBACjG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sCAAsC,CACjE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,+BAA+B,EAAE;gBAClC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,iBAAiB,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBACxG,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,yEAAyE,CACpG,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,kDAAkD,EAAE;gBACrD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,IAAA,4BAAoB,EAAC,YAAY,CAAC,EAAE,YAAY,CAAC,CAAC;gBAC3G,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;oBACrD,EAAE,CAAC,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAChC,CAAC,CAAC,CAAC;gBACH,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sCAAsC,CACjE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,kEAAkE,EAAE;gBACrE,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CACtD,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAC1B,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAC3B,CAAC;gBACF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAC,EAAE,EAAE,KAAK,EAAE,EAAE;oBAC5D,EAAE,CAAC,IAAI,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;oBAC9B,EAAE,CAAC,MAAM,GAAG,IAAA,gCAAgB,EAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;oBACvD,EAAE,CAAC,iBAAiB,GAAG,2BAAc,CAAC,IAAI,CAAC,WAAW,CAAC;gBACzD,CAAC,CAAC,CAAC;gBACH,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,mDAAmD,CAAC,CACtF,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,gEAAgE,EAAE;gBACnE,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAE9C,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,IAAA,4BAAoB,EAAC,YAAY,CAAC,EAAE,CAAC,CAAC,CAAC;gBAChG,MAAM,cAAc,GAAG,IAAA,6BAAoB,EAAC,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;gBACrG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,cAAc,CAAC;gBAEpD,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;gBAC1F,MAAM,YAAY,GAAG,IAAA,2BAAkB,EAAC,cAAc,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,aAAc,CAAC,CAAC;gBAC5F,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,YAAY,EAAE,cAAc,CAAC,CAAC,CAAC;gBACvE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;gBACxC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAEhD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,EACzD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,0EAA0E,CACrG,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,MAAM,EAAE;YACf,EAAE,CAAC,gCAAgC,EAAE;gBACnC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBACrD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EAChE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,4CAA4C,CACvE,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,oCAAoC,EAAE;gBACvC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBAEvF,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CACtD,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAC1B,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAC3B,CAAC;gBAEF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;gBAErD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,SAAS,CAAC;gBAC/C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAI,CAAC,sBAAsB,CAAC,CAAC,EAAE;oBAC7B,SAAS,EAAE,UAAU,CAAC,IAAI,CAAC,SAAS;oBACpC,UAAU,EAAE,UAAU,CAAC,IAAI,CAAC,UAAW;iBACxC,CAAC,EACJ,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,4BAA4B,CACvD,CAAC;gBACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,qDAAqD,EAAE;gBACxD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBAC9E,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EACvE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,wDAAwD,CACnF,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,qCAAqC,EAAE;gBACxC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBAC9E,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBACpE,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;gBAChD,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;YACtC,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,yCAAyC,EAAE;gBAC5C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;gBACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC/E,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;gBAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,CAAC,CAAC;gBACrE,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;gBAChD,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;YACtC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mCAAmC,EAAE;YACtC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YAEvF,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YAEjH,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,aAAa,GAAG,SAAS,CAAC;YAC9C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAI,CAAC,sBAAsB,CAAC,CAAC,EAAE;gBAC7B,SAAS,EAAE,UAAU,CAAC,IAAI,CAAC,SAAS;gBACpC,UAAU,EAAE,UAAU,CAAC,IAAI,CAAC,UAAW;aACxC,CAAC,EACJ,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,4BAA4B,CACvD,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE;YAC7C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,EAAE,CAAC;YACxC,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,qDAAqD,CAChF,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kEAAkE,EAAE;YACrE,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAI,CAAC,sBAAsB,CAAC,CAAC,EAAE;gBAC7B,UAAU,EAAE,2BAAc,CAAC,MAAM,CAAC,UAAW;gBAC7C,SAAS,EAAE,2BAAc,CAAC,MAAM,CAAC,SAAU;aAC5C,CAAC,EACJ,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,yDAAyD,CACpF,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6CAA6C,EAAE;YAChD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;YAChF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oDAAoD,CAC/E,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE;YAC5C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;YAC9C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,+CAA+C,CAC1E,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,wCAAwC,EAAE;YAC3C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAC5C,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;aAChD,CAAC,CAAC;YACH,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YACvF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;YAC9C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,wBAAwB,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,YAAY,CAC5F,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE;YAC7C,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAC5C,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;aAChD,CAAC,CAAC;YACH,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YAC3E,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;YAC9C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,0BAA0B,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,YAAY,CACxF,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mCAAmC,EAAE;YACtC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,MAAM,mBAAmB,GAAG,IAAI,sBAAc,CAAC,IAAA,uBAAY,EAAC,OAAO,CAAC,CAAC,CAAC;YACtE,MAAM,sBAAsB,GAAG,IAAA,wBAAW,EAAC,CAAC,YAAY,CAAC,EAAE,mBAAmB,CAAC,CAAC;YAChF,MAAM,SAAS,GAAG,IAAA,0BAAa,EAAC,sBAAsB,EAAE,mBAAmB,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACtG,SAAS,CAAC,yBAAyB,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAC;YAC9D,SAAS,CAAC,yBAAyB,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;YAE/D,MAAM,YAAY,GAAG,SAAS,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBACtD,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;aAChD,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;YAC9C,YAAY,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACtF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2CAA2C,CACtE,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gDAAgD,EAAE;YACnD,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAC5C,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;aAChD,CAAC,CAAC;YAEH,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,8BAAiB,CAAC,CAAC,CAAC;YAEpG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;YAC9C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACrD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oCAAoC,CAC/D,CAAC;YACF,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,wBAAwB,EAAE;QACjC,EAAE,CAAC,wCAAwC,EAAE;YAC3C,IAAI,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAC1F,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,MAAM,CAAC,CAAC;YACtD,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC5D,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,MAAM,CAAC,CAAC;YAC5C,IAAA,wCAA2B,EAAC,IAAI,EAAE,CAAC,EAAE,iBAAiB,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YACzE,IAAA,yCAA4B,EAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACtC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;YAChD,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAA,mCAAsB,EAAC,IAAI,EAAE,CAAC,EAAE,iBAAiB,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YACpE,IAAI,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAA6B,CAAC;YAC9D,MAAM,CAAC,EAAE,CAAC,IAAA,sCAA2B,EAAC,EAAE,EAAE,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC,CAAC;YAEhG,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,OAAO,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YACvF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,MAAM,CAAC,CAAC;YACtD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAC3C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,MAAM,CAAC,CAAC;YAC5C,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC5D,IAAA,wCAA2B,EAAC,IAAI,EAAE,CAAC,EAAE,iBAAiB,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YACzE,IAAA,yCAA4B,EAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACtC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;YAChD,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAA,mCAAsB,EAAC,IAAI,EAAE,CAAC,EAAE,iBAAiB,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YACpE,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAA6B,CAAC;YAC1D,MAAM,CAAC,EAAE,CAAC,IAAA,sCAA2B,EAAC,EAAE,EAAE,iBAAiB,EAAE,2BAAc,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC;QACnG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,UAAU,EAAE;YACb,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAC5F,IAAA,gDAAmC,EAAC,IAAI,EAAE,CAAC,EAAE,UAAU,CAAC,CAAC;YAEzD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAA,gDAAmC,EAAC,IAAI,EAAE,CAAC,EAAE,YAAY,CAAC,CAAC;YAE3D,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,MAAM,CAAC,CAAC;YAC5C,IAAA,gDAAmC,EAAC,IAAI,EAAE,CAAC,EAAE,aAAa,CAAC,CAAC;YAE5D,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,MAAM,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAA6B,CAAC;YAEhE,MAAM,aAAa,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YACrG,IAAA,mDAAsC,EAAC,aAAa,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;QAC/D,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iCAAiC,EAAE;QAC1C,EAAE,CAAC,sBAAsB,EAAE;YACzB,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YACjH,IAAI,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YACzF,IAAI,CAAC,uBAAuB,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YAC9C,IAAI,CAAC,uBAAuB,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;YAC/C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;YAErF,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YACrF,IAAI,CAAC,mBAAmB,CAAC,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,CAAC;YAC7C,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE,2BAAc,CAAC,KAAK,CAAC,CAAC;YACpD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;YACrF,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAC3C,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;QACxF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iBAAiB,EAAE;YACpB,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YACjH,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAC3F,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,EAC1C,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2BAA2B,CACtD,CAAC;YAEF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAErD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAE1C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,EAChF,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,+BAA+B,CAC1D,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,8CAA8C,EAAE;YACjD,MAAM,UAAU,GAAG,2BAAc,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YACjH,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAC3F,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAE3C,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC;YAC1D,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,SAAS,CAAC;YAC/C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,CAAC,EACpD,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oDAAoD,CAC/E,CAAC;YAEF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,cAAc,CAAC;YACpD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,aAAa,GAAG,SAAS,CAAC;YAC9C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC,EAAE,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,EAChF,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oDAAoD,CAC/E,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE;YAC5C,IAAI,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YACzF,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAE3C,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAChD,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,kBAAkB;aAClD,CAAC,CAAC;YAEH,MAAM,gBAAgB,GAAG,IAAI,sBAAc,CAAC,IAAA,uBAAY,EAAC,OAAO,CAAC,CAAC,CAAC;YACnE,MAAM,UAAU,GAAG,IAAA,wBAAW,EAAC,CAAC,YAAY,CAAC,EAAE,gBAAgB,CAAC,CAAC;YACjE,IAAI,GAAG,IAAA,0BAAa,EAAC,UAAU,EAAE,gBAAgB,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAChF,IAAI,CAAC,yBAAyB,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;YACtD,IAAI,CAAC,yBAAyB,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;YACvD,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;YAC5C,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;YAE7C,MAAM,YAAY,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBACjD,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,2BAA2B;aAC3D,CAAC,CAAC;YAEH,MAAM,MAAM,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAC3C,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,gBAAgB;aAChD,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,SAAS,CAAC;YAC/C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;YACrD,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;YAC/C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;YAC/C,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;YACpD,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;YAEpD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,EAC1C,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,+BAA+B,CAC1D,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uBAAuB,EAAE;YAC1B,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAC3F,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YAE3C,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE;gBAChD,UAAU,EAAE,mCAA2B;gBACvC,OAAO,EAAE,6BAAqB,CAAC,kBAAkB;aAClD,CAAC,CAAC;YAEH,WAAW,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,4BAAe,CAAC;YACvC,IAAI,CAAC,mCAAmC,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;YAE5D,MAAM,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,4BAA4B,EAAE;QACrC,EAAE,CAAC,kCAAkC,EAAE;YACrC,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAC3F,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,IAAI,CAAC,eAAe,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YAE1C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,EAC9B,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,oDAAoD,CAC/E,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,8BAA8B,EAAE;QACvC,EAAE,CAAC,4DAA4D,EAAE;YAC/D,MAAM,gBAAgB,GAAG,IAAI,sBAAc,CAAC,IAAA,uBAAY,EAAC,OAAO,CAAC,CAAC,CAAC;YACnE,MAAM,QAAQ,GAAG,IAAA,wBAAW,EAC1B,+BAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAC7B,gBAAgB,CACjB,CAAC;YAEF,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,QAAQ,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAClF,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE;gBAC5B,MAAM,UAAU,GAAG,IAAA,0BAAkB,EAAC,CAAC,CAAC,KAAK,CAAC,CAAC;gBAC/C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,UAAU,KAAK,YAAY;oBACzB,CAAC,CAAC,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC;oBACrD,CAAC,CAAC,UAAU,KAAK,MAAM;wBACvB,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,2BAAc,CAAC,IAAI,CAAC;wBACrD,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,2BAAc,CAAC,IAAI,CAAC,EAClD,CAAC,CAAM,EAAE,EAAE,CACT,IAAA,gBAAQ,EAAC,CAAC,CAAC,KAAK,CAAC,IAAI,UAAU,KAAK,WAAW;oBAC7C,CAAC,CAAC,CAAC,CAAC,OAAO,KAAK,6BAA6B,KAAK,gDAAgD;oBAClG,CAAC,CAAC,CAAC,CAAC,OAAO,KAAK,4BAA4B,KAAK,gDAAgD,CACtG,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,MAAM,wBAAwB,GAAG,IAAA,0BAAa,EAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,2BAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;YAC5G,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,wBAAwB,CAAC,kBAAkB,CAAC,CAAC,EAAE,2BAAc,CAAC,IAAI,CAAC,EACzE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2EAA2E,CACtG,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yFAAyF,EAAE;YAC5F,MAAM,EAAE,GAAG;gBACT,GAAG,EAAE;oBACH,UAAU,EAAE,OAAO;oBACnB,OAAO,EAAE,IAAI;oBACb,OAAO,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;iBAC/B;gBACD,KAAK,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;aAC7B,CAAC;YAEF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,qCAA4B,EAAC,EAAE,CAAC,EACtC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,OAAO,KAAK,sBAAsB,EAAE,CAAC,GAAG,CAAC,UAAU,eAAe,EAAE,CAAC,GAAG,CAAC,OAAO,4BAA4B,CACjH,CAAC;YAEF,EAAE,CAAC,GAAG,CAAC,UAAU,GAAG,mCAA2B,CAAC;YAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,qCAA4B,EAAC,EAAE,CAAC,EACtC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,OAAO,KAAK,sBAAsB,EAAE,CAAC,GAAG,CAAC,UAAU,eAAe,EAAE,CAAC,GAAG,CAAC,OAAO,4BAA4B,CACjH,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kFAAkF,EAAE;YACrF,MAAM,EAAE,GAAG;gBACT,GAAG,EAAE;oBACH,UAAU,EAAE,OAAO;oBACnB,OAAO,EAAE,IAAI;oBACb,OAAO,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;iBAC/B;gBACD,KAAK,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;aAC7B,CAAC;YAEF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,8BAAqB,EAAC,EAAE,CAAC,EAC/B,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sBAAsB,EAAE,CAAC,GAAG,CAAC,UAAU,eAAe,EAAE,CAAC,GAAG,CAAC,OAAO,YAAY,CAC3G,CAAC;YAEF,EAAE,CAAC,GAAG,CAAC,UAAU,GAAG,mCAA2B,CAAC;YAChD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,8BAAqB,EAAC,EAAE,CAAC,EAC/B,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sBAAsB,EAAE,CAAC,GAAG,CAAC,UAAU,eAAe,EAAE,CAAC,GAAG,CAAC,OAAO,YAAY,CAC3G,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gFAAgF,EAAE;YACnF,MAAM,eAAe,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACzC,iBAAiB,EAAE,oCAAuB,CAAC,CAAC,CAAC;gBAC7C,YAAY,EAAE,8BAAiB;gBAC/B,QAAQ,EAAE,0BAAa;aACxB,CAAC,CAAC,CAAC;YAEJ,IAAI,qBAAqB,GAAG;gBAC1B,kBAAkB,EAAE,oCAAuB;gBAC3C,cAAc,EAAE,gCAAmB;gBACnC,YAAY,EAAE,gCAAmB;aAClC,CAAC;YAEF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,+BAAsB,EAAC,eAAe,EAAE,qBAAqB,CAAC,EACpE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sCAAsC,CACjE,CAAC;YAEF,qBAAqB,GAAG;gBACtB,kBAAkB,EAAE,CAAC,sCAAyB,CAAC,CAAC,CAAC,EAAE,oCAAuB,CAAC,CAAC,CAAC,CAAC;gBAC9E,cAAc,EAAE,gCAAmB;gBACnC,YAAY,EAAE,8BAAiB;aAChC,CAAC;YACF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,+BAAsB,EAAC,eAAe,EAAE,qBAAqB,CAAC,EACpE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,qCAAqC,CAChE,CAAC;YAEF,qBAAqB,GAAG;gBACtB,kBAAkB,EAAE,CAAC,oCAAuB,CAAC,CAAC,CAAC,EAAE,oCAAuB,CAAC,CAAC,CAAC,CAAC;gBAC5E,cAAc,EAAE,gCAAmB;gBACnC,YAAY,EAAE,8BAAiB;aAChC,CAAC;YACF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,+BAAsB,EAAC,eAAe,EAAE,qBAAqB,CAAC,EACpE,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sCAAsC,CACjE,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2EAA2E,EAAE;YAC9E,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,uBAAc,EAAC,+BAAkB,EAAE,8BAAiB,CAAC,EAC3D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,sCAAsC,CACjE,CAAC;YAEF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,IAAA,uBAAc,EAAC,gCAAmB,EAAE,gCAAmB,CAAC,EAC9D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,2CAA2C,CACtE,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qDAAqD,EAAE;YACxD,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAA,0BAAiB,EACf,4BAAe,EACf,0BAAa,EACb;gBACE,SAAS,EAAE,oCAAuB,CAAC,CAAC,CAAC;gBACrC,QAAQ,EAAE,0BAAa;gBACvB,GAAG,EAAE,0BAAa;aACnB,EACD,IAAI,yBAAgB,EAAE,CACvB,EACH,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,mCAAmC,CAC9D,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2EAA2E,EAAE;YAC9E,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CACH,IAAA,mCAA0B,EAAC;gBACzB,UAAU,EAAE,8BAAiB;gBAC7B,iBAAiB,EAAE,oCAAuB,CAAC,CAAC,CAAC;gBAC7C,YAAY,EAAE,8BAAiB;aAChC,CAAC,EACJ,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,6BAA6B,CACxD,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE;YAC7B,MAAM,IAAI,GAAG,IAAA,0BAAa,EAAC,iBAAiB,EAAE,2BAAc,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;YAC3F,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,IAAI,CAAC,CAAC;YACpD,IAAI,CAAC,yBAAyB,CAAC,2BAAc,CAAC,KAAK,CAAC,CAAC;YACrD,MAAM,GAAG,GAAG;gBACV,UAAU,EAAE,OAAO;gBACnB,OAAO,EAAE,GAAG;gBACZ,OAAO,EAAE,8BAAiB;aAC3B,CAAC;YACF,IAAI,CAAC,2BAA2B,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,KAAK,EAAE,gCAAmB,EAAE,CAAC,CAAC;YACzE,IAAI,CAAC,wBAAwB,CAAC,CAAC,EAAE,EAAE,UAAU,EAAE,mCAA2B,EAAE,CAAC,CAAC;YAC9E,MAAM,MAAM,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;YAC7C,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YACrC,MAAM,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QACxD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import * as assert from 'assert';\n\nimport {\n  createPsbtFromHex,\n  getExternalChainCode,\n  getInternalChainCode,\n  isSegwit,\n  parsePsbtInput,\n  ProprietaryKeySubtype,\n  PSBT_PROPRIETARY_IDENTIFIER,\n  RootWalletKeys,\n  scriptTypeForChain,\n  UtxoTransaction,\n  verifySignatureWithUnspent,\n} from '../../../src/bitgo';\n\nimport { getKeyTriple, verifyFullySignedSignatures } from '../../../src/testutil';\nimport {\n  createTapInternalKey,\n  createTapOutputKey,\n  createTapTweak,\n  decodePsbtMusig2Nonce,\n  decodePsbtMusig2Participants,\n  encodePsbtMusig2PartialSig,\n  musig2PartialSign,\n  assertPsbtMusig2Nonces,\n  Musig2NonceStore,\n} from '../../../src/bitgo/Musig2';\nimport { scriptTypes2Of3, toXOnlyPublicKey } from '../../../src/bitgo/outputScripts';\nimport {\n  constructPsbt,\n  getUnspents,\n  invalidPartialSig,\n  invalidParticipantPubKeys,\n  invalidTapInputKey,\n  invalidTapOutputKey,\n  invalidTxHash,\n  dummyAggNonce,\n  validateNoncesKeyVals,\n  validatePartialSigKeyVals,\n  validateParticipantsKeyVals,\n  validatePsbtP2trMusig2Input,\n  validatePsbtP2trMusig2Output,\n  dummyParticipantPubKeys,\n  dummyPrivateKey,\n  dummyPubNonce,\n  dummyTapInternalKey,\n  dummyTapOutputKey,\n  dummyPartialSig,\n  validateFinalizedInput,\n  network,\n  validateParsedTaprootKeyPathTxInput,\n  validateParsedTaprootScriptPathTxInput,\n  validateParsedTaprootKeyPathPsbt,\n  validateParsedTaprootScriptPathPsbt,\n  rootWalletKeys,\n} from './Musig2Util';\n\nconst p2trMusig2Unspent = getUnspents(['p2trMusig2'], rootWalletKeys);\nconst outputType = 'p2trMusig2';\nconst CHANGE_INDEX = 100;\n\ndescribe('p2trMusig2', function () {\n  describe('p2trMusig2 key path', function () {\n    it(`create psbt, nonces, sign (internal verify) - success`, function () {\n      const walletKeys = rootWalletKeys.deriveForChainAndIndex(getExternalChainCode('p2trMusig2'), 0);\n      const unspents = getUnspents(\n        scriptTypes2Of3.map((t) => t),\n        rootWalletKeys\n      );\n      // WP creates PSBT during build API, serializes it, and sends the psbt to user\n      const buildPsbt = constructPsbt(unspents, rootWalletKeys, 'bitgo', 'user', outputType);\n      const buildPsbtSer = buildPsbt.toHex();\n\n      // User de-serialises the psbt, ands the user nonce, and sends it to the hsm so that it can add the bitgo nonce\n      const userPsbt = createPsbtFromHex(buildPsbtSer, network);\n      userPsbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      const userPsbtSer = userPsbt.toHex();\n\n      // HSM deserializes the user psbt, adds the deterministic bitgo nonce, and sends that back to the user\n      const bitgoPsbt = createPsbtFromHex(userPsbtSer, network);\n      bitgoPsbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true });\n      const bitgoPsbtSer = bitgoPsbt.toHex();\n\n      // User combines the psbt with the bitgo nonce, adds user signature, and sends half-signed to hsm\n      const bitgoPsbtDeser = createPsbtFromHex(bitgoPsbtSer, network);\n      userPsbt.combine(bitgoPsbtDeser);\n      userPsbt.signAllInputsHD(rootWalletKeys.user);\n      const userPsbtHalfSignedHex = userPsbt.toHex();\n\n      // WP de-serialises the psbt and validates user sig\n      const userPsbtDes = createPsbtFromHex(userPsbtHalfSignedHex, network);\n      assert.ok(userPsbtDes.validateTaprootMusig2SignaturesOfInput(4, walletKeys.user.publicKey));\n\n      // WP sends to hsm for signature and returns a fully signed psbt\n      const psbt = createPsbtFromHex(userPsbtHalfSignedHex, network);\n      psbt.signAllInputsHD(rootWalletKeys.bitgo, { deterministic: true });\n\n      unspents.forEach((unspent, index) => {\n        if (scriptTypeForChain(unspent.chain) !== 'p2trMusig2') {\n          assert.strictEqual(psbt.getProprietaryKeyVals(index).length, 0);\n          return;\n        }\n        validatePsbtP2trMusig2Input(psbt, index, unspent, 'keyPath');\n        validatePsbtP2trMusig2Output(psbt, 0);\n        validateParticipantsKeyVals(psbt, index, unspent);\n        validateNoncesKeyVals(psbt, index, unspent);\n        validatePartialSigKeyVals(psbt, index, unspent);\n      });\n\n      assert.ok(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      unspents.forEach((unspent, index) => {\n        validateFinalizedInput(psbt, index, unspent);\n      });\n      const tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n      assert.ok(verifyFullySignedSignatures(tx, unspents, rootWalletKeys, 'bitgo', 'user'));\n      unspents.map((unspent, inputIndex) => {\n        assert.deepStrictEqual(verifySignatureWithUnspent(tx, inputIndex, unspents, rootWalletKeys), [\n          true,\n          false,\n          true,\n        ]);\n      });\n    });\n\n    it(`parse tx`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'bitgo', 'user', outputType);\n      validateParsedTaprootKeyPathPsbt(psbt, 0, 'unsigned');\n\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      validateParsedTaprootKeyPathPsbt(psbt, 0, 'halfsigned');\n\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n      validateParsedTaprootKeyPathPsbt(psbt, 0, 'fullysigned');\n\n      psbt.finalizeAllInputs();\n      assert.throws(\n        () => parsePsbtInput(psbt.data.inputs[0]),\n        (e: any) => e.message === 'Finalized PSBT parsing is not supported'\n      );\n\n      const tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n      validateParsedTaprootKeyPathTxInput(psbt, tx);\n    });\n\n    describe('create nonce', function () {\n      it(`update with new nonce should be allowed`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n\n        let noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 1);\n        const userNonceKey = noncesKeyVals[0].key.keydata;\n        const userNonceValue = noncesKeyVals[0].value;\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n\n        noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 2);\n\n        noncesKeyVals = noncesKeyVals.filter((kv) => kv.key.keydata.equals(userNonceKey));\n        assert.strictEqual(noncesKeyVals.length, 1);\n        assert.ok(!noncesKeyVals[0].value.equals(userNonceValue));\n      });\n\n      it(`Cosigner nonce creation fail should not enforce the signer to recreate nonce`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n\n        const tapBip32Derivation = psbt.data.inputs[0].tapBip32Derivation;\n        psbt.data.inputs[0].tapBip32Derivation = undefined;\n\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo),\n          (e: any) => e.message === 'tapBip32Derivation is required to create nonce'\n        );\n\n        psbt.data.inputs[0].tapBip32Derivation = tapBip32Derivation;\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n        psbt.signAllInputsHD(rootWalletKeys.user);\n        psbt.signAllInputsHD(rootWalletKeys.bitgo);\n\n        const noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 2);\n\n        const partialSigKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n        });\n        assert.strictEqual(partialSigKeyVals.length, 2);\n\n        const participantKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n        });\n        assert.strictEqual(participantKeyVals.length, 1);\n      });\n\n      it('Cosigner can create a deterministic nonce', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true });\n\n        const noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 2);\n      });\n\n      it('Cosigner cannot create a deterministic nonce if there is no signer nonce', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true }),\n          (e: any) => e.message === 'No nonces found on input #0'\n        );\n\n        let noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 0);\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true }),\n          (e: any) => e.message === 'signer nonce must be set if cosigner nonce is to be derived deterministically'\n        );\n\n        noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 1);\n      });\n\n      it('Cosigner cannot add entropy to deterministic nonce creation', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        assert.throws(\n          () =>\n            psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, {\n              deterministic: true,\n              sessionId: Buffer.allocUnsafe(32),\n            }),\n          (e: any) => e.message === 'Cannot add extra entropy when generating a deterministic nonce'\n        );\n        const noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 1);\n      });\n\n      it('Signer cannot create a deterministic nonce', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user, { deterministic: true }),\n          (e: any) => e.message === `Only the cosigner's nonce can be set deterministically`\n        );\n        const noncesKeyVals = psbt.getProprietaryKeyVals(0, {\n          identifier: PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(noncesKeyVals.length, 0);\n      });\n\n      it(`skipped if tapInternalKey doesn't match participant pub keys agg`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.data.inputs[0].tapInternalKey = dummyTapInternalKey;\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === 'tapInternalKey and aggregated participant pub keys does not match'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if sessionId size is invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user, { sessionId: Buffer.allocUnsafe(33) }),\n          (e: any) => e.message === 'Invalid sessionId size 33'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if private key is missing`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user.neutered()),\n          (e: any) => e.message === 'private key is required to generate nonce'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if tapBip32Derivation is missing`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.data.inputs[0].tapBip32Derivation = [];\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === 'tapBip32Derivation is required to create nonce'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if participant pub keys is missing`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.data.inputs[0].unknownKeyVals = [];\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === 'Found 0 matching participant key value instead of 1'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);\n      });\n\n      it(`fails if participant pub keys keydata size is invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].key.keydata = Buffer.concat([keyVals[0].key.keydata, Buffer.from('dummy')]);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Invalid keydata size ${keyVals[0].key.keydata.length} for participant pub keys`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if participant keydata tapOutputKey in invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].key.keydata = Buffer.concat([dummyTapOutputKey, keyVals[0].key.keydata.subarray(32)]);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Invalid participants keydata tapOutputKey`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if participant keydata tapInternalKey in invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].key.keydata = Buffer.concat([keyVals[0].key.keydata.subarray(0, 32), dummyTapInternalKey]);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Invalid participants keydata tapInternalKey`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if tapInternalKey and aggregated participant pub keys don't match`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n\n        const walletKeys = rootWalletKeys.deriveForChainAndIndex(getInternalChainCode('p2trMusig2'), 1);\n        const tapInternalKey = createTapInternalKey([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);\n        const tapOutputKey = createTapOutputKey(tapInternalKey, psbt.data.inputs[0].tapMerkleRoot!);\n\n        keyVals[0].key.keydata = Buffer.concat([tapOutputKey, tapInternalKey]);\n        keyVals[0].value = Buffer.concat([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);\n\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `tapInternalKey and aggregated participant pub keys does not match`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if keydata size of participant pub keys is invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].key.keydata = Buffer.allocUnsafe(65);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Invalid keydata size 65 for participant pub keys`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if valuedata size of participant pub keys is invalid`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].value = Buffer.allocUnsafe(67);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Invalid valuedata size 67 for participant pub keys`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if duplicate participant pub keys found`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n        keyVals[0].value = Buffer.concat([keyVals[0].value.subarray(33), keyVals[0].value.subarray(33)]);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `Duplicate participant pub keys found`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if no fingerprint match`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.data.inputs[0].tapBip32Derivation?.forEach((bv) => (bv.masterFingerprint = Buffer.allocUnsafe(4)));\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === 'No bip32Derivation masterFingerprint matched the HD keyPair fingerprint'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if pubkey did not match tapBip32Derivation`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const walletKeys = rootWalletKeys.deriveForChainAndIndex(getInternalChainCode('p2trMusig2'), CHANGE_INDEX);\n        psbt.data.inputs[0].tapBip32Derivation?.forEach((bv) => {\n          bv.path = walletKeys.paths[2];\n        });\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === 'pubkey did not match bip32Derivation'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if root wallet key derive more than one tapBip32Derivation`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const walletKeys = rootWalletKeys.deriveForChainAndIndex(\n          p2trMusig2Unspent[0].chain,\n          p2trMusig2Unspent[0].index\n        );\n        psbt.data.inputs[0].tapBip32Derivation?.forEach((bv, index) => {\n          bv.path = walletKeys.paths[0];\n          bv.pubkey = toXOnlyPublicKey(walletKeys.publicKeys[0]);\n          bv.masterFingerprint = rootWalletKeys.user.fingerprint;\n        });\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message.startsWith('more than one matching derivation for fingerprint')\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n\n      it(`fails if derived wallet key does not match any participant key`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        const keyVals = psbt.getProprietaryKeyVals(0);\n\n        const walletKeys = rootWalletKeys.deriveForChainAndIndex(getInternalChainCode('p2trMusig2'), 1);\n        const tapInternalKey = createTapInternalKey([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);\n        psbt.data.inputs[0].tapInternalKey = tapInternalKey;\n\n        keyVals[0].value = Buffer.concat([walletKeys.user.publicKey, walletKeys.bitgo.publicKey]);\n        const tapOutputKey = createTapOutputKey(tapInternalKey, psbt.data.inputs[0].tapMerkleRoot!);\n        keyVals[0].key.keydata = Buffer.concat([tapOutputKey, tapInternalKey]);\n        psbt.data.inputs[0].unknownKeyVals = [];\n        psbt.addProprietaryKeyValToInput(0, keyVals[0]);\n\n        assert.throws(\n          () => psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user),\n          (e: any) => e.message === `participant plain pub key should match one bip32Derivation plain pub key`\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 1);\n      });\n    });\n\n    describe('sign', function () {\n      it(`fails if privateKey is missing`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n        assert.throws(\n          () => psbt.signTaprootInputHD(0, rootWalletKeys.user.neutered()),\n          (e: any) => e.message === 'privateKey is required to sign p2tr musig2'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n      });\n\n      it(`fails if tapInternalKey is missing`, function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n\n        const walletKeys = rootWalletKeys.deriveForChainAndIndex(\n          p2trMusig2Unspent[0].chain,\n          p2trMusig2Unspent[0].index\n        );\n\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n        psbt.data.inputs[0].tapInternalKey = undefined;\n        assert.throws(\n          () =>\n            psbt.signTaprootMusig2Input(0, {\n              publicKey: walletKeys.user.publicKey,\n              privateKey: walletKeys.user.privateKey!,\n            }),\n          (e: any) => e.message === 'not a taproot musig2 input'\n        );\n        assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n      });\n\n      it('only the cosigner can add a deterministic signature', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true });\n        assert.throws(\n          () => psbt.signInputHD(0, rootWalletKeys.user, { deterministic: true }),\n          (e: any) => e.message === 'can only add a deterministic signature on the cosigner'\n        );\n      });\n\n      it('cosigner can sign deterministically', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: true });\n        psbt.signAllInputsHD(rootWalletKeys.user);\n        psbt.signAllInputsHD(rootWalletKeys.bitgo, { deterministic: true });\n        assert.ok(psbt.validateSignaturesOfAllInputs());\n        assert.ok(psbt.finalizeAllInputs());\n      });\n\n      it('cosigner can sign non-deterministically', function () {\n        const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n        psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo, { deterministic: false });\n        psbt.signAllInputsHD(rootWalletKeys.user);\n        psbt.signAllInputsHD(rootWalletKeys.bitgo, { deterministic: false });\n        assert.ok(psbt.validateSignaturesOfAllInputs());\n        assert.ok(psbt.finalizeAllInputs());\n      });\n    });\n\n    it(`fails if tapMerkleRoot is missing`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n\n      const walletKeys = rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);\n\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      psbt.data.inputs[0].tapMerkleRoot = undefined;\n      assert.throws(\n        () =>\n          psbt.signTaprootMusig2Input(0, {\n            publicKey: walletKeys.user.publicKey,\n            privateKey: walletKeys.user.privateKey!,\n          }),\n        (e: any) => e.message === 'not a taproot musig2 input'\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n\n    it(`fails if participant pub keys is missing`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.data.inputs[0].unknownKeyVals = [];\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === 'Found 0 matching participant key value instead of 1'\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);\n    });\n\n    it(`fails if signer pub key is not matching any participant pub keys`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      assert.throws(\n        () =>\n          psbt.signTaprootMusig2Input(0, {\n            privateKey: rootWalletKeys.backup.privateKey!,\n            publicKey: rootWalletKeys.backup.publicKey!,\n          }),\n        (e: any) => e.message === 'signer pub key should match one of participant pub keys'\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n\n    it(`fails if more than 2 nonce key value exists`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.data.inputs[0].unknownKeyVals?.push(psbt.data.inputs[0].unknownKeyVals[2]);\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === 'Found 3 matching nonce key value instead of 1 or 2'\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 4);\n    });\n\n    it(`fails if 2 nonce key value do not exist`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.data.inputs[0].unknownKeyVals?.splice(2);\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === 'Found 1 matching nonce key value instead of 2'\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 2);\n    });\n\n    it(`fails if nonce keydata size is invalid`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      const keyVals = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      });\n      keyVals[1].key.keydata = Buffer.concat([keyVals[1].key.keydata, Buffer.from('dummy')]);\n      psbt.data.inputs[0].unknownKeyVals?.splice(2);\n      psbt.addProprietaryKeyValToInput(0, keyVals[1]);\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === `Invalid keydata size ${keyVals[1].key.keydata.length} for nonce`\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n\n    it(`fails if nonce valuedata size is invalid`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      const keyVals = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      });\n      keyVals[1].value = Buffer.concat([keyVals[1].value, Buffer.from('dummy')]);\n      psbt.data.inputs[0].unknownKeyVals?.splice(2);\n      psbt.addProprietaryKeyValToInput(0, keyVals[1]);\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === `Invalid valuedata size ${keyVals[1].value.length} for nonce`\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n\n    it(`fails if nonce keydata is invalid`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      const dummyRootWalletKeys = new RootWalletKeys(getKeyTriple('dummy'));\n      const dummyP2trMusig2Unspent = getUnspents(['p2trMusig2'], dummyRootWalletKeys);\n      const dummyPsbt = constructPsbt(dummyP2trMusig2Unspent, dummyRootWalletKeys, 'user', 'bitgo', 'p2sh');\n      dummyPsbt.setAllInputsMusig2NonceHD(dummyRootWalletKeys.user);\n      dummyPsbt.setAllInputsMusig2NonceHD(dummyRootWalletKeys.bitgo);\n\n      const dummyKeyVals = dummyPsbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      });\n\n      psbt.data.inputs[0].unknownKeyVals?.splice(1);\n      dummyKeyVals.forEach((kv, i) => psbt.addProprietaryKeyValToInput(0, dummyKeyVals[i]));\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === `Invalid nonce keydata participant pub key`\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n\n    it(`fails if nonce keydata tapOutputKey is invalid`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', 'p2sh');\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      const keyVals = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      });\n\n      keyVals[1].key.keydata = Buffer.concat([keyVals[1].key.keydata.subarray(0, 33), dummyTapOutputKey]);\n\n      psbt.data.inputs[0].unknownKeyVals?.splice(2);\n      psbt.addProprietaryKeyValToInput(0, keyVals[1]);\n      assert.throws(\n        () => psbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === `Invalid nonce keydata tapOutputKey`\n      );\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 3);\n    });\n  });\n\n  describe('p2trMusig2 script path', function () {\n    it(`psbt creation success and musig2 skips`, function () {\n      let psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'backup', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.backup);\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      psbt.signAllInputsHD(rootWalletKeys.backup);\n      validatePsbtP2trMusig2Input(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');\n      validatePsbtP2trMusig2Output(psbt, 0);\n      assert.ok(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      validateFinalizedInput(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');\n      let tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n      assert.ok(verifyFullySignedSignatures(tx, p2trMusig2Unspent, rootWalletKeys, 'user', 'backup'));\n\n      psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'bitgo', 'backup', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.backup);\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.backup);\n      assert.strictEqual(psbt.getProprietaryKeyVals(0).length, 0);\n      validatePsbtP2trMusig2Input(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');\n      validatePsbtP2trMusig2Output(psbt, 0);\n      assert.ok(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      validateFinalizedInput(psbt, 0, p2trMusig2Unspent[0], 'scriptPath');\n      tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n      assert.ok(verifyFullySignedSignatures(tx, p2trMusig2Unspent, rootWalletKeys, 'bitgo', 'backup'));\n    });\n\n    it(`parse tx`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'backup', outputType);\n      validateParsedTaprootScriptPathPsbt(psbt, 0, 'unsigned');\n\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      validateParsedTaprootScriptPathPsbt(psbt, 0, 'halfsigned');\n\n      psbt.signAllInputsHD(rootWalletKeys.backup);\n      validateParsedTaprootScriptPathPsbt(psbt, 0, 'fullysigned');\n\n      psbt.finalizeAllInputs();\n      const tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n\n      const psbtDuplicate = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'backup', outputType);\n      validateParsedTaprootScriptPathTxInput(psbtDuplicate, tx, 0);\n    });\n  });\n\n  describe('validate p2tr Musig2 signatures', function () {\n    it(`validate with pubkey`, function () {\n      const walletKeys = rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);\n      let psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2Nonce(walletKeys.user);\n      psbt.setAllInputsMusig2Nonce(walletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.user.publicKey));\n\n      psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setInputMusig2Nonce(0, walletKeys.user);\n      psbt.setInputMusig2NonceHD(0, rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.user.publicKey));\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n      assert.ok(psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey));\n    });\n\n    it(`fails if no sig`, function () {\n      const walletKeys = rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      assert.throws(\n        () => psbt.validateSignaturesOfAllInputs(),\n        (e: any) => e.message === `No signatures to validate`\n      );\n\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n\n      psbt.signAllInputsHD(rootWalletKeys.user);\n\n      assert.throws(\n        () => psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey),\n        (e: any) => e.message === `No signatures for this pubkey`\n      );\n    });\n\n    it(`fails if no tapInternalKey and tapMerkleRoot`, function () {\n      const walletKeys = rootWalletKeys.deriveForChainAndIndex(p2trMusig2Unspent[0].chain, p2trMusig2Unspent[0].index);\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n\n      const tapInternalKey = psbt.data.inputs[0].tapInternalKey;\n      psbt.data.inputs[0].tapInternalKey = undefined;\n      assert.throws(\n        () => psbt.validateTaprootMusig2SignaturesOfInput(0),\n        (e: any) => e.message === `both tapInternalKey and tapMerkleRoot are required`\n      );\n\n      psbt.data.inputs[0].tapInternalKey = tapInternalKey;\n      psbt.data.inputs[0].tapMerkleRoot = undefined;\n      assert.throws(\n        () => psbt.validateTaprootMusig2SignaturesOfInput(0, walletKeys.bitgo.publicKey),\n        (e: any) => e.message === `both tapInternalKey and tapMerkleRoot are required`\n      );\n    });\n\n    it(`fails if no nonce and sig pub key match`, function () {\n      let psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n\n      const partialSigs = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n      });\n\n      const myRootWalletKeys = new RootWalletKeys(getKeyTriple('dummy'));\n      const myUnspents = getUnspents(['p2trMusig2'], myRootWalletKeys);\n      psbt = constructPsbt(myUnspents, myRootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(myRootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(myRootWalletKeys.bitgo);\n      psbt.signAllInputsHD(myRootWalletKeys.user);\n      psbt.signAllInputsHD(myRootWalletKeys.bitgo);\n\n      const participants = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n      });\n\n      const nonces = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n      });\n\n      psbt.data.inputs[0].unknownKeyVals = undefined;\n      psbt.addProprietaryKeyValToInput(0, participants[0]);\n      psbt.addProprietaryKeyValToInput(0, nonces[0]);\n      psbt.addProprietaryKeyValToInput(0, nonces[1]);\n      psbt.addProprietaryKeyValToInput(0, partialSigs[0]);\n      psbt.addProprietaryKeyValToInput(0, partialSigs[1]);\n\n      assert.throws(\n        () => psbt.validateSignaturesOfAllInputs(),\n        (e: any) => e.message === `Found no pub nonce for pubkey`\n      );\n    });\n\n    it(`fails if no valid sig`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n      psbt.signAllInputsHD(rootWalletKeys.bitgo);\n\n      const partialSigs = psbt.getProprietaryKeyVals(0, {\n        identifier: PSBT_PROPRIETARY_IDENTIFIER,\n        subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n      });\n\n      partialSigs[1].value = dummyPartialSig;\n      psbt.addOrUpdateProprietaryKeyValToInput(0, partialSigs[1]);\n\n      assert.ok(!psbt.validateSignaturesOfAllInputs());\n    });\n  });\n\n  describe('finalizeTaprootMusig2Input', function () {\n    it('fails if invalid number for sigs', function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      psbt.signAllInputsHD(rootWalletKeys.user);\n\n      assert.throws(\n        () => psbt.finalizeAllInputs(),\n        (e: any) => e.message === `invalid number of partial signatures 1 to finalize`\n      );\n    });\n  });\n\n  describe('Psbt musig2 common functions', function () {\n    it('output script should match the scriptPubKey in the prevout', function () {\n      const myRootWalletKeys = new RootWalletKeys(getKeyTriple('dummy'));\n      const unspents = getUnspents(\n        scriptTypes2Of3.map((t) => t),\n        myRootWalletKeys\n      );\n\n      const psbt = constructPsbt(unspents, rootWalletKeys, 'user', 'bitgo', outputType);\n      unspents.forEach((u, index) => {\n        const scriptType = scriptTypeForChain(u.chain);\n        assert.throws(\n          () =>\n            scriptType === 'p2trMusig2'\n              ? psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user)\n              : scriptType === 'p2tr'\n              ? psbt.signTaprootInputHD(index, rootWalletKeys.user)\n              : psbt.signInputHD(index, rootWalletKeys.user),\n          (e: any) =>\n            isSegwit(u.chain) && scriptType !== 'p2shP2wsh'\n              ? e.message === `Witness script for input #${index} doesn't match the scriptPubKey in the prevout`\n              : e.message === `Redeem script for input #${index} doesn't match the scriptPubKey in the prevout`\n        );\n      });\n\n      const p2trMusig2ScriptPathPsbt = constructPsbt([unspents[4]], rootWalletKeys, 'user', 'backup', outputType);\n      assert.throws(\n        () => p2trMusig2ScriptPathPsbt.signTaprootInputHD(0, rootWalletKeys.user),\n        (e: any) => e.message === `Witness script for input #0 doesn't match the scriptPubKey in the prevout`\n      );\n    });\n\n    it(`decodePsbtMusig2ParticipantsKeyValData fails if invalid subtype or identifier is passed`, function () {\n      const kv = {\n        key: {\n          identifier: 'dummy',\n          subtype: 0x05,\n          keydata: Buffer.allocUnsafe(1),\n        },\n        value: Buffer.allocUnsafe(1),\n      };\n\n      assert.throws(\n        () => decodePsbtMusig2Participants(kv),\n        (e: any) =>\n          e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`\n      );\n\n      kv.key.identifier = PSBT_PROPRIETARY_IDENTIFIER;\n      assert.throws(\n        () => decodePsbtMusig2Participants(kv),\n        (e: any) =>\n          e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`\n      );\n    });\n\n    it(`decodePsbtMusig2NonceKeyValData fails if invalid subtype or identifier is passed`, function () {\n      const kv = {\n        key: {\n          identifier: 'dummy',\n          subtype: 0x05,\n          keydata: Buffer.allocUnsafe(1),\n        },\n        value: Buffer.allocUnsafe(1),\n      };\n\n      assert.throws(\n        () => decodePsbtMusig2Nonce(kv),\n        (e: any) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`\n      );\n\n      kv.key.identifier = PSBT_PROPRIETARY_IDENTIFIER;\n      assert.throws(\n        () => decodePsbtMusig2Nonce(kv),\n        (e: any) => e.message === `Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`\n      );\n    });\n\n    it(`validatePsbtMusig2NoncesKeyValData fails if participant pub keys are duplicate`, function () {\n      const nonceKeyValData = [0, 1].map((i) => ({\n        participantPubKey: dummyParticipantPubKeys[i],\n        tapOutputKey: dummyTapOutputKey,\n        pubNonce: dummyPubNonce,\n      }));\n\n      let participantKeyValData = {\n        participantPubKeys: dummyParticipantPubKeys,\n        tapInternalKey: dummyTapInternalKey,\n        tapOutputKey: invalidTapOutputKey,\n      };\n\n      assert.throws(\n        () => assertPsbtMusig2Nonces(nonceKeyValData, participantKeyValData),\n        (e: any) => e.message === `invalid size 1. Must use x-only key.`\n      );\n\n      participantKeyValData = {\n        participantPubKeys: [invalidParticipantPubKeys[0], dummyParticipantPubKeys[0]],\n        tapInternalKey: dummyTapInternalKey,\n        tapOutputKey: dummyTapOutputKey,\n      };\n      assert.throws(\n        () => assertPsbtMusig2Nonces(nonceKeyValData, participantKeyValData),\n        (e: any) => e.message === `invalid size 1. Must use plain key.`\n      );\n\n      participantKeyValData = {\n        participantPubKeys: [dummyParticipantPubKeys[0], dummyParticipantPubKeys[0]],\n        tapInternalKey: dummyTapInternalKey,\n        tapOutputKey: dummyTapOutputKey,\n      };\n      assert.throws(\n        () => assertPsbtMusig2Nonces(nonceKeyValData, participantKeyValData),\n        (e: any) => e.message === `Duplicate participant pub keys found`\n      );\n    });\n\n    it(`createTapTweak fails if invalid tapInternalKey or tapMerkleRoot is passed`, function () {\n      assert.throws(\n        () => createTapTweak(invalidTapInputKey, dummyTapOutputKey),\n        (e: any) => e.message === `invalid size 1. Must use x-only key.`\n      );\n\n      assert.throws(\n        () => createTapTweak(dummyTapInternalKey, invalidTapOutputKey),\n        (e: any) => e.message === `invalid size 1. Must use tap merkle root.`\n      );\n    });\n\n    it(`musig2PartialSign fails if invalid txHash is passed`, function () {\n      assert.throws(\n        () =>\n          musig2PartialSign(\n            dummyPrivateKey,\n            dummyPubNonce,\n            {\n              publicKey: dummyParticipantPubKeys[0],\n              aggNonce: dummyAggNonce,\n              msg: invalidTxHash,\n            },\n            new Musig2NonceStore()\n          ),\n        (e: any) => e.message === `invalid size 1. Must use tx hash.`\n      );\n    });\n\n    it(`encodePsbtMusig2PartialSigKeyKeyValData fails if invalid txHash is passed`, function () {\n      assert.throws(\n        () =>\n          encodePsbtMusig2PartialSig({\n            partialSig: invalidPartialSig,\n            participantPubKey: dummyParticipantPubKeys[0],\n            tapOutputKey: dummyTapOutputKey,\n          }),\n        (e: any) => e.message === `Invalid partialSig length 1`\n      );\n    });\n\n    it(`deleteProprietaryKeyVals`, function () {\n      const psbt = constructPsbt(p2trMusig2Unspent, rootWalletKeys, 'user', 'bitgo', outputType);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      const key = {\n        identifier: 'DUMMY',\n        subtype: 100,\n        keydata: dummyTapOutputKey,\n      };\n      psbt.addProprietaryKeyValToInput(0, { key, value: dummyTapInternalKey });\n      psbt.deleteProprietaryKeyVals(0, { identifier: PSBT_PROPRIETARY_IDENTIFIER });\n      const keyVal = psbt.getProprietaryKeyVals(0);\n      assert.strictEqual(keyVal.length, 1);\n      assert.strictEqual(keyVal[0].key.identifier, 'DUMMY');\n    });\n  });\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!