@bitgo-beta/utxo-staking 1.1.1-beta.91 → 1.1.1-beta.910
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/scripts/babylon-sync.d.ts +2 -0
- package/dist/cjs/scripts/babylon-sync.d.ts.map +1 -0
- package/dist/cjs/scripts/babylon-sync.js +126 -0
- package/dist/cjs/src/babylon/delegationMessage.d.ts +38 -0
- package/dist/cjs/src/babylon/delegationMessage.d.ts.map +1 -0
- package/dist/cjs/src/babylon/delegationMessage.js +285 -0
- package/dist/cjs/src/babylon/descriptor.d.ts +48 -0
- package/dist/cjs/src/babylon/descriptor.d.ts.map +1 -0
- package/dist/cjs/src/babylon/descriptor.js +97 -0
- package/dist/cjs/src/babylon/index.d.ts +8 -0
- package/dist/cjs/src/babylon/index.d.ts.map +1 -0
- package/dist/cjs/src/babylon/index.js +51 -0
- package/dist/cjs/src/babylon/network.d.ts +7 -0
- package/dist/cjs/src/babylon/network.d.ts.map +1 -0
- package/dist/cjs/src/babylon/network.js +66 -0
- package/dist/cjs/src/babylon/params.mainnet.json +152 -0
- package/dist/cjs/src/babylon/params.testnet.json +212 -0
- package/dist/cjs/src/babylon/parseDescriptor.d.ts +26 -0
- package/dist/cjs/src/babylon/parseDescriptor.d.ts.map +1 -0
- package/dist/cjs/src/babylon/parseDescriptor.js +159 -0
- package/dist/cjs/src/babylon/stakingManager.d.ts +6 -0
- package/dist/cjs/src/babylon/stakingManager.d.ts.map +1 -0
- package/dist/cjs/src/babylon/stakingManager.js +120 -0
- package/dist/cjs/src/babylon/stakingParams.d.ts +40 -0
- package/dist/cjs/src/babylon/stakingParams.d.ts.map +1 -0
- package/dist/cjs/src/babylon/stakingParams.js +132 -0
- package/dist/cjs/src/babylon/undelegation/UndelegationResponse.d.ts +24 -0
- package/dist/cjs/src/babylon/undelegation/UndelegationResponse.d.ts.map +1 -0
- package/dist/cjs/src/babylon/undelegation/UndelegationResponse.js +55 -0
- package/dist/cjs/src/babylon/undelegation/index.d.ts +3 -0
- package/dist/cjs/src/babylon/undelegation/index.d.ts.map +1 -0
- package/dist/{src → cjs/src/babylon/undelegation}/index.js +3 -4
- package/dist/cjs/src/babylon/undelegation/unbonding.d.ts +16 -0
- package/dist/cjs/src/babylon/undelegation/unbonding.d.ts.map +1 -0
- package/dist/cjs/src/babylon/undelegation/unbonding.js +111 -0
- package/dist/cjs/src/coreDao/descriptor.d.ts.map +1 -0
- package/dist/cjs/src/coreDao/descriptor.js +38 -0
- package/dist/cjs/src/coreDao/index.d.ts.map +1 -0
- package/dist/{src → cjs/src}/coreDao/index.js +1 -1
- package/dist/cjs/src/coreDao/opReturn.d.ts.map +1 -0
- package/dist/cjs/src/coreDao/opReturn.js +183 -0
- package/dist/{src → cjs/src}/coreDao/transaction.d.ts +16 -7
- package/dist/cjs/src/coreDao/transaction.d.ts.map +1 -0
- package/dist/cjs/src/coreDao/transaction.js +34 -0
- package/dist/cjs/src/index.d.ts +3 -0
- package/dist/cjs/src/index.d.ts.map +1 -0
- package/dist/cjs/src/index.js +39 -0
- package/dist/cjs/test/unit/babylon/bug71.d.ts +2 -0
- package/dist/cjs/test/unit/babylon/bug71.d.ts.map +1 -0
- package/dist/cjs/test/unit/babylon/bug71.js +108 -0
- package/dist/cjs/test/unit/babylon/key.utils.d.ts +6 -0
- package/dist/cjs/test/unit/babylon/key.utils.d.ts.map +1 -0
- package/dist/cjs/test/unit/babylon/key.utils.js +68 -0
- package/dist/cjs/test/unit/babylon/transactions.d.ts +2 -0
- package/dist/cjs/test/unit/babylon/transactions.d.ts.map +1 -0
- package/dist/cjs/test/unit/babylon/transactions.js +338 -0
- package/dist/cjs/test/unit/babylon/undelegation.d.ts +2 -0
- package/dist/cjs/test/unit/babylon/undelegation.d.ts.map +1 -0
- package/dist/cjs/test/unit/babylon/undelegation.js +156 -0
- package/dist/cjs/test/unit/babylon/vendor.utils.d.ts +13 -0
- package/dist/cjs/test/unit/babylon/vendor.utils.d.ts.map +1 -0
- package/dist/cjs/test/unit/babylon/vendor.utils.js +78 -0
- package/dist/cjs/test/unit/coreDao/descriptor.d.ts +2 -0
- package/dist/cjs/test/unit/coreDao/descriptor.d.ts.map +1 -0
- package/dist/cjs/test/unit/coreDao/descriptor.js +144 -0
- package/dist/cjs/test/unit/coreDao/opReturn.d.ts +2 -0
- package/dist/cjs/test/unit/coreDao/opReturn.d.ts.map +1 -0
- package/dist/cjs/test/unit/coreDao/opReturn.js +275 -0
- package/dist/cjs/test/unit/coreDao/utils.d.ts +5 -0
- package/dist/cjs/test/unit/coreDao/utils.d.ts.map +1 -0
- package/dist/cjs/test/unit/coreDao/utils.js +54 -0
- package/dist/cjs/test/unit/fixtures.utils.d.ts +9 -0
- package/dist/cjs/test/unit/fixtures.utils.d.ts.map +1 -0
- package/dist/cjs/test/unit/fixtures.utils.js +91 -0
- package/dist/cjs/tsconfig.tsbuildinfo +1 -0
- package/dist/esm/babylon/delegationMessage.d.ts +38 -0
- package/dist/esm/babylon/delegationMessage.d.ts.map +1 -0
- package/dist/esm/babylon/delegationMessage.js +239 -0
- package/dist/esm/babylon/descriptor.d.ts +48 -0
- package/dist/esm/babylon/descriptor.d.ts.map +1 -0
- package/dist/esm/babylon/descriptor.js +91 -0
- package/dist/esm/babylon/index.d.ts +8 -0
- package/dist/esm/babylon/index.d.ts.map +1 -0
- package/dist/esm/babylon/index.js +13 -0
- package/dist/esm/babylon/network.d.ts +7 -0
- package/dist/esm/babylon/network.d.ts.map +1 -0
- package/dist/esm/babylon/network.js +29 -0
- package/dist/esm/babylon/params.mainnet.json +152 -0
- package/dist/esm/babylon/params.testnet.json +212 -0
- package/dist/esm/babylon/parseDescriptor.d.ts +26 -0
- package/dist/esm/babylon/parseDescriptor.d.ts.map +1 -0
- package/dist/esm/babylon/parseDescriptor.js +155 -0
- package/dist/esm/babylon/stakingManager.d.ts +6 -0
- package/dist/esm/babylon/stakingManager.d.ts.map +1 -0
- package/dist/esm/babylon/stakingManager.js +83 -0
- package/dist/esm/babylon/stakingParams.d.ts +40 -0
- package/dist/esm/babylon/stakingParams.d.ts.map +1 -0
- package/dist/esm/babylon/stakingParams.js +86 -0
- package/dist/esm/babylon/undelegation/UndelegationResponse.d.ts +24 -0
- package/dist/esm/babylon/undelegation/UndelegationResponse.d.ts.map +1 -0
- package/dist/esm/babylon/undelegation/UndelegationResponse.js +18 -0
- package/dist/esm/babylon/undelegation/index.d.ts +3 -0
- package/dist/esm/babylon/undelegation/index.d.ts.map +1 -0
- package/dist/esm/babylon/undelegation/index.js +3 -0
- package/dist/esm/babylon/undelegation/unbonding.d.ts +16 -0
- package/dist/esm/babylon/undelegation/unbonding.d.ts.map +1 -0
- package/dist/esm/babylon/undelegation/unbonding.js +70 -0
- package/dist/esm/coreDao/descriptor.d.ts +18 -0
- package/dist/{src → esm}/coreDao/descriptor.d.ts.map +1 -1
- package/dist/esm/coreDao/descriptor.js +35 -0
- package/dist/esm/coreDao/index.d.ts +4 -0
- package/dist/esm/coreDao/index.js +4 -0
- package/dist/esm/coreDao/opReturn.d.ts +41 -0
- package/dist/esm/coreDao/opReturn.js +175 -0
- package/dist/esm/coreDao/transaction.d.ts +28 -0
- package/dist/esm/coreDao/transaction.d.ts.map +1 -0
- package/dist/esm/coreDao/transaction.js +30 -0
- package/dist/esm/index.d.ts +3 -0
- package/dist/{src → esm}/index.d.ts.map +1 -1
- package/dist/esm/index.js +3 -0
- package/package.json +42 -12
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -63
- package/dist/src/coreDao/descriptor.js +0 -37
- package/dist/src/coreDao/opReturn.js +0 -183
- package/dist/src/coreDao/transaction.d.ts.map +0 -1
- package/dist/src/coreDao/transaction.js +0 -26
- package/dist/src/index.d.ts +0 -3
- package/dist/src/transaction.d.ts +0 -32
- package/dist/src/transaction.d.ts.map +0 -1
- package/dist/src/transaction.js +0 -54
- /package/dist/{src → cjs/src}/coreDao/descriptor.d.ts +0 -0
- /package/dist/{src → cjs/src}/coreDao/index.d.ts +0 -0
- /package/dist/{src → cjs/src}/coreDao/opReturn.d.ts +0 -0
- /package/dist/{src → esm}/coreDao/index.d.ts.map +0 -0
- /package/dist/{src → esm}/coreDao/opReturn.d.ts.map +0 -0
|
@@ -0,0 +1,338 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const assert_1 = __importDefault(require("assert"));
|
|
40
|
+
const vendor = __importStar(require("@bitgo-beta/babylonlabs-io-btc-staking-ts"));
|
|
41
|
+
const bitcoinjslib = __importStar(require("bitcoinjs-lib"));
|
|
42
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
43
|
+
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
44
|
+
const descriptor_1 = require("@bitgo-beta/utxo-core/descriptor");
|
|
45
|
+
const testutil_1 = require("@bitgo-beta/utxo-core/testutil");
|
|
46
|
+
const babylonlabs_io_btc_staking_ts_1 = require("@bitgo-beta/babylonlabs-io-btc-staking-ts");
|
|
47
|
+
const babylon_1 = require("../../../src/babylon");
|
|
48
|
+
const parseDescriptor_1 = require("../../../src/babylon/parseDescriptor");
|
|
49
|
+
const fixtures_utils_1 = require("../fixtures.utils");
|
|
50
|
+
const key_utils_1 = require("./key.utils");
|
|
51
|
+
const vendor_utils_1 = require("./vendor.utils");
|
|
52
|
+
function getStakingTransactionTreeVendor(builder, amount, utxos, feeRateSatB, signers, descriptorBuilder) {
|
|
53
|
+
const staking = builder.createStakingTransaction(amount, utxos, feeRateSatB);
|
|
54
|
+
const stakingWithdraw = builder.createWithdrawStakingExpiredPsbt(staking.transaction, feeRateSatB);
|
|
55
|
+
const unbonding = builder.createUnbondingTransaction(staking.transaction);
|
|
56
|
+
const unbondingWithdraw = builder.createWithdrawEarlyUnbondedTransaction(unbonding.transaction, feeRateSatB);
|
|
57
|
+
const unbondingSlashing = builder.createUnbondingOutputSlashingPsbt(unbonding.transaction);
|
|
58
|
+
const signSequence = [signers.staker];
|
|
59
|
+
if ('finalityProvider' in signers) {
|
|
60
|
+
signSequence.push(signers.finalityProvider, ...signers.covenant);
|
|
61
|
+
}
|
|
62
|
+
const unbondingSlashingWithdraw = signSequence
|
|
63
|
+
? builder.createWithdrawSlashingPsbt((0, babylon_1.forceFinalizePsbt)((0, babylon_1.getSignedPsbt)(unbondingSlashing.psbt, descriptorBuilder.getUnbondingDescriptor(), signSequence, {
|
|
64
|
+
finalize: false,
|
|
65
|
+
}), builder.network).extractTransaction(), feeRateSatB)
|
|
66
|
+
: undefined;
|
|
67
|
+
const slashing = builder.createStakingOutputSlashingPsbt(staking.transaction);
|
|
68
|
+
const slashingSigned = signSequence
|
|
69
|
+
? (0, babylon_1.getSignedPsbt)(slashing.psbt, descriptorBuilder.getStakingDescriptor(), signSequence, {
|
|
70
|
+
finalize: false,
|
|
71
|
+
})
|
|
72
|
+
: undefined;
|
|
73
|
+
const slashingWithdraw = slashingSigned
|
|
74
|
+
? builder.createWithdrawSlashingPsbt((0, babylon_1.forceFinalizePsbt)(slashingSigned.toBuffer(), builder.network).extractTransaction(), feeRateSatB)
|
|
75
|
+
: undefined;
|
|
76
|
+
return {
|
|
77
|
+
staking,
|
|
78
|
+
stakingWithdraw,
|
|
79
|
+
unbonding,
|
|
80
|
+
unbondingWithdraw,
|
|
81
|
+
unbondingSlashing,
|
|
82
|
+
unbondingSlashingWithdraw,
|
|
83
|
+
slashing,
|
|
84
|
+
slashingSigned,
|
|
85
|
+
slashingSignedBase64: slashingSigned?.toBuffer().toString('base64'),
|
|
86
|
+
slashingWithdraw,
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
function spendStakingOutput(stakingTx, descriptorBuilder, type, changeAddress, { sequence }) {
|
|
90
|
+
const network = utxolib.networks.bitcoin;
|
|
91
|
+
const witnessUtxoNumber = stakingTx.transaction.outs[0];
|
|
92
|
+
const witnessUtxo = {
|
|
93
|
+
script: witnessUtxoNumber.script,
|
|
94
|
+
value: BigInt(witnessUtxoNumber.value),
|
|
95
|
+
};
|
|
96
|
+
const selectTapLeafScript = wasm_utxo_1.Miniscript.fromString(wasm_utxo_1.ast.formatNode(type === 'unstaking'
|
|
97
|
+
? descriptorBuilder.getStakingTimelockMiniscriptNode()
|
|
98
|
+
: descriptorBuilder.getUnbondingMiniscriptNode()), 'tap');
|
|
99
|
+
return (0, descriptor_1.createPsbt)({
|
|
100
|
+
network,
|
|
101
|
+
}, [
|
|
102
|
+
{
|
|
103
|
+
hash: stakingTx.transaction.getId(),
|
|
104
|
+
index: 0,
|
|
105
|
+
witnessUtxo,
|
|
106
|
+
descriptor: descriptorBuilder.getStakingDescriptor(),
|
|
107
|
+
selectTapLeafScript,
|
|
108
|
+
sequence,
|
|
109
|
+
},
|
|
110
|
+
], [
|
|
111
|
+
{
|
|
112
|
+
script: utxolib.address.toOutputScript(changeAddress, network),
|
|
113
|
+
value: BigInt(witnessUtxoNumber.value) - 1000n,
|
|
114
|
+
},
|
|
115
|
+
]);
|
|
116
|
+
}
|
|
117
|
+
function getTestnetStakingParamsWithCovenant(params, covenantKeys) {
|
|
118
|
+
return {
|
|
119
|
+
...params,
|
|
120
|
+
covenantNoCoordPks: covenantKeys.map((pk) => (0, key_utils_1.getXOnlyPubkey)(pk).toString('hex')),
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
function wpkhDescriptor(key) {
|
|
124
|
+
return wasm_utxo_1.Descriptor.fromString(wasm_utxo_1.ast.formatNode({ wpkh: key.publicKey.toString('hex') }), 'definite');
|
|
125
|
+
}
|
|
126
|
+
function mockUtxo(descriptor) {
|
|
127
|
+
const scriptPubKey = Buffer.from(descriptor.scriptPubkey());
|
|
128
|
+
const witnessScript = Buffer.from(descriptor.encode());
|
|
129
|
+
return {
|
|
130
|
+
rawTxHex: undefined,
|
|
131
|
+
txid: Buffer.alloc(32).fill(0x11).toString('hex'),
|
|
132
|
+
value: 666666,
|
|
133
|
+
vout: 0,
|
|
134
|
+
redeemScript: undefined,
|
|
135
|
+
witnessScript: witnessScript.toString('hex'),
|
|
136
|
+
scriptPubKey: scriptPubKey.toString('hex'),
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
function parseScript(key, script) {
|
|
140
|
+
if (!Buffer.isBuffer(script)) {
|
|
141
|
+
throw new Error('script must be a buffer');
|
|
142
|
+
}
|
|
143
|
+
const ms = wasm_utxo_1.Miniscript.fromBitcoinScript(script, 'tap');
|
|
144
|
+
return {
|
|
145
|
+
script: script.toString('hex'),
|
|
146
|
+
miniscript: ms.toString(),
|
|
147
|
+
miniscriptAst: wasm_utxo_1.ast.fromMiniscript(ms),
|
|
148
|
+
scriptASM: utxolib.script.toASM(script).split(/\s+/),
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
function parseScripts(scripts) {
|
|
152
|
+
if (typeof scripts !== 'object' || scripts === null) {
|
|
153
|
+
throw new Error('scripts must be an object');
|
|
154
|
+
}
|
|
155
|
+
return Object.fromEntries(Object.entries(scripts).map(([key, value]) => [key, parseScript(key, value)]));
|
|
156
|
+
}
|
|
157
|
+
async function assertScriptsEqualFixture(fixtureName, builder, scripts) {
|
|
158
|
+
await (0, fixtures_utils_1.assertEqualsFixture)(fixtureName, {
|
|
159
|
+
builder: (0, testutil_1.toPlainObject)(builder),
|
|
160
|
+
scripts: parseScripts(scripts),
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
function assertEqualScripts(descriptorBuilder, builder) {
|
|
164
|
+
for (const [key, script] of Object.entries(builder)) {
|
|
165
|
+
switch (key) {
|
|
166
|
+
case 'timelockScript':
|
|
167
|
+
(0, fixtures_utils_1.assertEqualsMiniscript)(script, descriptorBuilder.getStakingTimelockMiniscriptNode());
|
|
168
|
+
break;
|
|
169
|
+
case 'unbondingScript':
|
|
170
|
+
(0, fixtures_utils_1.assertEqualsMiniscript)(script, descriptorBuilder.getUnbondingMiniscriptNode());
|
|
171
|
+
break;
|
|
172
|
+
case 'slashingScript':
|
|
173
|
+
(0, fixtures_utils_1.assertEqualsMiniscript)(script, descriptorBuilder.getSlashingMiniscriptNode());
|
|
174
|
+
break;
|
|
175
|
+
case 'unbondingTimelockScript':
|
|
176
|
+
(0, fixtures_utils_1.assertEqualsMiniscript)(script, descriptorBuilder.getUnbondingTimelockMiniscriptNode());
|
|
177
|
+
break;
|
|
178
|
+
default:
|
|
179
|
+
throw new Error(`unexpected script key: ${key}`);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
function assertEqualOutputScript(outputInfo, descriptor) {
|
|
184
|
+
assert_1.default.strictEqual(outputInfo.scriptPubKey.toString('hex'), Buffer.from(descriptor.scriptPubkey()).toString('hex'));
|
|
185
|
+
}
|
|
186
|
+
function describeWithKeys(tag, finalityProviderKeys, covenantKeys, stakingParams, { signIntermediateTxs = false } = {}) {
|
|
187
|
+
const stakerKey = (0, key_utils_1.getECKey)('staker');
|
|
188
|
+
const covenantThreshold = stakingParams.covenantQuorum;
|
|
189
|
+
const stakingTimelock = stakingParams.minStakingTimeBlocks;
|
|
190
|
+
const unbondingTimelock = stakingParams.unbondingTime;
|
|
191
|
+
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);
|
|
192
|
+
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);
|
|
193
|
+
describe(`Babylon Staking [${tag}]`, function () {
|
|
194
|
+
it('generates expected staking scripts', async function () {
|
|
195
|
+
await assertScriptsEqualFixture(`test/fixtures/babylon/scripts.${tag}.json`, vendorBuilder, vendorBuilder.buildScripts());
|
|
196
|
+
});
|
|
197
|
+
it('matches inner taproot scripts', function () {
|
|
198
|
+
assertEqualScripts(descriptorBuilder, vendorBuilder.buildScripts());
|
|
199
|
+
});
|
|
200
|
+
it('matches output scripts', function () {
|
|
201
|
+
assertEqualOutputScript(vendor.deriveStakingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getStakingDescriptor());
|
|
202
|
+
assertEqualOutputScript(
|
|
203
|
+
/* I don't know why this is called deriveSlashingOutput */
|
|
204
|
+
vendor.deriveSlashingOutput(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getUnbondingTimelockDescriptor());
|
|
205
|
+
assertEqualOutputScript(vendor.deriveUnbondingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin), descriptorBuilder.getUnbondingDescriptor());
|
|
206
|
+
});
|
|
207
|
+
it('round-trip parseStakingDescriptor', function () {
|
|
208
|
+
const descriptor = descriptorBuilder.getStakingDescriptor();
|
|
209
|
+
const parsed = (0, parseDescriptor_1.parseStakingDescriptor)(descriptor);
|
|
210
|
+
(0, assert_1.default)(parsed);
|
|
211
|
+
assert_1.default.deepStrictEqual(parsed.slashingMiniscriptNode, descriptorBuilder.getSlashingMiniscriptNode());
|
|
212
|
+
assert_1.default.deepStrictEqual(parsed.unbondingMiniscriptNode, descriptorBuilder.getUnbondingMiniscriptNode());
|
|
213
|
+
assert_1.default.deepStrictEqual(parsed.timelockMiniscriptNode, descriptorBuilder.getStakingTimelockMiniscriptNode());
|
|
214
|
+
assert_1.default.strictEqual((0, parseDescriptor_1.parseStakingDescriptor)(descriptorBuilder.getUnbondingDescriptor()), null);
|
|
215
|
+
assert_1.default.strictEqual((0, parseDescriptor_1.parseStakingDescriptor)(descriptorBuilder.getUnbondingTimelockDescriptor()), null);
|
|
216
|
+
});
|
|
217
|
+
it('round-trip parseUnbondingDescriptor', function () {
|
|
218
|
+
const descriptor = descriptorBuilder.getUnbondingDescriptor();
|
|
219
|
+
const parsed = (0, parseDescriptor_1.parseUnbondingDescriptor)(descriptor);
|
|
220
|
+
(0, assert_1.default)(parsed);
|
|
221
|
+
assert_1.default.deepStrictEqual(parsed.slashingMiniscriptNode, descriptorBuilder.getSlashingMiniscriptNode());
|
|
222
|
+
assert_1.default.deepStrictEqual(parsed.unbondingTimelockMiniscriptNode, descriptorBuilder.getUnbondingTimelockMiniscriptNode());
|
|
223
|
+
assert_1.default.deepStrictEqual(parsed.stakerKey, descriptorBuilder.stakerKey);
|
|
224
|
+
assert_1.default.deepStrictEqual((0, babylon_1.sortedKeys)(parsed.finalityProviderKeys), (0, babylon_1.sortedKeys)(descriptorBuilder.finalityProviderKeys));
|
|
225
|
+
assert_1.default.deepStrictEqual((0, babylon_1.sortedKeys)(parsed.covenantKeys), (0, babylon_1.sortedKeys)(descriptorBuilder.covenantKeys));
|
|
226
|
+
assert_1.default.strictEqual(parsed.covenantThreshold, descriptorBuilder.covenantThreshold);
|
|
227
|
+
assert_1.default.strictEqual(parsed.unbondingTimeLock, descriptorBuilder.unbondingTimeLock);
|
|
228
|
+
assert_1.default.strictEqual((0, parseDescriptor_1.parseUnbondingDescriptor)(descriptorBuilder.getStakingDescriptor()), null);
|
|
229
|
+
assert_1.default.strictEqual((0, parseDescriptor_1.parseUnbondingDescriptor)(descriptorBuilder.getUnbondingTimelockDescriptor()), null);
|
|
230
|
+
});
|
|
231
|
+
describe('Transaction Sets', async function () {
|
|
232
|
+
const stakerMainWalletKey = (0, key_utils_1.getECKey)('stakerMainWallet');
|
|
233
|
+
const mainWallet = wpkhDescriptor(stakerMainWalletKey);
|
|
234
|
+
const amount = 55555;
|
|
235
|
+
const changeAddress = (0, descriptor_1.createAddressFromDescriptor)(mainWallet, undefined, utxolib.networks.bitcoin);
|
|
236
|
+
const feeRateSatB = 2;
|
|
237
|
+
const utxo = mockUtxo(mainWallet);
|
|
238
|
+
let stakingTx;
|
|
239
|
+
before('setup stakingTx', function () {
|
|
240
|
+
stakingTx = vendor.stakingTransaction(vendorBuilder.buildScripts(), amount, changeAddress, [mockUtxo(mainWallet)], bitcoinjslib.networks.bitcoin, feeRateSatB);
|
|
241
|
+
});
|
|
242
|
+
it('has expected transactions', async function () {
|
|
243
|
+
await (0, fixtures_utils_1.assertTransactionEqualsFixture)(`test/fixtures/babylon/stakingTransaction.${tag}.json`, stakingTx);
|
|
244
|
+
// simply one staking output and one change output
|
|
245
|
+
// nothing special
|
|
246
|
+
assert_1.default.deepStrictEqual(stakingTx.transaction.outs, [
|
|
247
|
+
{
|
|
248
|
+
script: Buffer.from(descriptorBuilder.getStakingDescriptor().scriptPubkey()),
|
|
249
|
+
value: amount,
|
|
250
|
+
},
|
|
251
|
+
{
|
|
252
|
+
script: utxolib.address.toOutputScript(changeAddress, utxolib.networks.bitcoin),
|
|
253
|
+
value: utxo.value - amount - stakingTx.fee,
|
|
254
|
+
},
|
|
255
|
+
]);
|
|
256
|
+
});
|
|
257
|
+
if (finalityProviderKeys.length !== 1) {
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
const finalityProvider = finalityProviderKeys[0];
|
|
261
|
+
it('has expected transactions (vendorStaking.Staking)', async function () {
|
|
262
|
+
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);
|
|
263
|
+
const txTree = getStakingTransactionTreeVendor(vendorStakingTxBuilder, amount, [utxo], feeRateSatB, signIntermediateTxs
|
|
264
|
+
? {
|
|
265
|
+
staker: stakerKey,
|
|
266
|
+
finalityProvider,
|
|
267
|
+
covenant: covenantKeys,
|
|
268
|
+
covenantThreshold: covenantThreshold,
|
|
269
|
+
}
|
|
270
|
+
: { staker: stakerKey }, descriptorBuilder);
|
|
271
|
+
await (0, fixtures_utils_1.assertTransactionEqualsFixture)(`test/fixtures/babylon/txTree.${tag}.json`, txTree);
|
|
272
|
+
});
|
|
273
|
+
it('creates MsgCreateBTCDelegation', async function () {
|
|
274
|
+
const fVendor = vendor_utils_1.getVendorMsgCreateBtcDelegation;
|
|
275
|
+
const fBitGo = vendor_utils_1.getBitGoUtxoStakingMsgCreateBtcDelegation;
|
|
276
|
+
for (const f of [fVendor, fBitGo]) {
|
|
277
|
+
await (0, fixtures_utils_1.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) => {
|
|
278
|
+
// The vendor library serializes the signature as BIP322, while
|
|
279
|
+
// our implementation serializes it as ECDSA.
|
|
280
|
+
// Strip the pop field from the MsgCreateBTCDelegation.
|
|
281
|
+
function stripPop(v) {
|
|
282
|
+
const vAny = v;
|
|
283
|
+
delete vAny['unsignedDelegationMsg']['value']['pop'];
|
|
284
|
+
}
|
|
285
|
+
stripPop(a);
|
|
286
|
+
stripPop(b);
|
|
287
|
+
assert_1.default.deepStrictEqual(a, b);
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
});
|
|
291
|
+
async function testCreateTransaction(type, params) {
|
|
292
|
+
const unstakingPsbt = spendStakingOutput(stakingTx, descriptorBuilder, type, changeAddress, params);
|
|
293
|
+
const wrappedPsbt = (0, descriptor_1.toWrappedPsbt)(unstakingPsbt);
|
|
294
|
+
params.signers.forEach((signer) => {
|
|
295
|
+
(0, assert_1.default)((0, descriptor_1.getNewSignatureCount)((0, descriptor_1.signWithKey)(wrappedPsbt, signer)) > 0);
|
|
296
|
+
});
|
|
297
|
+
if (!params.finalize) {
|
|
298
|
+
return;
|
|
299
|
+
}
|
|
300
|
+
wrappedPsbt.finalize();
|
|
301
|
+
const tx = (0, descriptor_1.toUtxoPsbt)(wrappedPsbt, utxolib.networks.bitcoin).extractTransaction();
|
|
302
|
+
await (0, fixtures_utils_1.assertTransactionEqualsFixture)(`test/fixtures/babylon/${type}Transaction.${tag}.json`, {
|
|
303
|
+
psbt: unstakingPsbt,
|
|
304
|
+
transaction: tx,
|
|
305
|
+
});
|
|
306
|
+
}
|
|
307
|
+
it('creates unstaking transaction', async function () {
|
|
308
|
+
await testCreateTransaction('unstaking', {
|
|
309
|
+
sequence: stakingParams.minStakingTimeBlocks,
|
|
310
|
+
signers: [stakerKey],
|
|
311
|
+
finalize: true,
|
|
312
|
+
});
|
|
313
|
+
});
|
|
314
|
+
it('creates unbonding transaction', async function () {
|
|
315
|
+
await testCreateTransaction('unbonding', signIntermediateTxs
|
|
316
|
+
? {
|
|
317
|
+
signers: [stakerKey, ...covenantKeys],
|
|
318
|
+
finalize: true,
|
|
319
|
+
}
|
|
320
|
+
: {
|
|
321
|
+
signers: [stakerKey],
|
|
322
|
+
finalize: false,
|
|
323
|
+
});
|
|
324
|
+
});
|
|
325
|
+
});
|
|
326
|
+
});
|
|
327
|
+
}
|
|
328
|
+
function describeWithKeysFromStakingParams(tag, finalityProviderKeys, stakingParams) {
|
|
329
|
+
describeWithKeys(tag, finalityProviderKeys, stakingParams.covenantNoCoordPks.map((pk) => (0, key_utils_1.fromXOnlyPublicKey)(Buffer.from(pk, 'hex'))), stakingParams);
|
|
330
|
+
}
|
|
331
|
+
function describeWithMockKeys(tag, stakingParams, finalityProviderKeys, covenantKeys) {
|
|
332
|
+
describeWithKeys(tag, finalityProviderKeys, covenantKeys, getTestnetStakingParamsWithCovenant(stakingParams, covenantKeys), {
|
|
333
|
+
signIntermediateTxs: true,
|
|
334
|
+
});
|
|
335
|
+
}
|
|
336
|
+
describeWithKeysFromStakingParams('testnet', [(0, key_utils_1.fromXOnlyPublicKey)(babylon_1.testnetFinalityProvider0)], (0, babylonlabs_io_btc_staking_ts_1.getBabylonParamByVersion)(5, (0, babylon_1.getStakingParams)('testnet')));
|
|
337
|
+
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));
|
|
338
|
+
//# 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-beta/babylonlabs-io-btc-staking-ts';\nimport * as bitcoinjslib from 'bitcoinjs-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { ECPairInterface } from '@bitgo-beta/utxo-lib';\nimport { ast, Descriptor, Miniscript } from '@bitgo/wasm-utxo';\nimport {\n  createAddressFromDescriptor,\n  createPsbt,\n  getNewSignatureCount,\n  signWithKey,\n  toUtxoPsbt,\n  toWrappedPsbt,\n} from '@bitgo-beta/utxo-core/descriptor';\nimport { toPlainObject } from '@bitgo-beta/utxo-core/testutil';\nimport { getBabylonParamByVersion } from '@bitgo-beta/babylonlabs-io-btc-staking-ts';\n\nimport {\n  BabylonDescriptorBuilder,\n  testnetFinalityProvider0,\n  getSignedPsbt,\n  getStakingParams,\n  toStakerInfo,\n  forceFinalizePsbt,\n  sortedKeys,\n} from '../../../src/babylon';\nimport { parseStakingDescriptor, parseUnbondingDescriptor } from '../../../src/babylon/parseDescriptor';\nimport {\n  normalize,\n  assertEqualsFixture,\n  assertEqualsMiniscript,\n  assertTransactionEqualsFixture,\n} 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 spendStakingOutput(\n  stakingTx: vendor.TransactionResult,\n  descriptorBuilder: BabylonDescriptorBuilder,\n  type: 'unstaking' | 'unbonding',\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  const selectTapLeafScript = Miniscript.fromString(\n    ast.formatNode(\n      type === 'unstaking'\n        ? descriptorBuilder.getStakingTimelockMiniscriptNode()\n        : descriptorBuilder.getUnbondingMiniscriptNode()\n    ),\n    'tap'\n  );\n  return createPsbt(\n    {\n      network,\n    },\n    [\n      {\n        hash: stakingTx.transaction.getId(),\n        index: 0,\n        witnessUtxo,\n        descriptor: descriptorBuilder.getStakingDescriptor(),\n        selectTapLeafScript,\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\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\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.getStakingTimelockMiniscriptNode());\n        break;\n      case 'unbondingScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getUnbondingMiniscriptNode());\n        break;\n      case 'slashingScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getSlashingMiniscriptNode());\n        break;\n      case 'unbondingTimelockScript':\n        assertEqualsMiniscript(script, descriptorBuilder.getUnbondingTimelockMiniscriptNode());\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        /* I don't know why this is called deriveSlashingOutput */\n        vendor.deriveSlashingOutput(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin),\n        descriptorBuilder.getUnbondingTimelockDescriptor()\n      );\n      assertEqualOutputScript(\n        vendor.deriveUnbondingOutputInfo(vendorBuilder.buildScripts(), bitcoinjslib.networks.bitcoin),\n        descriptorBuilder.getUnbondingDescriptor()\n      );\n    });\n\n    it('round-trip parseStakingDescriptor', function () {\n      const descriptor = descriptorBuilder.getStakingDescriptor();\n      const parsed = parseStakingDescriptor(descriptor);\n\n      assert(parsed);\n      assert.deepStrictEqual(parsed.slashingMiniscriptNode, descriptorBuilder.getSlashingMiniscriptNode());\n      assert.deepStrictEqual(parsed.unbondingMiniscriptNode, descriptorBuilder.getUnbondingMiniscriptNode());\n      assert.deepStrictEqual(parsed.timelockMiniscriptNode, descriptorBuilder.getStakingTimelockMiniscriptNode());\n\n      assert.strictEqual(parseStakingDescriptor(descriptorBuilder.getUnbondingDescriptor()), null);\n      assert.strictEqual(parseStakingDescriptor(descriptorBuilder.getUnbondingTimelockDescriptor()), null);\n    });\n\n    it('round-trip parseUnbondingDescriptor', function () {\n      const descriptor = descriptorBuilder.getUnbondingDescriptor();\n      const parsed = parseUnbondingDescriptor(descriptor);\n\n      assert(parsed);\n      assert.deepStrictEqual(parsed.slashingMiniscriptNode, descriptorBuilder.getSlashingMiniscriptNode());\n      assert.deepStrictEqual(\n        parsed.unbondingTimelockMiniscriptNode,\n        descriptorBuilder.getUnbondingTimelockMiniscriptNode()\n      );\n      assert.deepStrictEqual(parsed.stakerKey, descriptorBuilder.stakerKey);\n      assert.deepStrictEqual(\n        sortedKeys(parsed.finalityProviderKeys),\n        sortedKeys(descriptorBuilder.finalityProviderKeys)\n      );\n      assert.deepStrictEqual(sortedKeys(parsed.covenantKeys), sortedKeys(descriptorBuilder.covenantKeys));\n      assert.strictEqual(parsed.covenantThreshold, descriptorBuilder.covenantThreshold);\n      assert.strictEqual(parsed.unbondingTimeLock, descriptorBuilder.unbondingTimeLock);\n\n      assert.strictEqual(parseUnbondingDescriptor(descriptorBuilder.getStakingDescriptor()), null);\n      assert.strictEqual(parseUnbondingDescriptor(descriptorBuilder.getUnbondingTimelockDescriptor()), null);\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      async function testCreateTransaction(\n        type: 'unstaking' | 'unbonding',\n        params: {\n          sequence?: number;\n          signers: utxolib.ECPairInterface[];\n          finalize: boolean;\n        }\n      ) {\n        const unstakingPsbt = spendStakingOutput(stakingTx, descriptorBuilder, type, changeAddress, params);\n        const wrappedPsbt = toWrappedPsbt(unstakingPsbt);\n        params.signers.forEach((signer) => {\n          assert(getNewSignatureCount(signWithKey(wrappedPsbt, signer)) > 0);\n        });\n        if (!params.finalize) {\n          return;\n        }\n        wrappedPsbt.finalize();\n        const tx = toUtxoPsbt(wrappedPsbt, utxolib.networks.bitcoin).extractTransaction();\n        await assertTransactionEqualsFixture(`test/fixtures/babylon/${type}Transaction.${tag}.json`, {\n          psbt: unstakingPsbt,\n          transaction: tx,\n        });\n      }\n\n      it('creates unstaking transaction', async function () {\n        await testCreateTransaction('unstaking', {\n          sequence: stakingParams.minStakingTimeBlocks,\n          signers: [stakerKey],\n          finalize: true,\n        });\n      });\n\n      it('creates unbonding transaction', async function () {\n        await testCreateTransaction(\n          'unbonding',\n          signIntermediateTxs\n            ? {\n                signers: [stakerKey, ...covenantKeys],\n                finalize: true,\n              }\n            : {\n                signers: [stakerKey],\n                finalize: false,\n              }\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"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"undelegation.d.ts","sourceRoot":"","sources":["../../../../../test/unit/babylon/undelegation.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const assert_1 = __importDefault(require("assert"));
|
|
40
|
+
const fs_1 = require("fs");
|
|
41
|
+
const t = __importStar(require("io-ts"));
|
|
42
|
+
const PathReporter_1 = require("io-ts/lib/PathReporter");
|
|
43
|
+
const Either_1 = require("fp-ts/Either");
|
|
44
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
45
|
+
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
46
|
+
const babylon_1 = require("../../../src/babylon");
|
|
47
|
+
const undelegation_1 = require("../../../src/babylon/undelegation");
|
|
48
|
+
const parseDescriptor_1 = require("../../../src/babylon/parseDescriptor");
|
|
49
|
+
const fixtures_utils_1 = require("../fixtures.utils");
|
|
50
|
+
const BtcDelegation = t.type({
|
|
51
|
+
unbonding_time: t.number,
|
|
52
|
+
undelegation_response: undelegation_1.UndelegationResponse,
|
|
53
|
+
});
|
|
54
|
+
async function getFixture(txid) {
|
|
55
|
+
// As returned by https://babylon.nodes.guru/api#/Query/BTCDelegation
|
|
56
|
+
const BtcDelegationResponse = t.type({ btc_delegation: BtcDelegation }, 'BtcDelegationResponse');
|
|
57
|
+
const filename = __dirname + `/../../fixtures/babylon/rpc/btc_delegation/testnet.${txid}.json`;
|
|
58
|
+
const data = JSON.parse(await fs_1.promises.readFile(filename, 'utf8'));
|
|
59
|
+
const result = BtcDelegationResponse.decode(data);
|
|
60
|
+
if ((0, Either_1.isLeft)(result)) {
|
|
61
|
+
throw new Error(`Failed to decode fixture data for txid ${txid}: ${PathReporter_1.PathReporter.report(result).join(', ')}`);
|
|
62
|
+
}
|
|
63
|
+
return result.right.btc_delegation;
|
|
64
|
+
}
|
|
65
|
+
function toDescriptor(descriptor) {
|
|
66
|
+
return descriptor instanceof wasm_utxo_1.Descriptor
|
|
67
|
+
? descriptor
|
|
68
|
+
: wasm_utxo_1.Descriptor.fromStringDetectType(typeof descriptor === 'string' ? descriptor : wasm_utxo_1.ast.formatNode(descriptor));
|
|
69
|
+
}
|
|
70
|
+
function runTest(network, txid, descriptor) {
|
|
71
|
+
describe(`Unbonding transaction ${txid}`, function () {
|
|
72
|
+
let fixture;
|
|
73
|
+
let psbt;
|
|
74
|
+
let signatures;
|
|
75
|
+
before('should create a PSBT from the unbonding transaction', async function () {
|
|
76
|
+
fixture = await getFixture(txid);
|
|
77
|
+
const txBuffer = Buffer.from(fixture.undelegation_response.unbonding_tx_hex, 'hex');
|
|
78
|
+
const tx = utxolib.bitgo.createTransactionFromBuffer(txBuffer, network, {
|
|
79
|
+
amountType: 'bigint',
|
|
80
|
+
});
|
|
81
|
+
signatures = fixture.undelegation_response.covenant_unbonding_sig_list.map((sig) => (0, undelegation_1.toPartialSig)(sig));
|
|
82
|
+
psbt = (0, undelegation_1.toUnbondingPsbtWithSignatures)(tx, {
|
|
83
|
+
script: Buffer.from('5120b81e8691ab3c46bddf8ea43f776e4d64abf417a0fa5a4d09976c8aaadabbb6a0', 'hex'),
|
|
84
|
+
value: 100000n,
|
|
85
|
+
}, descriptor, signatures, utxolib.networks.testnet);
|
|
86
|
+
});
|
|
87
|
+
it('should have valid signatures', async function () {
|
|
88
|
+
(0, undelegation_1.assertValidSignatures)(psbt, 0, signatures);
|
|
89
|
+
});
|
|
90
|
+
it('should match the fixture', async function () {
|
|
91
|
+
await (0, fixtures_utils_1.assertTransactionEqualsFixture)('test/fixtures/babylon/unbonding.' + txid.substring(0, 4) + '.psbt.json', psbt);
|
|
92
|
+
});
|
|
93
|
+
it('can spend from unbonding output', async function () {
|
|
94
|
+
const parsed = (0, parseDescriptor_1.parseStakingDescriptor)(descriptor);
|
|
95
|
+
(0, assert_1.default)(parsed);
|
|
96
|
+
const descriptorBuilder = new babylon_1.BabylonDescriptorBuilder(parsed.stakerKey, parsed.finalityProviderKeys, parsed.covenantKeys, parsed.covenantThreshold, parsed.stakingTimeLock, fixture.unbonding_time);
|
|
97
|
+
assert_1.default.deepStrictEqual(descriptorBuilder.getUnbondingMiniscriptNode(), parsed.unbondingMiniscriptNode, 'Unbonding miniscript node does not match expected value');
|
|
98
|
+
(0, assert_1.default)(psbt.txOutputs.length === 1, 'Unbonding transaction should have exactly one output');
|
|
99
|
+
assert_1.default.deepStrictEqual(psbt.txOutputs[0].script, Buffer.from(descriptorBuilder.getUnbondingDescriptor().scriptPubkey()));
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
runTest(utxolib.networks.bitcoinPublicSignet, '5d277e1b29e5589074aea95ac8c8230fd911c2ec3c58774aafdef915619b772c', toDescriptor({
|
|
104
|
+
tr: [
|
|
105
|
+
'50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0',
|
|
106
|
+
[
|
|
107
|
+
{
|
|
108
|
+
and_v: [
|
|
109
|
+
{
|
|
110
|
+
and_v: [
|
|
111
|
+
{ 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' },
|
|
112
|
+
{ 'v:pk': 'd23c2c25e1fcf8fd1c21b9a402c19e2e309e531e45e92fb1e9805b6056b0cc76' },
|
|
113
|
+
],
|
|
114
|
+
},
|
|
115
|
+
{
|
|
116
|
+
multi_a: [
|
|
117
|
+
6,
|
|
118
|
+
'0aee0509b16db71c999238a4827db945526859b13c95487ab46725357c9a9f25',
|
|
119
|
+
'113c3a32a9d320b72190a04a020a0db3976ef36972673258e9a38a364f3dc3b0',
|
|
120
|
+
'17921cf156ccb4e73d428f996ed11b245313e37e27c978ac4d2cc21eca4672e4',
|
|
121
|
+
'3bb93dfc8b61887d771f3630e9a63e97cbafcfcc78556a474df83a31a0ef899c',
|
|
122
|
+
'40afaf47c4ffa56de86410d8e47baa2bb6f04b604f4ea24323737ddc3fe092df',
|
|
123
|
+
'79a71ffd71c503ef2e2f91bccfc8fcda7946f4653cef0d9f3dde20795ef3b9f0',
|
|
124
|
+
'd21faf78c6751a0d38e6bd8028b907ff07e9a869a43fc837d6b3f8dff6119a36',
|
|
125
|
+
'f5199efae3f28bb82476163a7e458c7ad445d9bffb0682d10d3bdb2cb41f8e8e',
|
|
126
|
+
'fa9d882d45f4060bdb8042183828cd87544f1ea997380e586cab77d5fd698737',
|
|
127
|
+
],
|
|
128
|
+
},
|
|
129
|
+
],
|
|
130
|
+
},
|
|
131
|
+
[
|
|
132
|
+
{
|
|
133
|
+
and_v: [
|
|
134
|
+
{ 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' },
|
|
135
|
+
{
|
|
136
|
+
multi_a: [
|
|
137
|
+
6,
|
|
138
|
+
'0aee0509b16db71c999238a4827db945526859b13c95487ab46725357c9a9f25',
|
|
139
|
+
'113c3a32a9d320b72190a04a020a0db3976ef36972673258e9a38a364f3dc3b0',
|
|
140
|
+
'17921cf156ccb4e73d428f996ed11b245313e37e27c978ac4d2cc21eca4672e4',
|
|
141
|
+
'3bb93dfc8b61887d771f3630e9a63e97cbafcfcc78556a474df83a31a0ef899c',
|
|
142
|
+
'40afaf47c4ffa56de86410d8e47baa2bb6f04b604f4ea24323737ddc3fe092df',
|
|
143
|
+
'79a71ffd71c503ef2e2f91bccfc8fcda7946f4653cef0d9f3dde20795ef3b9f0',
|
|
144
|
+
'd21faf78c6751a0d38e6bd8028b907ff07e9a869a43fc837d6b3f8dff6119a36',
|
|
145
|
+
'f5199efae3f28bb82476163a7e458c7ad445d9bffb0682d10d3bdb2cb41f8e8e',
|
|
146
|
+
'fa9d882d45f4060bdb8042183828cd87544f1ea997380e586cab77d5fd698737',
|
|
147
|
+
],
|
|
148
|
+
},
|
|
149
|
+
],
|
|
150
|
+
},
|
|
151
|
+
{ and_v: [{ 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' }, { older: 10000 }] },
|
|
152
|
+
],
|
|
153
|
+
],
|
|
154
|
+
],
|
|
155
|
+
}));
|
|
156
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"undelegation.js","sourceRoot":"","sources":["../../../../../test/unit/babylon/undelegation.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport { promises as fs } from 'fs';\n\nimport * as t from 'io-ts';\nimport { PathReporter } from 'io-ts/lib/PathReporter';\nimport { isLeft } from 'fp-ts/Either';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { ast, Descriptor } from '@bitgo/wasm-utxo';\nimport { PartialSig } from 'bip174/src/lib/interfaces';\n\nimport { BabylonDescriptorBuilder } from '../../../src/babylon';\nimport {\n  toPartialSig,\n  UndelegationResponse,\n  assertValidSignatures,\n  toUnbondingPsbtWithSignatures,\n} from '../../../src/babylon/undelegation';\nimport { parseStakingDescriptor } from '../../../src/babylon/parseDescriptor';\nimport { assertTransactionEqualsFixture } from '../fixtures.utils';\n\nconst BtcDelegation = t.type({\n  unbonding_time: t.number,\n  undelegation_response: UndelegationResponse,\n});\n\ntype BtcDelegation = t.TypeOf<typeof BtcDelegation>;\n\nasync function getFixture(txid: string): Promise<BtcDelegation> {\n  // As returned by https://babylon.nodes.guru/api#/Query/BTCDelegation\n  const BtcDelegationResponse = t.type({ btc_delegation: BtcDelegation }, 'BtcDelegationResponse');\n  const filename = __dirname + `/../../fixtures/babylon/rpc/btc_delegation/testnet.${txid}.json`;\n  const data = JSON.parse(await fs.readFile(filename, 'utf8'));\n  const result = BtcDelegationResponse.decode(data);\n  if (isLeft(result)) {\n    throw new Error(`Failed to decode fixture data for txid ${txid}: ${PathReporter.report(result).join(', ')}`);\n  }\n  return result.right.btc_delegation;\n}\n\ntype DescriptorLike = Descriptor | ast.DescriptorNode | string;\n\nfunction toDescriptor(descriptor: DescriptorLike): Descriptor {\n  return descriptor instanceof Descriptor\n    ? descriptor\n    : Descriptor.fromStringDetectType(typeof descriptor === 'string' ? descriptor : ast.formatNode(descriptor));\n}\n\nfunction runTest(network: utxolib.Network, txid: string, descriptor: Descriptor): void {\n  describe(`Unbonding transaction ${txid}`, function () {\n    let fixture: BtcDelegation;\n    let psbt: utxolib.bitgo.UtxoPsbt;\n    let signatures: PartialSig[];\n\n    before('should create a PSBT from the unbonding transaction', async function () {\n      fixture = await getFixture(txid);\n      const txBuffer = Buffer.from(fixture.undelegation_response.unbonding_tx_hex, 'hex');\n      const tx = utxolib.bitgo.createTransactionFromBuffer(txBuffer, network, {\n        amountType: 'bigint',\n      });\n      signatures = fixture.undelegation_response.covenant_unbonding_sig_list.map((sig) => toPartialSig(sig));\n\n      psbt = toUnbondingPsbtWithSignatures(\n        tx,\n        {\n          script: Buffer.from('5120b81e8691ab3c46bddf8ea43f776e4d64abf417a0fa5a4d09976c8aaadabbb6a0', 'hex'),\n          value: 100_000n,\n        },\n        descriptor,\n        signatures,\n        utxolib.networks.testnet\n      );\n    });\n\n    it('should have valid signatures', async function () {\n      assertValidSignatures(psbt, 0, signatures);\n    });\n\n    it('should match the fixture', async function () {\n      await assertTransactionEqualsFixture(\n        'test/fixtures/babylon/unbonding.' + txid.substring(0, 4) + '.psbt.json',\n        psbt\n      );\n    });\n\n    it('can spend from unbonding output', async function () {\n      const parsed = parseStakingDescriptor(descriptor);\n      assert(parsed);\n\n      const descriptorBuilder = new BabylonDescriptorBuilder(\n        parsed.stakerKey,\n        parsed.finalityProviderKeys,\n        parsed.covenantKeys,\n        parsed.covenantThreshold,\n        parsed.stakingTimeLock,\n        fixture.unbonding_time\n      );\n\n      assert.deepStrictEqual(\n        descriptorBuilder.getUnbondingMiniscriptNode(),\n        parsed.unbondingMiniscriptNode,\n        'Unbonding miniscript node does not match expected value'\n      );\n\n      assert(psbt.txOutputs.length === 1, 'Unbonding transaction should have exactly one output');\n\n      assert.deepStrictEqual(\n        psbt.txOutputs[0].script,\n        Buffer.from(descriptorBuilder.getUnbondingDescriptor().scriptPubkey())\n      );\n    });\n  });\n}\n\nrunTest(\n  utxolib.networks.bitcoinPublicSignet,\n  '5d277e1b29e5589074aea95ac8c8230fd911c2ec3c58774aafdef915619b772c',\n  toDescriptor({\n    tr: [\n      '50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0',\n      [\n        {\n          and_v: [\n            {\n              and_v: [\n                { 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' },\n                { 'v:pk': 'd23c2c25e1fcf8fd1c21b9a402c19e2e309e531e45e92fb1e9805b6056b0cc76' },\n              ],\n            },\n            {\n              multi_a: [\n                6,\n                '0aee0509b16db71c999238a4827db945526859b13c95487ab46725357c9a9f25',\n                '113c3a32a9d320b72190a04a020a0db3976ef36972673258e9a38a364f3dc3b0',\n                '17921cf156ccb4e73d428f996ed11b245313e37e27c978ac4d2cc21eca4672e4',\n                '3bb93dfc8b61887d771f3630e9a63e97cbafcfcc78556a474df83a31a0ef899c',\n                '40afaf47c4ffa56de86410d8e47baa2bb6f04b604f4ea24323737ddc3fe092df',\n                '79a71ffd71c503ef2e2f91bccfc8fcda7946f4653cef0d9f3dde20795ef3b9f0',\n                'd21faf78c6751a0d38e6bd8028b907ff07e9a869a43fc837d6b3f8dff6119a36',\n                'f5199efae3f28bb82476163a7e458c7ad445d9bffb0682d10d3bdb2cb41f8e8e',\n                'fa9d882d45f4060bdb8042183828cd87544f1ea997380e586cab77d5fd698737',\n              ],\n            },\n          ],\n        },\n        [\n          {\n            and_v: [\n              { 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' },\n              {\n                multi_a: [\n                  6,\n                  '0aee0509b16db71c999238a4827db945526859b13c95487ab46725357c9a9f25',\n                  '113c3a32a9d320b72190a04a020a0db3976ef36972673258e9a38a364f3dc3b0',\n                  '17921cf156ccb4e73d428f996ed11b245313e37e27c978ac4d2cc21eca4672e4',\n                  '3bb93dfc8b61887d771f3630e9a63e97cbafcfcc78556a474df83a31a0ef899c',\n                  '40afaf47c4ffa56de86410d8e47baa2bb6f04b604f4ea24323737ddc3fe092df',\n                  '79a71ffd71c503ef2e2f91bccfc8fcda7946f4653cef0d9f3dde20795ef3b9f0',\n                  'd21faf78c6751a0d38e6bd8028b907ff07e9a869a43fc837d6b3f8dff6119a36',\n                  'f5199efae3f28bb82476163a7e458c7ad445d9bffb0682d10d3bdb2cb41f8e8e',\n                  'fa9d882d45f4060bdb8042183828cd87544f1ea997380e586cab77d5fd698737',\n                ],\n              },\n            ],\n          },\n          { and_v: [{ 'v:pk': '1b443f34ddc1bbaef52a8c5162dfa5a84524636ec745292f949470369ee67391' }, { older: 10000 }] },\n        ],\n      ],\n    ],\n  })\n);\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import * as bitcoinjslib from 'bitcoinjs-lib';
|
|
2
|
+
import { ECPairInterface } from '@bitgo-beta/utxo-lib';
|
|
3
|
+
import * as vendor from '@bitgo-beta/babylonlabs-io-btc-staking-ts';
|
|
4
|
+
import * as babylonProtobuf from '@babylonlabs-io/babylon-proto-ts';
|
|
5
|
+
import { BabylonDescriptorBuilder, ValueWithTypeUrl } from '../../../src/babylon';
|
|
6
|
+
type Result = {
|
|
7
|
+
unsignedDelegationMsg: ValueWithTypeUrl<babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation>;
|
|
8
|
+
stakingTx: bitcoinjslib.Transaction;
|
|
9
|
+
};
|
|
10
|
+
export declare function getBitGoUtxoStakingMsgCreateBtcDelegation(network: bitcoinjslib.Network, stakerKey: ECPairInterface, finalityProvider: ECPairInterface, descriptorBuilder: BabylonDescriptorBuilder, stakingParams: vendor.VersionedStakingParams[], changeAddress: string, amount: number, utxo: vendor.UTXO, feeRateSatB: number, blockHeight: number): Promise<Result>;
|
|
11
|
+
export declare function getVendorMsgCreateBtcDelegation(network: bitcoinjslib.Network, stakerKey: ECPairInterface, finalityProvider: ECPairInterface, descriptorBuilder: BabylonDescriptorBuilder, stakingParams: vendor.VersionedStakingParams[], changeAddress: string, amount: number, utxo: vendor.UTXO, feeRateSatB: number, blockHeight: number): Promise<Result>;
|
|
12
|
+
export {};
|
|
13
|
+
//# sourceMappingURL=vendor.utils.d.ts.map
|