@bitgo-beta/utxo-core 0.0.0-semantic-release-managed
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/LICENSE +191 -0
- package/README.md +6 -0
- package/dist/src/Output.d.ts +43 -0
- package/dist/src/Output.d.ts.map +1 -0
- package/dist/src/Output.js +60 -0
- package/dist/src/bip322/index.d.ts +5 -0
- package/dist/src/bip322/index.d.ts.map +1 -0
- package/dist/src/bip322/index.js +21 -0
- package/dist/src/bip322/toSign.d.ts +25 -0
- package/dist/src/bip322/toSign.d.ts.map +1 -0
- package/dist/src/bip322/toSign.js +126 -0
- package/dist/src/bip322/toSpend.d.ts +22 -0
- package/dist/src/bip322/toSpend.d.ts.map +1 -0
- package/dist/src/bip322/toSpend.js +72 -0
- package/dist/src/bip322/utils.d.ts +18 -0
- package/dist/src/bip322/utils.d.ts.map +1 -0
- package/dist/src/bip322/utils.js +115 -0
- package/dist/src/bip322/verify.d.ts +12 -0
- package/dist/src/bip322/verify.d.ts.map +1 -0
- package/dist/src/bip322/verify.js +115 -0
- package/dist/src/bip32utils.d.ts +16 -0
- package/dist/src/bip32utils.d.ts.map +1 -0
- package/dist/src/bip32utils.js +71 -0
- package/dist/src/bip65/index.d.ts +2 -0
- package/dist/src/bip65/index.d.ts.map +1 -0
- package/dist/src/bip65/index.js +18 -0
- package/dist/src/bip65/locktime.d.ts +8 -0
- package/dist/src/bip65/locktime.d.ts.map +1 -0
- package/dist/src/bip65/locktime.js +37 -0
- package/dist/src/descriptor/DescriptorMap.d.ts +9 -0
- package/dist/src/descriptor/DescriptorMap.d.ts.map +1 -0
- package/dist/src/descriptor/DescriptorMap.js +9 -0
- package/dist/src/descriptor/Output.d.ts +23 -0
- package/dist/src/descriptor/Output.d.ts.map +1 -0
- package/dist/src/descriptor/Output.js +41 -0
- package/dist/src/descriptor/VirtualSize.d.ts +23 -0
- package/dist/src/descriptor/VirtualSize.d.ts.map +1 -0
- package/dist/src/descriptor/VirtualSize.js +100 -0
- package/dist/src/descriptor/address.d.ts +5 -0
- package/dist/src/descriptor/address.d.ts.map +1 -0
- package/dist/src/descriptor/address.js +48 -0
- package/dist/src/descriptor/derive.d.ts +13 -0
- package/dist/src/descriptor/derive.d.ts.map +1 -0
- package/dist/src/descriptor/derive.js +45 -0
- package/dist/src/descriptor/fromFixedScriptWallet.d.ts +16 -0
- package/dist/src/descriptor/fromFixedScriptWallet.d.ts.map +1 -0
- package/dist/src/descriptor/fromFixedScriptWallet.js +88 -0
- package/dist/src/descriptor/index.d.ts +11 -0
- package/dist/src/descriptor/index.d.ts.map +1 -0
- package/dist/src/descriptor/index.js +27 -0
- package/dist/src/descriptor/parse/PatternMatcher.d.ts +14 -0
- package/dist/src/descriptor/parse/PatternMatcher.d.ts.map +1 -0
- package/dist/src/descriptor/parse/PatternMatcher.js +60 -0
- package/dist/src/descriptor/psbt/assertSatisfiable.d.ts +20 -0
- package/dist/src/descriptor/psbt/assertSatisfiable.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/assertSatisfiable.js +74 -0
- package/dist/src/descriptor/psbt/createPsbt.d.ts +23 -0
- package/dist/src/descriptor/psbt/createPsbt.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/createPsbt.js +107 -0
- package/dist/src/descriptor/psbt/findDescriptors.d.ts +26 -0
- package/dist/src/descriptor/psbt/findDescriptors.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/findDescriptors.js +98 -0
- package/dist/src/descriptor/psbt/index.d.ts +7 -0
- package/dist/src/descriptor/psbt/index.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/index.js +23 -0
- package/dist/src/descriptor/psbt/parse.d.ts +27 -0
- package/dist/src/descriptor/psbt/parse.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/parse.js +87 -0
- package/dist/src/descriptor/psbt/sign.d.ts +26 -0
- package/dist/src/descriptor/psbt/sign.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/sign.js +42 -0
- package/dist/src/descriptor/psbt/wrap.d.ts +12 -0
- package/dist/src/descriptor/psbt/wrap.d.ts.map +1 -0
- package/dist/src/descriptor/psbt/wrap.js +76 -0
- package/dist/src/dustThreshold.d.ts +4 -0
- package/dist/src/dustThreshold.d.ts.map +1 -0
- package/dist/src/dustThreshold.js +134 -0
- package/dist/src/index.d.ts +10 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +49 -0
- package/dist/src/paygo/attestation.d.ts +11 -0
- package/dist/src/paygo/attestation.d.ts.map +1 -0
- package/dist/src/paygo/attestation.js +58 -0
- package/dist/src/paygo/index.d.ts +3 -0
- package/dist/src/paygo/index.d.ts.map +1 -0
- package/dist/src/paygo/index.js +19 -0
- package/dist/src/paygo/parsePayGoAttestation.d.ts +16 -0
- package/dist/src/paygo/parsePayGoAttestation.d.ts.map +1 -0
- package/dist/src/paygo/parsePayGoAttestation.js +50 -0
- package/dist/src/paygo/psbt/Errors.d.ts +22 -0
- package/dist/src/paygo/psbt/Errors.d.ts.map +1 -0
- package/dist/src/paygo/psbt/Errors.js +44 -0
- package/dist/src/paygo/psbt/index.d.ts +2 -0
- package/dist/src/paygo/psbt/index.d.ts.map +1 -0
- package/dist/src/paygo/psbt/index.js +18 -0
- package/dist/src/paygo/psbt/payGoAddressProof.d.ts +29 -0
- package/dist/src/paygo/psbt/payGoAddressProof.d.ts.map +1 -0
- package/dist/src/paygo/psbt/payGoAddressProof.js +124 -0
- package/dist/src/testutil/descriptor/descriptors.d.ts +13 -0
- package/dist/src/testutil/descriptor/descriptors.d.ts.map +1 -0
- package/dist/src/testutil/descriptor/descriptors.js +162 -0
- package/dist/src/testutil/descriptor/index.d.ts +4 -0
- package/dist/src/testutil/descriptor/index.d.ts.map +1 -0
- package/dist/src/testutil/descriptor/index.js +20 -0
- package/dist/src/testutil/descriptor/mock.utils.d.ts +36 -0
- package/dist/src/testutil/descriptor/mock.utils.d.ts.map +1 -0
- package/dist/src/testutil/descriptor/mock.utils.js +92 -0
- package/dist/src/testutil/descriptor/psbt.utils.d.ts +4 -0
- package/dist/src/testutil/descriptor/psbt.utils.d.ts.map +1 -0
- package/dist/src/testutil/descriptor/psbt.utils.js +21 -0
- package/dist/src/testutil/fixtures.utils.d.ts +15 -0
- package/dist/src/testutil/fixtures.utils.d.ts.map +1 -0
- package/dist/src/testutil/fixtures.utils.js +127 -0
- package/dist/src/testutil/generatePayGoAttestationProof.utils.d.ts +12 -0
- package/dist/src/testutil/generatePayGoAttestationProof.utils.d.ts.map +1 -0
- package/dist/src/testutil/generatePayGoAttestationProof.utils.js +38 -0
- package/dist/src/testutil/index.d.ts +6 -0
- package/dist/src/testutil/index.d.ts.map +1 -0
- package/dist/src/testutil/index.js +22 -0
- package/dist/src/testutil/key.utils.d.ts +16 -0
- package/dist/src/testutil/key.utils.d.ts.map +1 -0
- package/dist/src/testutil/key.utils.js +59 -0
- package/dist/src/testutil/toPlainObject.utils.d.ts +11 -0
- package/dist/src/testutil/toPlainObject.utils.d.ts.map +1 -0
- package/dist/src/testutil/toPlainObject.utils.js +89 -0
- package/dist/src/testutil/trimMessagePrefix.d.ts +11 -0
- package/dist/src/testutil/trimMessagePrefix.d.ts.map +1 -0
- package/dist/src/testutil/trimMessagePrefix.js +60 -0
- package/dist/src/xOnlyPubkey.d.ts +2 -0
- package/dist/src/xOnlyPubkey.d.ts.map +1 -0
- package/dist/src/xOnlyPubkey.js +18 -0
- package/package.json +63 -0
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ErrorMultiplePayGoProofAtPsbtIndex = exports.ErrorOutputIndexOutOfBounds = exports.ErrorPayGoAddressProofFailedVerification = exports.ErrorMultiplePayGoProof = exports.ErrorNoPayGoProof = exports.PayGoError = void 0;
|
|
4
|
+
class PayGoError extends Error {
|
|
5
|
+
constructor(message) {
|
|
6
|
+
super(message);
|
|
7
|
+
this.name = this.constructor.name;
|
|
8
|
+
}
|
|
9
|
+
}
|
|
10
|
+
exports.PayGoError = PayGoError;
|
|
11
|
+
class ErrorNoPayGoProof extends PayGoError {
|
|
12
|
+
constructor(outputIndex) {
|
|
13
|
+
super(`There is no paygo address proof encoded in the PSBT at output ${outputIndex}.`);
|
|
14
|
+
this.outputIndex = outputIndex;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
exports.ErrorNoPayGoProof = ErrorNoPayGoProof;
|
|
18
|
+
class ErrorMultiplePayGoProof extends PayGoError {
|
|
19
|
+
constructor() {
|
|
20
|
+
super('There are multiple paygo address proofs encoded in the PSBT. Something went wrong.');
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
exports.ErrorMultiplePayGoProof = ErrorMultiplePayGoProof;
|
|
24
|
+
class ErrorPayGoAddressProofFailedVerification extends PayGoError {
|
|
25
|
+
constructor() {
|
|
26
|
+
super('Cannot verify the paygo address signature with the provided pubkey.');
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
exports.ErrorPayGoAddressProofFailedVerification = ErrorPayGoAddressProofFailedVerification;
|
|
30
|
+
class ErrorOutputIndexOutOfBounds extends PayGoError {
|
|
31
|
+
constructor(outputIndex) {
|
|
32
|
+
super(`Output index ${outputIndex} is out of bounds for PSBT outputs.`);
|
|
33
|
+
this.outputIndex = outputIndex;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
exports.ErrorOutputIndexOutOfBounds = ErrorOutputIndexOutOfBounds;
|
|
37
|
+
class ErrorMultiplePayGoProofAtPsbtIndex extends PayGoError {
|
|
38
|
+
constructor(outputIndex) {
|
|
39
|
+
super(`There are multiple PayGo addresses in the PSBT output ${outputIndex}.`);
|
|
40
|
+
this.outputIndex = outputIndex;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
exports.ErrorMultiplePayGoProofAtPsbtIndex = ErrorMultiplePayGoProofAtPsbtIndex;
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3BheWdvL3BzYnQvRXJyb3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLE1BQWEsVUFBVyxTQUFRLEtBQUs7SUFDbkMsWUFBWSxPQUFlO1FBQ3pCLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNmLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUM7SUFDcEMsQ0FBQztDQUNGO0FBTEQsZ0NBS0M7QUFFRCxNQUFhLGlCQUFrQixTQUFRLFVBQVU7SUFDL0MsWUFBbUIsV0FBbUI7UUFDcEMsS0FBSyxDQUFDLGlFQUFpRSxXQUFXLEdBQUcsQ0FBQyxDQUFDO1FBRHRFLGdCQUFXLEdBQVgsV0FBVyxDQUFRO0lBRXRDLENBQUM7Q0FDRjtBQUpELDhDQUlDO0FBRUQsTUFBYSx1QkFBd0IsU0FBUSxVQUFVO0lBQ3JEO1FBQ0UsS0FBSyxDQUFDLG9GQUFvRixDQUFDLENBQUM7SUFDOUYsQ0FBQztDQUNGO0FBSkQsMERBSUM7QUFFRCxNQUFhLHdDQUF5QyxTQUFRLFVBQVU7SUFDdEU7UUFDRSxLQUFLLENBQUMscUVBQXFFLENBQUMsQ0FBQztJQUMvRSxDQUFDO0NBQ0Y7QUFKRCw0RkFJQztBQUVELE1BQWEsMkJBQTRCLFNBQVEsVUFBVTtJQUN6RCxZQUFtQixXQUFtQjtRQUNwQyxLQUFLLENBQUMsZ0JBQWdCLFdBQVcscUNBQXFDLENBQUMsQ0FBQztRQUR2RCxnQkFBVyxHQUFYLFdBQVcsQ0FBUTtJQUV0QyxDQUFDO0NBQ0Y7QUFKRCxrRUFJQztBQUVELE1BQWEsa0NBQW1DLFNBQVEsVUFBVTtJQUNoRSxZQUFtQixXQUFtQjtRQUNwQyxLQUFLLENBQUMseURBQXlELFdBQVcsR0FBRyxDQUFDLENBQUM7UUFEOUQsZ0JBQVcsR0FBWCxXQUFXLENBQVE7SUFFdEMsQ0FBQztDQUNGO0FBSkQsZ0ZBSUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgUGF5R29FcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IobWVzc2FnZTogc3RyaW5nKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gdGhpcy5jb25zdHJ1Y3Rvci5uYW1lO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBFcnJvck5vUGF5R29Qcm9vZiBleHRlbmRzIFBheUdvRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgb3V0cHV0SW5kZXg6IG51bWJlcikge1xuICAgIHN1cGVyKGBUaGVyZSBpcyBubyBwYXlnbyBhZGRyZXNzIHByb29mIGVuY29kZWQgaW4gdGhlIFBTQlQgYXQgb3V0cHV0ICR7b3V0cHV0SW5kZXh9LmApO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBFcnJvck11bHRpcGxlUGF5R29Qcm9vZiBleHRlbmRzIFBheUdvRXJyb3Ige1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICBzdXBlcignVGhlcmUgYXJlIG11bHRpcGxlIHBheWdvIGFkZHJlc3MgcHJvb2ZzIGVuY29kZWQgaW4gdGhlIFBTQlQuIFNvbWV0aGluZyB3ZW50IHdyb25nLicpO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBFcnJvclBheUdvQWRkcmVzc1Byb29mRmFpbGVkVmVyaWZpY2F0aW9uIGV4dGVuZHMgUGF5R29FcnJvciB7XG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKCdDYW5ub3QgdmVyaWZ5IHRoZSBwYXlnbyBhZGRyZXNzIHNpZ25hdHVyZSB3aXRoIHRoZSBwcm92aWRlZCBwdWJrZXkuJyk7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIEVycm9yT3V0cHV0SW5kZXhPdXRPZkJvdW5kcyBleHRlbmRzIFBheUdvRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgb3V0cHV0SW5kZXg6IG51bWJlcikge1xuICAgIHN1cGVyKGBPdXRwdXQgaW5kZXggJHtvdXRwdXRJbmRleH0gaXMgb3V0IG9mIGJvdW5kcyBmb3IgUFNCVCBvdXRwdXRzLmApO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBFcnJvck11bHRpcGxlUGF5R29Qcm9vZkF0UHNidEluZGV4IGV4dGVuZHMgUGF5R29FcnJvciB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyBvdXRwdXRJbmRleDogbnVtYmVyKSB7XG4gICAgc3VwZXIoYFRoZXJlIGFyZSBtdWx0aXBsZSBQYXlHbyBhZGRyZXNzZXMgaW4gdGhlIFBTQlQgb3V0cHV0ICR7b3V0cHV0SW5kZXh9LmApO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/paygo/psbt/index.ts"],"names":[],"mappings":"AAAA,cAAc,qBAAqB,CAAC"}
|
|
@@ -0,0 +1,18 @@
|
|
|
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 __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./payGoAddressProof"), exports);
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvcGF5Z28vcHNidC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsc0RBQW9DIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9wYXlHb0FkZHJlc3NQcm9vZic7XG4iXX0=
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
+
/** This function adds the entropy and signature into the PSBT output unknown key vals.
|
|
3
|
+
* We store the entropy so that we reconstruct the message <ENTROPY><ADDRESS><UUID>
|
|
4
|
+
* to later verify.
|
|
5
|
+
*
|
|
6
|
+
* @param psbt - PSBT that we need to encode our paygo address into
|
|
7
|
+
* @param outputIndex - the index of the address in our output
|
|
8
|
+
* @param sig - the signature that we want to encode
|
|
9
|
+
* @param entropy - the arbitrary entropy bytes from our vasp proof
|
|
10
|
+
*/
|
|
11
|
+
export declare function addPayGoAddressProof(psbt: utxolib.bitgo.UtxoPsbt, outputIndex: number, sig: Buffer, entropy: Buffer): void;
|
|
12
|
+
/** Verify the paygo address signature is valid using verification pub key.
|
|
13
|
+
*
|
|
14
|
+
* @param psbt - PSBT we want to verify that the paygo address is in
|
|
15
|
+
* @param outputIndex - we have the output index that address is in
|
|
16
|
+
* @param verificationPubkey - the pubkey signed by the HSM to verify our message
|
|
17
|
+
* @returns
|
|
18
|
+
*/
|
|
19
|
+
export declare function verifyPayGoAddressProof(psbt: utxolib.bitgo.UtxoPsbt, outputIndex: number, verificationPubkey: Buffer): void;
|
|
20
|
+
/** Get the output index of the paygo output if there is one. It does this by
|
|
21
|
+
* checking if the metadata is on one of the outputs of the PSBT. If there is
|
|
22
|
+
* no paygo output, return undefined
|
|
23
|
+
*
|
|
24
|
+
* @param psbt
|
|
25
|
+
* @returns number - the index of the output address
|
|
26
|
+
*/
|
|
27
|
+
export declare function getPayGoAddressProofOutputIndex(psbt: utxolib.bitgo.UtxoPsbt): number | undefined;
|
|
28
|
+
export declare function psbtOutputIncludesPaygoAddressProof(psbt: utxolib.bitgo.UtxoPsbt): boolean;
|
|
29
|
+
//# sourceMappingURL=payGoAddressProof.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"payGoAddressProof.d.ts","sourceRoot":"","sources":["../../../../src/paygo/psbt/payGoAddressProof.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAchD;;;;;;;;GAQG;AACH,wBAAgB,oBAAoB,CAClC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,WAAW,EAAE,MAAM,EACnB,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,GACd,IAAI,CASN;AAED;;;;;;GAMG;AACH,wBAAgB,uBAAuB,CACrC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,WAAW,EAAE,MAAM,EACnB,kBAAkB,EAAE,MAAM,GACzB,IAAI,CAiCN;AAED;;;;;;GAMG;AACH,wBAAgB,+BAA+B,CAAC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,MAAM,GAAG,SAAS,CAehG;AAED,wBAAgB,mCAAmC,CAAC,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAEzF"}
|
|
@@ -0,0 +1,124 @@
|
|
|
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.addPayGoAddressProof = addPayGoAddressProof;
|
|
37
|
+
exports.verifyPayGoAddressProof = verifyPayGoAddressProof;
|
|
38
|
+
exports.getPayGoAddressProofOutputIndex = getPayGoAddressProofOutputIndex;
|
|
39
|
+
exports.psbtOutputIncludesPaygoAddressProof = psbtOutputIncludesPaygoAddressProof;
|
|
40
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
41
|
+
const utils_1 = require("bip174/src/lib/utils");
|
|
42
|
+
const bip32utils_1 = require("../../bip32utils");
|
|
43
|
+
const attestation_1 = require("../attestation");
|
|
44
|
+
const Errors_1 = require("./Errors");
|
|
45
|
+
/** This function adds the entropy and signature into the PSBT output unknown key vals.
|
|
46
|
+
* We store the entropy so that we reconstruct the message <ENTROPY><ADDRESS><UUID>
|
|
47
|
+
* to later verify.
|
|
48
|
+
*
|
|
49
|
+
* @param psbt - PSBT that we need to encode our paygo address into
|
|
50
|
+
* @param outputIndex - the index of the address in our output
|
|
51
|
+
* @param sig - the signature that we want to encode
|
|
52
|
+
* @param entropy - the arbitrary entropy bytes from our vasp proof
|
|
53
|
+
*/
|
|
54
|
+
function addPayGoAddressProof(psbt, outputIndex, sig, entropy) {
|
|
55
|
+
utxolib.bitgo.addProprietaryKeyValuesFromUnknownKeyValues(psbt, 'output', outputIndex, {
|
|
56
|
+
key: {
|
|
57
|
+
identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
|
|
58
|
+
subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
|
|
59
|
+
keydata: entropy,
|
|
60
|
+
},
|
|
61
|
+
value: sig,
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
/** Verify the paygo address signature is valid using verification pub key.
|
|
65
|
+
*
|
|
66
|
+
* @param psbt - PSBT we want to verify that the paygo address is in
|
|
67
|
+
* @param outputIndex - we have the output index that address is in
|
|
68
|
+
* @param verificationPubkey - the pubkey signed by the HSM to verify our message
|
|
69
|
+
* @returns
|
|
70
|
+
*/
|
|
71
|
+
function verifyPayGoAddressProof(psbt, outputIndex, verificationPubkey) {
|
|
72
|
+
const psbtOutputs = (0, utils_1.checkForOutput)(psbt.data.outputs, outputIndex);
|
|
73
|
+
const stored = utxolib.bitgo.getProprietaryKeyValuesFromUnknownKeyValues(psbtOutputs, {
|
|
74
|
+
identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
|
|
75
|
+
subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
|
|
76
|
+
});
|
|
77
|
+
// assert stored length is 0 or 1
|
|
78
|
+
if (stored.length === 0) {
|
|
79
|
+
throw new Errors_1.ErrorNoPayGoProof(outputIndex);
|
|
80
|
+
}
|
|
81
|
+
else if (stored.length > 1) {
|
|
82
|
+
throw new Errors_1.ErrorMultiplePayGoProof();
|
|
83
|
+
}
|
|
84
|
+
// We get the signature and entropy from our PSBT unknown key vals
|
|
85
|
+
const signature = stored[0].value;
|
|
86
|
+
const entropy = stored[0].key.keydata;
|
|
87
|
+
// Get the the PayGo address from the txOutputs
|
|
88
|
+
const txOutputs = psbt.txOutputs;
|
|
89
|
+
if (outputIndex >= txOutputs.length) {
|
|
90
|
+
throw new Errors_1.ErrorOutputIndexOutOfBounds(outputIndex);
|
|
91
|
+
}
|
|
92
|
+
const output = txOutputs[outputIndex];
|
|
93
|
+
const addressFromOutput = utxolib.address.fromOutputScript(output.script, psbt.network);
|
|
94
|
+
// We construct our message <ENTROPY><ADDRESS><UUID>
|
|
95
|
+
const message = (0, attestation_1.createPayGoAttestationBuffer)(addressFromOutput, entropy, psbt.network);
|
|
96
|
+
// bip32utils.verifyMessage now takes in message as a Buffer
|
|
97
|
+
if (!(0, bip32utils_1.verifyMessage)(message, verificationPubkey, signature, utxolib.networks.bitcoin)) {
|
|
98
|
+
throw new Errors_1.ErrorPayGoAddressProofFailedVerification();
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
/** Get the output index of the paygo output if there is one. It does this by
|
|
102
|
+
* checking if the metadata is on one of the outputs of the PSBT. If there is
|
|
103
|
+
* no paygo output, return undefined
|
|
104
|
+
*
|
|
105
|
+
* @param psbt
|
|
106
|
+
* @returns number - the index of the output address
|
|
107
|
+
*/
|
|
108
|
+
function getPayGoAddressProofOutputIndex(psbt) {
|
|
109
|
+
const res = psbt.data.outputs.flatMap((output, outputIndex) => {
|
|
110
|
+
const proprietaryKeyVals = utxolib.bitgo.getPsbtOutputProprietaryKeyVals(output, {
|
|
111
|
+
identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
|
|
112
|
+
subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
|
|
113
|
+
});
|
|
114
|
+
if (proprietaryKeyVals.length > 1) {
|
|
115
|
+
throw new Errors_1.ErrorMultiplePayGoProofAtPsbtIndex(outputIndex);
|
|
116
|
+
}
|
|
117
|
+
return proprietaryKeyVals.length === 0 ? [] : [outputIndex];
|
|
118
|
+
});
|
|
119
|
+
return res.length === 0 ? undefined : res[0];
|
|
120
|
+
}
|
|
121
|
+
function psbtOutputIncludesPaygoAddressProof(psbt) {
|
|
122
|
+
return getPayGoAddressProofOutputIndex(psbt) !== undefined;
|
|
123
|
+
}
|
|
124
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"payGoAddressProof.js","sourceRoot":"","sources":["../../../../src/paygo/psbt/payGoAddressProof.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { checkForOutput } from 'bip174/src/lib/utils';\n\nimport { verifyMessage } from '../../bip32utils';\nimport { createPayGoAttestationBuffer } from '../attestation';\n\nimport {\n  ErrorMultiplePayGoProof,\n  ErrorMultiplePayGoProofAtPsbtIndex,\n  ErrorNoPayGoProof,\n  ErrorOutputIndexOutOfBounds,\n  ErrorPayGoAddressProofFailedVerification,\n} from './Errors';\n\n/** This function adds the entropy and signature into the PSBT output unknown key vals.\n * We store the entropy so that we reconstruct the message <ENTROPY><ADDRESS><UUID>\n * to later verify.\n *\n * @param psbt - PSBT that we need to encode our paygo address into\n * @param outputIndex - the index of the address in our output\n * @param sig - the signature that we want to encode\n * @param entropy - the arbitrary entropy bytes from our vasp proof\n */\nexport function addPayGoAddressProof(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  outputIndex: number,\n  sig: Buffer,\n  entropy: Buffer\n): void {\n  utxolib.bitgo.addProprietaryKeyValuesFromUnknownKeyValues(psbt, 'output', outputIndex, {\n    key: {\n      identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,\n      subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,\n      keydata: entropy,\n    },\n    value: sig,\n  });\n}\n\n/** Verify the paygo address signature is valid using verification pub key.\n *\n * @param psbt - PSBT we want to verify that the paygo address is in\n * @param outputIndex - we have the output index that address is in\n * @param verificationPubkey - the pubkey signed by the HSM to verify our message\n * @returns\n */\nexport function verifyPayGoAddressProof(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  outputIndex: number,\n  verificationPubkey: Buffer\n): void {\n  const psbtOutputs = checkForOutput(psbt.data.outputs, outputIndex);\n  const stored = utxolib.bitgo.getProprietaryKeyValuesFromUnknownKeyValues(psbtOutputs, {\n    identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,\n  });\n\n  // assert stored length is 0 or 1\n  if (stored.length === 0) {\n    throw new ErrorNoPayGoProof(outputIndex);\n  } else if (stored.length > 1) {\n    throw new ErrorMultiplePayGoProof();\n  }\n\n  // We get the signature and entropy from our PSBT unknown key vals\n  const signature = stored[0].value;\n  const entropy = stored[0].key.keydata;\n\n  // Get the the PayGo address from the txOutputs\n  const txOutputs = psbt.txOutputs;\n  if (outputIndex >= txOutputs.length) {\n    throw new ErrorOutputIndexOutOfBounds(outputIndex);\n  }\n  const output = txOutputs[outputIndex];\n  const addressFromOutput = utxolib.address.fromOutputScript(output.script, psbt.network);\n\n  // We construct our message <ENTROPY><ADDRESS><UUID>\n  const message = createPayGoAttestationBuffer(addressFromOutput, entropy, psbt.network);\n\n  // bip32utils.verifyMessage now takes in message as a Buffer\n  if (!verifyMessage(message, verificationPubkey, signature, utxolib.networks.bitcoin)) {\n    throw new ErrorPayGoAddressProofFailedVerification();\n  }\n}\n\n/** Get the output index of the paygo output if there is one. It does this by\n * checking if the metadata is on one of the outputs of the PSBT. If there is\n * no paygo output, return undefined\n *\n * @param psbt\n * @returns number - the index of the output address\n */\nexport function getPayGoAddressProofOutputIndex(psbt: utxolib.bitgo.UtxoPsbt): number | undefined {\n  const res = psbt.data.outputs.flatMap((output, outputIndex) => {\n    const proprietaryKeyVals = utxolib.bitgo.getPsbtOutputProprietaryKeyVals(output, {\n      identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,\n      subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,\n    });\n\n    if (proprietaryKeyVals.length > 1) {\n      throw new ErrorMultiplePayGoProofAtPsbtIndex(outputIndex);\n    }\n\n    return proprietaryKeyVals.length === 0 ? [] : [outputIndex];\n  });\n\n  return res.length === 0 ? undefined : res[0];\n}\n\nexport function psbtOutputIncludesPaygoAddressProof(psbt: utxolib.bitgo.UtxoPsbt): boolean {\n  return getPayGoAddressProofOutputIndex(psbt) !== undefined;\n}\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { Miniscript, Descriptor, ast } from '@bitgo/wasm-miniscript';
|
|
2
|
+
import { BIP32Interface } from '@bitgo-beta/utxo-lib';
|
|
3
|
+
import { DescriptorMap, PsbtParams } from '../../descriptor';
|
|
4
|
+
import { Triple, KeyTriple } from '../key.utils';
|
|
5
|
+
export declare function getDefaultXPubs(seed?: string): Triple<string>;
|
|
6
|
+
export declare function getUnspendableKey(): string;
|
|
7
|
+
export type DescriptorTemplate = 'Wsh2Of3' | 'Tr1Of3-NoKeyPath-Tree' | 'Tr1Of3-NoKeyPath-Tree-Plain' | 'Tr2Of3-NoKeyPath' | 'Wsh2Of2' | 'Wsh2Of3CltvDrop';
|
|
8
|
+
export declare function getPsbtParams(t: DescriptorTemplate): Partial<PsbtParams>;
|
|
9
|
+
export declare function containsKey(script: Miniscript | ast.MiniscriptNode, key: BIP32Interface | string): boolean;
|
|
10
|
+
export declare function getTapLeafScripts(d: Descriptor): string[];
|
|
11
|
+
export declare function getDescriptor(template: DescriptorTemplate, keys?: KeyTriple | string[], path?: string): Descriptor;
|
|
12
|
+
export declare function getDescriptorMap(template: DescriptorTemplate, keys?: KeyTriple | string[]): DescriptorMap;
|
|
13
|
+
//# sourceMappingURL=descriptors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"descriptors.d.ts","sourceRoot":"","sources":["../../../../src/testutil/descriptor/descriptors.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,wBAAwB,CAAC;AACrE,OAAO,EAAS,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAE7D,OAAO,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC7D,OAAO,EAAgB,MAAM,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAE/D,wBAAgB,eAAe,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAE7D;AAED,wBAAgB,iBAAiB,IAAI,MAAM,CAsB1C;AAMD,MAAM,MAAM,kBAAkB,GAC1B,SAAS,GACT,uBAAuB,GAEvB,6BAA6B,GAC7B,kBAAkB,GAClB,SAAS,GAMT,iBAAiB,CAAC;AAkCtB,wBAAgB,aAAa,CAAC,CAAC,EAAE,kBAAkB,GAAG,OAAO,CAAC,UAAU,CAAC,CAOxE;AAuED,wBAAgB,WAAW,CAAC,MAAM,EAAE,UAAU,GAAG,GAAG,CAAC,cAAc,EAAE,GAAG,EAAE,cAAc,GAAG,MAAM,GAAG,OAAO,CAQ1G;AAED,wBAAgB,iBAAiB,CAAC,CAAC,EAAE,UAAU,GAAG,MAAM,EAAE,CAIzD;AAED,wBAAgB,aAAa,CAC3B,QAAQ,EAAE,kBAAkB,EAC5B,IAAI,GAAE,SAAS,GAAG,MAAM,EAAsB,EAC9C,IAAI,SAAQ,GACX,UAAU,CAEZ;AAED,wBAAgB,gBAAgB,CAC9B,QAAQ,EAAE,kBAAkB,EAC5B,IAAI,GAAE,SAAS,GAAG,MAAM,EAAsB,GAC7C,aAAa,CAKf"}
|
|
@@ -0,0 +1,162 @@
|
|
|
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
|
+
exports.getDefaultXPubs = getDefaultXPubs;
|
|
7
|
+
exports.getUnspendableKey = getUnspendableKey;
|
|
8
|
+
exports.getPsbtParams = getPsbtParams;
|
|
9
|
+
exports.containsKey = containsKey;
|
|
10
|
+
exports.getTapLeafScripts = getTapLeafScripts;
|
|
11
|
+
exports.getDescriptor = getDescriptor;
|
|
12
|
+
exports.getDescriptorMap = getDescriptorMap;
|
|
13
|
+
const assert_1 = __importDefault(require("assert"));
|
|
14
|
+
const wasm_miniscript_1 = require("@bitgo/wasm-miniscript");
|
|
15
|
+
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
16
|
+
const key_utils_1 = require("../key.utils");
|
|
17
|
+
function getDefaultXPubs(seed) {
|
|
18
|
+
return (0, key_utils_1.getKeyTriple)(seed).map((k) => k.neutered().toBase58());
|
|
19
|
+
}
|
|
20
|
+
function getUnspendableKey() {
|
|
21
|
+
/*
|
|
22
|
+
https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki#constructing-and-spending-taproot-outputs
|
|
23
|
+
|
|
24
|
+
```
|
|
25
|
+
If one or more of the spending conditions consist of just a single key (after aggregation), the most likely one should
|
|
26
|
+
be made the internal key. If no such condition exists, it may be worthwhile adding one that consists of an aggregation
|
|
27
|
+
of all keys participating in all scripts combined; effectively adding an "everyone agrees" branch. If that is
|
|
28
|
+
inacceptable, pick as internal key a "Nothing Up My Sleeve" (NUMS) point, i.e., a point with unknown discrete
|
|
29
|
+
logarithm.
|
|
30
|
+
|
|
31
|
+
One example of such a point is H = lift_x(0x50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0) which is
|
|
32
|
+
constructed by taking the hash of the standard uncompressed encoding of the secp256k1 base point G as X coordinate.
|
|
33
|
+
In order to avoid leaking the information that key path spending is not possible it is recommended to pick a fresh
|
|
34
|
+
integer r in the range 0...n-1 uniformly at random and use H + rG as internal key. It is possible to prove that this
|
|
35
|
+
internal key does not have a known discrete logarithm with respect to G by revealing r to a verifier who can then
|
|
36
|
+
reconstruct how the internal key was created.
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
We could do the random integer trick here, but for internal testing it is sufficient to use the fixed point.
|
|
40
|
+
*/
|
|
41
|
+
return '50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0';
|
|
42
|
+
}
|
|
43
|
+
function toDescriptorMap(v) {
|
|
44
|
+
return new Map(Object.entries(v).map(([k, v]) => [k, wasm_miniscript_1.Descriptor.fromString(v, 'derivable')]));
|
|
45
|
+
}
|
|
46
|
+
function toXPub(k, path) {
|
|
47
|
+
if (typeof k === 'string') {
|
|
48
|
+
return k + '/' + path;
|
|
49
|
+
}
|
|
50
|
+
return k.neutered().toBase58() + '/' + path;
|
|
51
|
+
}
|
|
52
|
+
function toPlain(k, { xonly = false } = {}) {
|
|
53
|
+
if (typeof k === 'string') {
|
|
54
|
+
if (k.startsWith('xpub') || k.startsWith('xprv')) {
|
|
55
|
+
return toPlain(utxo_lib_1.bip32.fromBase58(k), { xonly });
|
|
56
|
+
}
|
|
57
|
+
return k;
|
|
58
|
+
}
|
|
59
|
+
return k.publicKey.subarray(xonly ? 1 : 0).toString('hex');
|
|
60
|
+
}
|
|
61
|
+
function toXOnly(k) {
|
|
62
|
+
return toPlain(k, { xonly: true });
|
|
63
|
+
}
|
|
64
|
+
function multiArgs(m, n, keys, path) {
|
|
65
|
+
if (n < m) {
|
|
66
|
+
throw new Error(`Cannot create ${m} of ${n} multisig`);
|
|
67
|
+
}
|
|
68
|
+
if (keys.length < n) {
|
|
69
|
+
throw new Error(`Not enough keys for ${m} of ${n} multisig: keys.length=${keys.length}`);
|
|
70
|
+
}
|
|
71
|
+
keys = keys.slice(0, n);
|
|
72
|
+
return [m, ...keys.map((k) => toXPub(k, path))];
|
|
73
|
+
}
|
|
74
|
+
function getPsbtParams(t) {
|
|
75
|
+
switch (t) {
|
|
76
|
+
case 'Wsh2Of3CltvDrop':
|
|
77
|
+
return { locktime: 1 };
|
|
78
|
+
default:
|
|
79
|
+
return {};
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
function getDescriptorNode(template, keys = getDefaultXPubs(), path = '0/*') {
|
|
83
|
+
switch (template) {
|
|
84
|
+
case 'Wsh2Of3':
|
|
85
|
+
return {
|
|
86
|
+
wsh: { multi: multiArgs(2, 3, keys, path) },
|
|
87
|
+
};
|
|
88
|
+
case 'Wsh2Of3CltvDrop':
|
|
89
|
+
const { locktime } = getPsbtParams(template);
|
|
90
|
+
(0, assert_1.default)(locktime);
|
|
91
|
+
return {
|
|
92
|
+
wsh: {
|
|
93
|
+
and_v: [{ 'r:after': locktime }, { multi: multiArgs(2, 3, keys, path) }],
|
|
94
|
+
},
|
|
95
|
+
};
|
|
96
|
+
case 'Wsh2Of2':
|
|
97
|
+
return {
|
|
98
|
+
wsh: { multi: multiArgs(2, 2, keys, path) },
|
|
99
|
+
};
|
|
100
|
+
case 'Tr2Of3-NoKeyPath':
|
|
101
|
+
return {
|
|
102
|
+
tr: [getUnspendableKey(), { multi_a: multiArgs(2, 3, keys, path) }],
|
|
103
|
+
};
|
|
104
|
+
case 'Tr1Of3-NoKeyPath-Tree':
|
|
105
|
+
return {
|
|
106
|
+
tr: [
|
|
107
|
+
getUnspendableKey(),
|
|
108
|
+
[{ pk: toXPub(keys[0], path) }, [{ pk: toXPub(keys[1], path) }, { pk: toXPub(keys[2], path) }]],
|
|
109
|
+
],
|
|
110
|
+
};
|
|
111
|
+
case 'Tr1Of3-NoKeyPath-Tree-Plain':
|
|
112
|
+
return {
|
|
113
|
+
tr: [getUnspendableKey(), [{ pk: toXOnly(keys[0]) }, [{ pk: toXOnly(keys[1]) }, { pk: toXOnly(keys[2]) }]]],
|
|
114
|
+
};
|
|
115
|
+
}
|
|
116
|
+
throw new Error(`Unknown descriptor template: ${template}`);
|
|
117
|
+
}
|
|
118
|
+
function getTapLeafScriptNodes(t) {
|
|
119
|
+
if (Array.isArray(t)) {
|
|
120
|
+
if (t.length !== 2) {
|
|
121
|
+
throw new Error(`expected tuple, got: ${JSON.stringify(t)}`);
|
|
122
|
+
}
|
|
123
|
+
return t.map((v) => (Array.isArray(v) ? getTapLeafScriptNodes(v) : v)).flat();
|
|
124
|
+
}
|
|
125
|
+
if (typeof t === 'object') {
|
|
126
|
+
const node = t;
|
|
127
|
+
if (!('tr' in node)) {
|
|
128
|
+
throw new Error(`TapLeafScripts are only supported for Taproot descriptors, got: ${t}`);
|
|
129
|
+
}
|
|
130
|
+
if (!Array.isArray(node.tr) || node.tr.length !== 2) {
|
|
131
|
+
throw new Error(`expected tuple, got: ${JSON.stringify(node.tr)}`);
|
|
132
|
+
}
|
|
133
|
+
const tapscript = node.tr[1];
|
|
134
|
+
if (!Array.isArray(tapscript)) {
|
|
135
|
+
throw new Error(`expected tapscript to be an array, got: ${JSON.stringify(tapscript)}`);
|
|
136
|
+
}
|
|
137
|
+
return getTapLeafScriptNodes(tapscript);
|
|
138
|
+
}
|
|
139
|
+
throw new Error(`Invalid input: ${JSON.stringify(t)}`);
|
|
140
|
+
}
|
|
141
|
+
function containsKey(script, key) {
|
|
142
|
+
if (script instanceof wasm_miniscript_1.Miniscript) {
|
|
143
|
+
script = wasm_miniscript_1.ast.fromMiniscript(script);
|
|
144
|
+
}
|
|
145
|
+
if ('pk' in script) {
|
|
146
|
+
return script.pk === toXOnly(key);
|
|
147
|
+
}
|
|
148
|
+
throw new Error(`Unsupported script type: ${JSON.stringify(script)}`);
|
|
149
|
+
}
|
|
150
|
+
function getTapLeafScripts(d) {
|
|
151
|
+
return getTapLeafScriptNodes(wasm_miniscript_1.ast.fromDescriptor(d)).map((n) => wasm_miniscript_1.Miniscript.fromString(wasm_miniscript_1.ast.formatNode(n), 'tap').toString());
|
|
152
|
+
}
|
|
153
|
+
function getDescriptor(template, keys = getDefaultXPubs(), path = '0/*') {
|
|
154
|
+
return wasm_miniscript_1.Descriptor.fromStringDetectType(wasm_miniscript_1.ast.formatNode(getDescriptorNode(template, keys, path)));
|
|
155
|
+
}
|
|
156
|
+
function getDescriptorMap(template, keys = getDefaultXPubs()) {
|
|
157
|
+
return toDescriptorMap({
|
|
158
|
+
external: getDescriptor(template, keys, '0/*').toString(),
|
|
159
|
+
internal: getDescriptor(template, keys, '1/*').toString(),
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"descriptors.js","sourceRoot":"","sources":["../../../../src/testutil/descriptor/descriptors.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport { Miniscript, Descriptor, ast } from '@bitgo/wasm-miniscript';\nimport { bip32, BIP32Interface } from '@bitgo-beta/utxo-lib';\n\nimport { DescriptorMap, PsbtParams } from '../../descriptor';\nimport { getKeyTriple, Triple, KeyTriple } from '../key.utils';\n\nexport function getDefaultXPubs(seed?: string): Triple<string> {\n  return getKeyTriple(seed).map((k) => k.neutered().toBase58()) as Triple<string>;\n}\n\nexport function getUnspendableKey(): string {\n  /*\n  https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki#constructing-and-spending-taproot-outputs\n\n  ```\n  If one or more of the spending conditions consist of just a single key (after aggregation), the most likely one should\n  be made the internal key. If no such condition exists, it may be worthwhile adding one that consists of an aggregation\n  of all keys participating in all scripts combined; effectively adding an \"everyone agrees\" branch. If that is\n  inacceptable, pick as internal key a \"Nothing Up My Sleeve\" (NUMS) point, i.e., a point with unknown discrete\n  logarithm.\n\n  One example of such a point is H = lift_x(0x50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0) which is\n  constructed by taking the hash of the standard uncompressed encoding of the secp256k1 base point G as X coordinate.\n  In order to avoid leaking the information that key path spending is not possible it is recommended to pick a fresh\n  integer r in the range 0...n-1 uniformly at random and use H + rG as internal key. It is possible to prove that this\n  internal key does not have a known discrete logarithm with respect to G by revealing r to a verifier who can then\n  reconstruct how the internal key was created.\n  ```\n\n  We could do the random integer trick here, but for internal testing it is sufficient to use the fixed point.\n  */\n  return '50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0';\n}\n\nfunction toDescriptorMap(v: Record<string, string>): DescriptorMap {\n  return new Map(Object.entries(v).map(([k, v]) => [k, Descriptor.fromString(v, 'derivable')]));\n}\n\nexport type DescriptorTemplate =\n  | 'Wsh2Of3'\n  | 'Tr1Of3-NoKeyPath-Tree'\n  // no xpubs, just plain keys\n  | 'Tr1Of3-NoKeyPath-Tree-Plain'\n  | 'Tr2Of3-NoKeyPath'\n  | 'Wsh2Of2'\n  /*\n   * This is a wrapped segwit 2of3 multisig that also uses a relative locktime with\n   * an OP_DROP (requiring a miniscript extension).\n   * It is basically what is used in CoreDao staking transactions.\n   */\n  | 'Wsh2Of3CltvDrop';\n\nfunction toXPub(k: BIP32Interface | string, path: string): string {\n  if (typeof k === 'string') {\n    return k + '/' + path;\n  }\n  return k.neutered().toBase58() + '/' + path;\n}\n\nfunction toPlain(k: BIP32Interface | string, { xonly = false } = {}): string {\n  if (typeof k === 'string') {\n    if (k.startsWith('xpub') || k.startsWith('xprv')) {\n      return toPlain(bip32.fromBase58(k), { xonly });\n    }\n    return k;\n  }\n  return k.publicKey.subarray(xonly ? 1 : 0).toString('hex');\n}\n\nfunction toXOnly(k: BIP32Interface | string): string {\n  return toPlain(k, { xonly: true });\n}\n\nfunction multiArgs(m: number, n: number, keys: BIP32Interface[] | string[], path: string): [number, ...string[]] {\n  if (n < m) {\n    throw new Error(`Cannot create ${m} of ${n} multisig`);\n  }\n  if (keys.length < n) {\n    throw new Error(`Not enough keys for ${m} of ${n} multisig: keys.length=${keys.length}`);\n  }\n  keys = keys.slice(0, n);\n  return [m, ...keys.map((k) => toXPub(k, path))];\n}\n\nexport function getPsbtParams(t: DescriptorTemplate): Partial<PsbtParams> {\n  switch (t) {\n    case 'Wsh2Of3CltvDrop':\n      return { locktime: 1 };\n    default:\n      return {};\n  }\n}\n\nfunction getDescriptorNode(\n  template: DescriptorTemplate,\n  keys: KeyTriple | string[] = getDefaultXPubs(),\n  path = '0/*'\n): ast.DescriptorNode {\n  switch (template) {\n    case 'Wsh2Of3':\n      return {\n        wsh: { multi: multiArgs(2, 3, keys, path) },\n      };\n    case 'Wsh2Of3CltvDrop':\n      const { locktime } = getPsbtParams(template);\n      assert(locktime);\n      return {\n        wsh: {\n          and_v: [{ 'r:after': locktime }, { multi: multiArgs(2, 3, keys, path) }],\n        },\n      };\n    case 'Wsh2Of2':\n      return {\n        wsh: { multi: multiArgs(2, 2, keys, path) },\n      };\n    case 'Tr2Of3-NoKeyPath':\n      return {\n        tr: [getUnspendableKey(), { multi_a: multiArgs(2, 3, keys, path) }],\n      };\n    case 'Tr1Of3-NoKeyPath-Tree':\n      return {\n        tr: [\n          getUnspendableKey(),\n          [{ pk: toXPub(keys[0], path) }, [{ pk: toXPub(keys[1], path) }, { pk: toXPub(keys[2], path) }]],\n        ],\n      };\n    case 'Tr1Of3-NoKeyPath-Tree-Plain':\n      return {\n        tr: [getUnspendableKey(), [{ pk: toXOnly(keys[0]) }, [{ pk: toXOnly(keys[1]) }, { pk: toXOnly(keys[2]) }]]],\n      };\n  }\n  throw new Error(`Unknown descriptor template: ${template}`);\n}\n\ntype TapTree = [TapTree, TapTree] | ast.MiniscriptNode;\n\nfunction getTapLeafScriptNodes(t: ast.DescriptorNode | TapTree): ast.MiniscriptNode[] {\n  if (Array.isArray(t)) {\n    if (t.length !== 2) {\n      throw new Error(`expected tuple, got: ${JSON.stringify(t)}`);\n    }\n    return t.map((v) => (Array.isArray(v) ? getTapLeafScriptNodes(v) : v)).flat();\n  }\n\n  if (typeof t === 'object') {\n    const node = t;\n    if (!('tr' in node)) {\n      throw new Error(`TapLeafScripts are only supported for Taproot descriptors, got: ${t}`);\n    }\n    if (!Array.isArray(node.tr) || node.tr.length !== 2) {\n      throw new Error(`expected tuple, got: ${JSON.stringify(node.tr)}`);\n    }\n    const tapscript = node.tr[1];\n    if (!Array.isArray(tapscript)) {\n      throw new Error(`expected tapscript to be an array, got: ${JSON.stringify(tapscript)}`);\n    }\n    return getTapLeafScriptNodes(tapscript);\n  }\n\n  throw new Error(`Invalid input: ${JSON.stringify(t)}`);\n}\n\nexport function containsKey(script: Miniscript | ast.MiniscriptNode, key: BIP32Interface | string): boolean {\n  if (script instanceof Miniscript) {\n    script = ast.fromMiniscript(script);\n  }\n  if ('pk' in script) {\n    return script.pk === toXOnly(key);\n  }\n  throw new Error(`Unsupported script type: ${JSON.stringify(script)}`);\n}\n\nexport function getTapLeafScripts(d: Descriptor): string[] {\n  return getTapLeafScriptNodes(ast.fromDescriptor(d)).map((n) =>\n    Miniscript.fromString(ast.formatNode(n), 'tap').toString()\n  );\n}\n\nexport function getDescriptor(\n  template: DescriptorTemplate,\n  keys: KeyTriple | string[] = getDefaultXPubs(),\n  path = '0/*'\n): Descriptor {\n  return Descriptor.fromStringDetectType(ast.formatNode(getDescriptorNode(template, keys, path)));\n}\n\nexport function getDescriptorMap(\n  template: DescriptorTemplate,\n  keys: KeyTriple | string[] = getDefaultXPubs()\n): DescriptorMap {\n  return toDescriptorMap({\n    external: getDescriptor(template, keys, '0/*').toString(),\n    internal: getDescriptor(template, keys, '1/*').toString(),\n  });\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/testutil/descriptor/index.ts"],"names":[],"mappings":"AAAA,cAAc,eAAe,CAAC;AAC9B,cAAc,cAAc,CAAC;AAC7B,cAAc,cAAc,CAAC"}
|
|
@@ -0,0 +1,20 @@
|
|
|
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 __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./descriptors"), exports);
|
|
18
|
+
__exportStar(require("./mock.utils"), exports);
|
|
19
|
+
__exportStar(require("./psbt.utils"), exports);
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdGVzdHV0aWwvZGVzY3JpcHRvci9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsZ0RBQThCO0FBQzlCLCtDQUE2QjtBQUM3QiwrQ0FBNkIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Rlc2NyaXB0b3JzJztcbmV4cG9ydCAqIGZyb20gJy4vbW9jay51dGlscyc7XG5leHBvcnQgKiBmcm9tICcuL3BzYnQudXRpbHMnO1xuIl19
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { Descriptor, Miniscript } from '@bitgo/wasm-miniscript';
|
|
2
|
+
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
3
|
+
import { PsbtParams, DerivedDescriptorTransactionInput } from '../../descriptor';
|
|
4
|
+
import { DescriptorTemplate } from './descriptors';
|
|
5
|
+
type MockOutputIdParams = {
|
|
6
|
+
hash?: string;
|
|
7
|
+
vout?: number;
|
|
8
|
+
};
|
|
9
|
+
type BaseMockDescriptorOutputParams = {
|
|
10
|
+
id?: MockOutputIdParams;
|
|
11
|
+
index?: number;
|
|
12
|
+
value?: bigint;
|
|
13
|
+
sequence?: number;
|
|
14
|
+
selectTapLeafScript?: Miniscript;
|
|
15
|
+
};
|
|
16
|
+
export declare function mockDerivedDescriptorWalletOutput(descriptor: Descriptor, outputParams?: BaseMockDescriptorOutputParams): DerivedDescriptorTransactionInput;
|
|
17
|
+
type MockInput = BaseMockDescriptorOutputParams & {
|
|
18
|
+
index: number;
|
|
19
|
+
descriptor: Descriptor;
|
|
20
|
+
selectTapLeafScript?: Miniscript;
|
|
21
|
+
};
|
|
22
|
+
type MockOutput = {
|
|
23
|
+
descriptor: Descriptor;
|
|
24
|
+
index: number;
|
|
25
|
+
value: bigint;
|
|
26
|
+
external?: boolean;
|
|
27
|
+
};
|
|
28
|
+
export declare function mockPsbt(inputs: MockInput[], outputs: MockOutput[], params?: Partial<PsbtParams>): utxolib.bitgo.UtxoPsbt;
|
|
29
|
+
export declare function mockPsbtDefault({ descriptorSelf, descriptorOther, params, }?: {
|
|
30
|
+
descriptorSelf?: Descriptor;
|
|
31
|
+
descriptorOther?: Descriptor;
|
|
32
|
+
params?: Partial<PsbtParams>;
|
|
33
|
+
}): utxolib.bitgo.UtxoPsbt;
|
|
34
|
+
export declare function mockPsbtDefaultWithDescriptorTemplate(t: DescriptorTemplate, params?: Partial<PsbtParams>): utxolib.bitgo.UtxoPsbt;
|
|
35
|
+
export {};
|
|
36
|
+
//# sourceMappingURL=mock.utils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mock.utils.d.ts","sourceRoot":"","sources":["../../../../src/testutil/descriptor/mock.utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AAChE,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EACL,UAAU,EAGV,iCAAiC,EAClC,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EAAE,kBAAkB,EAAkC,MAAM,eAAe,CAAC;AAEnF,KAAK,kBAAkB,GAAG;IAAE,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC;AAE3D,KAAK,8BAA8B,GAAG;IACpC,EAAE,CAAC,EAAE,kBAAkB,CAAC;IACxB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,mBAAmB,CAAC,EAAE,UAAU,CAAC;CAClC,CAAC;AAWF,wBAAgB,iCAAiC,CAC/C,UAAU,EAAE,UAAU,EACtB,YAAY,GAAE,8BAAmC,GAChD,iCAAiC,CAcnC;AAED,KAAK,SAAS,GAAG,8BAA8B,GAAG;IAChD,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,UAAU,CAAC;IACvB,mBAAmB,CAAC,EAAE,UAAU,CAAC;CAClC,CAAC;AAEF,KAAK,UAAU,GAAG;IAChB,UAAU,EAAE,UAAU,CAAC;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB,CAAC;AAMF,wBAAgB,QAAQ,CACtB,MAAM,EAAE,SAAS,EAAE,EACnB,OAAO,EAAE,UAAU,EAAE,EACrB,MAAM,GAAE,OAAO,CAAC,UAAU,CAAM,GAC/B,OAAO,CAAC,KAAK,CAAC,QAAQ,CAaxB;AAED,wBAAgB,eAAe,CAAC,EAC9B,cAA+D,EAC/D,eAAgE,EAChE,MAAW,GACZ,GAAE;IACD,cAAc,CAAC,EAAE,UAAU,CAAC;IAC5B,eAAe,CAAC,EAAE,UAAU,CAAC;IAC7B,MAAM,CAAC,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC;CACzB,GAAG,OAAO,CAAC,KAAK,CAAC,QAAQ,CAY9B;AAED,wBAAgB,qCAAqC,CACnD,CAAC,EAAE,kBAAkB,EACrB,MAAM,GAAE,OAAO,CAAC,UAAU,CAAM,GAC/B,OAAO,CAAC,KAAK,CAAC,QAAQ,CAMxB"}
|