@bitgo-beta/utxo-staking 1.1.1-beta.91 → 1.1.1-beta.911
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 @@
|
|
|
1
|
+
{"version":3,"file":"vendor.utils.d.ts","sourceRoot":"","sources":["../../../../../test/unit/babylon/vendor.utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,YAAY,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,KAAK,MAAM,MAAM,2CAA2C,CAAC;AACpE,OAAO,KAAK,eAAe,MAAM,kCAAkC,CAAC;AAIpE,OAAO,EACL,wBAAwB,EAGxB,gBAAgB,EACjB,MAAM,sBAAsB,CAAC;AAI9B,KAAK,MAAM,GAAG;IACZ,qBAAqB,EAAE,gBAAgB,CAAC,eAAe,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;IAC7F,SAAS,EAAE,YAAY,CAAC,WAAW,CAAC;CACrC,CAAC;AAEF,wBAAsB,yCAAyC,CAC7D,OAAO,EAAE,YAAY,CAAC,OAAO,EAC7B,SAAS,EAAE,eAAe,EAC1B,gBAAgB,EAAE,eAAe,EACjC,iBAAiB,EAAE,wBAAwB,EAC3C,aAAa,EAAE,MAAM,CAAC,sBAAsB,EAAE,EAC9C,aAAa,EAAE,MAAM,EACrB,MAAM,EAAE,MAAM,EACd,IAAI,EAAE,MAAM,CAAC,IAAI,EACjB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC,MAAM,CAAC,CAoBjB;AAED,wBAAsB,+BAA+B,CACnD,OAAO,EAAE,YAAY,CAAC,OAAO,EAC7B,SAAS,EAAE,eAAe,EAC1B,gBAAgB,EAAE,eAAe,EACjC,iBAAiB,EAAE,wBAAwB,EAC3C,aAAa,EAAE,MAAM,CAAC,sBAAsB,EAAE,EAC9C,aAAa,EAAE,MAAM,EACrB,MAAM,EAAE,MAAM,EACd,IAAI,EAAE,MAAM,CAAC,IAAI,EACjB,WAAW,EAAE,MAAM,EACnB,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC,MAAM,CAAC,CAuCjB"}
|
|
@@ -0,0 +1,78 @@
|
|
|
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
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.getBitGoUtxoStakingMsgCreateBtcDelegation = getBitGoUtxoStakingMsgCreateBtcDelegation;
|
|
37
|
+
exports.getVendorMsgCreateBtcDelegation = getVendorMsgCreateBtcDelegation;
|
|
38
|
+
const vendor = __importStar(require("@bitgo-beta/babylonlabs-io-btc-staking-ts"));
|
|
39
|
+
const babylonProtobuf = __importStar(require("@babylonlabs-io/babylon-proto-ts"));
|
|
40
|
+
const address_1 = require("bitcoinjs-lib/src/address");
|
|
41
|
+
const babylonlabs_io_btc_staking_ts_1 = require("@bitgo-beta/babylonlabs-io-btc-staking-ts");
|
|
42
|
+
const babylon_1 = require("../../../src/babylon");
|
|
43
|
+
const key_utils_1 = require("./key.utils");
|
|
44
|
+
async function getBitGoUtxoStakingMsgCreateBtcDelegation(network, stakerKey, finalityProvider, descriptorBuilder, stakingParams, changeAddress, amount, utxo, feeRateSatB, blockHeight) {
|
|
45
|
+
return await (0, babylon_1.createUnsignedPreStakeRegistrationBabylonTransactionWithBtcProvider)((0, babylon_1.getBtcProviderForECKey)(descriptorBuilder, stakerKey), network, {
|
|
46
|
+
address: changeAddress,
|
|
47
|
+
publicKeyNoCoordHex: (0, key_utils_1.getXOnlyPubkey)(stakerKey).toString('hex'),
|
|
48
|
+
}, {
|
|
49
|
+
finalityProviderPksNoCoordHex: [(0, key_utils_1.getXOnlyPubkey)(finalityProvider).toString('hex')],
|
|
50
|
+
stakingAmountSat: amount,
|
|
51
|
+
stakingTimelock: (0, babylonlabs_io_btc_staking_ts_1.getBabylonParamByBtcHeight)(blockHeight, stakingParams).minStakingTimeBlocks,
|
|
52
|
+
}, 800000, [utxo], feeRateSatB, (0, address_1.toBech32)(Buffer.from('test'), 0, 'bbn'), 'delegation:create', stakingParams);
|
|
53
|
+
}
|
|
54
|
+
async function getVendorMsgCreateBtcDelegation(network, stakerKey, finalityProvider, descriptorBuilder, stakingParams, changeAddress, amount, utxo, feeRateSatB, blockHeight) {
|
|
55
|
+
const babylonProvider = {
|
|
56
|
+
async signTransaction(msg) {
|
|
57
|
+
// return unsigned payload
|
|
58
|
+
return babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation.encode(msg.value).finish();
|
|
59
|
+
},
|
|
60
|
+
};
|
|
61
|
+
const manager = new vendor.BabylonBtcStakingManager(network, stakingParams, (0, babylon_1.getBtcProviderForECKey)(descriptorBuilder, stakerKey), babylonProvider);
|
|
62
|
+
const result = await manager.preStakeRegistrationBabylonTransaction({
|
|
63
|
+
address: changeAddress,
|
|
64
|
+
publicKeyNoCoordHex: (0, key_utils_1.getXOnlyPubkey)(stakerKey).toString('hex'),
|
|
65
|
+
}, {
|
|
66
|
+
finalityProviderPksNoCoordHex: [(0, key_utils_1.getXOnlyPubkey)(finalityProvider).toString('hex')],
|
|
67
|
+
stakingAmountSat: amount,
|
|
68
|
+
stakingTimelock: (0, babylonlabs_io_btc_staking_ts_1.getBabylonParamByBtcHeight)(blockHeight, stakingParams).minStakingTimeBlocks,
|
|
69
|
+
}, blockHeight, [utxo], feeRateSatB, (0, address_1.toBech32)(Buffer.from('test'), 0, 'bbn'));
|
|
70
|
+
return {
|
|
71
|
+
unsignedDelegationMsg: {
|
|
72
|
+
typeUrl: '/babylon.btcstaking.v1.MsgCreateBTCDelegation',
|
|
73
|
+
value: babylonProtobuf.btcstakingtx.MsgCreateBTCDelegation.decode(result.signedBabylonTx),
|
|
74
|
+
},
|
|
75
|
+
stakingTx: result.stakingTx,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVuZG9yLnV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vdGVzdC91bml0L2JhYnlsb24vdmVuZG9yLnV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBcUJBLDhGQStCQztBQUVELDBFQWtEQztBQXRHRCxrRkFBb0U7QUFDcEUsa0ZBQW9FO0FBQ3BFLHVEQUFxRDtBQUNyRCw2RkFBdUY7QUFFdkYsa0RBSzhCO0FBRTlCLDJDQUE2QztBQU90QyxLQUFLLFVBQVUseUNBQXlDLENBQzdELE9BQTZCLEVBQzdCLFNBQTBCLEVBQzFCLGdCQUFpQyxFQUNqQyxpQkFBMkMsRUFDM0MsYUFBOEMsRUFDOUMsYUFBcUIsRUFDckIsTUFBYyxFQUNkLElBQWlCLEVBQ2pCLFdBQW1CLEVBQ25CLFdBQW1CO0lBRW5CLE9BQU8sTUFBTSxJQUFBLDZFQUFtRSxFQUM5RSxJQUFBLGdDQUFzQixFQUFDLGlCQUFpQixFQUFFLFNBQVMsQ0FBQyxFQUNwRCxPQUFPLEVBQ1A7UUFDRSxPQUFPLEVBQUUsYUFBYTtRQUN0QixtQkFBbUIsRUFBRSxJQUFBLDBCQUFjLEVBQUMsU0FBUyxDQUFDLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQztLQUMvRCxFQUNEO1FBQ0UsNkJBQTZCLEVBQUUsQ0FBQyxJQUFBLDBCQUFjLEVBQUMsZ0JBQWdCLENBQUMsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDakYsZ0JBQWdCLEVBQUUsTUFBTTtRQUN4QixlQUFlLEVBQUUsSUFBQSwwREFBMEIsRUFBQyxXQUFXLEVBQUUsYUFBYSxDQUFDLENBQUMsb0JBQW9CO0tBQzdGLEVBQ0QsTUFBTyxFQUNQLENBQUMsSUFBSSxDQUFDLEVBQ04sV0FBVyxFQUNYLElBQUEsa0JBQVEsRUFBQyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxLQUFLLENBQUMsRUFDdkMsbUJBQW1CLEVBQ25CLGFBQWEsQ0FDZCxDQUFDO0FBQ0osQ0FBQztBQUVNLEtBQUssVUFBVSwrQkFBK0IsQ0FDbkQsT0FBNkIsRUFDN0IsU0FBMEIsRUFDMUIsZ0JBQWlDLEVBQ2pDLGlCQUEyQyxFQUMzQyxhQUE4QyxFQUM5QyxhQUFxQixFQUNyQixNQUFjLEVBQ2QsSUFBaUIsRUFDakIsV0FBbUIsRUFDbkIsV0FBbUI7SUFFbkIsTUFBTSxlQUFlLEdBQTJCO1FBQzlDLEtBQUssQ0FBQyxlQUFlLENBQUMsR0FBRztZQUN2QiwwQkFBMEI7WUFDMUIsT0FBTyxlQUFlLENBQUMsWUFBWSxDQUFDLHNCQUFzQixDQUFDLE1BQU0sQ0FDL0QsR0FBRyxDQUFDLEtBQTRELENBQ2pFLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDYixDQUFDO0tBQ0YsQ0FBQztJQUNGLE1BQU0sT0FBTyxHQUFHLElBQUksTUFBTSxDQUFDLHdCQUF3QixDQUNqRCxPQUFPLEVBQ1AsYUFBYSxFQUNiLElBQUEsZ0NBQXNCLEVBQUMsaUJBQWlCLEVBQUUsU0FBUyxDQUFDLEVBQ3BELGVBQWUsQ0FDaEIsQ0FBQztJQUVGLE1BQU0sTUFBTSxHQUFHLE1BQU0sT0FBTyxDQUFDLHNDQUFzQyxDQUNqRTtRQUNFLE9BQU8sRUFBRSxhQUFhO1FBQ3RCLG1CQUFtQixFQUFFLElBQUEsMEJBQWMsRUFBQyxTQUFTLENBQUMsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDO0tBQy9ELEVBQ0Q7UUFDRSw2QkFBNkIsRUFBRSxDQUFDLElBQUEsMEJBQWMsRUFBQyxnQkFBZ0IsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNqRixnQkFBZ0IsRUFBRSxNQUFNO1FBQ3hCLGVBQWUsRUFBRSxJQUFBLDBEQUEwQixFQUFDLFdBQVcsRUFBRSxhQUFhLENBQUMsQ0FBQyxvQkFBb0I7S0FDN0YsRUFDRCxXQUFXLEVBQ1gsQ0FBQyxJQUFJLENBQUMsRUFDTixXQUFXLEVBQ1gsSUFBQSxrQkFBUSxFQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxFQUFFLEtBQUssQ0FBQyxDQUN4QyxDQUFDO0lBRUYsT0FBTztRQUNMLHFCQUFxQixFQUFFO1lBQ3JCLE9BQU8sRUFBRSwrQ0FBK0M7WUFDeEQsS0FBSyxFQUFFLGVBQWUsQ0FBQyxZQUFZLENBQUMsc0JBQXNCLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxlQUFlLENBQUM7U0FDMUY7UUFDRCxTQUFTLEVBQUUsTUFBTSxDQUFDLFNBQVM7S0FDNUIsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBiaXRjb2luanNsaWIgZnJvbSAnYml0Y29pbmpzLWxpYic7XG5pbXBvcnQgeyBFQ1BhaXJJbnRlcmZhY2UgfSBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5pbXBvcnQgKiBhcyB2ZW5kb3IgZnJvbSAnQGJpdGdvLWJldGEvYmFieWxvbmxhYnMtaW8tYnRjLXN0YWtpbmctdHMnO1xuaW1wb3J0ICogYXMgYmFieWxvblByb3RvYnVmIGZyb20gJ0BiYWJ5bG9ubGFicy1pby9iYWJ5bG9uLXByb3RvLXRzJztcbmltcG9ydCB7IHRvQmVjaDMyIH0gZnJvbSAnYml0Y29pbmpzLWxpYi9zcmMvYWRkcmVzcyc7XG5pbXBvcnQgeyBnZXRCYWJ5bG9uUGFyYW1CeUJ0Y0hlaWdodCB9IGZyb20gJ0BiaXRnby1iZXRhL2JhYnlsb25sYWJzLWlvLWJ0Yy1zdGFraW5nLXRzJztcblxuaW1wb3J0IHtcbiAgQmFieWxvbkRlc2NyaXB0b3JCdWlsZGVyLFxuICBjcmVhdGVVbnNpZ25lZFByZVN0YWtlUmVnaXN0cmF0aW9uQmFieWxvblRyYW5zYWN0aW9uV2l0aEJ0Y1Byb3ZpZGVyLFxuICBnZXRCdGNQcm92aWRlckZvckVDS2V5LFxuICBWYWx1ZVdpdGhUeXBlVXJsLFxufSBmcm9tICcuLi8uLi8uLi9zcmMvYmFieWxvbic7XG5cbmltcG9ydCB7IGdldFhPbmx5UHVia2V5IH0gZnJvbSAnLi9rZXkudXRpbHMnO1xuXG50eXBlIFJlc3VsdCA9IHtcbiAgdW5zaWduZWREZWxlZ2F0aW9uTXNnOiBWYWx1ZVdpdGhUeXBlVXJsPGJhYnlsb25Qcm90b2J1Zi5idGNzdGFraW5ndHguTXNnQ3JlYXRlQlRDRGVsZWdhdGlvbj47XG4gIHN0YWtpbmdUeDogYml0Y29pbmpzbGliLlRyYW5zYWN0aW9uO1xufTtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldEJpdEdvVXR4b1N0YWtpbmdNc2dDcmVhdGVCdGNEZWxlZ2F0aW9uKFxuICBuZXR3b3JrOiBiaXRjb2luanNsaWIuTmV0d29yayxcbiAgc3Rha2VyS2V5OiBFQ1BhaXJJbnRlcmZhY2UsXG4gIGZpbmFsaXR5UHJvdmlkZXI6IEVDUGFpckludGVyZmFjZSxcbiAgZGVzY3JpcHRvckJ1aWxkZXI6IEJhYnlsb25EZXNjcmlwdG9yQnVpbGRlcixcbiAgc3Rha2luZ1BhcmFtczogdmVuZG9yLlZlcnNpb25lZFN0YWtpbmdQYXJhbXNbXSxcbiAgY2hhbmdlQWRkcmVzczogc3RyaW5nLFxuICBhbW91bnQ6IG51bWJlcixcbiAgdXR4bzogdmVuZG9yLlVUWE8sXG4gIGZlZVJhdGVTYXRCOiBudW1iZXIsXG4gIGJsb2NrSGVpZ2h0OiBudW1iZXJcbik6IFByb21pc2U8UmVzdWx0PiB7XG4gIHJldHVybiBhd2FpdCBjcmVhdGVVbnNpZ25lZFByZVN0YWtlUmVnaXN0cmF0aW9uQmFieWxvblRyYW5zYWN0aW9uV2l0aEJ0Y1Byb3ZpZGVyKFxuICAgIGdldEJ0Y1Byb3ZpZGVyRm9yRUNLZXkoZGVzY3JpcHRvckJ1aWxkZXIsIHN0YWtlcktleSksXG4gICAgbmV0d29yayxcbiAgICB7XG4gICAgICBhZGRyZXNzOiBjaGFuZ2VBZGRyZXNzLFxuICAgICAgcHVibGljS2V5Tm9Db29yZEhleDogZ2V0WE9ubHlQdWJrZXkoc3Rha2VyS2V5KS50b1N0cmluZygnaGV4JyksXG4gICAgfSxcbiAgICB7XG4gICAgICBmaW5hbGl0eVByb3ZpZGVyUGtzTm9Db29yZEhleDogW2dldFhPbmx5UHVia2V5KGZpbmFsaXR5UHJvdmlkZXIpLnRvU3RyaW5nKCdoZXgnKV0sXG4gICAgICBzdGFraW5nQW1vdW50U2F0OiBhbW91bnQsXG4gICAgICBzdGFraW5nVGltZWxvY2s6IGdldEJhYnlsb25QYXJhbUJ5QnRjSGVpZ2h0KGJsb2NrSGVpZ2h0LCBzdGFraW5nUGFyYW1zKS5taW5TdGFraW5nVGltZUJsb2NrcyxcbiAgICB9LFxuICAgIDgwMF8wMDAsXG4gICAgW3V0eG9dLFxuICAgIGZlZVJhdGVTYXRCLFxuICAgIHRvQmVjaDMyKEJ1ZmZlci5mcm9tKCd0ZXN0JyksIDAsICdiYm4nKSxcbiAgICAnZGVsZWdhdGlvbjpjcmVhdGUnLFxuICAgIHN0YWtpbmdQYXJhbXNcbiAgKTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFZlbmRvck1zZ0NyZWF0ZUJ0Y0RlbGVnYXRpb24oXG4gIG5ldHdvcms6IGJpdGNvaW5qc2xpYi5OZXR3b3JrLFxuICBzdGFrZXJLZXk6IEVDUGFpckludGVyZmFjZSxcbiAgZmluYWxpdHlQcm92aWRlcjogRUNQYWlySW50ZXJmYWNlLFxuICBkZXNjcmlwdG9yQnVpbGRlcjogQmFieWxvbkRlc2NyaXB0b3JCdWlsZGVyLFxuICBzdGFraW5nUGFyYW1zOiB2ZW5kb3IuVmVyc2lvbmVkU3Rha2luZ1BhcmFtc1tdLFxuICBjaGFuZ2VBZGRyZXNzOiBzdHJpbmcsXG4gIGFtb3VudDogbnVtYmVyLFxuICB1dHhvOiB2ZW5kb3IuVVRYTyxcbiAgZmVlUmF0ZVNhdEI6IG51bWJlcixcbiAgYmxvY2tIZWlnaHQ6IG51bWJlclxuKTogUHJvbWlzZTxSZXN1bHQ+IHtcbiAgY29uc3QgYmFieWxvblByb3ZpZGVyOiB2ZW5kb3IuQmFieWxvblByb3ZpZGVyID0ge1xuICAgIGFzeW5jIHNpZ25UcmFuc2FjdGlvbihtc2cpIHtcbiAgICAgIC8vIHJldHVybiB1bnNpZ25lZCBwYXlsb2FkXG4gICAgICByZXR1cm4gYmFieWxvblByb3RvYnVmLmJ0Y3N0YWtpbmd0eC5Nc2dDcmVhdGVCVENEZWxlZ2F0aW9uLmVuY29kZShcbiAgICAgICAgbXNnLnZhbHVlIGFzIGJhYnlsb25Qcm90b2J1Zi5idGNzdGFraW5ndHguTXNnQ3JlYXRlQlRDRGVsZWdhdGlvblxuICAgICAgKS5maW5pc2goKTtcbiAgICB9LFxuICB9O1xuICBjb25zdCBtYW5hZ2VyID0gbmV3IHZlbmRvci5CYWJ5bG9uQnRjU3Rha2luZ01hbmFnZXIoXG4gICAgbmV0d29yayxcbiAgICBzdGFraW5nUGFyYW1zLFxuICAgIGdldEJ0Y1Byb3ZpZGVyRm9yRUNLZXkoZGVzY3JpcHRvckJ1aWxkZXIsIHN0YWtlcktleSksXG4gICAgYmFieWxvblByb3ZpZGVyXG4gICk7XG5cbiAgY29uc3QgcmVzdWx0ID0gYXdhaXQgbWFuYWdlci5wcmVTdGFrZVJlZ2lzdHJhdGlvbkJhYnlsb25UcmFuc2FjdGlvbihcbiAgICB7XG4gICAgICBhZGRyZXNzOiBjaGFuZ2VBZGRyZXNzLFxuICAgICAgcHVibGljS2V5Tm9Db29yZEhleDogZ2V0WE9ubHlQdWJrZXkoc3Rha2VyS2V5KS50b1N0cmluZygnaGV4JyksXG4gICAgfSxcbiAgICB7XG4gICAgICBmaW5hbGl0eVByb3ZpZGVyUGtzTm9Db29yZEhleDogW2dldFhPbmx5UHVia2V5KGZpbmFsaXR5UHJvdmlkZXIpLnRvU3RyaW5nKCdoZXgnKV0sXG4gICAgICBzdGFraW5nQW1vdW50U2F0OiBhbW91bnQsXG4gICAgICBzdGFraW5nVGltZWxvY2s6IGdldEJhYnlsb25QYXJhbUJ5QnRjSGVpZ2h0KGJsb2NrSGVpZ2h0LCBzdGFraW5nUGFyYW1zKS5taW5TdGFraW5nVGltZUJsb2NrcyxcbiAgICB9LFxuICAgIGJsb2NrSGVpZ2h0LFxuICAgIFt1dHhvXSxcbiAgICBmZWVSYXRlU2F0QixcbiAgICB0b0JlY2gzMihCdWZmZXIuZnJvbSgndGVzdCcpLCAwLCAnYmJuJylcbiAgKTtcblxuICByZXR1cm4ge1xuICAgIHVuc2lnbmVkRGVsZWdhdGlvbk1zZzoge1xuICAgICAgdHlwZVVybDogJy9iYWJ5bG9uLmJ0Y3N0YWtpbmcudjEuTXNnQ3JlYXRlQlRDRGVsZWdhdGlvbicsXG4gICAgICB2YWx1ZTogYmFieWxvblByb3RvYnVmLmJ0Y3N0YWtpbmd0eC5Nc2dDcmVhdGVCVENEZWxlZ2F0aW9uLmRlY29kZShyZXN1bHQuc2lnbmVkQmFieWxvblR4KSxcbiAgICB9LFxuICAgIHN0YWtpbmdUeDogcmVzdWx0LnN0YWtpbmdUeCxcbiAgfTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"descriptor.d.ts","sourceRoot":"","sources":["../../../../../test/unit/coreDao/descriptor.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,144 @@
|
|
|
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
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
const assert = __importStar(require("assert"));
|
|
37
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
38
|
+
const wasm_utxo_1 = require("@bitgo/wasm-utxo");
|
|
39
|
+
const testutil_1 = require("@bitgo-beta/utxo-core/testutil");
|
|
40
|
+
const coreDao_1 = require("../../../src/coreDao");
|
|
41
|
+
const utils_1 = require("./utils");
|
|
42
|
+
describe('descriptor', function () {
|
|
43
|
+
const baseFixturePath = 'test/fixtures/coreDao/descriptor/';
|
|
44
|
+
const rootWalletKeys = utxolib.testutil.getDefaultWalletKeys();
|
|
45
|
+
const key1 = rootWalletKeys.triple[0];
|
|
46
|
+
const key2 = rootWalletKeys.triple[1];
|
|
47
|
+
const key3 = rootWalletKeys.triple[2];
|
|
48
|
+
const validLocktime = 2048;
|
|
49
|
+
it('should fail if m is longer than the number of keys or not at least 1', function () {
|
|
50
|
+
assert.throws(() => {
|
|
51
|
+
(0, coreDao_1.createMultiSigDescriptor)('sh', validLocktime, 3, [key1, key2], false);
|
|
52
|
+
});
|
|
53
|
+
assert.throws(() => {
|
|
54
|
+
(0, coreDao_1.createMultiSigDescriptor)('sh', validLocktime, 0, [key1, key2], false);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
it('should fail if locktime is invalid', function () {
|
|
58
|
+
assert.throws(() => {
|
|
59
|
+
(0, coreDao_1.createMultiSigDescriptor)('sh', 0, 2, [key1, key2], false);
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
async function runTestForParams(scriptType, m, keys) {
|
|
63
|
+
const fixturePath = baseFixturePath + `${scriptType}-${m}of${keys.length}`;
|
|
64
|
+
describe(`should create a ${m} of ${keys.length} multi-sig ${scriptType} descriptor`, function () {
|
|
65
|
+
it('has expected descriptor string', async function () {
|
|
66
|
+
const descriptorString = (0, coreDao_1.createMultiSigDescriptor)(scriptType, validLocktime, m, keys, false);
|
|
67
|
+
assert.strictEqual(descriptorString, await (0, testutil_1.getFixture)(fixturePath + `-string.txt`, descriptorString), descriptorString);
|
|
68
|
+
});
|
|
69
|
+
it('has expected AST', async function () {
|
|
70
|
+
const descriptor = wasm_utxo_1.Descriptor.fromString((0, coreDao_1.createMultiSigDescriptor)(scriptType, validLocktime, m, keys, false), 'derivable');
|
|
71
|
+
assert.deepStrictEqual(descriptor.node(), await (0, testutil_1.getFixture)(fixturePath + '-ast.json', descriptor.node()));
|
|
72
|
+
});
|
|
73
|
+
it('has expected asm', async function () {
|
|
74
|
+
const descriptor = wasm_utxo_1.Descriptor.fromString((0, coreDao_1.createMultiSigDescriptor)(scriptType, validLocktime, m, keys, false), 'derivable');
|
|
75
|
+
const asmString = descriptor.atDerivationIndex(0).toAsmString();
|
|
76
|
+
assert.strictEqual(asmString, await (0, testutil_1.getFixture)(fixturePath + '-asm.txt', asmString), asmString);
|
|
77
|
+
});
|
|
78
|
+
it('can be signed', async function () {
|
|
79
|
+
// Derive the script from the descriptor
|
|
80
|
+
const descriptor = wasm_utxo_1.Descriptor.fromString((0, coreDao_1.createMultiSigDescriptor)(scriptType, validLocktime, m, keys, false), 'derivable');
|
|
81
|
+
const descriptorAt0 = descriptor.atDerivationIndex(0);
|
|
82
|
+
const script = Buffer.from(descriptorAt0.scriptPubkey());
|
|
83
|
+
// Make the prevTx
|
|
84
|
+
const prevPsbt = utxolib.testutil.constructPsbt([{ scriptType: 'p2wsh', value: BigInt(1.1e8) }], [{ script: script.toString('hex'), value: BigInt(1e8) }], utxolib.networks.bitcoin, rootWalletKeys, 'fullsigned');
|
|
85
|
+
const prevTx = prevPsbt.finalizeAllInputs().extractTransaction();
|
|
86
|
+
// Create the PSBT and sign
|
|
87
|
+
const psbt = Object.assign(new utxolib.Psbt({ network: utxolib.networks.bitcoin }), {
|
|
88
|
+
locktime: validLocktime,
|
|
89
|
+
});
|
|
90
|
+
psbt.addInput({
|
|
91
|
+
hash: prevTx.getId(),
|
|
92
|
+
index: 0,
|
|
93
|
+
sequence: 0xfffffffe,
|
|
94
|
+
});
|
|
95
|
+
if (scriptType === 'sh-wsh') {
|
|
96
|
+
psbt.updateInput(0, { witnessUtxo: { script, value: BigInt(1e8) } });
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
psbt.updateInput(0, { nonWitnessUtxo: prevTx.toBuffer() });
|
|
100
|
+
}
|
|
101
|
+
psbt.addOutput({ script, value: BigInt(0.9e8) });
|
|
102
|
+
(0, utils_1.updateInputWithDescriptor)(psbt, 0, descriptorAt0);
|
|
103
|
+
keys.forEach((signer, i) => {
|
|
104
|
+
if (i >= m) {
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
psbt.signAllInputsHD(signer);
|
|
108
|
+
});
|
|
109
|
+
// Get the fully signed transaction and check
|
|
110
|
+
const signedTx = (0, utils_1.finalizePsbt)(psbt).extractTransaction().toBuffer();
|
|
111
|
+
assert.strictEqual(signedTx.toString('hex'), await (0, testutil_1.getFixture)(fixturePath + '-tx.txt', signedTx.toString('hex')));
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
}
|
|
115
|
+
runTestForParams('sh', 2, [key1, key2]);
|
|
116
|
+
runTestForParams('sh-wsh', 2, [key1, key2]);
|
|
117
|
+
runTestForParams('sh', 3, [key1, key2, key3]);
|
|
118
|
+
runTestForParams('wsh', 3, [key1, key2, key3]);
|
|
119
|
+
it('should recreate the script used in testnet staking transaction', function () {
|
|
120
|
+
// Source: https://mempool.space/testnet/address/2MxTi2EhHKgdJFKRTBttVGGxir9ZzjmKCXw
|
|
121
|
+
// 2 of 2 multisig
|
|
122
|
+
const timelock = 'fce4cb66';
|
|
123
|
+
const pubkey1 = '03ecb6d4b7f5d56962e547fc52dd588359f5729c0ba856d6978b84723895a16691';
|
|
124
|
+
const pubkey2 = '024aaea25d82b1db2be030a05b641d6302e48ed652b1ca9cb08a67267fcbb56747';
|
|
125
|
+
const redeemScriptASM = [
|
|
126
|
+
'OP_PUSHBYTES_4',
|
|
127
|
+
timelock,
|
|
128
|
+
'OP_CLTV',
|
|
129
|
+
'OP_DROP',
|
|
130
|
+
'OP_PUSHNUM_2',
|
|
131
|
+
'OP_PUSHBYTES_33',
|
|
132
|
+
pubkey1,
|
|
133
|
+
'OP_PUSHBYTES_33',
|
|
134
|
+
pubkey2,
|
|
135
|
+
'OP_PUSHNUM_2',
|
|
136
|
+
'OP_CHECKMULTISIG',
|
|
137
|
+
].join(' ');
|
|
138
|
+
const decodedTimelock = (0, coreDao_1.decodeTimelock)(Buffer.from(timelock, 'hex'));
|
|
139
|
+
const descriptor = (0, coreDao_1.createMultiSigDescriptor)('sh', decodedTimelock, 2, [Buffer.from(pubkey1, 'hex'), Buffer.from(pubkey2, 'hex')], false);
|
|
140
|
+
const descriptorASM = wasm_utxo_1.Descriptor.fromString(descriptor, 'definite').toAsmString();
|
|
141
|
+
assert.deepStrictEqual(redeemScriptASM, descriptorASM);
|
|
142
|
+
});
|
|
143
|
+
});
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"descriptor.js","sourceRoot":"","sources":["../../../../../test/unit/coreDao/descriptor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Descriptor } from '@bitgo/wasm-utxo';\nimport { getFixture } from '@bitgo-beta/utxo-core/testutil';\n\nimport { createMultiSigDescriptor, decodeTimelock } from '../../../src/coreDao';\n\nimport { finalizePsbt, updateInputWithDescriptor } from './utils';\n\ndescribe('descriptor', function () {\n  const baseFixturePath = 'test/fixtures/coreDao/descriptor/';\n  const rootWalletKeys = utxolib.testutil.getDefaultWalletKeys();\n  const key1 = rootWalletKeys.triple[0];\n  const key2 = rootWalletKeys.triple[1];\n  const key3 = rootWalletKeys.triple[2];\n  const validLocktime = 2048;\n\n  it('should fail if m is longer than the number of keys or not at least 1', function () {\n    assert.throws(() => {\n      createMultiSigDescriptor('sh', validLocktime, 3, [key1, key2], false);\n    });\n\n    assert.throws(() => {\n      createMultiSigDescriptor('sh', validLocktime, 0, [key1, key2], false);\n    });\n  });\n\n  it('should fail if locktime is invalid', function () {\n    assert.throws(() => {\n      createMultiSigDescriptor('sh', 0, 2, [key1, key2], false);\n    });\n  });\n\n  async function runTestForParams(scriptType: 'sh' | 'sh-wsh' | 'wsh', m: number, keys: utxolib.BIP32Interface[]) {\n    const fixturePath = baseFixturePath + `${scriptType}-${m}of${keys.length}`;\n    describe(`should create a ${m} of ${keys.length} multi-sig ${scriptType} descriptor`, function () {\n      it('has expected descriptor string', async function () {\n        const descriptorString = createMultiSigDescriptor(scriptType, validLocktime, m, keys, false);\n        assert.strictEqual(\n          descriptorString,\n          await getFixture(fixturePath + `-string.txt`, descriptorString),\n          descriptorString\n        );\n      });\n\n      it('has expected AST', async function () {\n        const descriptor = Descriptor.fromString(\n          createMultiSigDescriptor(scriptType, validLocktime, m, keys, false),\n          'derivable'\n        );\n\n        assert.deepStrictEqual(descriptor.node(), await getFixture(fixturePath + '-ast.json', descriptor.node()));\n      });\n\n      it('has expected asm', async function () {\n        const descriptor = Descriptor.fromString(\n          createMultiSigDescriptor(scriptType, validLocktime, m, keys, false),\n          'derivable'\n        );\n        const asmString = descriptor.atDerivationIndex(0).toAsmString();\n        assert.strictEqual(asmString, await getFixture(fixturePath + '-asm.txt', asmString), asmString);\n      });\n\n      it('can be signed', async function () {\n        // Derive the script from the descriptor\n        const descriptor = Descriptor.fromString(\n          createMultiSigDescriptor(scriptType, validLocktime, m, keys, false),\n          'derivable'\n        );\n        const descriptorAt0 = descriptor.atDerivationIndex(0);\n        const script = Buffer.from(descriptorAt0.scriptPubkey());\n\n        // Make the prevTx\n        const prevPsbt = utxolib.testutil.constructPsbt(\n          [{ scriptType: 'p2wsh', value: BigInt(1.1e8) }],\n          [{ script: script.toString('hex'), value: BigInt(1e8) }],\n          utxolib.networks.bitcoin,\n          rootWalletKeys,\n          'fullsigned'\n        );\n        const prevTx = prevPsbt.finalizeAllInputs().extractTransaction();\n\n        // Create the PSBT and sign\n        const psbt = Object.assign(new utxolib.Psbt({ network: utxolib.networks.bitcoin }), {\n          locktime: validLocktime,\n        });\n        psbt.addInput({\n          hash: prevTx.getId(),\n          index: 0,\n          sequence: 0xfffffffe,\n        });\n        if (scriptType === 'sh-wsh') {\n          psbt.updateInput(0, { witnessUtxo: { script, value: BigInt(1e8) } });\n        } else {\n          psbt.updateInput(0, { nonWitnessUtxo: prevTx.toBuffer() });\n        }\n        psbt.addOutput({ script, value: BigInt(0.9e8) });\n        updateInputWithDescriptor(psbt, 0, descriptorAt0);\n        keys.forEach((signer, i) => {\n          if (i >= m) {\n            return;\n          }\n          psbt.signAllInputsHD(signer);\n        });\n\n        // Get the fully signed transaction and check\n        const signedTx = finalizePsbt(psbt).extractTransaction().toBuffer();\n        assert.strictEqual(\n          signedTx.toString('hex'),\n          await getFixture(fixturePath + '-tx.txt', signedTx.toString('hex'))\n        );\n      });\n    });\n  }\n\n  runTestForParams('sh', 2, [key1, key2]);\n  runTestForParams('sh-wsh', 2, [key1, key2]);\n  runTestForParams('sh', 3, [key1, key2, key3]);\n  runTestForParams('wsh', 3, [key1, key2, key3]);\n\n  it('should recreate the script used in testnet staking transaction', function () {\n    // Source: https://mempool.space/testnet/address/2MxTi2EhHKgdJFKRTBttVGGxir9ZzjmKCXw\n    // 2 of 2 multisig\n    const timelock = 'fce4cb66';\n    const pubkey1 = '03ecb6d4b7f5d56962e547fc52dd588359f5729c0ba856d6978b84723895a16691';\n    const pubkey2 = '024aaea25d82b1db2be030a05b641d6302e48ed652b1ca9cb08a67267fcbb56747';\n    const redeemScriptASM = [\n      'OP_PUSHBYTES_4',\n      timelock,\n      'OP_CLTV',\n      'OP_DROP',\n      'OP_PUSHNUM_2',\n      'OP_PUSHBYTES_33',\n      pubkey1,\n      'OP_PUSHBYTES_33',\n      pubkey2,\n      'OP_PUSHNUM_2',\n      'OP_CHECKMULTISIG',\n    ].join(' ');\n\n    const decodedTimelock = decodeTimelock(Buffer.from(timelock, 'hex'));\n    const descriptor = createMultiSigDescriptor(\n      'sh',\n      decodedTimelock,\n      2,\n      [Buffer.from(pubkey1, 'hex'), Buffer.from(pubkey2, 'hex')],\n      false\n    );\n    const descriptorASM = Descriptor.fromString(descriptor, 'definite').toAsmString();\n    assert.deepStrictEqual(redeemScriptASM, descriptorASM);\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"opReturn.d.ts","sourceRoot":"","sources":["../../../../../test/unit/coreDao/opReturn.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,275 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
const assert_1 = __importDefault(require("assert"));
|
|
7
|
+
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
8
|
+
const testutil_1 = require("@bitgo-beta/utxo-core/testutil");
|
|
9
|
+
const coreDao_1 = require("../../../src/coreDao");
|
|
10
|
+
describe('OP_RETURN', function () {
|
|
11
|
+
const validVersion = 2;
|
|
12
|
+
const validChainId = coreDao_1.CORE_DAO_MAINNET_CHAIN_ID;
|
|
13
|
+
// random 20 byte buffers
|
|
14
|
+
const validDelegator = Buffer.alloc(20, utxo_lib_1.testutil.getKey('wasm-possum').publicKey);
|
|
15
|
+
const validValidator = Buffer.alloc(20, utxo_lib_1.testutil.getKey('possum-wasm').publicKey);
|
|
16
|
+
const validFee = 1;
|
|
17
|
+
const validRedeemScript = Buffer.from('522103a8295453660d5e212d55556666666666666666666666666666666666', 'hex');
|
|
18
|
+
const validTimelock = 800800;
|
|
19
|
+
let defaultScript;
|
|
20
|
+
before(async function () {
|
|
21
|
+
// https://docs.coredao.org/docs/Learn/products/btc-staking/design#op_return-output-1
|
|
22
|
+
const script = await (0, testutil_1.getFixture)('test/fixtures/coreDao/opReturn/documentation.txt', undefined);
|
|
23
|
+
(0, assert_1.default)(typeof script === 'string');
|
|
24
|
+
defaultScript = script;
|
|
25
|
+
});
|
|
26
|
+
describe('createCoreDaoOpReturnOutputScript', function () {
|
|
27
|
+
it('should throw if invalid parameters are passed', function () {
|
|
28
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
29
|
+
version: 292,
|
|
30
|
+
chainId: validChainId,
|
|
31
|
+
delegator: validDelegator,
|
|
32
|
+
validator: validValidator,
|
|
33
|
+
fee: validFee,
|
|
34
|
+
timelock: validTimelock,
|
|
35
|
+
}));
|
|
36
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
37
|
+
version: validVersion,
|
|
38
|
+
chainId: Buffer.alloc(32, 0),
|
|
39
|
+
delegator: validDelegator,
|
|
40
|
+
validator: validValidator,
|
|
41
|
+
fee: validFee,
|
|
42
|
+
timelock: validTimelock,
|
|
43
|
+
}));
|
|
44
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
45
|
+
version: validVersion,
|
|
46
|
+
chainId: validChainId,
|
|
47
|
+
delegator: Buffer.alloc(19, 0),
|
|
48
|
+
validator: validValidator,
|
|
49
|
+
fee: validFee,
|
|
50
|
+
timelock: validTimelock,
|
|
51
|
+
}));
|
|
52
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
53
|
+
version: validVersion,
|
|
54
|
+
chainId: validChainId,
|
|
55
|
+
delegator: validDelegator,
|
|
56
|
+
validator: Buffer.alloc(19, 0),
|
|
57
|
+
fee: validFee,
|
|
58
|
+
timelock: validTimelock,
|
|
59
|
+
}));
|
|
60
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
61
|
+
version: validVersion,
|
|
62
|
+
chainId: validChainId,
|
|
63
|
+
delegator: validDelegator,
|
|
64
|
+
validator: validValidator,
|
|
65
|
+
fee: 256,
|
|
66
|
+
timelock: validTimelock,
|
|
67
|
+
}));
|
|
68
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
69
|
+
version: validVersion,
|
|
70
|
+
chainId: validChainId,
|
|
71
|
+
delegator: validDelegator,
|
|
72
|
+
validator: validValidator,
|
|
73
|
+
fee: validFee,
|
|
74
|
+
timelock: -1,
|
|
75
|
+
}));
|
|
76
|
+
});
|
|
77
|
+
it('should throw if the length of the script is too long', function () {
|
|
78
|
+
assert_1.default.throws(() => (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
79
|
+
version: validVersion,
|
|
80
|
+
chainId: validChainId,
|
|
81
|
+
delegator: validDelegator,
|
|
82
|
+
validator: validValidator,
|
|
83
|
+
fee: validFee,
|
|
84
|
+
redeemScript: Buffer.alloc(100),
|
|
85
|
+
}));
|
|
86
|
+
});
|
|
87
|
+
it('should return a buffer with the correct length', function () {
|
|
88
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
89
|
+
version: validVersion,
|
|
90
|
+
chainId: validChainId,
|
|
91
|
+
delegator: validDelegator,
|
|
92
|
+
validator: validValidator,
|
|
93
|
+
fee: validFee,
|
|
94
|
+
timelock: validTimelock,
|
|
95
|
+
});
|
|
96
|
+
// Make sure that the first byte is the OP_RETURN opcode
|
|
97
|
+
assert_1.default.strictEqual(script[0], 0x6a);
|
|
98
|
+
// Make sure that the length of the script matches what is in the buffer
|
|
99
|
+
assert_1.default.strictEqual(
|
|
100
|
+
// We do not count the OP_RETURN opcode or the bytes for the length
|
|
101
|
+
script.length - 2, script[1]);
|
|
102
|
+
});
|
|
103
|
+
it('should have the correct placement of the values provided with a redeem script', function () {
|
|
104
|
+
// This should produce an Op_RETURN that needs the extra push bytes for the length
|
|
105
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
106
|
+
version: validVersion,
|
|
107
|
+
chainId: validChainId,
|
|
108
|
+
delegator: validDelegator,
|
|
109
|
+
validator: validValidator,
|
|
110
|
+
fee: validFee,
|
|
111
|
+
redeemScript: validRedeemScript,
|
|
112
|
+
});
|
|
113
|
+
// Make sure that the first byte is the OP_RETURN opcode
|
|
114
|
+
assert_1.default.strictEqual(script[0], 0x6a);
|
|
115
|
+
// Make sure that the length of the script matches what is in the buffer
|
|
116
|
+
assert_1.default.strictEqual(script[1], 0x4c);
|
|
117
|
+
assert_1.default.strictEqual(
|
|
118
|
+
// We do not count the OP_RETURN opcode or the length + pushbytes
|
|
119
|
+
script.length - 3, script[2]);
|
|
120
|
+
// Satoshi plus identifier
|
|
121
|
+
assert_1.default.deepStrictEqual(script.subarray(3, 7).toString('hex'), coreDao_1.CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));
|
|
122
|
+
// Make sure that the version is correct
|
|
123
|
+
assert_1.default.strictEqual(script[7], validVersion);
|
|
124
|
+
// Make sure that the chainId is correct
|
|
125
|
+
assert_1.default.deepStrictEqual(script.subarray(8, 10).toString('hex'), validChainId.toString('hex'));
|
|
126
|
+
// Make sure that the delegator is correct
|
|
127
|
+
assert_1.default.deepStrictEqual(script.subarray(10, 30).toString('hex'), validDelegator.toString('hex'));
|
|
128
|
+
// Make sure that the validator is correct
|
|
129
|
+
assert_1.default.deepStrictEqual(script.subarray(30, 50).toString('hex'), validValidator.toString('hex'));
|
|
130
|
+
// Make sure that the fee is correct
|
|
131
|
+
assert_1.default.strictEqual(script[50], validFee);
|
|
132
|
+
// Make sure that the redeemScript is correct
|
|
133
|
+
assert_1.default.deepStrictEqual(script.subarray(51, 51 + validRedeemScript.length).toString('hex'), validRedeemScript.toString('hex'));
|
|
134
|
+
});
|
|
135
|
+
it('should have the correct placement of the values provided with a blockheight timelock', function () {
|
|
136
|
+
// This should produce an Op_RETURN that needs the extra push bytes for the length
|
|
137
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
138
|
+
version: validVersion,
|
|
139
|
+
chainId: validChainId,
|
|
140
|
+
delegator: validDelegator,
|
|
141
|
+
validator: validValidator,
|
|
142
|
+
fee: validFee,
|
|
143
|
+
timelock: validTimelock,
|
|
144
|
+
});
|
|
145
|
+
// Make sure that the first byte is the OP_RETURN opcode
|
|
146
|
+
assert_1.default.strictEqual(script[0], 0x6a);
|
|
147
|
+
// Make sure that the length of the script matches what is in the buffer
|
|
148
|
+
assert_1.default.strictEqual(
|
|
149
|
+
// We do not count the OP_RETURN opcode or the length
|
|
150
|
+
script.length - 2, script[1]);
|
|
151
|
+
// Satoshi plus identifier
|
|
152
|
+
assert_1.default.deepStrictEqual(script.subarray(2, 6).toString('hex'), coreDao_1.CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));
|
|
153
|
+
// Make sure that the version is correct
|
|
154
|
+
assert_1.default.strictEqual(script[6], validVersion);
|
|
155
|
+
// Make sure that the chainId is correct
|
|
156
|
+
assert_1.default.deepStrictEqual(script.subarray(7, 9).toString('hex'), validChainId.toString('hex'));
|
|
157
|
+
// Make sure that the delegator is correct
|
|
158
|
+
assert_1.default.deepStrictEqual(script.subarray(9, 29).toString('hex'), validDelegator.toString('hex'));
|
|
159
|
+
// Make sure that the validator is correct
|
|
160
|
+
assert_1.default.deepStrictEqual(script.subarray(29, 49).toString('hex'), validValidator.toString('hex'));
|
|
161
|
+
// Make sure that the fee is correct
|
|
162
|
+
assert_1.default.strictEqual(script[49], validFee);
|
|
163
|
+
// Make sure that the redeemScript is correct
|
|
164
|
+
assert_1.default.deepStrictEqual(script.subarray(50, 54).toString('hex'), (0, coreDao_1.encodeTimelock)(validTimelock).toString('hex'));
|
|
165
|
+
assert_1.default.deepStrictEqual((0, coreDao_1.decodeTimelock)(script.subarray(50, 54)), validTimelock);
|
|
166
|
+
});
|
|
167
|
+
it('should have the correct placement of the values provided with a time timelock', function () {
|
|
168
|
+
// This should produce an Op_RETURN that needs the extra push bytes for the length
|
|
169
|
+
const timelock = new Date('2022-01-01T00:00:00Z').getTime() / 1000;
|
|
170
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
171
|
+
version: validVersion,
|
|
172
|
+
chainId: validChainId,
|
|
173
|
+
delegator: validDelegator,
|
|
174
|
+
validator: validValidator,
|
|
175
|
+
fee: validFee,
|
|
176
|
+
timelock,
|
|
177
|
+
});
|
|
178
|
+
// Make sure that the first byte is the OP_RETURN opcode
|
|
179
|
+
assert_1.default.strictEqual(script[0], 0x6a);
|
|
180
|
+
// Make sure that the length of the script matches what is in the buffer
|
|
181
|
+
assert_1.default.strictEqual(
|
|
182
|
+
// We do not count the OP_RETURN opcode or the length
|
|
183
|
+
script.length - 2, script[1]);
|
|
184
|
+
// Satoshi plus identifier
|
|
185
|
+
assert_1.default.deepStrictEqual(script.subarray(2, 6).toString('hex'), coreDao_1.CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));
|
|
186
|
+
// Make sure that the version is correct
|
|
187
|
+
assert_1.default.strictEqual(script[6], validVersion);
|
|
188
|
+
// Make sure that the chainId is correct
|
|
189
|
+
assert_1.default.deepStrictEqual(script.subarray(7, 9).toString('hex'), validChainId.toString('hex'));
|
|
190
|
+
// Make sure that the delegator is correct
|
|
191
|
+
assert_1.default.deepStrictEqual(script.subarray(9, 29).toString('hex'), validDelegator.toString('hex'));
|
|
192
|
+
// Make sure that the validator is correct
|
|
193
|
+
assert_1.default.deepStrictEqual(script.subarray(29, 49).toString('hex'), validValidator.toString('hex'));
|
|
194
|
+
// Make sure that the fee is correct
|
|
195
|
+
assert_1.default.strictEqual(script[49], validFee);
|
|
196
|
+
// Make sure that the redeemScript is correct
|
|
197
|
+
assert_1.default.deepStrictEqual(script.subarray(50, 54).toString('hex'), (0, coreDao_1.encodeTimelock)(timelock).toString('hex'));
|
|
198
|
+
assert_1.default.deepStrictEqual((0, coreDao_1.decodeTimelock)(script.subarray(50, 54)), timelock);
|
|
199
|
+
});
|
|
200
|
+
it('should recreate the example OP_RETURN correctly', function () {
|
|
201
|
+
assert_1.default.deepStrictEqual((0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
202
|
+
version: 1,
|
|
203
|
+
chainId: Buffer.from('045b', 'hex'),
|
|
204
|
+
delegator: Buffer.from('de60b7d0e6b758ca5dd8c61d377a2c5f1af51ec1', 'hex'),
|
|
205
|
+
validator: Buffer.from('a9e209f5ea0036c8c2f41078a3cebee57d8a47d5', 'hex'),
|
|
206
|
+
fee: 1,
|
|
207
|
+
redeemScript: Buffer.from('041f5e0e66b17576a914c4b8ae927ff2b9ce218e20bf06d425d6b68424fd88ac', 'hex'),
|
|
208
|
+
}).toString('hex'), defaultScript);
|
|
209
|
+
});
|
|
210
|
+
});
|
|
211
|
+
describe('parseCoreDaoOpReturnOutputScript', function () {
|
|
212
|
+
it('should parse a valid script with a timelock', function () {
|
|
213
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
214
|
+
version: validVersion,
|
|
215
|
+
chainId: validChainId,
|
|
216
|
+
delegator: validDelegator,
|
|
217
|
+
validator: validValidator,
|
|
218
|
+
fee: validFee,
|
|
219
|
+
timelock: validTimelock,
|
|
220
|
+
});
|
|
221
|
+
const parsed = (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(script);
|
|
222
|
+
assert_1.default.strictEqual(parsed.version, validVersion);
|
|
223
|
+
assert_1.default.deepStrictEqual(parsed.chainId, validChainId);
|
|
224
|
+
assert_1.default.deepStrictEqual(parsed.delegator, validDelegator);
|
|
225
|
+
assert_1.default.deepStrictEqual(parsed.validator, validValidator);
|
|
226
|
+
assert_1.default.strictEqual(parsed.fee, validFee);
|
|
227
|
+
(0, assert_1.default)('timelock' in parsed);
|
|
228
|
+
assert_1.default.deepStrictEqual(parsed.timelock, validTimelock);
|
|
229
|
+
});
|
|
230
|
+
it('should parse a valid script with a redeem script', function () {
|
|
231
|
+
const script = (0, coreDao_1.createCoreDaoOpReturnOutputScript)({
|
|
232
|
+
version: validVersion,
|
|
233
|
+
chainId: validChainId,
|
|
234
|
+
delegator: validDelegator,
|
|
235
|
+
validator: validValidator,
|
|
236
|
+
fee: validFee,
|
|
237
|
+
redeemScript: validRedeemScript,
|
|
238
|
+
});
|
|
239
|
+
const parsed = (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(script);
|
|
240
|
+
assert_1.default.strictEqual(parsed.version, validVersion);
|
|
241
|
+
assert_1.default.deepStrictEqual(parsed.chainId, validChainId);
|
|
242
|
+
assert_1.default.deepStrictEqual(parsed.delegator, validDelegator);
|
|
243
|
+
assert_1.default.deepStrictEqual(parsed.validator, validValidator);
|
|
244
|
+
assert_1.default.strictEqual(parsed.fee, validFee);
|
|
245
|
+
(0, assert_1.default)('redeemScript' in parsed);
|
|
246
|
+
assert_1.default.deepStrictEqual(parsed.redeemScript, validRedeemScript);
|
|
247
|
+
});
|
|
248
|
+
it('should parse valid opreturn script from testnet', async function () {
|
|
249
|
+
// Source: https://mempool.space/testnet/tx/66ed4cea26a410248a6d87f14b2bca514f33920c54d4af63ed46a903793115d5
|
|
250
|
+
const baseFixturePath = 'test/fixtures/coreDao/opReturn/66ed4cea26a410248a6d87f14b2bca514f33920c54d4af63ed46a903793115d5';
|
|
251
|
+
const opReturnHex = await (0, testutil_1.getFixture)(baseFixturePath + '.txt', undefined);
|
|
252
|
+
(0, assert_1.default)(typeof opReturnHex === 'string');
|
|
253
|
+
const parsed = (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(Buffer.from(opReturnHex, 'hex'));
|
|
254
|
+
const parsedFixture = await (0, testutil_1.getFixture)(baseFixturePath + '.json', JSON.parse((0, coreDao_1.toString)(parsed)));
|
|
255
|
+
assert_1.default.deepStrictEqual((0, coreDao_1.toString)(parsed), JSON.stringify(parsedFixture));
|
|
256
|
+
});
|
|
257
|
+
it('should fail if there is an invalid op-return', function () {
|
|
258
|
+
const script = defaultScript.replace('6a4c50', '6b4c50');
|
|
259
|
+
assert_1.default.throws(() => (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(Buffer.from(script, 'hex')));
|
|
260
|
+
});
|
|
261
|
+
it('should fail if the length is incorrect', function () {
|
|
262
|
+
const script = defaultScript.replace('4c50', '4c51');
|
|
263
|
+
assert_1.default.throws(() => (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(Buffer.from(script, 'hex')));
|
|
264
|
+
});
|
|
265
|
+
it('should fail if the satoshi+ identifier is incorrect', function () {
|
|
266
|
+
const script = defaultScript.replace('5341542b', '5341532b');
|
|
267
|
+
assert_1.default.throws(() => (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(Buffer.from(script, 'hex')));
|
|
268
|
+
});
|
|
269
|
+
it('should fail if the chainId is incorrect', function () {
|
|
270
|
+
const script = defaultScript.replace('045b', '0454');
|
|
271
|
+
assert_1.default.throws(() => (0, coreDao_1.parseCoreDaoOpReturnOutputScript)(Buffer.from(script, 'hex')));
|
|
272
|
+
});
|
|
273
|
+
});
|
|
274
|
+
});
|
|
275
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"opReturn.js","sourceRoot":"","sources":["../../../../../test/unit/coreDao/opReturn.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport { testutil } from '@bitgo-beta/utxo-lib';\nimport { getFixture } from '@bitgo-beta/utxo-core/testutil';\n\nimport {\n  CORE_DAO_MAINNET_CHAIN_ID,\n  CORE_DAO_SATOSHI_PLUS_IDENTIFIER,\n  createCoreDaoOpReturnOutputScript,\n  decodeTimelock,\n  encodeTimelock,\n  parseCoreDaoOpReturnOutputScript,\n  toString,\n} from '../../../src/coreDao';\n\ndescribe('OP_RETURN', function () {\n  const validVersion = 2;\n  const validChainId = CORE_DAO_MAINNET_CHAIN_ID;\n  // random 20 byte buffers\n  const validDelegator = Buffer.alloc(20, testutil.getKey('wasm-possum').publicKey);\n  const validValidator = Buffer.alloc(20, testutil.getKey('possum-wasm').publicKey);\n  const validFee = 1;\n  const validRedeemScript = Buffer.from('522103a8295453660d5e212d55556666666666666666666666666666666666', 'hex');\n  const validTimelock = 800800;\n  let defaultScript: string;\n\n  before(async function () {\n    // https://docs.coredao.org/docs/Learn/products/btc-staking/design#op_return-output-1\n    const script = await getFixture('test/fixtures/coreDao/opReturn/documentation.txt', undefined);\n    assert(typeof script === 'string');\n    defaultScript = script;\n  });\n\n  describe('createCoreDaoOpReturnOutputScript', function () {\n    it('should throw if invalid parameters are passed', function () {\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: 292,\n          chainId: validChainId,\n          delegator: validDelegator,\n          validator: validValidator,\n          fee: validFee,\n          timelock: validTimelock,\n        })\n      );\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: Buffer.alloc(32, 0),\n          delegator: validDelegator,\n          validator: validValidator,\n          fee: validFee,\n          timelock: validTimelock,\n        })\n      );\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: validChainId,\n          delegator: Buffer.alloc(19, 0),\n          validator: validValidator,\n          fee: validFee,\n          timelock: validTimelock,\n        })\n      );\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: validChainId,\n          delegator: validDelegator,\n          validator: Buffer.alloc(19, 0),\n          fee: validFee,\n          timelock: validTimelock,\n        })\n      );\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: validChainId,\n          delegator: validDelegator,\n          validator: validValidator,\n          fee: 256,\n          timelock: validTimelock,\n        })\n      );\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: validChainId,\n          delegator: validDelegator,\n          validator: validValidator,\n          fee: validFee,\n          timelock: -1,\n        })\n      );\n    });\n\n    it('should throw if the length of the script is too long', function () {\n      assert.throws(() =>\n        createCoreDaoOpReturnOutputScript({\n          version: validVersion,\n          chainId: validChainId,\n          delegator: validDelegator,\n          validator: validValidator,\n          fee: validFee,\n          redeemScript: Buffer.alloc(100),\n        })\n      );\n    });\n\n    it('should return a buffer with the correct length', function () {\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        timelock: validTimelock,\n      });\n      // Make sure that the first byte is the OP_RETURN opcode\n      assert.strictEqual(script[0], 0x6a);\n      // Make sure that the length of the script matches what is in the buffer\n      assert.strictEqual(\n        // We do not count the OP_RETURN opcode or the bytes for the length\n        script.length - 2,\n        script[1]\n      );\n    });\n\n    it('should have the correct placement of the values provided with a redeem script', function () {\n      // This should produce an Op_RETURN that needs the extra push bytes for the length\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        redeemScript: validRedeemScript,\n      });\n      // Make sure that the first byte is the OP_RETURN opcode\n      assert.strictEqual(script[0], 0x6a);\n      // Make sure that the length of the script matches what is in the buffer\n      assert.strictEqual(script[1], 0x4c);\n      assert.strictEqual(\n        // We do not count the OP_RETURN opcode or the length + pushbytes\n        script.length - 3,\n        script[2]\n      );\n      // Satoshi plus identifier\n      assert.deepStrictEqual(script.subarray(3, 7).toString('hex'), CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));\n      // Make sure that the version is correct\n      assert.strictEqual(script[7], validVersion);\n      // Make sure that the chainId is correct\n      assert.deepStrictEqual(script.subarray(8, 10).toString('hex'), validChainId.toString('hex'));\n      // Make sure that the delegator is correct\n      assert.deepStrictEqual(script.subarray(10, 30).toString('hex'), validDelegator.toString('hex'));\n      // Make sure that the validator is correct\n      assert.deepStrictEqual(script.subarray(30, 50).toString('hex'), validValidator.toString('hex'));\n      // Make sure that the fee is correct\n      assert.strictEqual(script[50], validFee);\n      // Make sure that the redeemScript is correct\n      assert.deepStrictEqual(\n        script.subarray(51, 51 + validRedeemScript.length).toString('hex'),\n        validRedeemScript.toString('hex')\n      );\n    });\n\n    it('should have the correct placement of the values provided with a blockheight timelock', function () {\n      // This should produce an Op_RETURN that needs the extra push bytes for the length\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        timelock: validTimelock,\n      });\n      // Make sure that the first byte is the OP_RETURN opcode\n      assert.strictEqual(script[0], 0x6a);\n      // Make sure that the length of the script matches what is in the buffer\n      assert.strictEqual(\n        // We do not count the OP_RETURN opcode or the length\n        script.length - 2,\n        script[1]\n      );\n      // Satoshi plus identifier\n      assert.deepStrictEqual(script.subarray(2, 6).toString('hex'), CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));\n      // Make sure that the version is correct\n      assert.strictEqual(script[6], validVersion);\n      // Make sure that the chainId is correct\n      assert.deepStrictEqual(script.subarray(7, 9).toString('hex'), validChainId.toString('hex'));\n      // Make sure that the delegator is correct\n      assert.deepStrictEqual(script.subarray(9, 29).toString('hex'), validDelegator.toString('hex'));\n      // Make sure that the validator is correct\n      assert.deepStrictEqual(script.subarray(29, 49).toString('hex'), validValidator.toString('hex'));\n      // Make sure that the fee is correct\n      assert.strictEqual(script[49], validFee);\n      // Make sure that the redeemScript is correct\n      assert.deepStrictEqual(script.subarray(50, 54).toString('hex'), encodeTimelock(validTimelock).toString('hex'));\n      assert.deepStrictEqual(decodeTimelock(script.subarray(50, 54)), validTimelock);\n    });\n\n    it('should have the correct placement of the values provided with a time timelock', function () {\n      // This should produce an Op_RETURN that needs the extra push bytes for the length\n      const timelock = new Date('2022-01-01T00:00:00Z').getTime() / 1000;\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        timelock,\n      });\n      // Make sure that the first byte is the OP_RETURN opcode\n      assert.strictEqual(script[0], 0x6a);\n      // Make sure that the length of the script matches what is in the buffer\n      assert.strictEqual(\n        // We do not count the OP_RETURN opcode or the length\n        script.length - 2,\n        script[1]\n      );\n      // Satoshi plus identifier\n      assert.deepStrictEqual(script.subarray(2, 6).toString('hex'), CORE_DAO_SATOSHI_PLUS_IDENTIFIER.toString('hex'));\n      // Make sure that the version is correct\n      assert.strictEqual(script[6], validVersion);\n      // Make sure that the chainId is correct\n      assert.deepStrictEqual(script.subarray(7, 9).toString('hex'), validChainId.toString('hex'));\n      // Make sure that the delegator is correct\n      assert.deepStrictEqual(script.subarray(9, 29).toString('hex'), validDelegator.toString('hex'));\n      // Make sure that the validator is correct\n      assert.deepStrictEqual(script.subarray(29, 49).toString('hex'), validValidator.toString('hex'));\n      // Make sure that the fee is correct\n      assert.strictEqual(script[49], validFee);\n      // Make sure that the redeemScript is correct\n      assert.deepStrictEqual(script.subarray(50, 54).toString('hex'), encodeTimelock(timelock).toString('hex'));\n      assert.deepStrictEqual(decodeTimelock(script.subarray(50, 54)), timelock);\n    });\n\n    it('should recreate the example OP_RETURN correctly', function () {\n      assert.deepStrictEqual(\n        createCoreDaoOpReturnOutputScript({\n          version: 1,\n          chainId: Buffer.from('045b', 'hex'),\n          delegator: Buffer.from('de60b7d0e6b758ca5dd8c61d377a2c5f1af51ec1', 'hex'),\n          validator: Buffer.from('a9e209f5ea0036c8c2f41078a3cebee57d8a47d5', 'hex'),\n          fee: 1,\n          redeemScript: Buffer.from('041f5e0e66b17576a914c4b8ae927ff2b9ce218e20bf06d425d6b68424fd88ac', 'hex'),\n        }).toString('hex'),\n        defaultScript\n      );\n    });\n  });\n\n  describe('parseCoreDaoOpReturnOutputScript', function () {\n    it('should parse a valid script with a timelock', function () {\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        timelock: validTimelock,\n      });\n      const parsed = parseCoreDaoOpReturnOutputScript(script);\n      assert.strictEqual(parsed.version, validVersion);\n      assert.deepStrictEqual(parsed.chainId, validChainId);\n      assert.deepStrictEqual(parsed.delegator, validDelegator);\n      assert.deepStrictEqual(parsed.validator, validValidator);\n      assert.strictEqual(parsed.fee, validFee);\n      assert('timelock' in parsed);\n      assert.deepStrictEqual(parsed.timelock, validTimelock);\n    });\n\n    it('should parse a valid script with a redeem script', function () {\n      const script = createCoreDaoOpReturnOutputScript({\n        version: validVersion,\n        chainId: validChainId,\n        delegator: validDelegator,\n        validator: validValidator,\n        fee: validFee,\n        redeemScript: validRedeemScript,\n      });\n      const parsed = parseCoreDaoOpReturnOutputScript(script);\n      assert.strictEqual(parsed.version, validVersion);\n      assert.deepStrictEqual(parsed.chainId, validChainId);\n      assert.deepStrictEqual(parsed.delegator, validDelegator);\n      assert.deepStrictEqual(parsed.validator, validValidator);\n      assert.strictEqual(parsed.fee, validFee);\n      assert('redeemScript' in parsed);\n      assert.deepStrictEqual(parsed.redeemScript, validRedeemScript);\n    });\n\n    it('should parse valid opreturn script from testnet', async function () {\n      // Source: https://mempool.space/testnet/tx/66ed4cea26a410248a6d87f14b2bca514f33920c54d4af63ed46a903793115d5\n      const baseFixturePath =\n        'test/fixtures/coreDao/opReturn/66ed4cea26a410248a6d87f14b2bca514f33920c54d4af63ed46a903793115d5';\n      const opReturnHex = await getFixture(baseFixturePath + '.txt', undefined);\n      assert(typeof opReturnHex === 'string');\n      const parsed = parseCoreDaoOpReturnOutputScript(Buffer.from(opReturnHex, 'hex'));\n      const parsedFixture = await getFixture(baseFixturePath + '.json', JSON.parse(toString(parsed)));\n      assert.deepStrictEqual(toString(parsed), JSON.stringify(parsedFixture));\n    });\n\n    it('should fail if there is an invalid op-return', function () {\n      const script = defaultScript.replace('6a4c50', '6b4c50');\n      assert.throws(() => parseCoreDaoOpReturnOutputScript(Buffer.from(script, 'hex')));\n    });\n\n    it('should fail if the length is incorrect', function () {\n      const script = defaultScript.replace('4c50', '4c51');\n      assert.throws(() => parseCoreDaoOpReturnOutputScript(Buffer.from(script, 'hex')));\n    });\n\n    it('should fail if the satoshi+ identifier is incorrect', function () {\n      const script = defaultScript.replace('5341542b', '5341532b');\n      assert.throws(() => parseCoreDaoOpReturnOutputScript(Buffer.from(script, 'hex')));\n    });\n\n    it('should fail if the chainId is incorrect', function () {\n      const script = defaultScript.replace('045b', '0454');\n      assert.throws(() => parseCoreDaoOpReturnOutputScript(Buffer.from(script, 'hex')));\n    });\n  });\n});\n"]}
|