@bitgo-beta/utxo-lib 8.0.3-beta.52 → 8.0.3-beta.521
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/README.md +19 -16
- package/dist/src/address.js +5 -5
- package/dist/src/addressFormat.d.ts +1 -1
- package/dist/src/addressFormat.d.ts.map +1 -1
- package/dist/src/addressFormat.js +9 -9
- package/dist/src/bitgo/Musig2.js +19 -19
- package/dist/src/bitgo/PsbtUtil.d.ts +17 -0
- package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -1
- package/dist/src/bitgo/PsbtUtil.js +57 -3
- package/dist/src/bitgo/Unspent.d.ts +2 -2
- package/dist/src/bitgo/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/Unspent.js +3 -3
- package/dist/src/bitgo/UtxoPsbt.d.ts +16 -4
- package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoPsbt.js +118 -76
- package/dist/src/bitgo/UtxoTransaction.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoTransaction.js +9 -9
- package/dist/src/bitgo/bitcoincash/address.js +6 -6
- package/dist/src/bitgo/bitcoincash/index.js +6 -2
- package/dist/src/bitgo/dash/DashTransaction.js +3 -3
- package/dist/src/bitgo/dash/index.js +6 -2
- package/dist/src/bitgo/index.d.ts +3 -1
- package/dist/src/bitgo/index.d.ts.map +1 -1
- package/dist/src/bitgo/index.js +9 -3
- package/dist/src/bitgo/keyutil.d.ts +9 -0
- package/dist/src/bitgo/keyutil.d.ts.map +1 -1
- package/dist/src/bitgo/keyutil.js +24 -2
- package/dist/src/bitgo/legacysafe/index.d.ts +15 -0
- package/dist/src/bitgo/legacysafe/index.d.ts.map +1 -0
- package/dist/src/bitgo/legacysafe/index.js +61 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts +1 -1
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts.map +1 -1
- package/dist/src/bitgo/litecoin/LitecoinTransaction.js +2 -2
- package/dist/src/bitgo/litecoin/index.js +6 -2
- package/dist/src/bitgo/outputScripts.d.ts +7 -6
- package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
- package/dist/src/bitgo/outputScripts.js +29 -14
- package/dist/src/bitgo/parseInput.d.ts +11 -7
- package/dist/src/bitgo/parseInput.d.ts.map +1 -1
- package/dist/src/bitgo/parseInput.js +23 -7
- package/dist/src/bitgo/psbt/fromHalfSigned.js +5 -5
- package/dist/src/bitgo/signature.d.ts +2 -2
- package/dist/src/bitgo/signature.d.ts.map +1 -1
- package/dist/src/bitgo/signature.js +17 -8
- package/dist/src/bitgo/transaction.d.ts +13 -0
- package/dist/src/bitgo/transaction.d.ts.map +1 -1
- package/dist/src/bitgo/transaction.js +31 -11
- package/dist/src/bitgo/transactionAmounts.d.ts +9 -0
- package/dist/src/bitgo/transactionAmounts.d.ts.map +1 -0
- package/dist/src/bitgo/transactionAmounts.js +33 -0
- package/dist/src/bitgo/types.d.ts +9 -2
- package/dist/src/bitgo/types.d.ts.map +1 -1
- package/dist/src/bitgo/types.js +11 -2
- package/dist/src/bitgo/wallet/Psbt.d.ts +48 -18
- package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Psbt.js +129 -29
- package/dist/src/bitgo/wallet/Unspent.d.ts +23 -15
- package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Unspent.js +59 -54
- package/dist/src/bitgo/wallet/WalletKeys.d.ts +1 -1
- package/dist/src/bitgo/wallet/WalletKeys.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/WalletOutput.d.ts +17 -1
- package/dist/src/bitgo/wallet/WalletOutput.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/WalletOutput.js +68 -27
- package/dist/src/bitgo/wallet/WalletScripts.js +2 -2
- package/dist/src/bitgo/wallet/WalletUnspentSigner.js +7 -7
- package/dist/src/bitgo/wallet/chains.d.ts +2 -2
- package/dist/src/bitgo/wallet/chains.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/index.d.ts +2 -0
- package/dist/src/bitgo/wallet/index.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/index.js +8 -2
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.d.ts +50 -0
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.d.ts.map +1 -0
- package/dist/src/bitgo/wallet/psbt/PsbtOutputs.js +87 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.d.ts +32 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.d.ts.map +1 -0
- package/dist/src/bitgo/wallet/psbt/RootNodes.js +125 -0
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts +0 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.js +5 -14
- package/dist/src/bitgo/zcash/ZcashTransaction.d.ts +1 -1
- package/dist/src/bitgo/zcash/ZcashTransaction.d.ts.map +1 -1
- package/dist/src/bitgo/zcash/ZcashTransaction.js +15 -15
- package/dist/src/bitgo/zcash/ZcashTransactionBuilder.js +4 -4
- package/dist/src/bitgo/zcash/address.js +3 -3
- package/dist/src/bitgo/zcash/hashZip0244.d.ts +1 -1
- package/dist/src/bitgo/zcash/hashZip0244.d.ts.map +1 -1
- package/dist/src/bitgo/zcash/hashZip0244.js +3 -3
- package/dist/src/bitgo/zcash/index.js +6 -2
- package/dist/src/index.js +6 -2
- package/dist/src/networks.d.ts +2 -2
- package/dist/src/networks.d.ts.map +1 -1
- package/dist/src/networks.js +47 -1
- package/dist/src/noble_ecc.d.ts +7 -7
- package/dist/src/noble_ecc.d.ts.map +1 -1
- package/dist/src/noble_ecc.js +4 -4
- package/dist/src/testutil/index.js +6 -2
- package/dist/src/testutil/keys.d.ts +4 -1
- package/dist/src/testutil/keys.d.ts.map +1 -1
- package/dist/src/testutil/keys.js +18 -3
- package/dist/src/testutil/mock.d.ts +1 -1
- package/dist/src/testutil/mock.d.ts.map +1 -1
- package/dist/src/testutil/mock.js +15 -15
- package/dist/src/testutil/psbt.d.ts +37 -15
- package/dist/src/testutil/psbt.d.ts.map +1 -1
- package/dist/src/testutil/psbt.js +41 -31
- package/dist/src/testutil/transaction.d.ts +15 -6
- package/dist/src/testutil/transaction.d.ts.map +1 -1
- package/dist/src/testutil/transaction.js +20 -20
- package/dist/src/transaction_builder.d.ts.map +1 -1
- package/dist/src/transaction_builder.js +12 -17
- package/package.json +7 -7
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.getDefaultWalletKeys = exports.getDefaultCosigner = exports.getKeyName = exports.getKeyTriple = exports.getKey = void 0;
|
|
3
|
+
exports.getDefaultWalletKeys = exports.getDefaultCosigner = exports.getKeyName = exports.getUncompressedKeyTriple = exports.getKeyTriple = exports.getKey = void 0;
|
|
4
4
|
const bip32_1 = require("bip32");
|
|
5
5
|
const crypto = require("crypto");
|
|
6
6
|
const bitgo_1 = require("../bitgo");
|
|
7
7
|
const noble_ecc_1 = require("../noble_ecc");
|
|
8
|
-
const
|
|
8
|
+
const networks_1 = require("../networks");
|
|
9
|
+
const bip32 = (0, bip32_1.BIP32Factory)(noble_ecc_1.ecc);
|
|
9
10
|
function getKey(seed) {
|
|
10
11
|
return bip32.fromSeed(crypto.createHash('sha256').update(seed).digest());
|
|
11
12
|
}
|
|
@@ -14,6 +15,20 @@ function getKeyTriple(seed) {
|
|
|
14
15
|
return [getKey(seed + '.0'), getKey(seed + '.1'), getKey(seed + '.2')];
|
|
15
16
|
}
|
|
16
17
|
exports.getKeyTriple = getKeyTriple;
|
|
18
|
+
function getUncompressedKey(input) {
|
|
19
|
+
// Using input for deterministic randomness
|
|
20
|
+
return noble_ecc_1.ECPair.makeRandom({
|
|
21
|
+
compressed: false,
|
|
22
|
+
network: networks_1.networks.testnet,
|
|
23
|
+
rng: () => {
|
|
24
|
+
return Buffer.alloc(32, input);
|
|
25
|
+
},
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
function getUncompressedKeyTriple(inputs) {
|
|
29
|
+
return [getUncompressedKey(inputs[0]), getUncompressedKey(inputs[1]), getUncompressedKey(inputs[2])];
|
|
30
|
+
}
|
|
31
|
+
exports.getUncompressedKeyTriple = getUncompressedKeyTriple;
|
|
17
32
|
function getKeyName(triple, k) {
|
|
18
33
|
return ['user', 'backup', 'bitgo'][triple.indexOf(k)];
|
|
19
34
|
}
|
|
@@ -37,4 +52,4 @@ function getDefaultWalletKeys() {
|
|
|
37
52
|
return new bitgo_1.RootWalletKeys(getKeyTriple('default'));
|
|
38
53
|
}
|
|
39
54
|
exports.getDefaultWalletKeys = getDefaultWalletKeys;
|
|
40
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoia2V5cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy90ZXN0dXRpbC9rZXlzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLGlDQUErRDtBQUMvRCxpQ0FBaUM7QUFHakMsb0NBQTBDO0FBQzFDLDRDQUE0RDtBQUM1RCwwQ0FBdUM7QUFFdkMsTUFBTSxLQUFLLEdBQWEsSUFBQSxvQkFBWSxFQUFDLGVBQUcsQ0FBQyxDQUFDO0FBSzFDLFNBQWdCLE1BQU0sQ0FBQyxJQUFZO0lBQ2pDLE9BQU8sS0FBSyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO0FBQzNFLENBQUM7QUFGRCx3QkFFQztBQUVELFNBQWdCLFlBQVksQ0FBQyxJQUFZO0lBQ3ZDLE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxFQUFFLE1BQU0sQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDLEVBQUUsTUFBTSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUMsQ0FBQyxDQUFDO0FBQ3pFLENBQUM7QUFGRCxvQ0FFQztBQUVELFNBQVMsa0JBQWtCLENBQUMsS0FBSztJQUMvQiwyQ0FBMkM7SUFDM0MsT0FBTyxrQkFBTSxDQUFDLFVBQVUsQ0FBQztRQUN2QixVQUFVLEVBQUUsS0FBSztRQUNqQixPQUFPLEVBQUUsbUJBQVEsQ0FBQyxPQUFPO1FBQ3pCLEdBQUcsRUFBRSxHQUFXLEVBQUU7WUFDaEIsT0FBTyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztRQUNqQyxDQUFDO0tBQ0YsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVELFNBQWdCLHdCQUF3QixDQUFDLE1BQXNCO0lBQzdELE9BQU8sQ0FBQyxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3ZHLENBQUM7QUFGRCw0REFFQztBQUVELFNBQWdCLFVBQVUsQ0FBQyxNQUE4QixFQUFFLENBQWlCO0lBQzFFLE9BQU8sQ0FBQyxNQUFNLEVBQUUsUUFBUSxFQUFFLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUN4RCxDQUFDO0FBRkQsZ0NBRUM7QUFFRCxTQUFnQixrQkFBa0IsQ0FBSSxNQUFpQixFQUFFLE1BQVM7SUFDaEUsTUFBTSxFQUFFLEdBQUcsQ0FBQyxDQUFJLEVBQUUsQ0FBSSxFQUFFLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNoRyxNQUFNLENBQUMsSUFBSSxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsR0FBRyxNQUFNLENBQUM7SUFDckMsSUFBSSxFQUFFLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxFQUFFO1FBQ3BCLE9BQU8sS0FBSyxDQUFDO0tBQ2Q7SUFDRCxJQUFJLEVBQUUsQ0FBQyxNQUFNLEVBQUUsTUFBTSxDQUFDLEVBQUU7UUFDdEIsT0FBTyxLQUFLLENBQUM7S0FDZDtJQUNELElBQUksRUFBRSxDQUFDLE1BQU0sRUFBRSxLQUFLLENBQUMsRUFBRTtRQUNyQixPQUFPLElBQUksQ0FBQztLQUNiO0lBQ0QsTUFBTSxJQUFJLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO0FBQzNDLENBQUM7QUFiRCxnREFhQztBQUVELFNBQWdCLG9CQUFvQjtJQUNsQyxPQUFPLElBQUksc0JBQWMsQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQztBQUNyRCxDQUFDO0FBRkQsb0RBRUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBCSVAzMkFQSSwgQklQMzJGYWN0b3J5LCBCSVAzMkludGVyZmFjZSB9IGZyb20gJ2JpcDMyJztcbmltcG9ydCAqIGFzIGNyeXB0byBmcm9tICdjcnlwdG8nO1xuXG5pbXBvcnQgeyBUcmlwbGUgfSBmcm9tICcuLi9iaXRnbyc7XG5pbXBvcnQgeyBSb290V2FsbGV0S2V5cyB9IGZyb20gJy4uL2JpdGdvJztcbmltcG9ydCB7IGVjYywgRUNQYWlyLCBFQ1BhaXJJbnRlcmZhY2UgfSBmcm9tICcuLi9ub2JsZV9lY2MnO1xuaW1wb3J0IHsgbmV0d29ya3MgfSBmcm9tICcuLi9uZXR3b3Jrcyc7XG5cbmNvbnN0IGJpcDMyOiBCSVAzMkFQSSA9IEJJUDMyRmFjdG9yeShlY2MpO1xuXG5leHBvcnQgdHlwZSBLZXlUcmlwbGUgPSBUcmlwbGU8QklQMzJJbnRlcmZhY2U+O1xuZXhwb3J0IHR5cGUgVW5jb21wcmVzc2VkS2V5VHJpcGxlID0gVHJpcGxlPEVDUGFpckludGVyZmFjZT47XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRLZXkoc2VlZDogc3RyaW5nKTogQklQMzJJbnRlcmZhY2Uge1xuICByZXR1cm4gYmlwMzIuZnJvbVNlZWQoY3J5cHRvLmNyZWF0ZUhhc2goJ3NoYTI1NicpLnVwZGF0ZShzZWVkKS5kaWdlc3QoKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRLZXlUcmlwbGUoc2VlZDogc3RyaW5nKTogS2V5VHJpcGxlIHtcbiAgcmV0dXJuIFtnZXRLZXkoc2VlZCArICcuMCcpLCBnZXRLZXkoc2VlZCArICcuMScpLCBnZXRLZXkoc2VlZCArICcuMicpXTtcbn1cblxuZnVuY3Rpb24gZ2V0VW5jb21wcmVzc2VkS2V5KGlucHV0KSB7XG4gIC8vIFVzaW5nIGlucHV0IGZvciBkZXRlcm1pbmlzdGljIHJhbmRvbW5lc3NcbiAgcmV0dXJuIEVDUGFpci5tYWtlUmFuZG9tKHtcbiAgICBjb21wcmVzc2VkOiBmYWxzZSxcbiAgICBuZXR3b3JrOiBuZXR3b3Jrcy50ZXN0bmV0LFxuICAgIHJuZzogKCk6IEJ1ZmZlciA9PiB7XG4gICAgICByZXR1cm4gQnVmZmVyLmFsbG9jKDMyLCBpbnB1dCk7XG4gICAgfSxcbiAgfSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRVbmNvbXByZXNzZWRLZXlUcmlwbGUoaW5wdXRzOiBUcmlwbGU8bnVtYmVyPik6IFVuY29tcHJlc3NlZEtleVRyaXBsZSB7XG4gIHJldHVybiBbZ2V0VW5jb21wcmVzc2VkS2V5KGlucHV0c1swXSksIGdldFVuY29tcHJlc3NlZEtleShpbnB1dHNbMV0pLCBnZXRVbmNvbXByZXNzZWRLZXkoaW5wdXRzWzJdKV07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRLZXlOYW1lKHRyaXBsZTogVHJpcGxlPEJJUDMySW50ZXJmYWNlPiwgazogQklQMzJJbnRlcmZhY2UpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICByZXR1cm4gWyd1c2VyJywgJ2JhY2t1cCcsICdiaXRnbyddW3RyaXBsZS5pbmRleE9mKGspXTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldERlZmF1bHRDb3NpZ25lcjxUPihrZXlzZXQ6IFRyaXBsZTxUPiwgc2lnbmVyOiBUKTogVCB7XG4gIGNvbnN0IGVxID0gKGE6IFQsIGI6IFQpID0+IGEgPT09IGIgfHwgKEJ1ZmZlci5pc0J1ZmZlcihhKSAmJiBCdWZmZXIuaXNCdWZmZXIoYikgJiYgYS5lcXVhbHMoYikpO1xuICBjb25zdCBbdXNlciwgYmFja3VwLCBiaXRnb10gPSBrZXlzZXQ7XG4gIGlmIChlcShzaWduZXIsIHVzZXIpKSB7XG4gICAgcmV0dXJuIGJpdGdvO1xuICB9XG4gIGlmIChlcShzaWduZXIsIGJhY2t1cCkpIHtcbiAgICByZXR1cm4gYml0Z287XG4gIH1cbiAgaWYgKGVxKHNpZ25lciwgYml0Z28pKSB7XG4gICAgcmV0dXJuIHVzZXI7XG4gIH1cbiAgdGhyb3cgbmV3IEVycm9yKGBzaWduZXIgbm90IGluIHB1YmtleXNgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldERlZmF1bHRXYWxsZXRLZXlzKCk6IFJvb3RXYWxsZXRLZXlzIHtcbiAgcmV0dXJuIG5ldyBSb290V2FsbGV0S2V5cyhnZXRLZXlUcmlwbGUoJ2RlZmF1bHQnKSk7XG59XG4iXX0=
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
import { BIP32Interface } from 'bip32';
|
|
3
3
|
import { Network } from '../networks';
|
|
4
4
|
import { ChainCode, NonWitnessWalletUnspent, outputScripts, RootWalletKeys, Unspent, UnspentWithPrevTx, UtxoTransaction, WalletUnspent } from '../bitgo';
|
|
5
|
-
export
|
|
5
|
+
export type InputType = outputScripts.ScriptType2Of3;
|
|
6
6
|
export declare function mockPrevTx(vout: number, outputScript: Buffer, value: bigint, network: Network): UtxoTransaction<bigint>;
|
|
7
7
|
export declare const replayProtectionKeyPair: BIP32Interface;
|
|
8
8
|
export declare function isReplayProtectionUnspent<TNumber extends bigint | number>(u: Unspent<TNumber>, network: Network): boolean;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mock.d.ts","sourceRoot":"","sources":["../../../src/testutil/mock.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,cAAc,EAAE,MAAM,OAAO,CAAC;AAGvC,OAAO,EAAc,OAAO,EAAY,MAAM,aAAa,CAAC;AAE5D,OAAO,EACL,SAAS,EAMT,uBAAuB,EACvB,aAAa,EACb,cAAc,EAEd,OAAO,EACP,iBAAiB,EACjB,eAAe,EACf,aAAa,EACd,MAAM,UAAU,CAAC;AAMlB,
|
|
1
|
+
{"version":3,"file":"mock.d.ts","sourceRoot":"","sources":["../../../src/testutil/mock.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,cAAc,EAAE,MAAM,OAAO,CAAC;AAGvC,OAAO,EAAc,OAAO,EAAY,MAAM,aAAa,CAAC;AAE5D,OAAO,EACL,SAAS,EAMT,uBAAuB,EACvB,aAAa,EACb,cAAc,EAEd,OAAO,EACP,iBAAiB,EACjB,eAAe,EACf,aAAa,EACd,MAAM,UAAU,CAAC;AAMlB,MAAM,MAAM,SAAS,GAAG,aAAa,CAAC,cAAc,CAAC;AAErD,wBAAgB,UAAU,CACxB,IAAI,EAAE,MAAM,EACZ,YAAY,EAAE,MAAM,EACpB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,OAAO,GACf,eAAe,CAAC,MAAM,CAAC,CA8BzB;AAED,eAAO,MAAM,uBAAuB,gBAA8B,CAAC;AAGnE,wBAAgB,yBAAyB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACvE,CAAC,EAAE,OAAO,CAAC,OAAO,CAAC,EACnB,OAAO,EAAE,OAAO,GACf,OAAO,CAET;AAED,wBAAgB,2BAA2B,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACzE,OAAO,EAAE,OAAO,EAChB,KAAK,EAAE,OAAO,EACd,EAAE,GAA6B,EAAE,IAAQ,EAAE,GAAE;IAAE,GAAG,CAAC,EAAE,cAAc,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAO,GACxF,iBAAiB,CAAC,OAAO,CAAC,CAI5B;AAED,wBAAgB,iBAAiB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC/D,OAAO,EAAE,OAAO,EAChB,KAAK,EAAE,OAAO,EACd,EACE,KAAS,EACT,KAAS,EACT,IAA6B,EAC7B,IAAQ,EACR,EAAE,GACH,GAAE;IAAE,KAAK,CAAC,EAAE,SAAS,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,cAAc,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,EAAE,CAAC,EAAE,MAAM,CAAA;CAAO,GAC/F,aAAa,CAAC,OAAO,CAAC,GAAG,uBAAuB,CAAC,OAAO,CAAC,CA0B3D;AAED,wBAAgB,YAAY,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,cAAc,EAAE,cAAc,EAC9B,gBAAgB,EAAE,CAAC,SAAS,GAAG,aAAa,CAAC,kBAAkB,CAAC,EAAE,EAClE,gBAAgB,EAAE,OAAO,EACzB,OAAO,EAAE,OAAO,GACf,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,aAAa,CAAC,OAAO,CAAC,CAAC,EAAE,CAgB/C"}
|
|
@@ -10,8 +10,8 @@ const address_1 = require("../address");
|
|
|
10
10
|
const outputScripts_1 = require("../bitgo/outputScripts");
|
|
11
11
|
const keys_1 = require("./keys");
|
|
12
12
|
function mockPrevTx(vout, outputScript, value, network) {
|
|
13
|
-
const psbtFromNetwork = bitgo_1.createPsbtForNetwork({ network });
|
|
14
|
-
const keypair = keys_1.getKey('mock-prev-tx');
|
|
13
|
+
const psbtFromNetwork = (0, bitgo_1.createPsbtForNetwork)({ network });
|
|
14
|
+
const keypair = (0, keys_1.getKey)('mock-prev-tx');
|
|
15
15
|
const pubkey = keypair.publicKey;
|
|
16
16
|
assert(keypair.privateKey);
|
|
17
17
|
const payment = utxolib.payments.p2wpkh({ pubkey });
|
|
@@ -40,29 +40,29 @@ function mockPrevTx(vout, outputScript, value, network) {
|
|
|
40
40
|
return psbtFromNetwork.extractTransaction();
|
|
41
41
|
}
|
|
42
42
|
exports.mockPrevTx = mockPrevTx;
|
|
43
|
-
exports.replayProtectionKeyPair = keys_1.getKey('replay-protection');
|
|
44
|
-
const replayProtectionScriptPubKey = outputScripts_1.createOutputScriptP2shP2pk(exports.replayProtectionKeyPair.publicKey).scriptPubKey;
|
|
43
|
+
exports.replayProtectionKeyPair = (0, keys_1.getKey)('replay-protection');
|
|
44
|
+
const replayProtectionScriptPubKey = (0, outputScripts_1.createOutputScriptP2shP2pk)(exports.replayProtectionKeyPair.publicKey).scriptPubKey;
|
|
45
45
|
function isReplayProtectionUnspent(u, network) {
|
|
46
|
-
return u.address === address_1.fromOutputScript(replayProtectionScriptPubKey, network);
|
|
46
|
+
return u.address === (0, address_1.fromOutputScript)(replayProtectionScriptPubKey, network);
|
|
47
47
|
}
|
|
48
48
|
exports.isReplayProtectionUnspent = isReplayProtectionUnspent;
|
|
49
49
|
function mockReplayProtectionUnspent(network, value, { key = exports.replayProtectionKeyPair, vout = 0 } = {}) {
|
|
50
|
-
const outputScript = outputScripts_1.createOutputScriptP2shP2pk(key.publicKey).scriptPubKey;
|
|
50
|
+
const outputScript = (0, outputScripts_1.createOutputScriptP2shP2pk)(key.publicKey).scriptPubKey;
|
|
51
51
|
const prevTransaction = mockPrevTx(vout, outputScript, BigInt(value), network);
|
|
52
|
-
return { ...bitgo_1.fromOutputWithPrevTx(prevTransaction, vout), value };
|
|
52
|
+
return { ...(0, bitgo_1.fromOutputWithPrevTx)(prevTransaction, vout), value };
|
|
53
53
|
}
|
|
54
54
|
exports.mockReplayProtectionUnspent = mockReplayProtectionUnspent;
|
|
55
|
-
function mockWalletUnspent(network, value, { chain = 0, index = 0, keys = keys_1.getDefaultWalletKeys(), vout = 0, id, } = {}) {
|
|
55
|
+
function mockWalletUnspent(network, value, { chain = 0, index = 0, keys = (0, keys_1.getDefaultWalletKeys)(), vout = 0, id, } = {}) {
|
|
56
56
|
const derivedKeys = keys.deriveForChainAndIndex(chain, index);
|
|
57
|
-
const address = address_1.fromOutputScript(outputScripts_1.createOutputScript2of3(derivedKeys.publicKeys, bitgo_1.scriptTypeForChain(chain)).scriptPubKey, network);
|
|
57
|
+
const address = (0, address_1.fromOutputScript)((0, outputScripts_1.createOutputScript2of3)(derivedKeys.publicKeys, (0, bitgo_1.scriptTypeForChain)(chain)).scriptPubKey, network);
|
|
58
58
|
if (id && typeof id === 'string') {
|
|
59
59
|
return { id, address, chain, index, value };
|
|
60
60
|
}
|
|
61
61
|
else {
|
|
62
|
-
const prevTransaction = mockPrevTx(vout, outputScripts_1.createOutputScript2of3(derivedKeys.publicKeys, bitgo_1.scriptTypeForChain(chain), network).scriptPubKey, BigInt(value), network);
|
|
63
|
-
const unspent = bitgo_1.isSegwit(chain) || networks_1.getMainnet(network) === networks_1.networks.zcash
|
|
64
|
-
? bitgo_1.fromOutput(prevTransaction, vout)
|
|
65
|
-
: bitgo_1.fromOutputWithPrevTx(prevTransaction, vout);
|
|
62
|
+
const prevTransaction = mockPrevTx(vout, (0, outputScripts_1.createOutputScript2of3)(derivedKeys.publicKeys, (0, bitgo_1.scriptTypeForChain)(chain), network).scriptPubKey, BigInt(value), network);
|
|
63
|
+
const unspent = (0, bitgo_1.isSegwit)(chain) || (0, networks_1.getMainnet)(network) === networks_1.networks.zcash
|
|
64
|
+
? (0, bitgo_1.fromOutput)(prevTransaction, vout)
|
|
65
|
+
: (0, bitgo_1.fromOutputWithPrevTx)(prevTransaction, vout);
|
|
66
66
|
return {
|
|
67
67
|
...unspent,
|
|
68
68
|
chain,
|
|
@@ -77,7 +77,7 @@ function mockUnspents(rootWalletKeys, inputScriptTypes, testOutputAmount, networ
|
|
|
77
77
|
if (bitgo_1.outputScripts.isScriptType2Of3(t)) {
|
|
78
78
|
return mockWalletUnspent(network, testOutputAmount, {
|
|
79
79
|
keys: rootWalletKeys,
|
|
80
|
-
chain: bitgo_1.getExternalChainCode(t),
|
|
80
|
+
chain: (0, bitgo_1.getExternalChainCode)(t),
|
|
81
81
|
vout: i,
|
|
82
82
|
});
|
|
83
83
|
}
|
|
@@ -91,4 +91,4 @@ function mockUnspents(rootWalletKeys, inputScriptTypes, testOutputAmount, networ
|
|
|
91
91
|
});
|
|
92
92
|
}
|
|
93
93
|
exports.mockUnspents = mockUnspents;
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mock.js","sourceRoot":"","sources":["../../../src/testutil/mock.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport { BIP32Interface } from 'bip32';\nimport * as noble from '@noble/secp256k1';\nimport * as utxolib from '..';\nimport { getMainnet, Network, networks } from '../networks';\n\nimport {\n  ChainCode,\n  createPsbtForNetwork,\n  fromOutput,\n  fromOutputWithPrevTx,\n  getExternalChainCode,\n  isSegwit,\n  NonWitnessWalletUnspent,\n  outputScripts,\n  RootWalletKeys,\n  scriptTypeForChain,\n  Unspent,\n  UnspentWithPrevTx,\n  UtxoTransaction,\n  WalletUnspent,\n} from '../bitgo';\nimport { fromOutputScript } from '../address';\nimport { createOutputScript2of3, createOutputScriptP2shP2pk } from '../bitgo/outputScripts';\n\nimport { getDefaultWalletKeys, getKey } from './keys';\n\nexport type InputType = outputScripts.ScriptType2Of3;\n\nexport function mockPrevTx(\n  vout: number,\n  outputScript: Buffer,\n  value: bigint,\n  network: Network\n): UtxoTransaction<bigint> {\n  const psbtFromNetwork = createPsbtForNetwork({ network });\n\n  const keypair = getKey('mock-prev-tx');\n  const pubkey = keypair.publicKey;\n  assert(keypair.privateKey);\n  const payment = utxolib.payments.p2wpkh({ pubkey });\n  const destOutput = payment.output;\n  if (!destOutput) throw new Error('Impossible, payment we just constructed has no output');\n\n  for (let index = 0; index <= vout; index++) {\n    if (index === vout) {\n      psbtFromNetwork.addOutput({ script: outputScript, value });\n    } else {\n      psbtFromNetwork.addOutput({ script: destOutput, value });\n    }\n  }\n  psbtFromNetwork.addInput({\n    hash: Buffer.alloc(32, 0x01),\n    index: 0,\n    witnessUtxo: { script: destOutput, value: value * (BigInt(vout) + BigInt(1)) + BigInt(1000) },\n  });\n  psbtFromNetwork.signInput(0, {\n    publicKey: pubkey,\n    sign: (hash: Buffer, lowR?: boolean) =>\n      Buffer.from(noble.signSync(hash, keypair.privateKey as Buffer, { canonical: !lowR, der: false })),\n  });\n  psbtFromNetwork.validateSignaturesOfAllInputs();\n  psbtFromNetwork.finalizeAllInputs();\n  return psbtFromNetwork.extractTransaction();\n}\n\nexport const replayProtectionKeyPair = getKey('replay-protection');\nconst replayProtectionScriptPubKey = createOutputScriptP2shP2pk(replayProtectionKeyPair.publicKey).scriptPubKey;\n\nexport function isReplayProtectionUnspent<TNumber extends bigint | number>(\n  u: Unspent<TNumber>,\n  network: Network\n): boolean {\n  return u.address === fromOutputScript(replayProtectionScriptPubKey, network);\n}\n\nexport function mockReplayProtectionUnspent<TNumber extends number | bigint>(\n  network: Network,\n  value: TNumber,\n  { key = replayProtectionKeyPair, vout = 0 }: { key?: BIP32Interface; vout?: number } = {}\n): UnspentWithPrevTx<TNumber> {\n  const outputScript = createOutputScriptP2shP2pk(key.publicKey).scriptPubKey;\n  const prevTransaction = mockPrevTx(vout, outputScript, BigInt(value), network);\n  return { ...fromOutputWithPrevTx(prevTransaction, vout), value };\n}\n\nexport function mockWalletUnspent<TNumber extends number | bigint>(\n  network: Network,\n  value: TNumber,\n  {\n    chain = 0,\n    index = 0,\n    keys = getDefaultWalletKeys(),\n    vout = 0,\n    id,\n  }: { chain?: ChainCode; index?: number; keys?: RootWalletKeys; vout?: number; id?: string } = {}\n): WalletUnspent<TNumber> | NonWitnessWalletUnspent<TNumber> {\n  const derivedKeys = keys.deriveForChainAndIndex(chain, index);\n  const address = fromOutputScript(\n    createOutputScript2of3(derivedKeys.publicKeys, scriptTypeForChain(chain)).scriptPubKey,\n    network\n  );\n  if (id && typeof id === 'string') {\n    return { id, address, chain, index, value };\n  } else {\n    const prevTransaction = mockPrevTx(\n      vout,\n      createOutputScript2of3(derivedKeys.publicKeys, scriptTypeForChain(chain), network).scriptPubKey,\n      BigInt(value),\n      network\n    );\n    const unspent =\n      isSegwit(chain) || getMainnet(network) === networks.zcash\n        ? fromOutput(prevTransaction, vout)\n        : fromOutputWithPrevTx(prevTransaction, vout);\n    return {\n      ...unspent,\n      chain,\n      index,\n      value,\n    };\n  }\n}\n\nexport function mockUnspents<TNumber extends number | bigint>(\n  rootWalletKeys: RootWalletKeys,\n  inputScriptTypes: (InputType | outputScripts.ScriptTypeP2shP2pk)[],\n  testOutputAmount: TNumber,\n  network: Network\n): (Unspent<TNumber> | WalletUnspent<TNumber>)[] {\n  return inputScriptTypes.map((t, i): Unspent<TNumber> => {\n    if (outputScripts.isScriptType2Of3(t)) {\n      return mockWalletUnspent(network, testOutputAmount, {\n        keys: rootWalletKeys,\n        chain: getExternalChainCode(t),\n        vout: i,\n      });\n    } else if (t === outputScripts.scriptTypeP2shP2pk) {\n      return mockReplayProtectionUnspent(network, testOutputAmount, {\n        key: replayProtectionKeyPair,\n        vout: i,\n      });\n    }\n    throw new Error(`invalid input type ${t}`);\n  });\n}\n"]}
|
|
94
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mock.js","sourceRoot":"","sources":["../../../src/testutil/mock.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport { BIP32Interface } from 'bip32';\nimport * as noble from '@noble/secp256k1';\nimport * as utxolib from '..';\nimport { getMainnet, Network, networks } from '../networks';\n\nimport {\n  ChainCode,\n  createPsbtForNetwork,\n  fromOutput,\n  fromOutputWithPrevTx,\n  getExternalChainCode,\n  isSegwit,\n  NonWitnessWalletUnspent,\n  outputScripts,\n  RootWalletKeys,\n  scriptTypeForChain,\n  Unspent,\n  UnspentWithPrevTx,\n  UtxoTransaction,\n  WalletUnspent,\n} from '../bitgo';\nimport { fromOutputScript } from '../address';\nimport { createOutputScript2of3, createOutputScriptP2shP2pk } from '../bitgo/outputScripts';\n\nimport { getDefaultWalletKeys, getKey } from './keys';\n\nexport type InputType = outputScripts.ScriptType2Of3;\n\nexport function mockPrevTx(\n  vout: number,\n  outputScript: Buffer,\n  value: bigint,\n  network: Network\n): UtxoTransaction<bigint> {\n  const psbtFromNetwork = createPsbtForNetwork({ network });\n\n  const keypair = getKey('mock-prev-tx');\n  const pubkey = keypair.publicKey;\n  assert(keypair.privateKey);\n  const payment = utxolib.payments.p2wpkh({ pubkey });\n  const destOutput = payment.output;\n  if (!destOutput) throw new Error('Impossible, payment we just constructed has no output');\n\n  for (let index = 0; index <= vout; index++) {\n    if (index === vout) {\n      psbtFromNetwork.addOutput({ script: outputScript, value });\n    } else {\n      psbtFromNetwork.addOutput({ script: destOutput, value });\n    }\n  }\n  psbtFromNetwork.addInput({\n    hash: Buffer.alloc(32, 0x01),\n    index: 0,\n    witnessUtxo: { script: destOutput, value: value * (BigInt(vout) + BigInt(1)) + BigInt(1000) },\n  });\n  psbtFromNetwork.signInput(0, {\n    publicKey: pubkey,\n    sign: (hash: Buffer, lowR?: boolean) =>\n      Buffer.from(noble.signSync(hash, keypair.privateKey as Buffer, { canonical: !lowR, der: false })),\n  });\n  psbtFromNetwork.validateSignaturesOfAllInputs();\n  psbtFromNetwork.finalizeAllInputs();\n  return psbtFromNetwork.extractTransaction();\n}\n\nexport const replayProtectionKeyPair = getKey('replay-protection');\nconst replayProtectionScriptPubKey = createOutputScriptP2shP2pk(replayProtectionKeyPair.publicKey).scriptPubKey;\n\nexport function isReplayProtectionUnspent<TNumber extends bigint | number>(\n  u: Unspent<TNumber>,\n  network: Network\n): boolean {\n  return u.address === fromOutputScript(replayProtectionScriptPubKey, network);\n}\n\nexport function mockReplayProtectionUnspent<TNumber extends number | bigint>(\n  network: Network,\n  value: TNumber,\n  { key = replayProtectionKeyPair, vout = 0 }: { key?: BIP32Interface; vout?: number } = {}\n): UnspentWithPrevTx<TNumber> {\n  const outputScript = createOutputScriptP2shP2pk(key.publicKey).scriptPubKey;\n  const prevTransaction = mockPrevTx(vout, outputScript, BigInt(value), network);\n  return { ...fromOutputWithPrevTx(prevTransaction, vout), value };\n}\n\nexport function mockWalletUnspent<TNumber extends number | bigint>(\n  network: Network,\n  value: TNumber,\n  {\n    chain = 0,\n    index = 0,\n    keys = getDefaultWalletKeys(),\n    vout = 0,\n    id,\n  }: { chain?: ChainCode; index?: number; keys?: RootWalletKeys; vout?: number; id?: string } = {}\n): WalletUnspent<TNumber> | NonWitnessWalletUnspent<TNumber> {\n  const derivedKeys = keys.deriveForChainAndIndex(chain, index);\n  const address = fromOutputScript(\n    createOutputScript2of3(derivedKeys.publicKeys, scriptTypeForChain(chain)).scriptPubKey,\n    network\n  );\n  if (id && typeof id === 'string') {\n    return { id, address, chain, index, value };\n  } else {\n    const prevTransaction = mockPrevTx(\n      vout,\n      createOutputScript2of3(derivedKeys.publicKeys, scriptTypeForChain(chain), network).scriptPubKey,\n      BigInt(value),\n      network\n    );\n    const unspent =\n      isSegwit(chain) || getMainnet(network) === networks.zcash\n        ? fromOutput(prevTransaction, vout)\n        : fromOutputWithPrevTx(prevTransaction, vout);\n    return {\n      ...unspent,\n      chain,\n      index,\n      value,\n    };\n  }\n}\n\nexport function mockUnspents<TNumber extends number | bigint>(\n  rootWalletKeys: RootWalletKeys,\n  inputScriptTypes: (InputType | outputScripts.ScriptTypeP2shP2pk)[],\n  testOutputAmount: TNumber,\n  network: Network\n): (Unspent<TNumber> | WalletUnspent<TNumber>)[] {\n  return inputScriptTypes.map((t, i): Unspent<TNumber> => {\n    if (outputScripts.isScriptType2Of3(t)) {\n      return mockWalletUnspent(network, testOutputAmount, {\n        keys: rootWalletKeys,\n        chain: getExternalChainCode(t),\n        vout: i,\n      });\n    } else if (t === outputScripts.scriptTypeP2shP2pk) {\n      return mockReplayProtectionUnspent(network, testOutputAmount, {\n        key: replayProtectionKeyPair,\n        vout: i,\n      });\n    }\n    throw new Error(`invalid input type ${t}`);\n  });\n}\n"]}
|
|
@@ -1,30 +1,31 @@
|
|
|
1
1
|
import { ScriptType, ScriptType2Of3 } from '../bitgo/outputScripts';
|
|
2
|
-
import { KeyName, RootWalletKeys, Unspent, UtxoPsbt, UtxoTransaction
|
|
2
|
+
import { KeyName, RootWalletKeys, Unspent, UtxoPsbt, UtxoTransaction } from '../bitgo';
|
|
3
3
|
import { Network } from '../networks';
|
|
4
4
|
/**
|
|
5
5
|
* input script type and value.
|
|
6
6
|
* use p2trMusig2 for p2trMusig2 script path.
|
|
7
7
|
* use taprootKeyPathSpend for p2trMusig2 key path.
|
|
8
8
|
*/
|
|
9
|
-
export
|
|
10
|
-
export
|
|
9
|
+
export type InputScriptType = ScriptType | 'taprootKeyPathSpend';
|
|
10
|
+
export type OutputScriptType = ScriptType2Of3;
|
|
11
11
|
/**
|
|
12
12
|
* input script type and value
|
|
13
13
|
*/
|
|
14
|
-
export
|
|
14
|
+
export type Input = {
|
|
15
15
|
scriptType: InputScriptType;
|
|
16
16
|
value: bigint;
|
|
17
|
-
}
|
|
17
|
+
};
|
|
18
18
|
/**
|
|
19
|
-
*
|
|
20
|
-
* set isInternalAddress=true for internal output address
|
|
19
|
+
* Set isInternalAddress=true for internal output address
|
|
21
20
|
*/
|
|
22
|
-
export
|
|
23
|
-
address?: string;
|
|
24
|
-
scriptType?: OutputScriptType;
|
|
21
|
+
export type Output = {
|
|
25
22
|
value: bigint;
|
|
26
23
|
isInternalAddress?: boolean;
|
|
27
|
-
}
|
|
24
|
+
} & ({
|
|
25
|
+
scriptType: OutputScriptType;
|
|
26
|
+
} | {
|
|
27
|
+
address: string;
|
|
28
|
+
});
|
|
28
29
|
/**
|
|
29
30
|
* array of supported input script types.
|
|
30
31
|
* use p2trMusig2 for p2trMusig2 script path.
|
|
@@ -53,19 +54,40 @@ export declare function getSigners(inputType: InputScriptType): {
|
|
|
53
54
|
* signs with first or second signature for single input.
|
|
54
55
|
* p2shP2pk is signed only with first sign.
|
|
55
56
|
*/
|
|
56
|
-
export declare function signPsbtInput(psbt: UtxoPsbt, input: Input, inputIndex: number, rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned'
|
|
57
|
+
export declare function signPsbtInput(psbt: UtxoPsbt, input: Input, inputIndex: number, rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', params?: {
|
|
58
|
+
signers?: {
|
|
59
|
+
signerName: KeyName;
|
|
60
|
+
cosignerName?: KeyName;
|
|
61
|
+
};
|
|
62
|
+
deterministic?: boolean;
|
|
63
|
+
skipNonWitnessUtxo?: boolean;
|
|
64
|
+
}): void;
|
|
57
65
|
/**
|
|
58
66
|
* signs with first or second signature for all inputs.
|
|
59
67
|
* p2shP2pk is signed only with first sign.
|
|
60
68
|
*/
|
|
61
|
-
export declare function signAllPsbtInputs(psbt: UtxoPsbt, inputs: Input[], rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned'
|
|
69
|
+
export declare function signAllPsbtInputs(psbt: UtxoPsbt, inputs: Input[], rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', params?: {
|
|
70
|
+
signers?: {
|
|
71
|
+
signerName: KeyName;
|
|
72
|
+
cosignerName?: KeyName;
|
|
73
|
+
};
|
|
74
|
+
deterministic?: boolean;
|
|
75
|
+
skipNonWitnessUtxo?: boolean;
|
|
76
|
+
}): void;
|
|
62
77
|
/**
|
|
63
78
|
* construct psbt for given inputs, outputs, network and root wallet keys.
|
|
64
79
|
*/
|
|
65
|
-
export declare function constructPsbt(inputs: Input[], outputs: Output[], network: Network, rootWalletKeys: RootWalletKeys, sign: 'unsigned' | 'halfsigned' | 'fullsigned'
|
|
80
|
+
export declare function constructPsbt(inputs: Input[], outputs: Output[], network: Network, rootWalletKeys: RootWalletKeys, sign: 'unsigned' | 'halfsigned' | 'fullsigned', params?: {
|
|
81
|
+
signers?: {
|
|
82
|
+
signerName: KeyName;
|
|
83
|
+
cosignerName?: KeyName;
|
|
84
|
+
};
|
|
85
|
+
deterministic?: boolean;
|
|
86
|
+
skipNonWitnessUtxo?: boolean;
|
|
87
|
+
}): UtxoPsbt;
|
|
66
88
|
/**
|
|
67
89
|
* Verifies signatures of fully signed tx (with taproot key path support).
|
|
68
90
|
* NOTE: taproot key path tx can only be built and signed with PSBT.
|
|
69
91
|
*/
|
|
70
|
-
export declare function verifyFullySignedSignatures(tx: UtxoTransaction<bigint>, unspents:
|
|
92
|
+
export declare function verifyFullySignedSignatures(tx: UtxoTransaction<bigint>, unspents: Unspent<bigint>[], walletKeys: RootWalletKeys, signer: KeyName, cosigner: KeyName): boolean;
|
|
71
93
|
//# sourceMappingURL=psbt.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../src/testutil/psbt.ts"],"names":[],"mappings":"AAEA,OAAO,EAEL,UAAU,EACV,cAAc,EAGf,MAAM,wBAAwB,CAAC;AAChC,OAAO,
|
|
1
|
+
{"version":3,"file":"psbt.d.ts","sourceRoot":"","sources":["../../../src/testutil/psbt.ts"],"names":[],"mappings":"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"}
|
|
@@ -21,11 +21,11 @@ exports.outputScriptTypes = outputScripts_1.scriptTypes2Of3;
|
|
|
21
21
|
*/
|
|
22
22
|
function toUnspent(input, index, network, rootWalletKeys) {
|
|
23
23
|
if (input.scriptType === 'p2shP2pk') {
|
|
24
|
-
return mock_1.mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });
|
|
24
|
+
return (0, mock_1.mockReplayProtectionUnspent)(network, input.value, { key: rootWalletKeys['user'], vout: index });
|
|
25
25
|
}
|
|
26
26
|
else {
|
|
27
|
-
const chain = bitgo_1.getInternalChainCode(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);
|
|
28
|
-
return mock_1.mockWalletUnspent(network, input.value, {
|
|
27
|
+
const chain = (0, bitgo_1.getInternalChainCode)(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);
|
|
28
|
+
return (0, mock_1.mockWalletUnspent)(network, input.value, {
|
|
29
29
|
chain,
|
|
30
30
|
vout: index,
|
|
31
31
|
keys: rootWalletKeys,
|
|
@@ -51,18 +51,27 @@ exports.getSigners = getSigners;
|
|
|
51
51
|
* signs with first or second signature for single input.
|
|
52
52
|
* p2shP2pk is signed only with first sign.
|
|
53
53
|
*/
|
|
54
|
-
function signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign) {
|
|
55
|
-
|
|
54
|
+
function signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, params) {
|
|
55
|
+
function signPsbt(psbt, signFunc, skipNonWitnessUtxo) {
|
|
56
|
+
if (skipNonWitnessUtxo) {
|
|
57
|
+
(0, bitgo_1.withUnsafeNonSegwit)(psbt, signFunc);
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
signFunc();
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
const { signers, deterministic, skipNonWitnessUtxo } = params !== null && params !== void 0 ? params : {};
|
|
64
|
+
const { signerName, cosignerName } = signers ? signers : getSigners(input.scriptType);
|
|
56
65
|
if (sign === 'halfsigned') {
|
|
57
66
|
if (input.scriptType === 'p2shP2pk') {
|
|
58
|
-
psbt.signInput(inputIndex, rootWalletKeys[signerName]);
|
|
67
|
+
signPsbt(psbt, () => psbt.signInput(inputIndex, rootWalletKeys[signerName]), skipNonWitnessUtxo);
|
|
59
68
|
}
|
|
60
69
|
else {
|
|
61
|
-
psbt.signInputHD(inputIndex, rootWalletKeys[signerName]);
|
|
70
|
+
signPsbt(psbt, () => psbt.signInputHD(inputIndex, rootWalletKeys[signerName]), skipNonWitnessUtxo);
|
|
62
71
|
}
|
|
63
72
|
}
|
|
64
|
-
if (sign === 'fullsigned' && cosignerName) {
|
|
65
|
-
psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName]);
|
|
73
|
+
if (sign === 'fullsigned' && cosignerName && input.scriptType !== 'p2shP2pk') {
|
|
74
|
+
signPsbt(psbt, () => psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName], { deterministic }), skipNonWitnessUtxo);
|
|
66
75
|
}
|
|
67
76
|
}
|
|
68
77
|
exports.signPsbtInput = signPsbtInput;
|
|
@@ -70,50 +79,51 @@ exports.signPsbtInput = signPsbtInput;
|
|
|
70
79
|
* signs with first or second signature for all inputs.
|
|
71
80
|
* p2shP2pk is signed only with first sign.
|
|
72
81
|
*/
|
|
73
|
-
function signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign) {
|
|
74
|
-
|
|
75
|
-
|
|
82
|
+
function signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, params) {
|
|
83
|
+
const { signers, deterministic, skipNonWitnessUtxo } = params !== null && params !== void 0 ? params : {};
|
|
84
|
+
inputs.forEach((input, inputIndex) => {
|
|
85
|
+
signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, { signers, deterministic, skipNonWitnessUtxo });
|
|
76
86
|
});
|
|
77
87
|
}
|
|
78
88
|
exports.signAllPsbtInputs = signAllPsbtInputs;
|
|
79
89
|
/**
|
|
80
90
|
* construct psbt for given inputs, outputs, network and root wallet keys.
|
|
81
91
|
*/
|
|
82
|
-
function constructPsbt(inputs, outputs, network, rootWalletKeys, sign) {
|
|
92
|
+
function constructPsbt(inputs, outputs, network, rootWalletKeys, sign, params) {
|
|
93
|
+
const { signers, deterministic, skipNonWitnessUtxo } = params !== null && params !== void 0 ? params : {};
|
|
83
94
|
const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));
|
|
84
95
|
const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));
|
|
85
96
|
assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');
|
|
86
|
-
|
|
87
|
-
const psbt = bitgo_1.createPsbtForNetwork({ network });
|
|
97
|
+
const psbt = (0, bitgo_1.createPsbtForNetwork)({ network });
|
|
88
98
|
const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys));
|
|
89
99
|
unspents.forEach((u, i) => {
|
|
90
|
-
const { signerName, cosignerName } = getSigners(inputs[i].scriptType);
|
|
91
|
-
if (bitgo_1.isWalletUnspent(u) && cosignerName) {
|
|
92
|
-
bitgo_1.addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signerName, cosignerName);
|
|
100
|
+
const { signerName, cosignerName } = signers ? signers : getSigners(inputs[i].scriptType);
|
|
101
|
+
if ((0, bitgo_1.isWalletUnspent)(u) && cosignerName) {
|
|
102
|
+
(0, bitgo_1.addWalletUnspentToPsbt)(psbt, u, rootWalletKeys, signerName, cosignerName, { skipNonWitnessUtxo });
|
|
93
103
|
}
|
|
94
104
|
else {
|
|
95
|
-
const { redeemScript } = outputScripts_1.createOutputScriptP2shP2pk(rootWalletKeys[signerName].publicKey);
|
|
105
|
+
const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(rootWalletKeys[signerName].publicKey);
|
|
96
106
|
assert(redeemScript);
|
|
97
|
-
bitgo_1.addReplayProtectionUnspentToPsbt(psbt, u, redeemScript);
|
|
107
|
+
(0, bitgo_1.addReplayProtectionUnspentToPsbt)(psbt, u, redeemScript, { skipNonWitnessUtxo });
|
|
98
108
|
}
|
|
99
109
|
});
|
|
100
110
|
outputs.forEach((output, i) => {
|
|
101
|
-
if (output
|
|
102
|
-
bitgo_1.addWalletOutputToPsbt(psbt, rootWalletKeys, output.isInternalAddress ? bitgo_1.getInternalChainCode(output.scriptType) : bitgo_1.getExternalChainCode(output.scriptType), i, output.value);
|
|
111
|
+
if ('scriptType' in output) {
|
|
112
|
+
(0, bitgo_1.addWalletOutputToPsbt)(psbt, rootWalletKeys, output.isInternalAddress ? (0, bitgo_1.getInternalChainCode)(output.scriptType) : (0, bitgo_1.getExternalChainCode)(output.scriptType), i, output.value);
|
|
103
113
|
}
|
|
104
114
|
else if (output.address) {
|
|
105
115
|
const { address, value } = output;
|
|
106
|
-
psbt.addOutput({ script: address_1.toOutputScript(address, network), value });
|
|
116
|
+
psbt.addOutput({ script: (0, address_1.toOutputScript)(address, network), value });
|
|
107
117
|
}
|
|
108
118
|
});
|
|
109
119
|
if (sign === 'unsigned') {
|
|
110
120
|
return psbt;
|
|
111
121
|
}
|
|
112
122
|
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);
|
|
113
|
-
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo']);
|
|
114
|
-
signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned');
|
|
123
|
+
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic });
|
|
124
|
+
signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', { signers, skipNonWitnessUtxo });
|
|
115
125
|
if (sign === 'fullsigned') {
|
|
116
|
-
signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign);
|
|
126
|
+
signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, { signers, deterministic, skipNonWitnessUtxo });
|
|
117
127
|
}
|
|
118
128
|
return psbt;
|
|
119
129
|
}
|
|
@@ -123,14 +133,14 @@ exports.constructPsbt = constructPsbt;
|
|
|
123
133
|
* NOTE: taproot key path tx can only be built and signed with PSBT.
|
|
124
134
|
*/
|
|
125
135
|
function verifyFullySignedSignatures(tx, unspents, walletKeys, signer, cosigner) {
|
|
126
|
-
const prevOutputs = unspents.map((u) => bitgo_1.toOutput(u, tx.network));
|
|
136
|
+
const prevOutputs = unspents.map((u) => (0, bitgo_1.toOutput)(u, tx.network));
|
|
127
137
|
return unspents.every((u, index) => {
|
|
128
|
-
if (bitgo_1.parseSignatureScript2Of3(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {
|
|
129
|
-
const result = bitgo_1.getSignatureVerifications(tx, index, u.value, undefined, prevOutputs);
|
|
138
|
+
if ((0, bitgo_1.parseSignatureScript2Of3)(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {
|
|
139
|
+
const result = (0, bitgo_1.getSignatureVerifications)(tx, index, u.value, undefined, prevOutputs);
|
|
130
140
|
return result.length === 1 && result[0].signature;
|
|
131
141
|
}
|
|
132
142
|
else {
|
|
133
|
-
const result = bitgo_1.verifySignatureWithUnspent(tx, index, unspents, walletKeys);
|
|
143
|
+
const result = (0, bitgo_1.verifySignatureWithUnspent)(tx, index, unspents, walletKeys);
|
|
134
144
|
if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {
|
|
135
145
|
return result[0] && !result[1] && result[2];
|
|
136
146
|
}
|
|
@@ -144,4 +154,4 @@ function verifyFullySignedSignatures(tx, unspents, walletKeys, signer, cosigner)
|
|
|
144
154
|
});
|
|
145
155
|
}
|
|
146
156
|
exports.verifyFullySignedSignatures = verifyFullySignedSignatures;
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../src/testutil/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport {\n  createOutputScriptP2shP2pk,\n  ScriptType,\n  ScriptType2Of3,\n  scriptTypeP2shP2pk,\n  scriptTypes2Of3,\n} from '../bitgo/outputScripts';\nimport {\n  addReplayProtectionUnspentToPsbt,\n  addWalletOutputToPsbt,\n  addWalletUnspentToPsbt,\n  createPsbtForNetwork,\n  getExternalChainCode,\n  getSignatureVerifications,\n  isWalletUnspent,\n  KeyName,\n  parseSignatureScript2Of3,\n  getInternalChainCode,\n  RootWalletKeys,\n  toOutput,\n  Unspent,\n  UtxoPsbt,\n  UtxoTransaction,\n  verifySignatureWithUnspent,\n  WalletUnspent,\n} from '../bitgo';\nimport { Network } from '../networks';\nimport { mockReplayProtectionUnspent, mockWalletUnspent } from './mock';\nimport { toOutputScript } from '../address';\n\n/**\n * input script type and value.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport type InputScriptType = ScriptType | 'taprootKeyPathSpend';\nexport type OutputScriptType = ScriptType2Of3;\n\n/**\n * input script type and value\n */\nexport interface Input {\n  scriptType: InputScriptType;\n  value: bigint;\n}\n\n/**\n * should set either address or scriptType, never both.\n * set isInternalAddress=true for internal output address\n */\nexport interface Output {\n  address?: string;\n  scriptType?: OutputScriptType;\n  value: bigint;\n  isInternalAddress?: boolean;\n}\n\n/**\n * array of supported input script types.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport const inputScriptTypes = [...scriptTypes2Of3, 'taprootKeyPathSpend', scriptTypeP2shP2pk] as const;\n\n/**\n * array of supported output script types.\n */\nexport const outputScriptTypes = scriptTypes2Of3;\n\n/**\n * create unspent object from input script type, index, network and root wallet key.\n */\nexport function toUnspent(\n  input: Input,\n  index: number,\n  network: Network,\n  rootWalletKeys: RootWalletKeys\n): Unspent<bigint> {\n  if (input.scriptType === 'p2shP2pk') {\n    return mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });\n  } else {\n    const chain = getInternalChainCode(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);\n    return mockWalletUnspent(network, input.value, {\n      chain,\n      vout: index,\n      keys: rootWalletKeys,\n      index,\n    });\n  }\n}\n\n/**\n * returns signer and cosigner names for InputScriptType.\n * user and undefined as signer and cosigner respectively for p2shP2pk.\n * user and backup as signer and cosigner respectively for p2trMusig2.\n * user and bitgo as signer and cosigner respectively for other input script types.\n */\nexport function getSigners(inputType: InputScriptType): { signerName: KeyName; cosignerName?: KeyName } {\n  return {\n    signerName: 'user',\n    cosignerName: inputType === 'p2shP2pk' ? undefined : inputType === 'p2trMusig2' ? 'backup' : 'bitgo',\n  };\n}\n\n/**\n * signs with first or second signature for single input.\n * p2shP2pk is signed only with first sign.\n */\nexport function signPsbtInput(\n  psbt: UtxoPsbt,\n  input: Input,\n  inputIndex: number,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned'\n): void {\n  const { signerName, cosignerName } = getSigners(input.scriptType);\n  if (sign === 'halfsigned') {\n    if (input.scriptType === 'p2shP2pk') {\n      psbt.signInput(inputIndex, rootWalletKeys[signerName]);\n    } else {\n      psbt.signInputHD(inputIndex, rootWalletKeys[signerName]);\n    }\n  }\n  if (sign === 'fullsigned' && cosignerName) {\n    psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName]);\n  }\n}\n\n/**\n * signs with first or second signature for all inputs.\n * p2shP2pk is signed only with first sign.\n */\nexport function signAllPsbtInputs(\n  psbt: UtxoPsbt,\n  inputs: Input[],\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned'\n): void {\n  inputs.forEach((input, index) => {\n    signPsbtInput(psbt, input, index, rootWalletKeys, sign);\n  });\n}\n\n/**\n * construct psbt for given inputs, outputs, network and root wallet keys.\n */\nexport function constructPsbt(\n  inputs: Input[],\n  outputs: Output[],\n  network: Network,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'unsigned' | 'halfsigned' | 'fullsigned'\n): UtxoPsbt {\n  const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));\n  const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));\n  assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');\n  assert(\n    !outputs.some((o) => (o.scriptType && o.address) || (!o.scriptType && !o.address)),\n    'only either output script type or address should be provided'\n  );\n\n  const psbt = createPsbtForNetwork({ network });\n  const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys));\n\n  unspents.forEach((u, i) => {\n    const { signerName, cosignerName } = getSigners(inputs[i].scriptType);\n    if (isWalletUnspent(u) && cosignerName) {\n      addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signerName, cosignerName);\n    } else {\n      const { redeemScript } = createOutputScriptP2shP2pk(rootWalletKeys[signerName].publicKey);\n      assert(redeemScript);\n      addReplayProtectionUnspentToPsbt(psbt, u, redeemScript);\n    }\n  });\n\n  outputs.forEach((output, i) => {\n    if (output.scriptType) {\n      addWalletOutputToPsbt(\n        psbt,\n        rootWalletKeys,\n        output.isInternalAddress ? getInternalChainCode(output.scriptType) : getExternalChainCode(output.scriptType),\n        i,\n        output.value\n      );\n    } else if (output.address) {\n      const { address, value } = output;\n      psbt.addOutput({ script: toOutputScript(address, network), value });\n    }\n  });\n\n  if (sign === 'unsigned') {\n    return psbt;\n  }\n\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo']);\n\n  signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned');\n\n  if (sign === 'fullsigned') {\n    signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign);\n  }\n\n  return psbt;\n}\n\n/**\n * Verifies signatures of fully signed tx (with taproot key path support).\n * NOTE: taproot key path tx can only be built and signed with PSBT.\n */\nexport function verifyFullySignedSignatures(\n  tx: UtxoTransaction<bigint>,\n  unspents: WalletUnspent<bigint>[],\n  walletKeys: RootWalletKeys,\n  signer: KeyName,\n  cosigner: KeyName\n): boolean {\n  const prevOutputs = unspents.map((u) => toOutput(u, tx.network));\n  return unspents.every((u, index) => {\n    if (parseSignatureScript2Of3(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {\n      const result = getSignatureVerifications(tx, index, u.value, undefined, prevOutputs);\n      return result.length === 1 && result[0].signature;\n    } else {\n      const result = verifySignatureWithUnspent(tx, index, unspents, walletKeys);\n      if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {\n        return result[0] && !result[1] && result[2];\n      } else if ((signer === 'user' && cosigner === 'backup') || (signer === 'backup' && cosigner === 'user')) {\n        return result[0] && result[1] && !result[2];\n      } else {\n        return !result[0] && result[1] && result[2];\n      }\n    }\n  });\n}\n"]}
|
|
157
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../src/testutil/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport {\n  createOutputScriptP2shP2pk,\n  ScriptType,\n  ScriptType2Of3,\n  scriptTypeP2shP2pk,\n  scriptTypes2Of3,\n} from '../bitgo/outputScripts';\nimport {\n  addReplayProtectionUnspentToPsbt,\n  addWalletOutputToPsbt,\n  addWalletUnspentToPsbt,\n  createPsbtForNetwork,\n  getExternalChainCode,\n  getInternalChainCode,\n  getSignatureVerifications,\n  isWalletUnspent,\n  KeyName,\n  parseSignatureScript2Of3,\n  RootWalletKeys,\n  toOutput,\n  Unspent,\n  UtxoPsbt,\n  UtxoTransaction,\n  verifySignatureWithUnspent,\n  withUnsafeNonSegwit,\n} from '../bitgo';\nimport { Network } from '../networks';\nimport { mockReplayProtectionUnspent, mockWalletUnspent } from './mock';\nimport { toOutputScript } from '../address';\n\n/**\n * input script type and value.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport type InputScriptType = ScriptType | 'taprootKeyPathSpend';\nexport type OutputScriptType = ScriptType2Of3;\n\n/**\n * input script type and value\n */\nexport type Input = {\n  scriptType: InputScriptType;\n  value: bigint;\n};\n\n/**\n * Set isInternalAddress=true for internal output address\n */\nexport type Output = {\n  value: bigint;\n  isInternalAddress?: boolean;\n} & ({ scriptType: OutputScriptType } | { address: string });\n\n/**\n * array of supported input script types.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport const inputScriptTypes = [...scriptTypes2Of3, 'taprootKeyPathSpend', scriptTypeP2shP2pk] as const;\n\n/**\n * array of supported output script types.\n */\nexport const outputScriptTypes = scriptTypes2Of3;\n\n/**\n * create unspent object from input script type, index, network and root wallet key.\n */\nexport function toUnspent(\n  input: Input,\n  index: number,\n  network: Network,\n  rootWalletKeys: RootWalletKeys\n): Unspent<bigint> {\n  if (input.scriptType === 'p2shP2pk') {\n    return mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });\n  } else {\n    const chain = getInternalChainCode(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);\n    return mockWalletUnspent(network, input.value, {\n      chain,\n      vout: index,\n      keys: rootWalletKeys,\n      index,\n    });\n  }\n}\n\n/**\n * returns signer and cosigner names for InputScriptType.\n * user and undefined as signer and cosigner respectively for p2shP2pk.\n * user and backup as signer and cosigner respectively for p2trMusig2.\n * user and bitgo as signer and cosigner respectively for other input script types.\n */\nexport function getSigners(inputType: InputScriptType): { signerName: KeyName; cosignerName?: KeyName } {\n  return {\n    signerName: 'user',\n    cosignerName: inputType === 'p2shP2pk' ? undefined : inputType === 'p2trMusig2' ? 'backup' : 'bitgo',\n  };\n}\n\n/**\n * signs with first or second signature for single input.\n * p2shP2pk is signed only with first sign.\n */\nexport function signPsbtInput(\n  psbt: UtxoPsbt,\n  input: Input,\n  inputIndex: number,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n    skipNonWitnessUtxo?: boolean;\n  }\n): void {\n  function signPsbt(psbt: UtxoPsbt, signFunc: () => void, skipNonWitnessUtxo?: boolean) {\n    if (skipNonWitnessUtxo) {\n      withUnsafeNonSegwit(psbt, signFunc);\n    } else {\n      signFunc();\n    }\n  }\n\n  const { signers, deterministic, skipNonWitnessUtxo } = params ?? {};\n  const { signerName, cosignerName } = signers ? signers : getSigners(input.scriptType);\n  if (sign === 'halfsigned') {\n    if (input.scriptType === 'p2shP2pk') {\n      signPsbt(psbt, () => psbt.signInput(inputIndex, rootWalletKeys[signerName]), skipNonWitnessUtxo);\n    } else {\n      signPsbt(psbt, () => psbt.signInputHD(inputIndex, rootWalletKeys[signerName]), skipNonWitnessUtxo);\n    }\n  }\n  if (sign === 'fullsigned' && cosignerName && input.scriptType !== 'p2shP2pk') {\n    signPsbt(\n      psbt,\n      () => psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName], { deterministic }),\n      skipNonWitnessUtxo\n    );\n  }\n}\n\n/**\n * signs with first or second signature for all inputs.\n * p2shP2pk is signed only with first sign.\n */\nexport function signAllPsbtInputs(\n  psbt: UtxoPsbt,\n  inputs: Input[],\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n    skipNonWitnessUtxo?: boolean;\n  }\n): void {\n  const { signers, deterministic, skipNonWitnessUtxo } = params ?? {};\n  inputs.forEach((input, inputIndex) => {\n    signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, { signers, deterministic, skipNonWitnessUtxo });\n  });\n}\n\n/**\n * construct psbt for given inputs, outputs, network and root wallet keys.\n */\nexport function constructPsbt(\n  inputs: Input[],\n  outputs: Output[],\n  network: Network,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'unsigned' | 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n    skipNonWitnessUtxo?: boolean;\n  }\n): UtxoPsbt {\n  const { signers, deterministic, skipNonWitnessUtxo } = params ?? {};\n  const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));\n  const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));\n  assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');\n\n  const psbt = createPsbtForNetwork({ network });\n  const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys));\n\n  unspents.forEach((u, i) => {\n    const { signerName, cosignerName } = signers ? signers : getSigners(inputs[i].scriptType);\n    if (isWalletUnspent(u) && cosignerName) {\n      addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signerName, cosignerName, { skipNonWitnessUtxo });\n    } else {\n      const { redeemScript } = createOutputScriptP2shP2pk(rootWalletKeys[signerName].publicKey);\n      assert(redeemScript);\n      addReplayProtectionUnspentToPsbt(psbt, u, redeemScript, { skipNonWitnessUtxo });\n    }\n  });\n\n  outputs.forEach((output, i) => {\n    if ('scriptType' in output) {\n      addWalletOutputToPsbt(\n        psbt,\n        rootWalletKeys,\n        output.isInternalAddress ? getInternalChainCode(output.scriptType) : getExternalChainCode(output.scriptType),\n        i,\n        output.value\n      );\n    } else if (output.address) {\n      const { address, value } = output;\n      psbt.addOutput({ script: toOutputScript(address, network), value });\n    }\n  });\n\n  if (sign === 'unsigned') {\n    return psbt;\n  }\n\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic });\n\n  signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', { signers, skipNonWitnessUtxo });\n\n  if (sign === 'fullsigned') {\n    signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, { signers, deterministic, skipNonWitnessUtxo });\n  }\n\n  return psbt;\n}\n\n/**\n * Verifies signatures of fully signed tx (with taproot key path support).\n * NOTE: taproot key path tx can only be built and signed with PSBT.\n */\nexport function verifyFullySignedSignatures(\n  tx: UtxoTransaction<bigint>,\n  unspents: Unspent<bigint>[],\n  walletKeys: RootWalletKeys,\n  signer: KeyName,\n  cosigner: KeyName\n): boolean {\n  const prevOutputs = unspents.map((u) => toOutput(u, tx.network));\n  return unspents.every((u, index) => {\n    if (parseSignatureScript2Of3(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {\n      const result = getSignatureVerifications(tx, index, u.value, undefined, prevOutputs);\n      return result.length === 1 && result[0].signature;\n    } else {\n      const result = verifySignatureWithUnspent(tx, index, unspents, walletKeys);\n      if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {\n        return result[0] && !result[1] && result[2];\n      } else if ((signer === 'user' && cosigner === 'backup') || (signer === 'backup' && cosigner === 'user')) {\n        return result[0] && result[1] && !result[2];\n      } else {\n        return !result[0] && result[1] && result[2];\n      }\n    }\n  });\n}\n"]}
|
|
@@ -4,8 +4,8 @@ import { Network } from '../networks';
|
|
|
4
4
|
/**
|
|
5
5
|
* input script type and value.
|
|
6
6
|
*/
|
|
7
|
-
export
|
|
8
|
-
export
|
|
7
|
+
export type TxnInputScriptType = Exclude<ScriptType, 'p2trMusig2'>;
|
|
8
|
+
export type TxnOutputScriptType = ScriptType2Of3;
|
|
9
9
|
/**
|
|
10
10
|
* output script type and value
|
|
11
11
|
*/
|
|
@@ -30,7 +30,7 @@ export declare const txnInputScriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh"
|
|
|
30
30
|
/**
|
|
31
31
|
* array of supported output script types.
|
|
32
32
|
*/
|
|
33
|
-
export declare const txnOutputScriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh", "p2tr"];
|
|
33
|
+
export declare const txnOutputScriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh", "p2tr", "p2trMusig2"];
|
|
34
34
|
/**
|
|
35
35
|
* create unspent object from input script type, index, network and root wallet key.
|
|
36
36
|
*/
|
|
@@ -48,14 +48,23 @@ export declare function getTxnSigners(inputType: TxnInputScriptType): {
|
|
|
48
48
|
* signs with first or second signature for single input.
|
|
49
49
|
* p2shP2pk is signed only with first sign.
|
|
50
50
|
*/
|
|
51
|
-
export declare function signTxnInput<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, input: TxnInput<TNumber>, inputIndex: number, rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned'
|
|
51
|
+
export declare function signTxnInput<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, input: TxnInput<TNumber>, inputIndex: number, rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', signers?: {
|
|
52
|
+
signerName: KeyName;
|
|
53
|
+
cosignerName?: KeyName;
|
|
54
|
+
}): void;
|
|
52
55
|
/**
|
|
53
56
|
* signs with first or second signature for all inputs.
|
|
54
57
|
* p2shP2pk is signed only with first sign.
|
|
55
58
|
*/
|
|
56
|
-
export declare function signAllTxnInputs<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, inputs: TxnInput<TNumber>[], rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned'
|
|
59
|
+
export declare function signAllTxnInputs<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, inputs: TxnInput<TNumber>[], rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', signers?: {
|
|
60
|
+
signerName: KeyName;
|
|
61
|
+
cosignerName?: KeyName;
|
|
62
|
+
}): void;
|
|
57
63
|
/**
|
|
58
64
|
* construct transaction for given inputs, outputs, network and root wallet keys.
|
|
59
65
|
*/
|
|
60
|
-
export declare function constructTxnBuilder<TNumber extends number | bigint>(inputs: TxnInput<TNumber>[], outputs: TxnOutput<TNumber>[], network: Network, rootWalletKeys: RootWalletKeys, sign: 'unsigned' | 'halfsigned' | 'fullsigned'
|
|
66
|
+
export declare function constructTxnBuilder<TNumber extends number | bigint>(inputs: TxnInput<TNumber>[], outputs: TxnOutput<TNumber>[], network: Network, rootWalletKeys: RootWalletKeys, sign: 'unsigned' | 'halfsigned' | 'fullsigned', signers?: {
|
|
67
|
+
signerName: KeyName;
|
|
68
|
+
cosignerName?: KeyName;
|
|
69
|
+
}): UtxoTransactionBuilder<TNumber>;
|
|
61
70
|
//# sourceMappingURL=transaction.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/testutil/transaction.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,cAAc,
|
|
1
|
+
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/testutil/transaction.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,cAAc,EAAuC,MAAM,wBAAwB,CAAC;AACzG,OAAO,EAGL,OAAO,EAEP,cAAc,EACd,OAAO,EACP,sBAAsB,EAOvB,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,OAAO,EAAE,MAAM,aAAa,CAAC;AAGtC;;GAEG;AACH,MAAM,MAAM,kBAAkB,GAAG,OAAO,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;AACnE,MAAM,MAAM,mBAAmB,GAAG,cAAc,CAAC;AAEjD;;GAEG;AACH,MAAM,WAAW,QAAQ,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM;IACvD,UAAU,EAAE,kBAAkB,CAAC;IAC/B,KAAK,EAAE,OAAO,CAAC;CAChB;AAED;;;GAGG;AACH,MAAM,WAAW,SAAS,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM;IACxD,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,mBAAmB,CAAC;IACjC,KAAK,EAAE,OAAO,CAAC;IACf,iBAAiB,CAAC,EAAE,OAAO,CAAC;CAC7B;AAED;;GAEG;AACH,eAAO,MAAM,mBAAmB,6DAAsE,CAAC;AAEvG;;GAEG;AACH,eAAO,MAAM,oBAAoB,+DAAkB,CAAC;AAEpD;;GAEG;AACH,wBAAgB,YAAY,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,EACxB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,OAAO,EAChB,cAAc,EAAE,cAAc,GAC7B,OAAO,CAAC,OAAO,CAAC,CAWlB;AAED;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,SAAS,EAAE,kBAAkB,GAAG;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,CAK5G;AAED;;;GAGG;AACH,wBAAgB,YAAY,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,GAAG,EAAE,sBAAsB,CAAC,OAAO,CAAC,EACpC,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,EACxB,UAAU,EAAE,MAAM,EAClB,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,YAAY,GAAG,YAAY,EACjC,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,IAAI,CAuBN;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC9D,GAAG,EAAE,sBAAsB,CAAC,OAAO,CAAC,EACpC,MAAM,EAAE,QAAQ,CAAC,OAAO,CAAC,EAAE,EAC3B,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,YAAY,GAAG,YAAY,EACjC,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,IAAI,CAIN;AAED;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACjE,MAAM,EAAE,QAAQ,CAAC,OAAO,CAAC,EAAE,EAC3B,OAAO,EAAE,SAAS,CAAC,OAAO,CAAC,EAAE,EAC7B,OAAO,EAAE,OAAO,EAChB,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,UAAU,GAAG,YAAY,GAAG,YAAY,EAC9C,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,sBAAsB,CAAC,OAAO,CAAC,CA2CjC"}
|