@bitgo-beta/utxo-staking 1.1.1-beta.826 → 1.1.1-beta.827
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.map +1 -0
- package/dist/cjs/src/babylon/delegationMessage.js +285 -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.map +1 -0
- package/dist/{src → cjs/src}/babylon/index.js +1 -1
- package/dist/cjs/src/babylon/network.d.ts.map +1 -0
- package/dist/{src → cjs/src}/babylon/network.js +1 -1
- 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.map +1 -0
- package/dist/cjs/src/babylon/stakingManager.js +120 -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.map +1 -0
- package/dist/{src → cjs/src}/babylon/undelegation/UndelegationResponse.js +1 -1
- package/dist/cjs/src/babylon/undelegation/index.d.ts.map +1 -0
- package/dist/{src → cjs/src}/babylon/undelegation/index.js +1 -1
- 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/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.map +1 -0
- package/dist/{src → cjs/src}/index.js +1 -1
- 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.js +239 -0
- package/dist/esm/babylon/descriptor.d.ts +48 -0
- package/dist/esm/babylon/descriptor.js +91 -0
- package/dist/esm/babylon/index.d.ts +6 -0
- package/dist/esm/babylon/index.js +11 -0
- package/dist/esm/babylon/network.d.ts +7 -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.js +155 -0
- package/dist/esm/babylon/stakingManager.d.ts +6 -0
- package/dist/esm/babylon/stakingManager.js +83 -0
- package/dist/esm/babylon/stakingParams.d.ts +40 -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.js +18 -0
- package/dist/esm/babylon/undelegation/index.d.ts +3 -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.js +70 -0
- package/dist/esm/coreDao/descriptor.d.ts +18 -0
- 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.js +30 -0
- package/dist/esm/index.d.ts +3 -0
- package/dist/esm/index.js +3 -0
- package/package.json +26 -10
- package/dist/src/babylon/delegationMessage.js +0 -285
- package/dist/src/babylon/descriptor.js +0 -97
- package/dist/src/babylon/parseDescriptor.js +0 -159
- package/dist/src/babylon/stakingManager.js +0 -120
- package/dist/src/babylon/stakingParams.js +0 -132
- package/dist/src/babylon/undelegation/unbonding.js +0 -111
- package/dist/src/coreDao/descriptor.js +0 -38
- package/dist/src/coreDao/opReturn.js +0 -183
- package/dist/src/coreDao/transaction.js +0 -34
- /package/dist/{src → cjs/src}/babylon/delegationMessage.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/descriptor.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/index.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/network.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/params.mainnet.json +0 -0
- /package/dist/{src → cjs/src}/babylon/params.testnet.json +0 -0
- /package/dist/{src → cjs/src}/babylon/parseDescriptor.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/stakingManager.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/stakingParams.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/undelegation/UndelegationResponse.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/undelegation/index.d.ts +0 -0
- /package/dist/{src → cjs/src}/babylon/undelegation/unbonding.d.ts +0 -0
- /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 → cjs/src}/coreDao/transaction.d.ts +0 -0
- /package/dist/{src → cjs/src}/index.d.ts +0 -0
- /package/dist/{src → esm}/babylon/delegationMessage.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/descriptor.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/index.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/network.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/parseDescriptor.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/stakingManager.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/stakingParams.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/undelegation/UndelegationResponse.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/undelegation/index.d.ts.map +0 -0
- /package/dist/{src → esm}/babylon/undelegation/unbonding.d.ts.map +0 -0
- /package/dist/{src → esm}/coreDao/descriptor.d.ts.map +0 -0
- /package/dist/{src → esm}/coreDao/index.d.ts.map +0 -0
- /package/dist/{src → esm}/coreDao/opReturn.d.ts.map +0 -0
- /package/dist/{src → esm}/coreDao/transaction.d.ts.map +0 -0
- /package/dist/{src → esm}/index.d.ts.map +0 -0
|
@@ -1,285 +0,0 @@
|
|
|
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
|
-
exports.getSignedPsbt = getSignedPsbt;
|
|
40
|
-
exports.forceFinalizePsbt = forceFinalizePsbt;
|
|
41
|
-
exports.getBtcProviderForECKey = getBtcProviderForECKey;
|
|
42
|
-
exports.toStakerInfo = toStakerInfo;
|
|
43
|
-
exports.createStaking = createStaking;
|
|
44
|
-
exports.toStakingTransaction = toStakingTransaction;
|
|
45
|
-
exports.createDelegationMessageWithTransaction = createDelegationMessageWithTransaction;
|
|
46
|
-
exports.createUnsignedPreStakeRegistrationBabylonTransactionWithBtcProvider = createUnsignedPreStakeRegistrationBabylonTransactionWithBtcProvider;
|
|
47
|
-
/**
|
|
48
|
-
* https://github.com/babylonlabs-io/babylon/blob/v1.99.0-snapshot.250211/x/btcstaking/types/validate_parsed_message.go
|
|
49
|
-
*/
|
|
50
|
-
const assert_1 = __importDefault(require("assert"));
|
|
51
|
-
const bip322_js_1 = require("bip322-js");
|
|
52
|
-
const vendor = __importStar(require("@bitgo-beta/babylonlabs-io-btc-staking-ts"));
|
|
53
|
-
const bitcoinjslib = __importStar(require("bitcoinjs-lib"));
|
|
54
|
-
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
55
|
-
const utxo_core_1 = require("@bitgo-beta/utxo-core");
|
|
56
|
-
const descriptor_1 = require("@bitgo-beta/utxo-core/descriptor");
|
|
57
|
-
const stakingManager_1 = require("./stakingManager");
|
|
58
|
-
const stakingParams_1 = require("./stakingParams");
|
|
59
|
-
const network_1 = require("./network");
|
|
60
|
-
/**
|
|
61
|
-
* Decode a hex or base64 encoded string and check if the length is valid.
|
|
62
|
-
* @param v
|
|
63
|
-
* @param encoding
|
|
64
|
-
*/
|
|
65
|
-
function decodeCheck(v, encoding) {
|
|
66
|
-
const result = Buffer.from(v, encoding);
|
|
67
|
-
if (result.toString(encoding).length !== v.length) {
|
|
68
|
-
throw new Error(`Invalid ${encoding} encoding`);
|
|
69
|
-
}
|
|
70
|
-
return result;
|
|
71
|
-
}
|
|
72
|
-
/**
|
|
73
|
-
* Convert a Buffer or string to a base64 encoded string.
|
|
74
|
-
* @param v
|
|
75
|
-
*/
|
|
76
|
-
function toBase64(v) {
|
|
77
|
-
if (typeof v === 'string') {
|
|
78
|
-
for (const encoding of ['base64', 'hex']) {
|
|
79
|
-
try {
|
|
80
|
-
return toBase64(decodeCheck(v, encoding));
|
|
81
|
-
}
|
|
82
|
-
catch (e) {
|
|
83
|
-
// try next
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
throw new Error(`Invalid base64 or hex encoding: ${v}`);
|
|
87
|
-
}
|
|
88
|
-
return v.toString('base64');
|
|
89
|
-
}
|
|
90
|
-
function getSignedPsbt(psbt, descriptor, signers, { finalize = false }) {
|
|
91
|
-
const wrappedPsbt = (0, descriptor_1.toWrappedPsbt)(psbt.toBuffer());
|
|
92
|
-
const signedInputs = psbt.data.inputs.flatMap((input, i) => {
|
|
93
|
-
(0, assert_1.default)(input.witnessUtxo);
|
|
94
|
-
if (Buffer.from(descriptor.scriptPubkey()).equals(input.witnessUtxo.script)) {
|
|
95
|
-
wrappedPsbt.updateInputWithDescriptor(i, descriptor);
|
|
96
|
-
const signResults = signers.map((signer) => {
|
|
97
|
-
(0, assert_1.default)(signer.privateKey);
|
|
98
|
-
return wrappedPsbt.signWithPrv(signer.privateKey);
|
|
99
|
-
});
|
|
100
|
-
return [[i, signResults]];
|
|
101
|
-
}
|
|
102
|
-
return [];
|
|
103
|
-
});
|
|
104
|
-
(0, assert_1.default)(signedInputs.length > 0);
|
|
105
|
-
if (finalize) {
|
|
106
|
-
wrappedPsbt.finalize();
|
|
107
|
-
}
|
|
108
|
-
return bitcoinjslib.Psbt.fromBuffer(Buffer.from(wrappedPsbt.serialize()));
|
|
109
|
-
}
|
|
110
|
-
/**
|
|
111
|
-
* Utility method to work around a bug in btc-staking-ts
|
|
112
|
-
* https://github.com/babylonlabs-io/btc-staking-ts/issues/71
|
|
113
|
-
* @param v
|
|
114
|
-
* @param network
|
|
115
|
-
*/
|
|
116
|
-
function forceFinalizePsbt(v, network) {
|
|
117
|
-
if (v instanceof utxolib.Psbt) {
|
|
118
|
-
v = v.toBuffer();
|
|
119
|
-
}
|
|
120
|
-
if (v instanceof bitcoinjslib.Psbt) {
|
|
121
|
-
v = v.toBuffer();
|
|
122
|
-
}
|
|
123
|
-
const psbt = bitcoinjslib.Psbt.fromBuffer(v, { network: (0, network_1.toBitcoinJsNetwork)(network) });
|
|
124
|
-
// this only works with certain bitcoinjslib versions
|
|
125
|
-
psbt.finalizeAllInputs();
|
|
126
|
-
return psbt;
|
|
127
|
-
}
|
|
128
|
-
function getBtcProviderForECKey(descriptorBuilder, stakerKey) {
|
|
129
|
-
function signWithDescriptor(psbt, descriptor, key) {
|
|
130
|
-
psbt = getSignedPsbt(psbt, descriptor, [key], { finalize: false });
|
|
131
|
-
// BUG: we need to blindly finalize here even though we have not fully signed
|
|
132
|
-
psbt.finalizeAllInputs();
|
|
133
|
-
return psbt;
|
|
134
|
-
}
|
|
135
|
-
function signBip322Simple(message) {
|
|
136
|
-
// Get the script public key from the staking descriptor
|
|
137
|
-
const scriptPubKey = Buffer.from(descriptorBuilder.getStakingDescriptor().scriptPubkey());
|
|
138
|
-
const toSpendTx = bip322_js_1.BIP322.buildToSpendTx(message, scriptPubKey);
|
|
139
|
-
// Get the to_spend txid
|
|
140
|
-
const toSpendTxId = toSpendTx.getId();
|
|
141
|
-
// Create PSBT object for constructing the transaction
|
|
142
|
-
const toSignPsbt = new bitcoinjslib.Psbt();
|
|
143
|
-
toSignPsbt.setVersion(2); // nVersion = 0
|
|
144
|
-
toSignPsbt.setLocktime(0); // nLockTime = 0
|
|
145
|
-
toSignPsbt.addInput({
|
|
146
|
-
hash: toSpendTxId,
|
|
147
|
-
index: 0,
|
|
148
|
-
sequence: descriptorBuilder.stakingTimeLock,
|
|
149
|
-
witnessUtxo: {
|
|
150
|
-
script: scriptPubKey,
|
|
151
|
-
value: 0,
|
|
152
|
-
},
|
|
153
|
-
});
|
|
154
|
-
// Sign the PSBT with the staker key
|
|
155
|
-
const wrappedPsbt = (0, descriptor_1.toWrappedPsbt)(toSignPsbt.toBuffer());
|
|
156
|
-
wrappedPsbt.updateInputWithDescriptor(0, descriptorBuilder.getStakingDescriptor());
|
|
157
|
-
(0, descriptor_1.signWithKey)(wrappedPsbt, stakerKey);
|
|
158
|
-
wrappedPsbt.finalize();
|
|
159
|
-
// Encode the witness data and return
|
|
160
|
-
return bip322_js_1.BIP322.encodeWitness(bitcoinjslib.Psbt.fromBuffer(Buffer.from(wrappedPsbt.serialize())));
|
|
161
|
-
}
|
|
162
|
-
return {
|
|
163
|
-
/**
|
|
164
|
-
* @param message
|
|
165
|
-
* @param type
|
|
166
|
-
* @returns Base64 encoded string
|
|
167
|
-
*/
|
|
168
|
-
async signMessage(message, type) {
|
|
169
|
-
switch (type) {
|
|
170
|
-
case 'ecdsa':
|
|
171
|
-
return toBase64(stakerKey.sign(Buffer.from(message, 'hex')));
|
|
172
|
-
case 'bip322-simple':
|
|
173
|
-
return toBase64(signBip322Simple(message));
|
|
174
|
-
default:
|
|
175
|
-
throw new Error(`unexpected type: ${type}`);
|
|
176
|
-
}
|
|
177
|
-
},
|
|
178
|
-
async signPsbt(psbtHex, options) {
|
|
179
|
-
const psbt = bitcoinjslib.Psbt.fromHex(psbtHex);
|
|
180
|
-
switch (options.action.name) {
|
|
181
|
-
case 'sign-btc-slashing-transaction':
|
|
182
|
-
return signWithDescriptor(psbt, descriptorBuilder.getStakingDescriptor(), stakerKey).toHex();
|
|
183
|
-
case 'sign-btc-unbonding-slashing-transaction':
|
|
184
|
-
return signWithDescriptor(psbt, descriptorBuilder.getUnbondingDescriptor(), stakerKey).toHex();
|
|
185
|
-
default:
|
|
186
|
-
throw new Error(`unexpected signing step: ${options.action.name}`);
|
|
187
|
-
}
|
|
188
|
-
},
|
|
189
|
-
/**
|
|
190
|
-
* This function is only used by btc-staking-ts to create a staking expansion registration
|
|
191
|
-
* transaction, which we do not currently support.
|
|
192
|
-
*/
|
|
193
|
-
async getTransactionHex(txid) {
|
|
194
|
-
throw new Error(`Unsupported operation getTransactionHex (txid=${txid})`);
|
|
195
|
-
},
|
|
196
|
-
};
|
|
197
|
-
}
|
|
198
|
-
/**
|
|
199
|
-
* @param stakingKey - this is the single-sig key that is used for co-signing the staking output
|
|
200
|
-
* @param changeAddress - this is unrelated to the staking key and is used for the change output
|
|
201
|
-
*/
|
|
202
|
-
function toStakerInfo(stakingKey, changeAddress) {
|
|
203
|
-
if (typeof stakingKey === 'object' && 'publicKey' in stakingKey) {
|
|
204
|
-
stakingKey = stakingKey.publicKey;
|
|
205
|
-
}
|
|
206
|
-
if (typeof stakingKey === 'string') {
|
|
207
|
-
stakingKey = Buffer.from(stakingKey, 'hex');
|
|
208
|
-
}
|
|
209
|
-
return {
|
|
210
|
-
publicKeyNoCoordHex: (0, utxo_core_1.toXOnlyPublicKey)(stakingKey).toString('hex'),
|
|
211
|
-
address: changeAddress,
|
|
212
|
-
};
|
|
213
|
-
}
|
|
214
|
-
function createStaking(network, blockHeight, stakerBtcInfo, stakingInput, versionedParams = (0, stakingParams_1.getStakingParams)(network)) {
|
|
215
|
-
if (blockHeight === 0) {
|
|
216
|
-
throw new Error('Babylon BTC tip height cannot be 0');
|
|
217
|
-
}
|
|
218
|
-
// Get the Babylon params based on the BTC tip height from Babylon chain
|
|
219
|
-
const params = vendor.getBabylonParamByBtcHeight(blockHeight, versionedParams);
|
|
220
|
-
return new vendor.Staking((0, network_1.toBitcoinJsNetwork)(network), stakerBtcInfo, params, stakingInput.finalityProviderPksNoCoordHex, stakingInput.stakingTimelock);
|
|
221
|
-
}
|
|
222
|
-
function toStakingTransactionFromPsbt(psbt) {
|
|
223
|
-
if (!(psbt instanceof utxolib.bitgo.UtxoPsbt)) {
|
|
224
|
-
psbt = utxolib.bitgo.createPsbtFromBuffer(psbt.toBuffer(), utxolib.networks.bitcoin);
|
|
225
|
-
}
|
|
226
|
-
if (psbt instanceof utxolib.bitgo.UtxoPsbt) {
|
|
227
|
-
// only utxolib.bitgo.UtxoPsbt has the getUnsignedTx method
|
|
228
|
-
return bitcoinjslib.Transaction.fromHex(psbt.getUnsignedTx().toHex());
|
|
229
|
-
}
|
|
230
|
-
throw new Error('illegal state');
|
|
231
|
-
}
|
|
232
|
-
function toStakingTransaction(tx) {
|
|
233
|
-
if (tx instanceof bitcoinjslib.Psbt || tx instanceof utxolib.Psbt) {
|
|
234
|
-
return toStakingTransactionFromPsbt(tx);
|
|
235
|
-
}
|
|
236
|
-
return bitcoinjslib.Transaction.fromHex(tx.toHex());
|
|
237
|
-
}
|
|
238
|
-
/**
|
|
239
|
-
* As of babylonlabs-io/btc-staking-ts v1.5.7, the BTC delegation message creation functions support two message types:
|
|
240
|
-
* - MsgCreateBTCDelegation
|
|
241
|
-
* - MsgBtcStakeExpand
|
|
242
|
-
*
|
|
243
|
-
* BitGo still only supports MsgCreateBTCDelegation, so we need to check the message type here.
|
|
244
|
-
*
|
|
245
|
-
* @param msg - the message to check
|
|
246
|
-
* @return `true` if the message is of type MsgCreateBTCDelegation
|
|
247
|
-
*/
|
|
248
|
-
function isMsgBtcStakeExpand(msg) {
|
|
249
|
-
return 'previousStakingTxHash' in msg;
|
|
250
|
-
}
|
|
251
|
-
/*
|
|
252
|
-
* This is mostly lifted from
|
|
253
|
-
* https://github.com/babylonlabs-io/btc-staking-ts/blob/v0.4.0-rc.2/src/staking/manager.ts#L100-L172
|
|
254
|
-
*
|
|
255
|
-
* The difference is that here we are returning an _unsigned_ delegation message.
|
|
256
|
-
*/
|
|
257
|
-
async function createDelegationMessageWithTransaction(channel, manager, staking, stakingAmountSat, transaction, babylonAddress) {
|
|
258
|
-
if (!vendor.isValidBabylonAddress(babylonAddress)) {
|
|
259
|
-
throw new Error('Invalid Babylon address');
|
|
260
|
-
}
|
|
261
|
-
// Create delegation message without including inclusion proof
|
|
262
|
-
const msg = await manager.createBtcDelegationMsg(channel, staking, {
|
|
263
|
-
stakingTimelock: staking.stakingTimelock,
|
|
264
|
-
finalityProviderPksNoCoordHex: staking.finalityProviderPksNoCoordHex,
|
|
265
|
-
stakingAmountSat,
|
|
266
|
-
}, toStakingTransaction(transaction), babylonAddress, staking.stakerInfo, staking.params);
|
|
267
|
-
// It shouldn't be possible for us to create a MsgBtcStakeExpand here because that only gets created when
|
|
268
|
-
// we pass channel = delegation:expand into createBtcDelegationMsg, which we cannot do.
|
|
269
|
-
if (isMsgBtcStakeExpand(msg.value)) {
|
|
270
|
-
throw new Error('MsgBtcStakeExpand is not supported');
|
|
271
|
-
}
|
|
272
|
-
return { ...msg, value: msg.value };
|
|
273
|
-
}
|
|
274
|
-
async function createUnsignedPreStakeRegistrationBabylonTransactionWithBtcProvider(btcProvider, network, stakerBtcInfo, stakingInput, babylonBtcTipHeight, inputUTXOs, feeRateSatB, babylonAddress, channel, stakingParams = (0, stakingParams_1.getStakingParams)(network)) {
|
|
275
|
-
if (inputUTXOs.length === 0) {
|
|
276
|
-
throw new Error('No input UTXOs provided');
|
|
277
|
-
}
|
|
278
|
-
const manager = (0, stakingManager_1.createStakingManager)(network, btcProvider, stakingParams);
|
|
279
|
-
const staking = createStaking(network, babylonBtcTipHeight, stakerBtcInfo, stakingInput, stakingParams);
|
|
280
|
-
// Create unsigned staking transaction
|
|
281
|
-
const { transaction } = staking.createStakingTransaction(stakingInput.stakingAmountSat, inputUTXOs, feeRateSatB);
|
|
282
|
-
const unsignedDelegationMsg = await createDelegationMessageWithTransaction(channel, manager, staking, stakingInput.stakingAmountSat, transaction, babylonAddress);
|
|
283
|
-
return { unsignedDelegationMsg, stakingTx: transaction };
|
|
284
|
-
}
|
|
285
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delegationMessage.js","sourceRoot":"","sources":["../../../src/babylon/delegationMessage.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * https://github.com/babylonlabs-io/babylon/blob/v1.99.0-snapshot.250211/x/btcstaking/types/validate_parsed_message.go\n */\nimport assert from 'assert';\n\nimport { BIP322 } from 'bip322-js';\nimport * as vendor from '@bitgo-beta/babylonlabs-io-btc-staking-ts';\nimport * as babylonProtobuf from '@babylonlabs-io/babylon-proto-ts';\nimport * as bitcoinjslib from 'bitcoinjs-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Descriptor } from '@bitgo/wasm-utxo';\nimport { toXOnlyPublicKey } from '@bitgo-beta/utxo-core';\nimport { signWithKey, toWrappedPsbt } from '@bitgo-beta/utxo-core/descriptor';\n\nimport { BabylonDescriptorBuilder } from './descriptor';\nimport { createStakingManager } from './stakingManager';\nimport { getStakingParams } from './stakingParams';\nimport { BabylonNetworkLike, toBitcoinJsNetwork } from './network';\n\nexport type ValueWithTypeUrl<T> = { typeUrl: string; value: T };\n\n/**\n * Decode a hex or base64 encoded string and check if the length is valid.\n * @param v\n * @param encoding\n */\nfunction decodeCheck(v: string, encoding: 'hex' | 'base64') {\n  const result = Buffer.from(v, encoding);\n  if (result.toString(encoding).length !== v.length) {\n    throw new Error(`Invalid ${encoding} encoding`);\n  }\n  return result;\n}\n\n/**\n * Convert a Buffer or string to a base64 encoded string.\n * @param v\n */\nfunction toBase64(v: Buffer | string) {\n  if (typeof v === 'string') {\n    for (const encoding of ['base64', 'hex'] as const) {\n      try {\n        return toBase64(decodeCheck(v, encoding));\n      } catch (e) {\n        // try next\n      }\n    }\n    throw new Error(`Invalid base64 or hex encoding: ${v}`);\n  }\n\n  return v.toString('base64');\n}\n\nexport function getSignedPsbt(\n  psbt: bitcoinjslib.Psbt,\n  descriptor: Descriptor,\n  signers: utxolib.ECPairInterface[],\n  { finalize = false }\n): bitcoinjslib.Psbt {\n  const wrappedPsbt = toWrappedPsbt(psbt.toBuffer());\n  const signedInputs = psbt.data.inputs.flatMap((input, i) => {\n    assert(input.witnessUtxo);\n    if (Buffer.from(descriptor.scriptPubkey()).equals(input.witnessUtxo.script)) {\n      wrappedPsbt.updateInputWithDescriptor(i, descriptor);\n      const signResults = signers.map((signer) => {\n        assert(signer.privateKey);\n        return wrappedPsbt.signWithPrv(signer.privateKey);\n      });\n      return [[i, signResults]];\n    }\n    return [];\n  });\n  assert(signedInputs.length > 0);\n  if (finalize) {\n    wrappedPsbt.finalize();\n  }\n  return bitcoinjslib.Psbt.fromBuffer(Buffer.from(wrappedPsbt.serialize()));\n}\n\n/**\n * Utility method to work around a bug in btc-staking-ts\n * https://github.com/babylonlabs-io/btc-staking-ts/issues/71\n * @param v\n * @param network\n */\nexport function forceFinalizePsbt(\n  v: Buffer | utxolib.Psbt | bitcoinjslib.Psbt,\n  network: BabylonNetworkLike\n): bitcoinjslib.Psbt {\n  if (v instanceof utxolib.Psbt) {\n    v = v.toBuffer();\n  }\n  if (v instanceof bitcoinjslib.Psbt) {\n    v = v.toBuffer();\n  }\n  const psbt = bitcoinjslib.Psbt.fromBuffer(v, { network: toBitcoinJsNetwork(network) });\n  // this only works with certain bitcoinjslib versions\n  psbt.finalizeAllInputs();\n  return psbt;\n}\n\nexport function getBtcProviderForECKey(\n  descriptorBuilder: BabylonDescriptorBuilder,\n  stakerKey: utxolib.ECPairInterface\n): vendor.BtcProvider {\n  function signWithDescriptor(\n    psbt: bitcoinjslib.Psbt,\n    descriptor: Descriptor,\n    key: utxolib.ECPairInterface\n  ): bitcoinjslib.Psbt {\n    psbt = getSignedPsbt(psbt, descriptor, [key], { finalize: false });\n    // BUG: we need to blindly finalize here even though we have not fully signed\n    psbt.finalizeAllInputs();\n    return psbt;\n  }\n\n  function signBip322Simple(message: string): string {\n    // Get the script public key from the staking descriptor\n    const scriptPubKey = Buffer.from(descriptorBuilder.getStakingDescriptor().scriptPubkey());\n    const toSpendTx = BIP322.buildToSpendTx(message, scriptPubKey);\n\n    // Get the to_spend txid\n    const toSpendTxId = toSpendTx.getId();\n\n    // Create PSBT object for constructing the transaction\n    const toSignPsbt = new bitcoinjslib.Psbt();\n    toSignPsbt.setVersion(2); // nVersion = 0\n    toSignPsbt.setLocktime(0); // nLockTime = 0\n    toSignPsbt.addInput({\n      hash: toSpendTxId,\n      index: 0,\n      sequence: descriptorBuilder.stakingTimeLock,\n      witnessUtxo: {\n        script: scriptPubKey,\n        value: 0,\n      },\n    });\n\n    // Sign the PSBT with the staker key\n    const wrappedPsbt = toWrappedPsbt(toSignPsbt.toBuffer());\n    wrappedPsbt.updateInputWithDescriptor(0, descriptorBuilder.getStakingDescriptor());\n    signWithKey(wrappedPsbt, stakerKey);\n    wrappedPsbt.finalize();\n\n    // Encode the witness data and return\n    return BIP322.encodeWitness(bitcoinjslib.Psbt.fromBuffer(Buffer.from(wrappedPsbt.serialize())));\n  }\n\n  return {\n    /**\n     * @param message\n     * @param type\n     * @returns Base64 encoded string\n     */\n    async signMessage(message: string, type: 'ecdsa' | 'bip322-simple'): Promise<string> {\n      switch (type) {\n        case 'ecdsa':\n          return toBase64(stakerKey.sign(Buffer.from(message, 'hex')));\n        case 'bip322-simple':\n          return toBase64(signBip322Simple(message));\n        default:\n          throw new Error(`unexpected type: ${type}`);\n      }\n    },\n\n    async signPsbt(psbtHex: string, options: vendor.SignPsbtOptions): Promise<string> {\n      const psbt = bitcoinjslib.Psbt.fromHex(psbtHex);\n      switch (options.action.name) {\n        case 'sign-btc-slashing-transaction':\n          return signWithDescriptor(psbt, descriptorBuilder.getStakingDescriptor(), stakerKey).toHex();\n        case 'sign-btc-unbonding-slashing-transaction':\n          return signWithDescriptor(psbt, descriptorBuilder.getUnbondingDescriptor(), stakerKey).toHex();\n        default:\n          throw new Error(`unexpected signing step: ${options.action.name}`);\n      }\n    },\n\n    /**\n     * This function is only used by btc-staking-ts to create a staking expansion registration\n     * transaction, which we do not currently support.\n     */\n    async getTransactionHex(txid: string): Promise<string> {\n      throw new Error(`Unsupported operation getTransactionHex (txid=${txid})`);\n    },\n  };\n}\ntype Result = {\n  unsignedDelegationMsg: ValueWithTypeUrl<babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation>;\n  stakingTx: bitcoinjslib.Transaction;\n};\n\n/**\n * @param stakingKey - this is the single-sig key that is used for co-signing the staking output\n * @param changeAddress - this is unrelated to the staking key and is used for the change output\n */\nexport function toStakerInfo(\n  stakingKey: utxolib.ECPairInterface | Buffer | string,\n  changeAddress: string\n): vendor.StakerInfo {\n  if (typeof stakingKey === 'object' && 'publicKey' in stakingKey) {\n    stakingKey = stakingKey.publicKey;\n  }\n  if (typeof stakingKey === 'string') {\n    stakingKey = Buffer.from(stakingKey, 'hex');\n  }\n  return {\n    publicKeyNoCoordHex: toXOnlyPublicKey(stakingKey).toString('hex'),\n    address: changeAddress,\n  };\n}\n\nexport function createStaking(\n  network: BabylonNetworkLike,\n  blockHeight: number,\n  stakerBtcInfo: vendor.StakerInfo,\n  stakingInput: vendor.StakingInputs,\n  versionedParams: vendor.VersionedStakingParams[] = getStakingParams(network)\n): vendor.Staking {\n  if (blockHeight === 0) {\n    throw new Error('Babylon BTC tip height cannot be 0');\n  }\n\n  // Get the Babylon params based on the BTC tip height from Babylon chain\n  const params = vendor.getBabylonParamByBtcHeight(blockHeight, versionedParams);\n\n  return new vendor.Staking(\n    toBitcoinJsNetwork(network),\n    stakerBtcInfo,\n    params,\n    stakingInput.finalityProviderPksNoCoordHex,\n    stakingInput.stakingTimelock\n  );\n}\n\ntype TransactionLike =\n  | bitcoinjslib.Psbt\n  | bitcoinjslib.Transaction\n  | utxolib.Transaction\n  | utxolib.bitgo.UtxoTransaction<bigint | number>\n  | utxolib.Psbt\n  | utxolib.bitgo.UtxoPsbt;\n\nfunction toStakingTransactionFromPsbt(\n  psbt: bitcoinjslib.Psbt | utxolib.Psbt | utxolib.bitgo.UtxoPsbt\n): bitcoinjslib.Transaction {\n  if (!(psbt instanceof utxolib.bitgo.UtxoPsbt)) {\n    psbt = utxolib.bitgo.createPsbtFromBuffer(psbt.toBuffer(), utxolib.networks.bitcoin);\n  }\n  if (psbt instanceof utxolib.bitgo.UtxoPsbt) {\n    // only utxolib.bitgo.UtxoPsbt has the getUnsignedTx method\n    return bitcoinjslib.Transaction.fromHex(psbt.getUnsignedTx().toHex());\n  }\n  throw new Error('illegal state');\n}\n\nexport function toStakingTransaction(tx: TransactionLike): bitcoinjslib.Transaction {\n  if (tx instanceof bitcoinjslib.Psbt || tx instanceof utxolib.Psbt) {\n    return toStakingTransactionFromPsbt(tx);\n  }\n  return bitcoinjslib.Transaction.fromHex(tx.toHex());\n}\n\n/**\n * As of babylonlabs-io/btc-staking-ts v1.5.7, the BTC delegation message creation functions support two message types:\n * - MsgCreateBTCDelegation\n * - MsgBtcStakeExpand\n *\n * BitGo still only supports MsgCreateBTCDelegation, so we need to check the message type here.\n *\n * @param msg - the message to check\n * @return `true` if the message is of type MsgCreateBTCDelegation\n */\nfunction isMsgBtcStakeExpand(\n  msg: babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation | babylonProtobuf.btcstakingtx.MsgBtcStakeExpand\n) {\n  return 'previousStakingTxHash' in msg;\n}\n\n/*\n * This is mostly lifted from\n * https://github.com/babylonlabs-io/btc-staking-ts/blob/v0.4.0-rc.2/src/staking/manager.ts#L100-L172\n *\n * The difference is that here we are returning an _unsigned_ delegation message.\n */\nexport async function createDelegationMessageWithTransaction(\n  channel: 'delegation:create' | 'delegation:register',\n  manager: vendor.BabylonBtcStakingManager,\n  staking: vendor.Staking,\n  stakingAmountSat: number,\n  transaction: TransactionLike,\n  babylonAddress: string\n): Promise<ValueWithTypeUrl<babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation>> {\n  if (!vendor.isValidBabylonAddress(babylonAddress)) {\n    throw new Error('Invalid Babylon address');\n  }\n  // Create delegation message without including inclusion proof\n  const msg = await manager.createBtcDelegationMsg(\n    channel,\n    staking,\n    {\n      stakingTimelock: staking.stakingTimelock,\n      finalityProviderPksNoCoordHex: staking.finalityProviderPksNoCoordHex,\n      stakingAmountSat,\n    },\n    toStakingTransaction(transaction),\n    babylonAddress,\n    staking.stakerInfo,\n    staking.params\n  );\n\n  // It shouldn't be possible for us to create a MsgBtcStakeExpand here because that only gets created when\n  // we pass channel = delegation:expand into createBtcDelegationMsg, which we cannot do.\n  if (isMsgBtcStakeExpand(msg.value)) {\n    throw new Error('MsgBtcStakeExpand is not supported');\n  }\n\n  return { ...msg, value: msg.value as babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation };\n}\n\nexport async function createUnsignedPreStakeRegistrationBabylonTransactionWithBtcProvider(\n  btcProvider: vendor.BtcProvider,\n  network: bitcoinjslib.Network,\n  stakerBtcInfo: vendor.StakerInfo,\n  stakingInput: vendor.StakingInputs,\n  babylonBtcTipHeight: number,\n  inputUTXOs: vendor.UTXO[],\n  feeRateSatB: number,\n  babylonAddress: string,\n  channel: 'delegation:create' | 'delegation:register',\n  stakingParams: vendor.VersionedStakingParams[] = getStakingParams(network)\n): Promise<Result> {\n  if (inputUTXOs.length === 0) {\n    throw new Error('No input UTXOs provided');\n  }\n  const manager = createStakingManager(network, btcProvider, stakingParams);\n  const staking = createStaking(network, babylonBtcTipHeight, stakerBtcInfo, stakingInput, stakingParams);\n  // Create unsigned staking transaction\n  const { transaction } = staking.createStakingTransaction(stakingInput.stakingAmountSat, inputUTXOs, feeRateSatB);\n  const unsignedDelegationMsg = await createDelegationMessageWithTransaction(\n    channel,\n    manager,\n    staking,\n    stakingInput.stakingAmountSat,\n    transaction,\n    babylonAddress\n  );\n  return { unsignedDelegationMsg, stakingTx: transaction };\n}\n"]}
|
|
@@ -1,97 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* https://github.com/babylonlabs-io/babylon/tree/main/docs
|
|
4
|
-
* https://github.com/babylonlabs-io/babylon/blob/main/docs/staking-script.md
|
|
5
|
-
* https://github.com/babylonlabs-io/babylon/blob/v1.99.0-snapshot.250211/btcstaking/staking.go
|
|
6
|
-
*/
|
|
7
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
-
exports.BabylonDescriptorBuilder = void 0;
|
|
9
|
-
exports.getUnspendableKey = getUnspendableKey;
|
|
10
|
-
exports.sortedKeys = sortedKeys;
|
|
11
|
-
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
12
|
-
function getUnspendableKey() {
|
|
13
|
-
// https://github.com/babylonlabs-io/btc-staking-ts/blob/v0.4.0-rc.2/src/constants/internalPubkey.ts
|
|
14
|
-
return '50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0';
|
|
15
|
-
}
|
|
16
|
-
// Helper functions for creating miniscript nodes
|
|
17
|
-
function pk(b) {
|
|
18
|
-
return { 'v:pk': b.toString('hex') };
|
|
19
|
-
}
|
|
20
|
-
function sortedKeys(keys) {
|
|
21
|
-
return [...keys].sort((a, b) => a.compare(b));
|
|
22
|
-
}
|
|
23
|
-
function multiArgs(threshold, keys) {
|
|
24
|
-
return [threshold, ...sortedKeys(keys).map((k) => k.toString('hex'))];
|
|
25
|
-
}
|
|
26
|
-
function taprootScriptOnlyFromAst(n) {
|
|
27
|
-
return wasm_utxo_1.Descriptor.fromString(wasm_utxo_1.ast.formatNode({ tr: [getUnspendableKey(), n] }), 'definite');
|
|
28
|
-
}
|
|
29
|
-
class BabylonDescriptorBuilder {
|
|
30
|
-
constructor(stakerKey, finalityProviderKeys, covenantKeys, covenantThreshold, stakingTimeLock, unbondingTimeLock) {
|
|
31
|
-
this.stakerKey = stakerKey;
|
|
32
|
-
this.finalityProviderKeys = finalityProviderKeys;
|
|
33
|
-
this.covenantKeys = covenantKeys;
|
|
34
|
-
this.covenantThreshold = covenantThreshold;
|
|
35
|
-
this.stakingTimeLock = stakingTimeLock;
|
|
36
|
-
this.unbondingTimeLock = unbondingTimeLock;
|
|
37
|
-
}
|
|
38
|
-
static fromParams(params) {
|
|
39
|
-
return new BabylonDescriptorBuilder(params.stakerKey, params.finalityProviderKeys, params.covenantNoCoordPks.map((k) => Buffer.from(k, 'hex')), params.covenantQuorum, params.minStakingTimeBlocks, params.unbondingTime);
|
|
40
|
-
}
|
|
41
|
-
/** Spend path with the staker key and the staking timelock */
|
|
42
|
-
getStakingTimelockMiniscriptNode() {
|
|
43
|
-
return { and_v: [pk(this.stakerKey), { older: this.stakingTimeLock }] };
|
|
44
|
-
}
|
|
45
|
-
/** Spend path with the staker key and the unbonding timelock */
|
|
46
|
-
getUnbondingTimelockMiniscriptNode() {
|
|
47
|
-
return { and_v: [pk(this.stakerKey), { older: this.unbondingTimeLock }] };
|
|
48
|
-
}
|
|
49
|
-
/** Spend path with the staker key and the covenant keys */
|
|
50
|
-
getUnbondingMiniscriptNode() {
|
|
51
|
-
return { and_v: [pk(this.stakerKey), { multi_a: multiArgs(this.covenantThreshold, this.covenantKeys) }] };
|
|
52
|
-
}
|
|
53
|
-
/** Spend path with the finality provider keys and the covenant keys */
|
|
54
|
-
getSlashingMiniscriptNode() {
|
|
55
|
-
return {
|
|
56
|
-
and_v: [
|
|
57
|
-
{
|
|
58
|
-
and_v: [
|
|
59
|
-
pk(this.stakerKey),
|
|
60
|
-
this.finalityProviderKeys.length === 1
|
|
61
|
-
? { 'v:pk': this.finalityProviderKeys[0].toString('hex') }
|
|
62
|
-
: { 'v:multi_a': multiArgs(1, this.finalityProviderKeys) },
|
|
63
|
-
],
|
|
64
|
-
},
|
|
65
|
-
{ multi_a: multiArgs(this.covenantThreshold, this.covenantKeys) },
|
|
66
|
-
],
|
|
67
|
-
};
|
|
68
|
-
}
|
|
69
|
-
/**
|
|
70
|
-
* Creates a descriptor for a staking output.
|
|
71
|
-
*
|
|
72
|
-
* Three spend paths:
|
|
73
|
-
* - the slashing script,
|
|
74
|
-
* - the unbonding script,
|
|
75
|
-
* - the timelocked unstaking script.
|
|
76
|
-
*/
|
|
77
|
-
getStakingDescriptor() {
|
|
78
|
-
return taprootScriptOnlyFromAst([
|
|
79
|
-
this.getSlashingMiniscriptNode(),
|
|
80
|
-
[this.getUnbondingMiniscriptNode(), this.getStakingTimelockMiniscriptNode()],
|
|
81
|
-
]);
|
|
82
|
-
}
|
|
83
|
-
/**
|
|
84
|
-
* Creates a descriptor for the timelocked unbonding script.
|
|
85
|
-
*/
|
|
86
|
-
getUnbondingTimelockDescriptor() {
|
|
87
|
-
return taprootScriptOnlyFromAst(this.getUnbondingTimelockMiniscriptNode());
|
|
88
|
-
}
|
|
89
|
-
/**
|
|
90
|
-
* Creates a descriptor with two script paths: the slashing script and the timelocked unbonding script.
|
|
91
|
-
*/
|
|
92
|
-
getUnbondingDescriptor() {
|
|
93
|
-
return taprootScriptOnlyFromAst([this.getSlashingMiniscriptNode(), this.getUnbondingTimelockMiniscriptNode()]);
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
exports.BabylonDescriptorBuilder = BabylonDescriptorBuilder;
|
|
97
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"descriptor.js","sourceRoot":"","sources":["../../../src/babylon/descriptor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * https://github.com/babylonlabs-io/babylon/tree/main/docs\n * https://github.com/babylonlabs-io/babylon/blob/main/docs/staking-script.md\n * https://github.com/babylonlabs-io/babylon/blob/v1.99.0-snapshot.250211/btcstaking/staking.go\n */\n\nimport { Descriptor, ast } from '@bitgo/wasm-utxo';\nimport { StakingParams } from '@bitgo-beta/babylonlabs-io-btc-staking-ts';\n\nexport function getUnspendableKey(): string {\n  // https://github.com/babylonlabs-io/btc-staking-ts/blob/v0.4.0-rc.2/src/constants/internalPubkey.ts\n  return '50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0';\n}\n\n// Helper functions for creating miniscript nodes\nfunction pk(b: Buffer): ast.MiniscriptNode {\n  return { 'v:pk': b.toString('hex') };\n}\n\nexport function sortedKeys(keys: Buffer[]): Buffer[] {\n  return [...keys].sort((a, b) => a.compare(b));\n}\n\nfunction multiArgs(threshold: number, keys: Buffer[]): [number, ...string[]] {\n  return [threshold, ...sortedKeys(keys).map((k) => k.toString('hex'))];\n}\n\nfunction taprootScriptOnlyFromAst(n: ast.TapTreeNode): Descriptor {\n  return Descriptor.fromString(ast.formatNode({ tr: [getUnspendableKey(), n] }), 'definite');\n}\n\nexport class BabylonDescriptorBuilder {\n  constructor(\n    public stakerKey: Buffer,\n    public finalityProviderKeys: Buffer[],\n    public covenantKeys: Buffer[],\n    public covenantThreshold: number,\n    public stakingTimeLock: number,\n    public unbondingTimeLock: number\n  ) {}\n\n  static fromParams(\n    params: {\n      stakerKey: Buffer;\n      finalityProviderKeys: Buffer[];\n    } & StakingParams\n  ): BabylonDescriptorBuilder {\n    return new BabylonDescriptorBuilder(\n      params.stakerKey,\n      params.finalityProviderKeys,\n      params.covenantNoCoordPks.map((k) => Buffer.from(k, 'hex')),\n      params.covenantQuorum,\n      params.minStakingTimeBlocks,\n      params.unbondingTime\n    );\n  }\n\n  /** Spend path with the staker key and the staking timelock */\n  getStakingTimelockMiniscriptNode(): ast.MiniscriptNode {\n    return { and_v: [pk(this.stakerKey), { older: this.stakingTimeLock }] };\n  }\n\n  /** Spend path with the staker key and the unbonding timelock */\n  getUnbondingTimelockMiniscriptNode(): ast.MiniscriptNode {\n    return { and_v: [pk(this.stakerKey), { older: this.unbondingTimeLock }] };\n  }\n\n  /** Spend path with the staker key and the covenant keys */\n  getUnbondingMiniscriptNode(): ast.MiniscriptNode {\n    return { and_v: [pk(this.stakerKey), { multi_a: multiArgs(this.covenantThreshold, this.covenantKeys) }] };\n  }\n\n  /** Spend path with the finality provider keys and the covenant keys */\n  getSlashingMiniscriptNode(): ast.MiniscriptNode {\n    return {\n      and_v: [\n        {\n          and_v: [\n            pk(this.stakerKey),\n            this.finalityProviderKeys.length === 1\n              ? { 'v:pk': this.finalityProviderKeys[0].toString('hex') }\n              : { 'v:multi_a': multiArgs(1, this.finalityProviderKeys) },\n          ],\n        },\n        { multi_a: multiArgs(this.covenantThreshold, this.covenantKeys) },\n      ],\n    };\n  }\n\n  /**\n   * Creates a descriptor for a staking output.\n   *\n   * Three spend paths:\n   * - the slashing script,\n   * - the unbonding script,\n   * - the timelocked unstaking script.\n   */\n  getStakingDescriptor(): Descriptor {\n    return taprootScriptOnlyFromAst([\n      this.getSlashingMiniscriptNode(),\n      [this.getUnbondingMiniscriptNode(), this.getStakingTimelockMiniscriptNode()],\n    ]);\n  }\n\n  /**\n   * Creates a descriptor for the timelocked unbonding script.\n   */\n  getUnbondingTimelockDescriptor(): Descriptor {\n    return taprootScriptOnlyFromAst(this.getUnbondingTimelockMiniscriptNode());\n  }\n\n  /**\n   * Creates a descriptor with two script paths: the slashing script and the timelocked unbonding script.\n   */\n  getUnbondingDescriptor(): Descriptor {\n    return taprootScriptOnlyFromAst([this.getSlashingMiniscriptNode(), this.getUnbondingTimelockMiniscriptNode()]);\n  }\n}\n"]}
|
|
@@ -1,159 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.parseStakingDescriptor = parseStakingDescriptor;
|
|
4
|
-
exports.parseUnbondingDescriptor = parseUnbondingDescriptor;
|
|
5
|
-
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
6
|
-
const descriptor_1 = require("@bitgo-beta/utxo-core/descriptor");
|
|
7
|
-
const descriptor_2 = require("./descriptor");
|
|
8
|
-
function parseMulti(multi) {
|
|
9
|
-
if (!Array.isArray(multi) || multi.length < 1) {
|
|
10
|
-
throw new Error('Invalid multi structure: not an array or empty');
|
|
11
|
-
}
|
|
12
|
-
const [threshold, ...keys] = multi;
|
|
13
|
-
if (typeof threshold !== 'number') {
|
|
14
|
-
throw new Error('Invalid multi structure: threshold is not a number');
|
|
15
|
-
}
|
|
16
|
-
if (!keys.every((k) => typeof k === 'string')) {
|
|
17
|
-
throw new Error('Invalid multi structure: not all keys are strings');
|
|
18
|
-
}
|
|
19
|
-
return [threshold, keys];
|
|
20
|
-
}
|
|
21
|
-
function parseUnilateralTimelock(node, matcher) {
|
|
22
|
-
const pattern = {
|
|
23
|
-
and_v: [{ 'v:pk': { $var: 'key' } }, { older: { $var: 'timelock' } }],
|
|
24
|
-
};
|
|
25
|
-
const match = matcher.match(node, pattern);
|
|
26
|
-
if (!match) {
|
|
27
|
-
return null;
|
|
28
|
-
}
|
|
29
|
-
if (typeof match.key !== 'string') {
|
|
30
|
-
throw new Error('key must be a string');
|
|
31
|
-
}
|
|
32
|
-
if (typeof match.timelock !== 'number') {
|
|
33
|
-
throw new Error('timelock must be a number');
|
|
34
|
-
}
|
|
35
|
-
return { key: match.key, timelock: match.timelock };
|
|
36
|
-
}
|
|
37
|
-
function parseSlashingNode(slashingNode, matcher) {
|
|
38
|
-
const slashingPattern = {
|
|
39
|
-
and_v: [
|
|
40
|
-
{
|
|
41
|
-
and_v: [{ 'v:pk': { $var: 'stakerKey' } }, { $var: 'finalityProviderKeyOrMulti' }],
|
|
42
|
-
},
|
|
43
|
-
{ multi_a: { $var: 'covenantMulti' } },
|
|
44
|
-
],
|
|
45
|
-
};
|
|
46
|
-
const slashingMatch = matcher.match(slashingNode, slashingPattern);
|
|
47
|
-
if (!slashingMatch) {
|
|
48
|
-
throw new Error('Slashing node does not match expected pattern');
|
|
49
|
-
}
|
|
50
|
-
if (typeof slashingMatch.stakerKey !== 'string') {
|
|
51
|
-
throw new Error('stakerKey must be a string');
|
|
52
|
-
}
|
|
53
|
-
const [covenantThreshold, covenantKeyStrings] = parseMulti(slashingMatch.covenantMulti);
|
|
54
|
-
const covenantKeys = covenantKeyStrings.map((k) => Buffer.from(k, 'hex'));
|
|
55
|
-
let finalityProviderKeys;
|
|
56
|
-
const fpKeyOrMulti = slashingMatch.finalityProviderKeyOrMulti;
|
|
57
|
-
if ('v:pk' in fpKeyOrMulti) {
|
|
58
|
-
finalityProviderKeys = [Buffer.from(fpKeyOrMulti['v:pk'], 'hex')];
|
|
59
|
-
}
|
|
60
|
-
else if ('v:multi_a' in fpKeyOrMulti) {
|
|
61
|
-
const [threshold, keyStrings] = parseMulti(fpKeyOrMulti['v:multi_a']);
|
|
62
|
-
if (threshold !== 1) {
|
|
63
|
-
throw new Error('Finality provider multi threshold must be 1');
|
|
64
|
-
}
|
|
65
|
-
finalityProviderKeys = keyStrings.map((k) => Buffer.from(k, 'hex'));
|
|
66
|
-
}
|
|
67
|
-
else {
|
|
68
|
-
throw new Error('Invalid finality provider key structure');
|
|
69
|
-
}
|
|
70
|
-
return {
|
|
71
|
-
stakerKey: slashingMatch.stakerKey,
|
|
72
|
-
finalityProviderKeys,
|
|
73
|
-
covenantKeys,
|
|
74
|
-
covenantThreshold,
|
|
75
|
-
};
|
|
76
|
-
}
|
|
77
|
-
/**
|
|
78
|
-
* @return parsed staking descriptor components or null if the descriptor does not match the expected staking pattern.
|
|
79
|
-
*/
|
|
80
|
-
function parseStakingDescriptor(descriptor) {
|
|
81
|
-
const pattern = {
|
|
82
|
-
tr: [
|
|
83
|
-
(0, descriptor_2.getUnspendableKey)(),
|
|
84
|
-
[{ $var: 'slashingMiniscriptNode' }, [{ $var: 'unbondingMiniscriptNode' }, { $var: 'timelockMiniscriptNode' }]],
|
|
85
|
-
],
|
|
86
|
-
};
|
|
87
|
-
const matcher = new descriptor_1.PatternMatcher();
|
|
88
|
-
const descriptorNode = descriptor instanceof wasm_utxo_1.Descriptor ? wasm_utxo_1.ast.fromDescriptor(descriptor) : descriptor;
|
|
89
|
-
const result = matcher.match(descriptorNode, pattern);
|
|
90
|
-
if (!result) {
|
|
91
|
-
return null;
|
|
92
|
-
}
|
|
93
|
-
const slashingNode = result.slashingMiniscriptNode;
|
|
94
|
-
const unbondingNode = result.unbondingMiniscriptNode;
|
|
95
|
-
const timelockNode = result.timelockMiniscriptNode;
|
|
96
|
-
// Verify slashing node shape: and_v([and_v([pk, pk/multi_a]), multi_a])
|
|
97
|
-
const { stakerKey: stakerKey1, finalityProviderKeys, covenantKeys, covenantThreshold, } = parseSlashingNode(slashingNode, matcher);
|
|
98
|
-
// Verify unbonding node shape: and_v([pk, multi_a])
|
|
99
|
-
const unbondingPattern = {
|
|
100
|
-
and_v: [{ 'v:pk': { $var: 'stakerKey2' } }, { multi_a: { $var: 'covenantMulti2' } }],
|
|
101
|
-
};
|
|
102
|
-
const unbondingMatch = matcher.match(unbondingNode, unbondingPattern);
|
|
103
|
-
if (!unbondingMatch) {
|
|
104
|
-
throw new Error('Unbonding node does not match expected pattern');
|
|
105
|
-
}
|
|
106
|
-
// Verify unbonding timelock node shape: and_v([pk, older])
|
|
107
|
-
const unilateralTimelock = parseUnilateralTimelock(timelockNode, matcher);
|
|
108
|
-
if (!unilateralTimelock) {
|
|
109
|
-
return null;
|
|
110
|
-
}
|
|
111
|
-
const { key: stakerKey3, timelock: stakingTimeLock } = unilateralTimelock;
|
|
112
|
-
// Verify all staker keys are the same
|
|
113
|
-
if (stakerKey1 !== unbondingMatch.stakerKey2 || unbondingMatch.stakerKey2 !== stakerKey3) {
|
|
114
|
-
throw new Error('Staker keys must be identical across all nodes');
|
|
115
|
-
}
|
|
116
|
-
const stakerKey = Buffer.from(stakerKey1, 'hex');
|
|
117
|
-
return {
|
|
118
|
-
stakerKey,
|
|
119
|
-
finalityProviderKeys,
|
|
120
|
-
covenantKeys,
|
|
121
|
-
covenantThreshold,
|
|
122
|
-
stakingTimeLock,
|
|
123
|
-
slashingMiniscriptNode: slashingNode,
|
|
124
|
-
unbondingMiniscriptNode: unbondingNode,
|
|
125
|
-
timelockMiniscriptNode: timelockNode,
|
|
126
|
-
};
|
|
127
|
-
}
|
|
128
|
-
function parseUnbondingDescriptor(descriptor) {
|
|
129
|
-
const pattern = {
|
|
130
|
-
tr: [(0, descriptor_2.getUnspendableKey)(), [{ $var: 'slashingMiniscriptNode' }, { $var: 'unbondingTimelockMiniscriptNode' }]],
|
|
131
|
-
};
|
|
132
|
-
const matcher = new descriptor_1.PatternMatcher();
|
|
133
|
-
const descriptorNode = descriptor instanceof wasm_utxo_1.Descriptor ? wasm_utxo_1.ast.fromDescriptor(descriptor) : descriptor;
|
|
134
|
-
const result = matcher.match(descriptorNode, pattern);
|
|
135
|
-
if (!result) {
|
|
136
|
-
return null;
|
|
137
|
-
}
|
|
138
|
-
const slashingNode = result.slashingMiniscriptNode;
|
|
139
|
-
const unbondingTimelockNode = result.unbondingTimelockMiniscriptNode;
|
|
140
|
-
const { stakerKey: stakerKey1, finalityProviderKeys, covenantKeys, covenantThreshold, } = parseSlashingNode(slashingNode, matcher);
|
|
141
|
-
const unilateralTimelock = parseUnilateralTimelock(unbondingTimelockNode, matcher);
|
|
142
|
-
if (!unilateralTimelock) {
|
|
143
|
-
return null;
|
|
144
|
-
}
|
|
145
|
-
const { key: stakerKey2, timelock: unbondingTimeLock } = unilateralTimelock;
|
|
146
|
-
if (stakerKey1 !== stakerKey2) {
|
|
147
|
-
throw new Error('Staker keys must be identical across all nodes');
|
|
148
|
-
}
|
|
149
|
-
return {
|
|
150
|
-
stakerKey: Buffer.from(stakerKey1, 'hex'),
|
|
151
|
-
finalityProviderKeys,
|
|
152
|
-
covenantKeys,
|
|
153
|
-
covenantThreshold,
|
|
154
|
-
unbondingTimeLock,
|
|
155
|
-
slashingMiniscriptNode: slashingNode,
|
|
156
|
-
unbondingTimelockMiniscriptNode: unbondingTimelockNode,
|
|
157
|
-
};
|
|
158
|
-
}
|
|
159
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseDescriptor.js","sourceRoot":"","sources":["../../../src/babylon/parseDescriptor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Descriptor, ast } from '@bitgo/wasm-utxo';\nimport { PatternMatcher, Pattern } from '@bitgo-beta/utxo-core/descriptor';\n\nimport { getUnspendableKey } from './descriptor';\n\nexport type ParsedStakingDescriptor = {\n  stakerKey: Buffer;\n  finalityProviderKeys: Buffer[];\n  covenantKeys: Buffer[];\n  covenantThreshold: number;\n  stakingTimeLock: number;\n  slashingMiniscriptNode: ast.MiniscriptNode;\n  unbondingMiniscriptNode: ast.MiniscriptNode;\n  timelockMiniscriptNode: ast.MiniscriptNode;\n};\n\nfunction parseMulti(multi: unknown): [number, string[]] {\n  if (!Array.isArray(multi) || multi.length < 1) {\n    throw new Error('Invalid multi structure: not an array or empty');\n  }\n  const [threshold, ...keys] = multi;\n  if (typeof threshold !== 'number') {\n    throw new Error('Invalid multi structure: threshold is not a number');\n  }\n  if (!keys.every((k) => typeof k === 'string')) {\n    throw new Error('Invalid multi structure: not all keys are strings');\n  }\n  return [threshold, keys];\n}\n\nfunction parseUnilateralTimelock(\n  node: ast.MiniscriptNode,\n  matcher: PatternMatcher\n): { key: string; timelock: number } | null {\n  const pattern: Pattern = {\n    and_v: [{ 'v:pk': { $var: 'key' } }, { older: { $var: 'timelock' } }],\n  };\n  const match = matcher.match(node, pattern);\n  if (!match) {\n    return null;\n  }\n  if (typeof match.key !== 'string') {\n    throw new Error('key must be a string');\n  }\n  if (typeof match.timelock !== 'number') {\n    throw new Error('timelock must be a number');\n  }\n  return { key: match.key, timelock: match.timelock };\n}\n\nfunction parseSlashingNode(\n  slashingNode: ast.MiniscriptNode,\n  matcher: PatternMatcher\n): {\n  stakerKey: string;\n  finalityProviderKeys: Buffer[];\n  covenantKeys: Buffer[];\n  covenantThreshold: number;\n} {\n  const slashingPattern: Pattern = {\n    and_v: [\n      {\n        and_v: [{ 'v:pk': { $var: 'stakerKey' } }, { $var: 'finalityProviderKeyOrMulti' }],\n      },\n      { multi_a: { $var: 'covenantMulti' } },\n    ],\n  };\n\n  const slashingMatch = matcher.match(slashingNode, slashingPattern);\n  if (!slashingMatch) {\n    throw new Error('Slashing node does not match expected pattern');\n  }\n\n  if (typeof slashingMatch.stakerKey !== 'string') {\n    throw new Error('stakerKey must be a string');\n  }\n\n  const [covenantThreshold, covenantKeyStrings] = parseMulti(slashingMatch.covenantMulti);\n  const covenantKeys = covenantKeyStrings.map((k) => Buffer.from(k, 'hex'));\n\n  let finalityProviderKeys: Buffer[];\n  const fpKeyOrMulti = slashingMatch.finalityProviderKeyOrMulti as ast.MiniscriptNode;\n  if ('v:pk' in fpKeyOrMulti) {\n    finalityProviderKeys = [Buffer.from(fpKeyOrMulti['v:pk'], 'hex')];\n  } else if ('v:multi_a' in fpKeyOrMulti) {\n    const [threshold, keyStrings] = parseMulti(fpKeyOrMulti['v:multi_a']);\n    if (threshold !== 1) {\n      throw new Error('Finality provider multi threshold must be 1');\n    }\n    finalityProviderKeys = keyStrings.map((k) => Buffer.from(k, 'hex'));\n  } else {\n    throw new Error('Invalid finality provider key structure');\n  }\n\n  return {\n    stakerKey: slashingMatch.stakerKey,\n    finalityProviderKeys,\n    covenantKeys,\n    covenantThreshold,\n  };\n}\n\n/**\n * @return parsed staking descriptor components or null if the descriptor does not match the expected staking pattern.\n */\nexport function parseStakingDescriptor(descriptor: Descriptor | ast.DescriptorNode): ParsedStakingDescriptor | null {\n  const pattern: Pattern = {\n    tr: [\n      getUnspendableKey(),\n      [{ $var: 'slashingMiniscriptNode' }, [{ $var: 'unbondingMiniscriptNode' }, { $var: 'timelockMiniscriptNode' }]],\n    ],\n  };\n\n  const matcher = new PatternMatcher();\n  const descriptorNode = descriptor instanceof Descriptor ? ast.fromDescriptor(descriptor) : descriptor;\n  const result = matcher.match(descriptorNode, pattern);\n\n  if (!result) {\n    return null;\n  }\n\n  const slashingNode = result.slashingMiniscriptNode as ast.MiniscriptNode;\n  const unbondingNode = result.unbondingMiniscriptNode as ast.MiniscriptNode;\n  const timelockNode = result.timelockMiniscriptNode as ast.MiniscriptNode;\n\n  // Verify slashing node shape: and_v([and_v([pk, pk/multi_a]), multi_a])\n  const {\n    stakerKey: stakerKey1,\n    finalityProviderKeys,\n    covenantKeys,\n    covenantThreshold,\n  } = parseSlashingNode(slashingNode, matcher);\n\n  // Verify unbonding node shape: and_v([pk, multi_a])\n  const unbondingPattern: Pattern = {\n    and_v: [{ 'v:pk': { $var: 'stakerKey2' } }, { multi_a: { $var: 'covenantMulti2' } }],\n  };\n\n  const unbondingMatch = matcher.match(unbondingNode, unbondingPattern);\n  if (!unbondingMatch) {\n    throw new Error('Unbonding node does not match expected pattern');\n  }\n\n  // Verify unbonding timelock node shape: and_v([pk, older])\n  const unilateralTimelock = parseUnilateralTimelock(timelockNode, matcher);\n  if (!unilateralTimelock) {\n    return null;\n  }\n\n  const { key: stakerKey3, timelock: stakingTimeLock } = unilateralTimelock;\n\n  // Verify all staker keys are the same\n  if (stakerKey1 !== unbondingMatch.stakerKey2 || unbondingMatch.stakerKey2 !== stakerKey3) {\n    throw new Error('Staker keys must be identical across all nodes');\n  }\n\n  const stakerKey = Buffer.from(stakerKey1, 'hex');\n\n  return {\n    stakerKey,\n    finalityProviderKeys,\n    covenantKeys,\n    covenantThreshold,\n    stakingTimeLock,\n    slashingMiniscriptNode: slashingNode,\n    unbondingMiniscriptNode: unbondingNode,\n    timelockMiniscriptNode: timelockNode,\n  };\n}\n\nexport type ParsedUnbondingDescriptor = {\n  stakerKey: Buffer;\n  finalityProviderKeys: Buffer[];\n  covenantKeys: Buffer[];\n  covenantThreshold: number;\n  unbondingTimeLock: number;\n  slashingMiniscriptNode: ast.MiniscriptNode;\n  unbondingTimelockMiniscriptNode: ast.MiniscriptNode;\n};\n\nexport function parseUnbondingDescriptor(\n  descriptor: Descriptor | ast.DescriptorNode\n): ParsedUnbondingDescriptor | null {\n  const pattern: Pattern = {\n    tr: [getUnspendableKey(), [{ $var: 'slashingMiniscriptNode' }, { $var: 'unbondingTimelockMiniscriptNode' }]],\n  };\n\n  const matcher = new PatternMatcher();\n  const descriptorNode = descriptor instanceof Descriptor ? ast.fromDescriptor(descriptor) : descriptor;\n  const result = matcher.match(descriptorNode, pattern);\n\n  if (!result) {\n    return null;\n  }\n\n  const slashingNode = result.slashingMiniscriptNode as ast.MiniscriptNode;\n  const unbondingTimelockNode = result.unbondingTimelockMiniscriptNode as ast.MiniscriptNode;\n\n  const {\n    stakerKey: stakerKey1,\n    finalityProviderKeys,\n    covenantKeys,\n    covenantThreshold,\n  } = parseSlashingNode(slashingNode, matcher);\n\n  const unilateralTimelock = parseUnilateralTimelock(unbondingTimelockNode, matcher);\n  if (!unilateralTimelock) {\n    return null;\n  }\n\n  const { key: stakerKey2, timelock: unbondingTimeLock } = unilateralTimelock;\n\n  if (stakerKey1 !== stakerKey2) {\n    throw new Error('Staker keys must be identical across all nodes');\n  }\n\n  return {\n    stakerKey: Buffer.from(stakerKey1, 'hex'),\n    finalityProviderKeys,\n    covenantKeys,\n    covenantThreshold,\n    unbondingTimeLock,\n    slashingMiniscriptNode: slashingNode,\n    unbondingTimelockMiniscriptNode: unbondingTimelockNode,\n  };\n}\n"]}
|