PHP WebShell
Текущая директория: /opt/BitGoJS/modules/utxo-staking/dist/test/unit/babylon
Просмотр файла: transactions.js
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const assert_1 = __importDefault(require("assert"));
const vendor = __importStar(require("@bitgo/babylonlabs-io-btc-staking-ts"));
const bitcoinjslib = __importStar(require("bitcoinjs-lib"));
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const wasm_miniscript_1 = require("@bitgo/wasm-miniscript");
const descriptor_1 = require("@bitgo/utxo-core/descriptor");
const testutil_1 = require("@bitgo/utxo-core/testutil");
const babylonlabs_io_btc_staking_ts_1 = require("@bitgo/babylonlabs-io-btc-staking-ts");
const babylon_1 = require("../../../src/babylon");
const fixtures_utils_1 = require("../fixtures.utils");
const key_utils_1 = require("./key.utils");
const vendor_utils_1 = require("./vendor.utils");
function getStakingTransactionTreeVendor(builder, amount, utxos, feeRateSatB, signers, descriptorBuilder) {
const staking = builder.createStakingTransaction(amount, utxos, feeRateSatB);
const stakingWithdraw = builder.createWithdrawStakingExpiredPsbt(staking.transaction, feeRateSatB);
const unbonding = builder.createUnbondingTransaction(staking.transaction);
const unbondingWithdraw = builder.createWithdrawEarlyUnbondedTransaction(unbonding.transaction, feeRateSatB);
const unbondingSlashing = builder.createUnbondingOutputSlashingPsbt(unbonding.transaction);
const signSequence = [signers.staker];
if ('finalityProvider' in signers) {
signSequence.push(signers.finalityProvider, ...signers.covenant);
}
const unbondingSlashingWithdraw = signSequence
? builder.createWithdrawSlashingPsbt((0, babylon_1.forceFinalizePsbt)((0, babylon_1.getSignedPsbt)(unbondingSlashing.psbt, descriptorBuilder.getUnbondingDescriptor(), signSequence, {
finalize: false,
}), builder.network).extractTransaction(), feeRateSatB)
: undefined;
const slashing = builder.createStakingOutputSlashingPsbt(staking.transaction);
const slashingSigned = signSequence
? (0, babylon_1.getSignedPsbt)(slashing.psbt, descriptorBuilder.getStakingDescriptor(), signSequence, {
finalize: false,
})
: undefined;
const slashingWithdraw = slashingSigned
? builder.createWithdrawSlashingPsbt((0, babylon_1.forceFinalizePsbt)(slashingSigned.toBuffer(), builder.network).extractTransaction(), feeRateSatB)
: undefined;
return {
staking,
stakingWithdraw,
unbonding,
unbondingWithdraw,
unbondingSlashing,
unbondingSlashingWithdraw,
slashing,
slashingSigned,
slashingSignedBase64: slashingSigned?.toBuffer().toString('base64'),
slashingWithdraw,
};
}
function createUnstakingTransaction(stakingTx, stakingDescriptor, changeAddress, { sequence }) {
const network = utxolib.networks.bitcoin;
const witnessUtxoNumber = stakingTx.transaction.outs[0];
const witnessUtxo = {
script: witnessUtxoNumber.script,
value: BigInt(witnessUtxoNumber.value),
};
return (0, descriptor_1.createPsbt)({
network,
}, [
{
hash: stakingTx.transaction.getId(),
index: 0,
witnessUtxo,
descriptor: stakingDescriptor,
sequence,
},
], [
{
script: utxolib.address.toOutputScript(changeAddress, network),
value: BigInt(witnessUtxoNumber.value) - 1000n,
},
]);
}
function getTestnetStakingParamsWithCovenant(params, covenantKeys) {
return {
...params,
covenantNoCoordPks: covenantKeys.map((pk) => (0, key_utils_1.getXOnlyPubkey)(pk).toString('hex')),
};
}
function wpkhDescriptor(key) {
return wasm_miniscript_1.Descriptor.fromString(wasm_miniscript_1.ast.formatNode({ wpkh: key.publicKey.toString('hex') }), 'definite');
}
function mockUtxo(descriptor) {
const scriptPubKey = Buffer.from(descriptor.scriptPubkey());
const witnessScript = Buffer.from(descriptor.encode());
return {
rawTxHex: undefined,
txid: Buffer.alloc(32).fill(0x11).toString('hex'),
value: 666666,
vout: 0,
redeemScript: undefined,
witnessScript: witnessScript.toString('hex'),
scriptPubKey: scriptPubKey.toString('hex'),
};
}
function parseScript(key, script) {
if (!Buffer.isBuffer(script)) {
throw new Error('script must be a buffer');
}
const ms = wasm_miniscript_1.Miniscript.fromBitcoinScript(script, 'tap');
return {
script: script.toString('hex'),
miniscript: ms.toString(),
miniscriptAst: wasm_miniscript_1.ast.fromMiniscript(ms),
scriptASM: utxolib.script.toASM(script).split(/\s+/),
};
}
function parseScripts(scripts) {
if (typeof scripts !== 'object' || scripts === null) {
throw new Error('scripts must be an object');
}
return Object.fromEntries(Object.entries(scripts).map(([key, value]) => [key, parseScript(key, value)]));
}
async function assertEqualsFixture(fixtureName, value, n = fixtures_utils_1.normalize, eq = assert_1.default.deepStrictEqual) {
value = n(value);
eq(await (0, testutil_1.getFixture)(fixtureName, value), value);
}
async function assertScriptsEqualFixture(fixtureName, builder, scripts) {
await assertEqualsFixture(fixtureName, {
builder: (0, testutil_1.toPlainObject)(builder),
scripts: parseScripts(scripts),
});
}
async function assertTransactionEqualsFixture(fixtureName, tx) {
await assertEqualsFixture(fixtureName, (0, fixtures_utils_1.normalize)(tx));
}
function assertEqualsMiniscript(script, miniscript) {
const ms = wasm_miniscript_1.Miniscript.fromBitcoinScript(script, 'tap');
assert_1.default.deepStrictEqual(wasm_miniscript_1.ast.fromMiniscript(ms), miniscript);
assert_1.default.deepStrictEqual(script.toString('hex'), Buffer.from(wasm_miniscript_1.Miniscript.fromString(wasm_miniscript_1.ast.formatNode(miniscript), 'tap').encode()).toString('hex'));
}
function assertEqualScripts(descriptorBuilder, builder) {
for (const [key, script] of Object.entries(builder)) {
switch (key) {
case 'timelockScript':
assertEqualsMiniscript(script, descriptorBuilder.getTimelockMiniscript());
break;
case 'unbondingScript':
assertEqualsMiniscript(script, descriptorBuilder.getUnbondingMiniscript());
break;
case 'slashingScript':
assertEqualsMiniscript(script, descriptorBuilder.getSlashingMiniscript());
break;
case 'unbondingTimelockScript':
assertEqualsMiniscript(script, descriptorBuilder.getUnbondingTimelockMiniscript());
break;
default:
throw new Error(`unexpected script key: ${key}`);
}
}
}
function assertEqualOutputScript(outputInfo, descriptor) {
assert_1.default.strictEqual(outputInfo.scriptPubKey.toString('hex'), Buffer.from(descriptor.scriptPubkey()).toString('hex'));
}
function describeWithKeys(tag, finalityProviderKeys, covenantKeys, stakingParams, { signIntermediateTxs = false } = {}) {
const stakerKey = (0, key_utils_1.getECKey)('staker');
const covenantThreshold = stakingParams.covenantQuorum;
const stakingTimelock = stakingParams.minStakingTimeBlocks;
const unbondingTimelock = stakingParams.unbondingTime;
const vendorBuilder = new vendor.StakingScriptData((0, key_utils_1.getXOnlyPubkey)(stakerKey), finalityProviderKeys.map(key_utils_1.getXOnlyPubkey), covenantKeys.map(key_utils_1.getXOnlyPubkey), covenantThreshold, stakingTimelock, unbondingTimelock);
const descriptorBuilder = new babylon_1.BabylonDescriptorBuilder((0, key_utils_1.getXOnlyPubkey)(stakerKey), finalityProviderKeys.map(key_utils_1.getXOnlyPubkey), covenantKeys.map(key_utils_1.getXOnlyPubkey), covenantThreshold, stakingTimelock, unbondingTimelock);
describe(`Babylon Staking [${tag}]`, function () {
it('generates expected staking scripts', async function () {
await assertScriptsEqualFixture(`test/fixtures/babylon/scripts.${tag}.json`, vendorBuilder, vendorBuilder.buildScripts());
});
it('matches inner taproot scripts', function () {
assertEqualScripts(descriptorBuilder, vendorBuilder.buildScripts());
});
it('matches output scripts', function () {
assertEqualOutputScript(vendor.deriveStakingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getStakingDescriptor());
assertEqualOutputScript(vendor.deriveSlashingOutput(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getSlashingDescriptor());
assertEqualOutputScript(vendor.deriveUnbondingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getUnbondingDescriptor());
});
describe('Transaction Sets', async function () {
const stakerMainWalletKey = (0, key_utils_1.getECKey)('stakerMainWallet');
const mainWallet = wpkhDescriptor(stakerMainWalletKey);
const amount = 55555;
const changeAddress = (0, descriptor_1.createAddressFromDescriptor)(mainWallet, undefined, utxolib.networks.bitcoin);
const feeRateSatB = 2;
const utxo = mockUtxo(mainWallet);
let stakingTx;
before('setup stakingTx', function () {
stakingTx = vendor.stakingTransaction(vendorBuilder.buildScripts(), amount, changeAddress, [mockUtxo(mainWallet)], bitcoinjslib.networks.bitcoin, feeRateSatB);
});
it('has expected transactions', async function () {
await assertTransactionEqualsFixture(`test/fixtures/babylon/stakingTransaction.${tag}.json`, stakingTx);
// simply one staking output and one change output
// nothing special
assert_1.default.deepStrictEqual(stakingTx.transaction.outs, [
{
script: Buffer.from(descriptorBuilder.getStakingDescriptor().scriptPubkey()),
value: amount,
},
{
script: utxolib.address.toOutputScript(changeAddress, utxolib.networks.bitcoin),
value: utxo.value - amount - stakingTx.fee,
},
]);
});
if (finalityProviderKeys.length !== 1) {
return;
}
const finalityProvider = finalityProviderKeys[0];
it('has expected transactions (vendorStaking.Staking)', async function () {
const vendorStakingTxBuilder = new vendor.Staking(bitcoinjslib.networks.bitcoin, (0, babylon_1.toStakerInfo)(stakerKey, changeAddress), stakingParams, (0, key_utils_1.getXOnlyPubkey)(finalityProvider).toString('hex'), stakingParams.minStakingTimeBlocks);
const txTree = getStakingTransactionTreeVendor(vendorStakingTxBuilder, amount, [utxo], feeRateSatB, signIntermediateTxs
? {
staker: stakerKey,
finalityProvider,
covenant: covenantKeys,
covenantThreshold: covenantThreshold,
}
: { staker: stakerKey }, descriptorBuilder);
await assertTransactionEqualsFixture(`test/fixtures/babylon/txTree.${tag}.json`, txTree);
});
it('creates MsgCreateBTCDelegation', async function () {
const fVendor = vendor_utils_1.getVendorMsgCreateBtcDelegation;
const fBitGo = vendor_utils_1.getBitGoUtxoStakingMsgCreateBtcDelegation;
for (const f of [fVendor, fBitGo]) {
await assertEqualsFixture(`test/fixtures/babylon/msgCreateBTCDelegation.${tag}.json`, await f(bitcoinjslib.networks.bitcoin, stakerKey, finalityProvider, descriptorBuilder, [{ ...stakingParams, version: 0, btcActivationHeight: 0 }], changeAddress, amount, utxo, feeRateSatB, 800000), fixtures_utils_1.normalize, (a, b) => {
// The vendor library serializes the signature as BIP322, while
// our implementation serializes it as ECDSA.
// Strip the pop field from the MsgCreateBTCDelegation.
function stripPop(v) {
const vAny = v;
delete vAny['unsignedDelegationMsg']['value']['pop'];
}
stripPop(a);
stripPop(b);
assert_1.default.deepStrictEqual(a, b);
});
}
});
it('creates unstaking transaction', async function () {
const unstaking = createUnstakingTransaction(stakingTx, descriptorBuilder.getStakingDescriptor(), changeAddress, { sequence: stakingParams.minStakingTimeBlocks });
const wrappedPsbt = (0, descriptor_1.toWrappedPsbt)(unstaking);
(0, assert_1.default)((0, descriptor_1.getNewSignatureCount)((0, descriptor_1.signWithKey)(wrappedPsbt, stakerKey)) > 0);
wrappedPsbt.finalize();
const tx = (0, descriptor_1.toUtxoPsbt)(wrappedPsbt, utxolib.networks.bitcoin).extractTransaction();
await assertTransactionEqualsFixture(`test/fixtures/babylon/unstakingTransaction.${tag}.json`, {
transaction: tx,
});
});
});
});
}
function describeWithKeysFromStakingParams(tag, finalityProviderKeys, stakingParams) {
describeWithKeys(tag, finalityProviderKeys, stakingParams.covenantNoCoordPks.map((pk) => (0, key_utils_1.fromXOnlyPublicKey)(Buffer.from(pk, 'hex'))), stakingParams);
}
function describeWithMockKeys(tag, stakingParams, finalityProviderKeys, covenantKeys) {
describeWithKeys(tag, finalityProviderKeys, covenantKeys, getTestnetStakingParamsWithCovenant(stakingParams, covenantKeys), {
signIntermediateTxs: true,
});
}
describeWithKeysFromStakingParams('testnet', [(0, key_utils_1.fromXOnlyPublicKey)(babylon_1.testnetFinalityProvider0)], (0, babylonlabs_io_btc_staking_ts_1.getBabylonParamByVersion)(5, (0, babylon_1.getStakingParams)('testnet')));
describeWithMockKeys('testnetMock', (0, babylonlabs_io_btc_staking_ts_1.getBabylonParamByVersion)(5, (0, babylon_1.getStakingParams)('testnet')), (0, key_utils_1.getECKeys)('finalityProvider', 1), (0, key_utils_1.getECKeys)('covenant', 9));
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactions.js","sourceRoot":"","sources":["../../../../test/unit/babylon/transactions.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as vendor from '@bitgo/babylonlabs-io-btc-staking-ts';\nimport * as bitcoinjslib from 'bitcoinjs-lib';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport { ECPairInterface } from '@bitgo/utxo-lib';\nimport { ast, Descriptor, Miniscript } from '@bitgo/wasm-miniscript';\nimport {\n  createAddressFromDescriptor,\n  createPsbt,\n  getNewSignatureCount,\n  signWithKey,\n  toUtxoPsbt,\n  toWrappedPsbt,\n} from '@bitgo/utxo-core/descriptor';\nimport { getFixture, toPlainObject } from '@bitgo/utxo-core/testutil';\nimport { getBabylonParamByVersion } from '@bitgo/babylonlabs-io-btc-staking-ts';\n\nimport {\n  BabylonDescriptorBuilder,\n  testnetFinalityProvider0,\n  getSignedPsbt,\n  getStakingParams,\n  toStakerInfo,\n  forceFinalizePsbt,\n} from '../../../src/babylon';\nimport { normalize } from '../fixtures.utils';\n\nimport { fromXOnlyPublicKey, getECKey, getECKeys, getXOnlyPubkey } from './key.utils';\nimport { getBitGoUtxoStakingMsgCreateBtcDelegation, getVendorMsgCreateBtcDelegation } from './vendor.utils';\n\ntype WithFee<T> = T & { fee: number };\ntype TransactionWithFee = WithFee<{ transaction: bitcoinjslib.Transaction }>;\ntype PsbtWithFee = WithFee<{ psbt: bitcoinjslib.Psbt }>;\n\ntype TransactionTree = {\n  staking: TransactionWithFee;\n  stakingWithdraw: PsbtWithFee;\n\n  unbonding: TransactionWithFee;\n  unbondingWithdraw: PsbtWithFee;\n  unbondingSlashing: PsbtWithFee;\n  unbondingSlashingWithdraw: PsbtWithFee | undefined;\n\n  slashing: PsbtWithFee;\n  slashingWithdraw: PsbtWithFee | undefined;\n  slashingSigned: bitcoinjslib.Psbt | undefined;\n  slashingSignedBase64: string | undefined;\n};\n\nfunction getStakingTransactionTreeVendor(\n  builder: vendor.Staking,\n  amount: number,\n  utxos: vendor.UTXO[],\n  feeRateSatB: number,\n  signers:\n    | {\n        staker: ECPairInterface;\n        finalityProvider: ECPairInterface;\n        covenant: ECPairInterface[];\n        covenantThreshold: number;\n      }\n    | { staker: ECPairInterface },\n  descriptorBuilder: BabylonDescriptorBuilder\n): TransactionTree {\n  const staking = builder.createStakingTransaction(amount, utxos, feeRateSatB);\n  const stakingWithdraw = builder.createWithdrawStakingExpiredPsbt(staking.transaction, feeRateSatB);\n  const unbonding = builder.createUnbondingTransaction(staking.transaction);\n  const unbondingWithdraw = builder.createWithdrawEarlyUnbondedTransaction(unbonding.transaction, feeRateSatB);\n  const unbondingSlashing = builder.createUnbondingOutputSlashingPsbt(unbonding.transaction);\n  const signSequence = [signers.staker];\n  if ('finalityProvider' in signers) {\n    signSequence.push(signers.finalityProvider, ...signers.covenant);\n  }\n  const unbondingSlashingWithdraw = signSequence\n    ? builder.createWithdrawSlashingPsbt(\n        forceFinalizePsbt(\n          getSignedPsbt(unbondingSlashing.psbt, descriptorBuilder.getUnbondingDescriptor(), signSequence, {\n            finalize: false,\n          }),\n          builder.network\n        ).extractTransaction(),\n        feeRateSatB\n      )\n    : undefined;\n  const slashing = builder.createStakingOutputSlashingPsbt(staking.transaction);\n  const slashingSigned = signSequence\n    ? getSignedPsbt(slashing.psbt, descriptorBuilder.getStakingDescriptor(), signSequence, {\n        finalize: false,\n      })\n    : undefined;\n  const slashingWithdraw = slashingSigned\n    ? builder.createWithdrawSlashingPsbt(\n        forceFinalizePsbt(slashingSigned.toBuffer(), builder.network).extractTransaction(),\n        feeRateSatB\n      )\n    : undefined;\n\n  return {\n    staking,\n    stakingWithdraw,\n    unbonding,\n    unbondingWithdraw,\n    unbondingSlashing,\n    unbondingSlashingWithdraw,\n    slashing,\n    slashingSigned,\n    slashingSignedBase64: slashingSigned?.toBuffer().toString('base64'),\n    slashingWithdraw,\n  };\n}\n\nfunction createUnstakingTransaction(\n  stakingTx: vendor.TransactionResult,\n  stakingDescriptor: Descriptor,\n  changeAddress: string,\n  { sequence }: { sequence: number }\n): utxolib.Psbt {\n  const network = utxolib.networks.bitcoin;\n  const witnessUtxoNumber = stakingTx.transaction.outs[0];\n  const witnessUtxo = {\n    script: witnessUtxoNumber.script,\n    value: BigInt(witnessUtxoNumber.value),\n  };\n  return createPsbt(\n    {\n      network,\n    },\n    [\n      {\n        hash: stakingTx.transaction.getId(),\n        index: 0,\n        witnessUtxo,\n        descriptor: stakingDescriptor,\n        sequence,\n      },\n    ],\n    [\n      {\n        script: utxolib.address.toOutputScript(changeAddress, network),\n        value: BigInt(witnessUtxoNumber.value) - 1000n,\n      },\n    ]\n  );\n}\n\nfunction getTestnetStakingParamsWithCovenant(\n  params: vendor.StakingParams,\n  covenantKeys: ECPairInterface[]\n): vendor.StakingParams {\n  return {\n    ...params,\n    covenantNoCoordPks: covenantKeys.map((pk) => getXOnlyPubkey(pk).toString('hex')),\n  };\n}\n\nfunction wpkhDescriptor(key: utxolib.ECPairInterface): Descriptor {\n  return Descriptor.fromString(ast.formatNode({ wpkh: key.publicKey.toString('hex') }), 'definite');\n}\n\nfunction mockUtxo(descriptor: Descriptor): vendor.UTXO {\n  const scriptPubKey = Buffer.from(descriptor.scriptPubkey());\n  const witnessScript = Buffer.from(descriptor.encode());\n  return {\n    rawTxHex: undefined,\n    txid: Buffer.alloc(32).fill(0x11).toString('hex'),\n    value: 666_666,\n    vout: 0,\n    redeemScript: undefined,\n    witnessScript: witnessScript.toString('hex'),\n    scriptPubKey: scriptPubKey.toString('hex'),\n  };\n}\n\nfunction parseScript(key: string, script: unknown) {\n  if (!Buffer.isBuffer(script)) {\n    throw new Error('script must be a buffer');\n  }\n  const ms = Miniscript.fromBitcoinScript(script, 'tap');\n  return {\n    script: script.toString('hex'),\n    miniscript: ms.toString(),\n    miniscriptAst: ast.fromMiniscript(ms),\n    scriptASM: utxolib.script.toASM(script).split(/\\s+/),\n  };\n}\n\nfunction parseScripts(scripts: unknown) {\n  if (typeof scripts !== 'object' || scripts === null) {\n    throw new Error('scripts must be an object');\n  }\n  return Object.fromEntries(Object.entries(scripts).map(([key, value]) => [key, parseScript(key, value)]));\n}\n\ntype EqualsAssertion = typeof assert.deepStrictEqual;\n\nasync function assertEqualsFixture(\n  fixtureName: string,\n  value: unknown,\n  n = normalize,\n  eq: EqualsAssertion = assert.deepStrictEqual\n): Promise<void> {\n  value = n(value);\n  eq(await getFixture(fixtureName, value), value);\n}\n\nasync function assertScriptsEqualFixture(\n  fixtureName: string,\n  builder: vendor.StakingScriptData,\n  scripts: unknown\n): Promise<void> {\n  await assertEqualsFixture(fixtureName, {\n    builder: toPlainObject(builder),\n    scripts: parseScripts(scripts),\n  });\n}\n\nasync function assertTransactionEqualsFixture(fixtureName: string, tx: unknown): Promise<void> {\n  await assertEqualsFixture(fixtureName, normalize(tx));\n}\n\nfunction assertEqualsMiniscript(script: Buffer, miniscript: ast.MiniscriptNode): void {\n  const ms = Miniscript.fromBitcoinScript(script, 'tap');\n  assert.deepStrictEqual(ast.fromMiniscript(ms), miniscript);\n  assert.deepStrictEqual(\n    script.toString('hex'),\n    Buffer.from(Miniscript.fromString(ast.formatNode(miniscript), 'tap').encode()).toString('hex')\n  );\n}\n\nfunction assertEqualScripts(descriptorBuilder: BabylonDescriptorBuilder, builder: vendor.StakingScripts) {\n  for (const [key, script] of Object.entries(builder) as [keyof vendor.StakingScripts, Buffer][]) {\n    switch (key) {\n      case 'timelockScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getTimelockMiniscript());\n        break;\n      case 'unbondingScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getUnbondingMiniscript());\n        break;\n      case 'slashingScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getSlashingMiniscript());\n        break;\n      case 'unbondingTimelockScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getUnbondingTimelockMiniscript());\n        break;\n      default:\n        throw new Error(`unexpected script key: ${key}`);\n    }\n  }\n}\n\nfunction assertEqualOutputScript(outputInfo: { scriptPubKey: Buffer }, descriptor: Descriptor) {\n  assert.strictEqual(outputInfo.scriptPubKey.toString('hex'), Buffer.from(descriptor.scriptPubkey()).toString('hex'));\n}\n\nfunction describeWithKeys(\n  tag: string,\n  finalityProviderKeys: ECPairInterface[],\n  covenantKeys: ECPairInterface[],\n  stakingParams: vendor.StakingParams,\n  { signIntermediateTxs = false } = {}\n) {\n  const stakerKey = getECKey('staker') as ECPairInterface & { privateKey: Buffer };\n  const covenantThreshold = stakingParams.covenantQuorum;\n  const stakingTimelock = stakingParams.minStakingTimeBlocks;\n  const unbondingTimelock = stakingParams.unbondingTime;\n  const vendorBuilder = new vendor.StakingScriptData(\n    getXOnlyPubkey(stakerKey),\n    finalityProviderKeys.map(getXOnlyPubkey),\n    covenantKeys.map(getXOnlyPubkey),\n    covenantThreshold,\n    stakingTimelock,\n    unbondingTimelock\n  );\n\n  const descriptorBuilder = new BabylonDescriptorBuilder(\n    getXOnlyPubkey(stakerKey),\n    finalityProviderKeys.map(getXOnlyPubkey),\n    covenantKeys.map(getXOnlyPubkey),\n    covenantThreshold,\n    stakingTimelock,\n    unbondingTimelock\n  );\n\n  describe(`Babylon Staking [${tag}]`, function () {\n    it('generates expected staking scripts', async function () {\n      await assertScriptsEqualFixture(\n        `test/fixtures/babylon/scripts.${tag}.json`,\n        vendorBuilder,\n        vendorBuilder.buildScripts()\n      );\n    });\n\n    it('matches inner taproot scripts', function () {\n      assertEqualScripts(descriptorBuilder, vendorBuilder.buildScripts());\n    });\n\n    it('matches output scripts', function () {\n      assertEqualOutputScript(\n        vendor.deriveStakingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin),\n        descriptorBuilder.getStakingDescriptor()\n      );\n      assertEqualOutputScript(\n        vendor.deriveSlashingOutput(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin),\n        descriptorBuilder.getSlashingDescriptor()\n      );\n      assertEqualOutputScript(\n        vendor.deriveUnbondingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin),\n        descriptorBuilder.getUnbondingDescriptor()\n      );\n    });\n\n    describe('Transaction Sets', async function () {\n      const stakerMainWalletKey = getECKey('stakerMainWallet');\n      const mainWallet = wpkhDescriptor(stakerMainWalletKey);\n      const amount = 55_555;\n      const changeAddress = createAddressFromDescriptor(mainWallet, undefined, utxolib.networks.bitcoin);\n      const feeRateSatB = 2;\n      const utxo = mockUtxo(mainWallet);\n\n      let stakingTx: vendor.TransactionResult;\n\n      before('setup stakingTx', function () {\n        stakingTx = vendor.stakingTransaction(\n          vendorBuilder.buildScripts(),\n          amount,\n          changeAddress,\n          [mockUtxo(mainWallet)],\n          bitcoinjslib.networks.bitcoin,\n          feeRateSatB\n        );\n      });\n\n      it('has expected transactions', async function () {\n        await assertTransactionEqualsFixture(`test/fixtures/babylon/stakingTransaction.${tag}.json`, stakingTx);\n\n        // simply one staking output and one change output\n        // nothing special\n        assert.deepStrictEqual(stakingTx.transaction.outs, [\n          {\n            script: Buffer.from(descriptorBuilder.getStakingDescriptor().scriptPubkey()),\n            value: amount,\n          },\n          {\n            script: utxolib.address.toOutputScript(changeAddress, utxolib.networks.bitcoin),\n            value: utxo.value - amount - stakingTx.fee,\n          },\n        ]);\n      });\n\n      if (finalityProviderKeys.length !== 1) {\n        return;\n      }\n\n      const finalityProvider = finalityProviderKeys[0];\n\n      it('has expected transactions (vendorStaking.Staking)', async function (this: Mocha.Context) {\n        const vendorStakingTxBuilder = new vendor.Staking(\n          bitcoinjslib.networks.bitcoin,\n          toStakerInfo(stakerKey, changeAddress),\n          stakingParams,\n          getXOnlyPubkey(finalityProvider).toString('hex'),\n          stakingParams.minStakingTimeBlocks\n        );\n\n        const txTree = getStakingTransactionTreeVendor(\n          vendorStakingTxBuilder,\n          amount,\n          [utxo],\n          feeRateSatB,\n          signIntermediateTxs\n            ? {\n                staker: stakerKey,\n                finalityProvider,\n                covenant: covenantKeys,\n                covenantThreshold: covenantThreshold,\n              }\n            : { staker: stakerKey },\n          descriptorBuilder\n        );\n        await assertTransactionEqualsFixture(`test/fixtures/babylon/txTree.${tag}.json`, txTree);\n      });\n\n      it('creates MsgCreateBTCDelegation', async function () {\n        type F = typeof getVendorMsgCreateBtcDelegation;\n        const fVendor: F = getVendorMsgCreateBtcDelegation;\n        const fBitGo: F = getBitGoUtxoStakingMsgCreateBtcDelegation;\n\n        for (const f of [fVendor, fBitGo]) {\n          await assertEqualsFixture(\n            `test/fixtures/babylon/msgCreateBTCDelegation.${tag}.json`,\n            await f(\n              bitcoinjslib.networks.bitcoin,\n              stakerKey,\n              finalityProvider,\n              descriptorBuilder,\n              [{ ...stakingParams, version: 0, btcActivationHeight: 0 }],\n              changeAddress,\n              amount,\n              utxo,\n              feeRateSatB,\n              800_000\n            ),\n            normalize,\n            (a, b) => {\n              // The vendor library serializes the signature as BIP322, while\n              // our implementation serializes it as ECDSA.\n              // Strip the pop field from the MsgCreateBTCDelegation.\n              function stripPop(v: unknown) {\n                const vAny = v as any;\n                delete vAny['unsignedDelegationMsg']['value']['pop'];\n              }\n              stripPop(a);\n              stripPop(b);\n              assert.deepStrictEqual(a, b);\n            }\n          );\n        }\n      });\n\n      it('creates unstaking transaction', async function () {\n        const unstaking = createUnstakingTransaction(\n          stakingTx,\n          descriptorBuilder.getStakingDescriptor(),\n          changeAddress,\n          { sequence: stakingParams.minStakingTimeBlocks }\n        );\n        const wrappedPsbt = toWrappedPsbt(unstaking);\n        assert(getNewSignatureCount(signWithKey(wrappedPsbt, stakerKey)) > 0);\n        wrappedPsbt.finalize();\n        const tx = toUtxoPsbt(wrappedPsbt, utxolib.networks.bitcoin).extractTransaction();\n        await assertTransactionEqualsFixture(`test/fixtures/babylon/unstakingTransaction.${tag}.json`, {\n          transaction: tx,\n        });\n      });\n    });\n  });\n}\n\nfunction describeWithKeysFromStakingParams(\n  tag: string,\n  finalityProviderKeys: ECPairInterface[],\n  stakingParams: vendor.StakingParams\n) {\n  describeWithKeys(\n    tag,\n    finalityProviderKeys,\n    stakingParams.covenantNoCoordPks.map((pk) => fromXOnlyPublicKey(Buffer.from(pk, 'hex'))),\n    stakingParams\n  );\n}\n\nfunction describeWithMockKeys(\n  tag: string,\n  stakingParams: vendor.StakingParams,\n  finalityProviderKeys: ECPairInterface[],\n  covenantKeys: ECPairInterface[]\n) {\n  describeWithKeys(\n    tag,\n    finalityProviderKeys,\n    covenantKeys,\n    getTestnetStakingParamsWithCovenant(stakingParams, covenantKeys),\n    {\n      signIntermediateTxs: true,\n    }\n  );\n}\n\ndescribeWithKeysFromStakingParams(\n  'testnet',\n  [fromXOnlyPublicKey(testnetFinalityProvider0)],\n  getBabylonParamByVersion(5, getStakingParams('testnet'))\n);\ndescribeWithMockKeys(\n  'testnetMock',\n  getBabylonParamByVersion(5, getStakingParams('testnet')),\n  getECKeys('finalityProvider', 1),\n  getECKeys('covenant', 9)\n);\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!