@bitgo-beta/utxo-staking 1.1.1-beta.477 → 1.1.1-beta.478
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/src/babylon/descriptor.d.ts +1 -0
- package/dist/src/babylon/descriptor.d.ts.map +1 -1
- package/dist/src/babylon/descriptor.js +2 -1
- package/dist/src/babylon/parseDescriptor.d.ts +16 -2
- package/dist/src/babylon/parseDescriptor.d.ts.map +1 -1
- package/dist/src/babylon/parseDescriptor.js +114 -25
- package/package.json +5 -5
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
import { Descriptor, ast } from '@bitgo/wasm-miniscript';
|
|
7
7
|
import { StakingParams } from '@bitgo-beta/babylonlabs-io-btc-staking-ts';
|
|
8
8
|
export declare function getUnspendableKey(): string;
|
|
9
|
+
export declare function sortedKeys(keys: Buffer[]): Buffer[];
|
|
9
10
|
export declare class BabylonDescriptorBuilder {
|
|
10
11
|
stakerKey: Buffer;
|
|
11
12
|
finalityProviderKeys: Buffer[];
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"descriptor.d.ts","sourceRoot":"","sources":["../../../src/babylon/descriptor.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,2CAA2C,CAAC;AAE1E,wBAAgB,iBAAiB,IAAI,MAAM,CAG1C;
|
|
1
|
+
{"version":3,"file":"descriptor.d.ts","sourceRoot":"","sources":["../../../src/babylon/descriptor.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,2CAA2C,CAAC;AAE1E,wBAAgB,iBAAiB,IAAI,MAAM,CAG1C;AAOD,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,MAAM,EAAE,CAEnD;AAUD,qBAAa,wBAAwB;IAE1B,SAAS,EAAE,MAAM;IACjB,oBAAoB,EAAE,MAAM,EAAE;IAC9B,YAAY,EAAE,MAAM,EAAE;IACtB,iBAAiB,EAAE,MAAM;IACzB,eAAe,EAAE,MAAM;IACvB,iBAAiB,EAAE,MAAM;gBALzB,SAAS,EAAE,MAAM,EACjB,oBAAoB,EAAE,MAAM,EAAE,EAC9B,YAAY,EAAE,MAAM,EAAE,EACtB,iBAAiB,EAAE,MAAM,EACzB,eAAe,EAAE,MAAM,EACvB,iBAAiB,EAAE,MAAM;IAGlC,MAAM,CAAC,UAAU,CACf,MAAM,EAAE;QACN,SAAS,EAAE,MAAM,CAAC;QAClB,oBAAoB,EAAE,MAAM,EAAE,CAAC;KAChC,GAAG,aAAa,GAChB,wBAAwB;IAW3B,8DAA8D;IAC9D,gCAAgC,IAAI,GAAG,CAAC,cAAc;IAItD,gEAAgE;IAChE,kCAAkC,IAAI,GAAG,CAAC,cAAc;IAIxD,2DAA2D;IAC3D,0BAA0B,IAAI,GAAG,CAAC,cAAc;IAIhD,uEAAuE;IACvE,yBAAyB,IAAI,GAAG,CAAC,cAAc;IAgB/C;;;;;;;OAOG;IACH,oBAAoB,IAAI,UAAU;IAOlC;;OAEG;IACH,8BAA8B,IAAI,UAAU;IAI5C;;OAEG;IACH,sBAAsB,IAAI,UAAU;CAGrC"}
|
|
@@ -7,6 +7,7 @@
|
|
|
7
7
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
8
|
exports.BabylonDescriptorBuilder = void 0;
|
|
9
9
|
exports.getUnspendableKey = getUnspendableKey;
|
|
10
|
+
exports.sortedKeys = sortedKeys;
|
|
10
11
|
const wasm_miniscript_1 = require("@bitgo/wasm-miniscript");
|
|
11
12
|
function getUnspendableKey() {
|
|
12
13
|
// https://github.com/babylonlabs-io/btc-staking-ts/blob/v0.4.0-rc.2/src/constants/internalPubkey.ts
|
|
@@ -93,4 +94,4 @@ class BabylonDescriptorBuilder {
|
|
|
93
94
|
}
|
|
94
95
|
}
|
|
95
96
|
exports.BabylonDescriptorBuilder = BabylonDescriptorBuilder;
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
97
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"descriptor.js","sourceRoot":"","sources":["../../../src/babylon/descriptor.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;AAKH,8CAGC;AAOD,gCAEC;AAfD,4DAAyD;AAGzD,SAAgB,iBAAiB;IAC/B,oGAAoG;IACpG,OAAO,kEAAkE,CAAC;AAC5E,CAAC;AAED,iDAAiD;AACjD,SAAS,EAAE,CAAC,CAAS;IACnB,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;AACvC,CAAC;AAED,SAAgB,UAAU,CAAC,IAAc;IACvC,OAAO,CAAC,GAAG,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;AAChD,CAAC;AAED,SAAS,SAAS,CAAC,SAAiB,EAAE,IAAc;IAClD,OAAO,CAAC,SAAS,EAAE,GAAG,UAAU,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AACxE,CAAC;AAED,SAAS,wBAAwB,CAAC,CAAkB;IAClD,OAAO,4BAAU,CAAC,UAAU,CAAC,qBAAG,CAAC,UAAU,CAAC,EAAE,EAAE,EAAE,CAAC,iBAAiB,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,UAAU,CAAC,CAAC;AAC7F,CAAC;AAED,MAAa,wBAAwB;IACnC,YACS,SAAiB,EACjB,oBAA8B,EAC9B,YAAsB,EACtB,iBAAyB,EACzB,eAAuB,EACvB,iBAAyB;QALzB,cAAS,GAAT,SAAS,CAAQ;QACjB,yBAAoB,GAApB,oBAAoB,CAAU;QAC9B,iBAAY,GAAZ,YAAY,CAAU;QACtB,sBAAiB,GAAjB,iBAAiB,CAAQ;QACzB,oBAAe,GAAf,eAAe,CAAQ;QACvB,sBAAiB,GAAjB,iBAAiB,CAAQ;IAC/B,CAAC;IAEJ,MAAM,CAAC,UAAU,CACf,MAGiB;QAEjB,OAAO,IAAI,wBAAwB,CACjC,MAAM,CAAC,SAAS,EAChB,MAAM,CAAC,oBAAoB,EAC3B,MAAM,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,EAC3D,MAAM,CAAC,cAAc,EACrB,MAAM,CAAC,oBAAoB,EAC3B,MAAM,CAAC,aAAa,CACrB,CAAC;IACJ,CAAC;IAED,8DAA8D;IAC9D,gCAAgC;QAC9B,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,EAAE,CAAC;IAC1E,CAAC;IAED,gEAAgE;IAChE,kCAAkC;QAChC,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,iBAAiB,EAAE,CAAC,EAAE,CAAC;IAC5E,CAAC;IAED,2DAA2D;IAC3D,0BAA0B;QACxB,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,OAAO,EAAE,SAAS,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE,CAAC;IAC5G,CAAC;IAED,uEAAuE;IACvE,yBAAyB;QACvB,OAAO;YACL,KAAK,EAAE;gBACL;oBACE,KAAK,EAAE;wBACL,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC;wBAClB,IAAI,CAAC,oBAAoB,CAAC,MAAM,KAAK,CAAC;4BACpC,CAAC,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;4BAC1D,CAAC,CAAC,EAAE,WAAW,EAAE,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC,oBAAoB,CAAC,EAAE;qBAC7D;iBACF;gBACD,EAAE,OAAO,EAAE,SAAS,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,YAAY,CAAC,EAAE;aAClE;SACF,CAAC;IACJ,CAAC;IAED;;;;;;;OAOG;IACH,oBAAoB;QAClB,OAAO,wBAAwB,CAAC;YAC9B,IAAI,CAAC,yBAAyB,EAAE;YAChC,CAAC,IAAI,CAAC,0BAA0B,EAAE,EAAE,IAAI,CAAC,gCAAgC,EAAE,CAAC;SAC7E,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,8BAA8B;QAC5B,OAAO,wBAAwB,CAAC,IAAI,CAAC,kCAAkC,EAAE,CAAC,CAAC;IAC7E,CAAC;IAED;;OAEG;IACH,sBAAsB;QACpB,OAAO,wBAAwB,CAAC,CAAC,IAAI,CAAC,yBAAyB,EAAE,EAAE,IAAI,CAAC,kCAAkC,EAAE,CAAC,CAAC,CAAC;IACjH,CAAC;CACF;AAtFD,4DAsFC","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-miniscript';\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,5 +1,10 @@
|
|
|
1
1
|
import { Descriptor, ast } from '@bitgo/wasm-miniscript';
|
|
2
|
-
type ParsedStakingDescriptor = {
|
|
2
|
+
export type ParsedStakingDescriptor = {
|
|
3
|
+
stakerKey: Buffer;
|
|
4
|
+
finalityProviderKeys: Buffer[];
|
|
5
|
+
covenantKeys: Buffer[];
|
|
6
|
+
covenantThreshold: number;
|
|
7
|
+
stakingTimeLock: number;
|
|
3
8
|
slashingMiniscriptNode: ast.MiniscriptNode;
|
|
4
9
|
unbondingMiniscriptNode: ast.MiniscriptNode;
|
|
5
10
|
timelockMiniscriptNode: ast.MiniscriptNode;
|
|
@@ -8,5 +13,14 @@ type ParsedStakingDescriptor = {
|
|
|
8
13
|
* @return parsed staking descriptor components or null if the descriptor does not match the expected staking pattern.
|
|
9
14
|
*/
|
|
10
15
|
export declare function parseStakingDescriptor(descriptor: Descriptor | ast.DescriptorNode): ParsedStakingDescriptor | null;
|
|
11
|
-
export {
|
|
16
|
+
export type ParsedUnbondingDescriptor = {
|
|
17
|
+
stakerKey: Buffer;
|
|
18
|
+
finalityProviderKeys: Buffer[];
|
|
19
|
+
covenantKeys: Buffer[];
|
|
20
|
+
covenantThreshold: number;
|
|
21
|
+
unbondingTimeLock: number;
|
|
22
|
+
slashingMiniscriptNode: ast.MiniscriptNode;
|
|
23
|
+
unbondingTimelockMiniscriptNode: ast.MiniscriptNode;
|
|
24
|
+
};
|
|
25
|
+
export declare function parseUnbondingDescriptor(descriptor: Descriptor | ast.DescriptorNode): ParsedUnbondingDescriptor | null;
|
|
12
26
|
//# sourceMappingURL=parseDescriptor.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"parseDescriptor.d.ts","sourceRoot":"","sources":["../../../src/babylon/parseDescriptor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,wBAAwB,CAAC;AAKzD,
|
|
1
|
+
{"version":3,"file":"parseDescriptor.d.ts","sourceRoot":"","sources":["../../../src/babylon/parseDescriptor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,wBAAwB,CAAC;AAKzD,MAAM,MAAM,uBAAuB,GAAG;IACpC,SAAS,EAAE,MAAM,CAAC;IAClB,oBAAoB,EAAE,MAAM,EAAE,CAAC;IAC/B,YAAY,EAAE,MAAM,EAAE,CAAC;IACvB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,eAAe,EAAE,MAAM,CAAC;IACxB,sBAAsB,EAAE,GAAG,CAAC,cAAc,CAAC;IAC3C,uBAAuB,EAAE,GAAG,CAAC,cAAc,CAAC;IAC5C,sBAAsB,EAAE,GAAG,CAAC,cAAc,CAAC;CAC5C,CAAC;AAwFF;;GAEG;AACH,wBAAgB,sBAAsB,CAAC,UAAU,EAAE,UAAU,GAAG,GAAG,CAAC,cAAc,GAAG,uBAAuB,GAAG,IAAI,CA+DlH;AAED,MAAM,MAAM,yBAAyB,GAAG;IACtC,SAAS,EAAE,MAAM,CAAC;IAClB,oBAAoB,EAAE,MAAM,EAAE,CAAC;IAC/B,YAAY,EAAE,MAAM,EAAE,CAAC;IACvB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,iBAAiB,EAAE,MAAM,CAAC;IAC1B,sBAAsB,EAAE,GAAG,CAAC,cAAc,CAAC;IAC3C,+BAA+B,EAAE,GAAG,CAAC,cAAc,CAAC;CACrD,CAAC;AAEF,wBAAgB,wBAAwB,CACtC,UAAU,EAAE,UAAU,GAAG,GAAG,CAAC,cAAc,GAC1C,yBAAyB,GAAG,IAAI,CA2ClC"}
|
|
@@ -1,9 +1,79 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.parseStakingDescriptor = parseStakingDescriptor;
|
|
4
|
+
exports.parseUnbondingDescriptor = parseUnbondingDescriptor;
|
|
4
5
|
const wasm_miniscript_1 = require("@bitgo/wasm-miniscript");
|
|
5
6
|
const descriptor_1 = require("@bitgo-beta/utxo-core/descriptor");
|
|
6
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
|
+
}
|
|
7
77
|
/**
|
|
8
78
|
* @return parsed staking descriptor components or null if the descriptor does not match the expected staking pattern.
|
|
9
79
|
*/
|
|
@@ -24,18 +94,7 @@ function parseStakingDescriptor(descriptor) {
|
|
|
24
94
|
const unbondingNode = result.unbondingMiniscriptNode;
|
|
25
95
|
const timelockNode = result.timelockMiniscriptNode;
|
|
26
96
|
// Verify slashing node shape: and_v([and_v([pk, pk/multi_a]), multi_a])
|
|
27
|
-
const
|
|
28
|
-
and_v: [
|
|
29
|
-
{
|
|
30
|
-
and_v: [{ 'v:pk': { $var: 'stakerKey1' } }, { $var: 'finalityProviderKeyOrMulti' }],
|
|
31
|
-
},
|
|
32
|
-
{ multi_a: { $var: 'covenantMulti' } },
|
|
33
|
-
],
|
|
34
|
-
};
|
|
35
|
-
const slashingMatch = matcher.match(slashingNode, slashingPattern);
|
|
36
|
-
if (!slashingMatch) {
|
|
37
|
-
throw new Error('Slashing node does not match expected pattern');
|
|
38
|
-
}
|
|
97
|
+
const { stakerKey: stakerKey1, finalityProviderKeys, covenantKeys, covenantThreshold, } = parseSlashingNode(slashingNode, matcher);
|
|
39
98
|
// Verify unbonding node shape: and_v([pk, multi_a])
|
|
40
99
|
const unbondingPattern = {
|
|
41
100
|
and_v: [{ 'v:pk': { $var: 'stakerKey2' } }, { multi_a: { $var: 'covenantMulti2' } }],
|
|
@@ -45,26 +104,56 @@ function parseStakingDescriptor(descriptor) {
|
|
|
45
104
|
throw new Error('Unbonding node does not match expected pattern');
|
|
46
105
|
}
|
|
47
106
|
// Verify unbonding timelock node shape: and_v([pk, older])
|
|
48
|
-
const
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
const timelockMatch = matcher.match(timelockNode, timelockPattern);
|
|
52
|
-
if (!timelockMatch) {
|
|
53
|
-
throw new Error('Unbonding timelock node does not match expected pattern');
|
|
107
|
+
const unilateralTimelock = parseUnilateralTimelock(timelockNode, matcher);
|
|
108
|
+
if (!unilateralTimelock) {
|
|
109
|
+
return null;
|
|
54
110
|
}
|
|
111
|
+
const { key: stakerKey3, timelock: stakingTimeLock } = unilateralTimelock;
|
|
55
112
|
// Verify all staker keys are the same
|
|
56
|
-
if (
|
|
57
|
-
unbondingMatch.stakerKey2 !== timelockMatch.stakerKey3) {
|
|
113
|
+
if (stakerKey1 !== unbondingMatch.stakerKey2 || unbondingMatch.stakerKey2 !== stakerKey3) {
|
|
58
114
|
throw new Error('Staker keys must be identical across all nodes');
|
|
59
115
|
}
|
|
60
|
-
|
|
61
|
-
if (typeof timelockMatch.unbondingTimeLockValue !== 'number') {
|
|
62
|
-
throw new Error('Unbonding timelock value must be a number');
|
|
63
|
-
}
|
|
116
|
+
const stakerKey = Buffer.from(stakerKey1, 'hex');
|
|
64
117
|
return {
|
|
118
|
+
stakerKey,
|
|
119
|
+
finalityProviderKeys,
|
|
120
|
+
covenantKeys,
|
|
121
|
+
covenantThreshold,
|
|
122
|
+
stakingTimeLock,
|
|
65
123
|
slashingMiniscriptNode: slashingNode,
|
|
66
124
|
unbondingMiniscriptNode: unbondingNode,
|
|
67
125
|
timelockMiniscriptNode: timelockNode,
|
|
68
126
|
};
|
|
69
127
|
}
|
|
70
|
-
|
|
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_miniscript_1.Descriptor ? wasm_miniscript_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-miniscript';\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"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bitgo-beta/utxo-staking",
|
|
3
|
-
"version": "1.1.1-beta.
|
|
3
|
+
"version": "1.1.1-beta.478",
|
|
4
4
|
"description": "BitGo SDK for build UTXO staking transactions",
|
|
5
5
|
"main": "./dist/src/index.js",
|
|
6
6
|
"types": "./dist/src/index.d.ts",
|
|
@@ -43,9 +43,9 @@
|
|
|
43
43
|
},
|
|
44
44
|
"dependencies": {
|
|
45
45
|
"@babylonlabs-io/babylon-proto-ts": "1.0.0",
|
|
46
|
-
"@bitgo-beta/babylonlabs-io-btc-staking-ts": "0.4.0-beta.
|
|
47
|
-
"@bitgo-beta/utxo-core": "1.8.1-beta.
|
|
48
|
-
"@bitgo-beta/utxo-lib": "8.0.3-beta.
|
|
46
|
+
"@bitgo-beta/babylonlabs-io-btc-staking-ts": "0.4.0-beta.283",
|
|
47
|
+
"@bitgo-beta/utxo-core": "1.8.1-beta.154",
|
|
48
|
+
"@bitgo-beta/utxo-lib": "8.0.3-beta.1038",
|
|
49
49
|
"@bitgo/wasm-miniscript": "2.0.0-beta.7",
|
|
50
50
|
"bip174": "npm:@bitgo-forks/bip174@3.1.0-master.4",
|
|
51
51
|
"bip322-js": "^2.0.0",
|
|
@@ -57,5 +57,5 @@
|
|
|
57
57
|
"devDependencies": {
|
|
58
58
|
"yargs": "^17.7.2"
|
|
59
59
|
},
|
|
60
|
-
"gitHead": "
|
|
60
|
+
"gitHead": "a95cb987d191427c2ed015279bd80f39c45f6a04"
|
|
61
61
|
}
|