utxo-lib 1.0.9 → 1.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +19 -16
- package/dist/src/address.d.ts.map +1 -1
- package/dist/src/address.js +11 -1
- package/dist/src/addressFormat.d.ts +1 -1
- package/dist/src/addressFormat.d.ts.map +1 -1
- package/dist/src/addressFormat.js +1 -1
- package/dist/src/bitgo/Musig2.d.ts +115 -17
- package/dist/src/bitgo/Musig2.d.ts.map +1 -1
- package/dist/src/bitgo/Musig2.js +283 -100
- package/dist/src/bitgo/PsbtUtil.d.ts +59 -0
- package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -0
- package/dist/src/bitgo/PsbtUtil.js +91 -0
- package/dist/src/bitgo/UtxoPsbt.d.ts +180 -47
- package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoPsbt.js +651 -107
- package/dist/src/bitgo/bitcoincash/address.js +2 -2
- package/dist/src/bitgo/index.d.ts +11 -0
- package/dist/src/bitgo/index.d.ts.map +1 -1
- package/dist/src/bitgo/index.js +6 -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/LitecoinPsbt.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.js +17 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts +16 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.js +46 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.js +15 -0
- package/dist/src/bitgo/litecoin/index.d.ts +4 -0
- package/dist/src/bitgo/litecoin/index.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/index.js +16 -0
- package/dist/src/bitgo/outputScripts.d.ts +3 -1
- package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
- package/dist/src/bitgo/outputScripts.js +18 -10
- package/dist/src/bitgo/parseInput.d.ts +49 -20
- package/dist/src/bitgo/parseInput.d.ts.map +1 -1
- package/dist/src/bitgo/parseInput.js +109 -25
- package/dist/src/bitgo/psbt/fromHalfSigned.d.ts.map +1 -1
- package/dist/src/bitgo/psbt/fromHalfSigned.js +9 -6
- package/dist/src/bitgo/signature.d.ts +3 -3
- package/dist/src/bitgo/signature.d.ts.map +1 -1
- package/dist/src/bitgo/signature.js +48 -16
- package/dist/src/bitgo/transaction.d.ts +18 -3
- package/dist/src/bitgo/transaction.d.ts.map +1 -1
- package/dist/src/bitgo/transaction.js +28 -15
- package/dist/src/bitgo/types.d.ts +2 -0
- package/dist/src/bitgo/types.d.ts.map +1 -1
- package/dist/src/bitgo/types.js +1 -1
- package/dist/src/bitgo/wallet/Psbt.d.ts +104 -12
- package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Psbt.js +285 -70
- package/dist/src/bitgo/wallet/Unspent.d.ts +28 -0
- package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Unspent.js +172 -68
- 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 +64 -23
- 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/chains.js +1 -1
- 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 +6 -14
- package/dist/src/musig.d.ts +0 -1
- package/dist/src/musig.d.ts.map +1 -1
- package/dist/src/musig.js +15 -29
- package/dist/src/networks.d.ts +1 -2
- package/dist/src/networks.d.ts.map +1 -1
- package/dist/src/networks.js +22 -29
- package/dist/src/noble_ecc.d.ts.map +1 -1
- package/dist/src/noble_ecc.js +7 -3
- package/dist/src/payments/p2tr.d.ts.map +1 -1
- package/dist/src/payments/p2tr.js +15 -9
- package/dist/src/taproot.d.ts +16 -0
- package/dist/src/taproot.d.ts.map +1 -1
- package/dist/src/taproot.js +44 -2
- package/dist/src/testutil/index.d.ts +2 -0
- package/dist/src/testutil/index.d.ts.map +1 -1
- package/dist/src/testutil/index.js +3 -1
- package/dist/src/testutil/keys.d.ts +3 -0
- package/dist/src/testutil/keys.d.ts.map +1 -1
- package/dist/src/testutil/keys.js +17 -2
- 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 +12 -4
- package/dist/src/testutil/psbt.d.ts +89 -0
- package/dist/src/testutil/psbt.d.ts.map +1 -0
- package/dist/src/testutil/psbt.js +150 -0
- package/dist/src/testutil/transaction.d.ts +70 -0
- package/dist/src/testutil/transaction.d.ts.map +1 -0
- package/dist/src/testutil/transaction.js +107 -0
- package/package.json +6 -6
package/dist/src/bitgo/Musig2.js
CHANGED
@@ -1,55 +1,112 @@
|
|
1
1
|
"use strict";
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
-
exports.
|
4
|
-
const UtxoPsbt_1 = require("./UtxoPsbt");
|
3
|
+
exports.musig2DeterministicSign = exports.createMusig2DeterministicNonce = exports.getSigHashTypeFromSigs = exports.assertPsbtMusig2Nonces = exports.assertPsbtMusig2Participants = exports.parsePsbtMusig2PartialSigs = exports.parsePsbtMusig2Nonces = exports.parsePsbtMusig2Participants = exports.createMusig2SigningSession = exports.musig2AggregateSigs = exports.musig2PartialSigVerify = exports.musig2PartialSign = exports.createTapTweak = exports.createAggregateNonce = exports.createTapOutputKey = exports.createTapInternalKey = exports.decodePsbtMusig2PartialSig = exports.decodePsbtMusig2Nonce = exports.decodePsbtMusig2Participants = exports.encodePsbtMusig2PartialSig = exports.encodePsbtMusig2PubNonce = exports.encodePsbtMusig2Participants = exports.Musig2NonceStore = void 0;
|
5
4
|
const outputScripts_1 = require("./outputScripts");
|
6
5
|
const noble_ecc_1 = require("../noble_ecc");
|
7
6
|
const taproot_1 = require("../taproot");
|
8
|
-
const
|
7
|
+
const index_1 = require("../index");
|
8
|
+
const PsbtUtil_1 = require("./PsbtUtil");
|
9
|
+
/**
|
10
|
+
* Because musig uses reference-equal buffers to cache nonces, we wrap it here to allow using
|
11
|
+
* nonces that are byte-equal but not reference-equal.
|
12
|
+
*/
|
13
|
+
class Musig2NonceStore {
|
14
|
+
constructor() {
|
15
|
+
this.nonces = [];
|
16
|
+
}
|
17
|
+
/**
|
18
|
+
* Get original Buffer instance for nonce (which may be a copy).
|
19
|
+
* @return byte-equal buffer that is reference-equal to what was stored earlier in createMusig2Nonce
|
20
|
+
*/
|
21
|
+
getRef(nonce) {
|
22
|
+
for (const b of this.nonces) {
|
23
|
+
if (Buffer.from(b).equals(nonce)) {
|
24
|
+
return b;
|
25
|
+
}
|
26
|
+
}
|
27
|
+
throw new Error(`unknown nonce`);
|
28
|
+
}
|
29
|
+
/**
|
30
|
+
* Creates musig2 nonce and stores buffer reference.
|
31
|
+
* tapInternalkey, tapMerkleRoot, tapBip32Derivation for rootWalletKey are required per p2trMusig2 key path input.
|
32
|
+
* Also participant keys are required from psbt proprietary key values.
|
33
|
+
* Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6
|
34
|
+
* @param privateKey - signer private key
|
35
|
+
* @param publicKey - signer xy public key
|
36
|
+
* @param xOnlyPublicKey - tweaked aggregated key (tapOutputKey)
|
37
|
+
* @param sessionId Additional entropy. If provided it must either be a counter unique to this secret key,
|
38
|
+
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
39
|
+
*/
|
40
|
+
createMusig2Nonce(privateKey, publicKey, xOnlyPublicKey, txHash, sessionId) {
|
41
|
+
if (txHash.length != 32) {
|
42
|
+
throw new Error(`Invalid txHash size ${txHash}`);
|
43
|
+
}
|
44
|
+
const buf = noble_ecc_1.musig.nonceGen({ secretKey: privateKey, publicKey, xOnlyPublicKey, msg: txHash, sessionId });
|
45
|
+
this.nonces.push(buf);
|
46
|
+
return buf;
|
47
|
+
}
|
48
|
+
}
|
49
|
+
exports.Musig2NonceStore = Musig2NonceStore;
|
9
50
|
/**
|
10
51
|
* Psbt proprietary key val util function for participants pub keys. SubType is 0x01
|
11
52
|
* Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6
|
12
53
|
* @return x-only tapOutputKey||tapInternalKey as sub keydata, plain sigining participant keys as valuedata
|
13
54
|
*/
|
14
|
-
function
|
15
|
-
const keydata = [
|
16
|
-
const value =
|
55
|
+
function encodePsbtMusig2Participants(participants) {
|
56
|
+
const keydata = [participants.tapOutputKey, participants.tapInternalKey].map((pubkey) => outputScripts_1.checkXOnlyPublicKey(pubkey));
|
57
|
+
const value = participants.participantPubKeys.map((pubkey) => outputScripts_1.checkPlainPublicKey(pubkey));
|
17
58
|
const key = {
|
18
|
-
identifier:
|
19
|
-
subtype:
|
59
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
60
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
|
20
61
|
keydata: Buffer.concat(keydata),
|
21
62
|
};
|
22
63
|
return { key, value: Buffer.concat(value) };
|
23
64
|
}
|
24
|
-
exports.
|
65
|
+
exports.encodePsbtMusig2Participants = encodePsbtMusig2Participants;
|
25
66
|
/**
|
26
67
|
* Psbt proprietary key val util function for pub nonce. SubType is 0x02
|
27
68
|
* Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6
|
28
69
|
* @return plain-participantPubKey||x-only-tapOutputKey as sub keydata, 66 bytes of 2 pub nonces as valuedata
|
29
70
|
*/
|
30
|
-
function
|
31
|
-
if (
|
32
|
-
throw new Error(`Invalid pubNonces length ${
|
71
|
+
function encodePsbtMusig2PubNonce(nonce) {
|
72
|
+
if (nonce.pubNonce.length !== 66) {
|
73
|
+
throw new Error(`Invalid pubNonces length ${nonce.pubNonce.length}`);
|
74
|
+
}
|
75
|
+
const keydata = Buffer.concat([
|
76
|
+
outputScripts_1.checkPlainPublicKey(nonce.participantPubKey),
|
77
|
+
outputScripts_1.checkXOnlyPublicKey(nonce.tapOutputKey),
|
78
|
+
]);
|
79
|
+
const key = {
|
80
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
81
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
|
82
|
+
keydata,
|
83
|
+
};
|
84
|
+
return { key, value: nonce.pubNonce };
|
85
|
+
}
|
86
|
+
exports.encodePsbtMusig2PubNonce = encodePsbtMusig2PubNonce;
|
87
|
+
function encodePsbtMusig2PartialSig(partialSig) {
|
88
|
+
if (partialSig.partialSig.length !== 32 && partialSig.partialSig.length !== 33) {
|
89
|
+
throw new Error(`Invalid partialSig length ${partialSig.partialSig.length}`);
|
33
90
|
}
|
34
91
|
const keydata = Buffer.concat([
|
35
|
-
outputScripts_1.checkPlainPublicKey(
|
36
|
-
outputScripts_1.checkXOnlyPublicKey(
|
92
|
+
outputScripts_1.checkPlainPublicKey(partialSig.participantPubKey),
|
93
|
+
outputScripts_1.checkXOnlyPublicKey(partialSig.tapOutputKey),
|
37
94
|
]);
|
38
95
|
const key = {
|
39
|
-
identifier:
|
40
|
-
subtype:
|
96
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
97
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
|
41
98
|
keydata,
|
42
99
|
};
|
43
|
-
return { key, value:
|
100
|
+
return { key, value: partialSig.partialSig };
|
44
101
|
}
|
45
|
-
exports.
|
102
|
+
exports.encodePsbtMusig2PartialSig = encodePsbtMusig2PartialSig;
|
46
103
|
/**
|
47
104
|
* Decodes proprietary key value data for participant pub keys
|
48
105
|
* @param kv
|
49
106
|
*/
|
50
|
-
function
|
51
|
-
if (kv.key.identifier !==
|
52
|
-
kv.key.subtype !==
|
107
|
+
function decodePsbtMusig2Participants(kv) {
|
108
|
+
if (kv.key.identifier !== PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER ||
|
109
|
+
kv.key.subtype !== PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS) {
|
53
110
|
throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`);
|
54
111
|
}
|
55
112
|
const key = kv.key.keydata;
|
@@ -66,109 +123,235 @@ function decodePsbtMusig2ParticipantsKeyValData(kv) {
|
|
66
123
|
}
|
67
124
|
return { tapOutputKey: key.subarray(0, 32), tapInternalKey: key.subarray(32), participantPubKeys };
|
68
125
|
}
|
69
|
-
exports.
|
126
|
+
exports.decodePsbtMusig2Participants = decodePsbtMusig2Participants;
|
127
|
+
/**
|
128
|
+
* Decodes proprietary key value data for musig2 nonce
|
129
|
+
* @param kv
|
130
|
+
*/
|
131
|
+
function decodePsbtMusig2Nonce(kv) {
|
132
|
+
if (kv.key.identifier !== PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER || kv.key.subtype !== PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE) {
|
133
|
+
throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`);
|
134
|
+
}
|
135
|
+
const key = kv.key.keydata;
|
136
|
+
if (key.length !== 65) {
|
137
|
+
throw new Error(`Invalid keydata size ${key.length} for nonce`);
|
138
|
+
}
|
139
|
+
const value = kv.value;
|
140
|
+
if (value.length !== 66) {
|
141
|
+
throw new Error(`Invalid valuedata size ${value.length} for nonce`);
|
142
|
+
}
|
143
|
+
return { participantPubKey: key.subarray(0, 33), tapOutputKey: key.subarray(33), pubNonce: value };
|
144
|
+
}
|
145
|
+
exports.decodePsbtMusig2Nonce = decodePsbtMusig2Nonce;
|
146
|
+
/**
|
147
|
+
* Decodes proprietary key value data for musig2 partial sig
|
148
|
+
* @param kv
|
149
|
+
*/
|
150
|
+
function decodePsbtMusig2PartialSig(kv) {
|
151
|
+
if (kv.key.identifier !== PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER ||
|
152
|
+
kv.key.subtype !== PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG) {
|
153
|
+
throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for partial sig`);
|
154
|
+
}
|
155
|
+
const key = kv.key.keydata;
|
156
|
+
if (key.length !== 65) {
|
157
|
+
throw new Error(`Invalid keydata size ${key.length} for partial sig`);
|
158
|
+
}
|
159
|
+
const value = kv.value;
|
160
|
+
if (value.length !== 32 && value.length !== 33) {
|
161
|
+
throw new Error(`Invalid valuedata size ${value.length} for partial sig`);
|
162
|
+
}
|
163
|
+
return { participantPubKey: key.subarray(0, 33), tapOutputKey: key.subarray(33), partialSig: value };
|
164
|
+
}
|
165
|
+
exports.decodePsbtMusig2PartialSig = decodePsbtMusig2PartialSig;
|
70
166
|
function createTapInternalKey(plainPubKeys) {
|
71
|
-
|
72
|
-
return Buffer.from(noble_ecc_1.musig.getXOnlyPubkey(noble_ecc_1.musig.keyAgg(noble_ecc_1.musig.keySort(plainPubKeys))));
|
167
|
+
return Buffer.from(noble_ecc_1.musig.getXOnlyPubkey(noble_ecc_1.musig.keyAgg(plainPubKeys)));
|
73
168
|
}
|
74
169
|
exports.createTapInternalKey = createTapInternalKey;
|
75
170
|
function createTapOutputKey(internalPubKey, tapTreeRoot) {
|
76
|
-
|
77
|
-
throw new Error(`Invalid tapTreeRoot size ${tapTreeRoot.length}`);
|
78
|
-
}
|
79
|
-
return Buffer.from(taproot_1.tapTweakPubkey(noble_ecc_1.ecc, outputScripts_1.toXOnlyPublicKey(internalPubKey), tapTreeRoot).xOnlyPubkey);
|
171
|
+
return Buffer.from(taproot_1.tapTweakPubkey(noble_ecc_1.ecc, outputScripts_1.toXOnlyPublicKey(internalPubKey), outputScripts_1.checkTapMerkleRoot(tapTreeRoot)).xOnlyPubkey);
|
80
172
|
}
|
81
173
|
exports.createTapOutputKey = createTapOutputKey;
|
82
|
-
function
|
83
|
-
|
84
|
-
|
174
|
+
function createAggregateNonce(pubNonces) {
|
175
|
+
return Buffer.from(noble_ecc_1.musig.nonceAgg(pubNonces));
|
176
|
+
}
|
177
|
+
exports.createAggregateNonce = createAggregateNonce;
|
178
|
+
function createTapTweak(tapInternalKey, tapMerkleRoot) {
|
179
|
+
return Buffer.from(taproot_1.calculateTapTweak(outputScripts_1.checkXOnlyPublicKey(tapInternalKey), outputScripts_1.checkTapMerkleRoot(tapMerkleRoot)));
|
180
|
+
}
|
181
|
+
exports.createTapTweak = createTapTweak;
|
182
|
+
function startMusig2SigningSession(aggNonce, hash, publicKeys, tweak) {
|
183
|
+
return noble_ecc_1.musig.startSigningSession(aggNonce, hash, publicKeys, { tweak, xOnly: true });
|
184
|
+
}
|
185
|
+
function musig2PartialSign(privateKey, publicNonce, sessionKey, nonceStore) {
|
186
|
+
outputScripts_1.checkTxHash(Buffer.from(sessionKey.msg));
|
187
|
+
return Buffer.from(noble_ecc_1.musig.partialSign({
|
188
|
+
secretKey: privateKey,
|
189
|
+
publicNonce: nonceStore.getRef(publicNonce),
|
190
|
+
sessionKey,
|
191
|
+
}));
|
192
|
+
}
|
193
|
+
exports.musig2PartialSign = musig2PartialSign;
|
194
|
+
function musig2PartialSigVerify(sig, publicKey, publicNonce, sessionKey) {
|
195
|
+
outputScripts_1.checkTxHash(Buffer.from(sessionKey.msg));
|
196
|
+
return noble_ecc_1.musig.partialVerify({ sig, publicKey, publicNonce, sessionKey });
|
197
|
+
}
|
198
|
+
exports.musig2PartialSigVerify = musig2PartialSigVerify;
|
199
|
+
function musig2AggregateSigs(sigs, sessionKey) {
|
200
|
+
return Buffer.from(noble_ecc_1.musig.signAgg(sigs, sessionKey));
|
201
|
+
}
|
202
|
+
exports.musig2AggregateSigs = musig2AggregateSigs;
|
203
|
+
/** @return session key that can be used to reference the session later */
|
204
|
+
function createMusig2SigningSession(sessionArgs) {
|
205
|
+
outputScripts_1.checkTxHash(sessionArgs.txHash);
|
206
|
+
const aggNonce = createAggregateNonce(sessionArgs.pubNonces);
|
207
|
+
const tweak = createTapTweak(sessionArgs.internalPubKey, sessionArgs.tapTreeRoot);
|
208
|
+
return startMusig2SigningSession(aggNonce, sessionArgs.txHash, sessionArgs.pubKeys, tweak);
|
209
|
+
}
|
210
|
+
exports.createMusig2SigningSession = createMusig2SigningSession;
|
211
|
+
/**
|
212
|
+
* @returns psbt proprietary key for musig2 participant key value data
|
213
|
+
* If no key value exists, undefined is returned.
|
214
|
+
*/
|
215
|
+
function parsePsbtMusig2Participants(input) {
|
216
|
+
const participantsKeyVals = PsbtUtil_1.getPsbtInputProprietaryKeyVals(input, {
|
217
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
218
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,
|
85
219
|
});
|
86
|
-
if (!
|
87
|
-
|
220
|
+
if (!participantsKeyVals.length) {
|
221
|
+
return undefined;
|
88
222
|
}
|
89
|
-
|
90
|
-
|
91
|
-
return bipDv.pubkey.equals(outputScripts_1.toXOnlyPublicKey(publicKey));
|
92
|
-
});
|
93
|
-
if (myDerivation.length !== 1) {
|
94
|
-
throw new Error('root wallet key should derive one tapBip32Derivation');
|
223
|
+
if (participantsKeyVals.length > 1) {
|
224
|
+
throw new Error(`Found ${participantsKeyVals.length} matching participant key value instead of 1`);
|
95
225
|
}
|
96
|
-
return
|
226
|
+
return decodePsbtMusig2Participants(participantsKeyVals[0]);
|
97
227
|
}
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
228
|
+
exports.parsePsbtMusig2Participants = parsePsbtMusig2Participants;
|
229
|
+
/**
|
230
|
+
* @returns psbt proprietary key for musig2 public nonce key value data
|
231
|
+
* If no key value exists, undefined is returned.
|
232
|
+
*/
|
233
|
+
function parsePsbtMusig2Nonces(input) {
|
234
|
+
const nonceKeyVals = PsbtUtil_1.getPsbtInputProprietaryKeyVals(input, {
|
235
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
236
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
|
237
|
+
});
|
238
|
+
if (!nonceKeyVals.length) {
|
239
|
+
return undefined;
|
105
240
|
}
|
106
|
-
if (
|
107
|
-
throw new Error(
|
241
|
+
if (nonceKeyVals.length > 2) {
|
242
|
+
throw new Error(`Found ${nonceKeyVals.length} matching nonce key value instead of 1 or 2`);
|
108
243
|
}
|
109
|
-
|
110
|
-
|
111
|
-
|
244
|
+
return nonceKeyVals.map((kv) => decodePsbtMusig2Nonce(kv));
|
245
|
+
}
|
246
|
+
exports.parsePsbtMusig2Nonces = parsePsbtMusig2Nonces;
|
247
|
+
/**
|
248
|
+
* @returns psbt proprietary key for musig2 partial sig key value data
|
249
|
+
* If no key value exists, undefined is returned.
|
250
|
+
*/
|
251
|
+
function parsePsbtMusig2PartialSigs(input) {
|
252
|
+
const sigKeyVals = PsbtUtil_1.getPsbtInputProprietaryKeyVals(input, {
|
253
|
+
identifier: PsbtUtil_1.PSBT_PROPRIETARY_IDENTIFIER,
|
254
|
+
subtype: PsbtUtil_1.ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,
|
112
255
|
});
|
113
|
-
if (
|
114
|
-
|
256
|
+
if (!sigKeyVals.length) {
|
257
|
+
return undefined;
|
115
258
|
}
|
116
|
-
|
259
|
+
if (sigKeyVals.length > 2) {
|
260
|
+
throw new Error(`Found ${sigKeyVals.length} matching partial signature key value instead of 1 or 2`);
|
261
|
+
}
|
262
|
+
return sigKeyVals.map((kv) => decodePsbtMusig2PartialSig(kv));
|
263
|
+
}
|
264
|
+
exports.parsePsbtMusig2PartialSigs = parsePsbtMusig2PartialSigs;
|
265
|
+
/**
|
266
|
+
* Assert musig2 participant key value data with tapInternalKey and tapMerkleRoot.
|
267
|
+
* <tapOutputKey><tapInputKey> => <participantKey1><participantKey2>
|
268
|
+
* Using tapMerkleRoot and 2 participant keys, the tapInputKey is validated and using tapMerkleRoot and tapInputKey,
|
269
|
+
* the tapOutputKey is validated.
|
270
|
+
*/
|
271
|
+
function assertPsbtMusig2Participants(participantKeyValData, tapInternalKey, tapMerkleRoot) {
|
272
|
+
outputScripts_1.checkXOnlyPublicKey(tapInternalKey);
|
273
|
+
outputScripts_1.checkTapMerkleRoot(tapMerkleRoot);
|
117
274
|
const participantPubKeys = participantKeyValData.participantPubKeys;
|
118
|
-
const
|
119
|
-
if (!
|
120
|
-
throw new Error('Invalid participants
|
275
|
+
const internalKey = createTapInternalKey(participantPubKeys);
|
276
|
+
if (!internalKey.equals(participantKeyValData.tapInternalKey)) {
|
277
|
+
throw new Error('Invalid participants keydata tapInternalKey');
|
121
278
|
}
|
122
|
-
const
|
123
|
-
if (!
|
124
|
-
throw new Error('Invalid participants
|
279
|
+
const outputKey = createTapOutputKey(internalKey, tapMerkleRoot);
|
280
|
+
if (!outputKey.equals(participantKeyValData.tapOutputKey)) {
|
281
|
+
throw new Error('Invalid participants keydata tapOutputKey');
|
125
282
|
}
|
126
|
-
if (!
|
283
|
+
if (!internalKey.equals(tapInternalKey)) {
|
127
284
|
throw new Error('tapInternalKey and aggregated participant pub keys does not match');
|
128
285
|
}
|
129
|
-
const derivedPubKey = deriveWalletPubKey(input.tapBip32Derivation, rootWalletKey);
|
130
|
-
const participantPubKey = participantPubKeys.find((pubKey) => pubKey.equals(derivedPubKey));
|
131
|
-
if (!Buffer.isBuffer(participantPubKey)) {
|
132
|
-
throw new Error('participant plain pub key should match one tapBip32Derivation plain pub key');
|
133
|
-
}
|
134
|
-
const { hash } = psbt.getTaprootHashForSigChecked(inputIndex);
|
135
|
-
const nonceGenArgs = {
|
136
|
-
sessionId,
|
137
|
-
publicKey: participantPubKey,
|
138
|
-
xOnlyPublicKey: tapOutputKey,
|
139
|
-
msg: hash,
|
140
|
-
secretKey: rootWalletKey.privateKey,
|
141
|
-
};
|
142
|
-
const pubNonces = Buffer.from(noble_ecc_1.musig.nonceGen(nonceGenArgs));
|
143
|
-
return encodePsbtMusig2PubNonceKeyValData({
|
144
|
-
participantPubKey,
|
145
|
-
tapOutputKey,
|
146
|
-
pubNonces,
|
147
|
-
});
|
148
286
|
}
|
287
|
+
exports.assertPsbtMusig2Participants = assertPsbtMusig2Participants;
|
149
288
|
/**
|
150
|
-
*
|
151
|
-
*
|
152
|
-
*
|
153
|
-
*
|
154
|
-
*
|
155
|
-
* @param rootWalletKey
|
156
|
-
* @param sessionId If provided it must either be a counter unique to this secret key,
|
157
|
-
* (converted to an array of 32 bytes), or 32 uniformly random bytes.
|
289
|
+
* Assert musig2 public nonce key value data with participant key value data
|
290
|
+
* (refer assertPsbtMusig2ParticipantsKeyValData).
|
291
|
+
* <participantKey1><tapOutputKey> => <pubNonce1>
|
292
|
+
* <participantKey2><tapOutputKey> => <pubNonce2>
|
293
|
+
* Checks against participant keys and tapOutputKey
|
158
294
|
*/
|
159
|
-
function
|
160
|
-
|
161
|
-
|
295
|
+
function assertPsbtMusig2Nonces(noncesKeyValData, participantKeyValData) {
|
296
|
+
outputScripts_1.checkXOnlyPublicKey(participantKeyValData.tapOutputKey);
|
297
|
+
participantKeyValData.participantPubKeys.forEach((kv) => outputScripts_1.checkPlainPublicKey(kv));
|
298
|
+
if (participantKeyValData.participantPubKeys[0].equals(participantKeyValData.participantPubKeys[1])) {
|
299
|
+
throw new Error(`Duplicate participant pub keys found`);
|
162
300
|
}
|
163
|
-
if (
|
164
|
-
throw new Error(`Invalid
|
301
|
+
if (noncesKeyValData.length > 2) {
|
302
|
+
throw new Error(`Invalid nonce key value count ${noncesKeyValData.length}`);
|
165
303
|
}
|
166
|
-
|
167
|
-
const
|
168
|
-
if (
|
169
|
-
|
304
|
+
noncesKeyValData.forEach((nonceKv) => {
|
305
|
+
const index = participantKeyValData.participantPubKeys.findIndex((pubKey) => nonceKv.participantPubKey.equals(pubKey));
|
306
|
+
if (index < 0) {
|
307
|
+
throw new Error('Invalid nonce keydata participant pub key');
|
308
|
+
}
|
309
|
+
if (!nonceKv.tapOutputKey.equals(participantKeyValData.tapOutputKey)) {
|
310
|
+
throw new Error('Invalid nonce keydata tapOutputKey');
|
170
311
|
}
|
171
312
|
});
|
172
313
|
}
|
173
|
-
exports.
|
174
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Musig2.js","sourceRoot":"","sources":["../../../src/bitgo/Musig2.ts"],"names":[],"mappings":"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","sourcesContent":["import { PSBT_PROPRIETARY_IDENTIFIER, ProprietaryKeyValueData, UtxoPsbt, ProprietaryKeySubtype } from './UtxoPsbt';\nimport { checkPlainPublicKey, checkXOnlyPublicKey, toXOnlyPublicKey } from './outputScripts';\nimport { BIP32Interface } from 'bip32';\nimport { ecc, musig } from '../noble_ecc';\nimport { Tuple } from './types';\nimport { tapTweakPubkey } from '../taproot';\nimport { TapBip32Derivation } from 'bip174/src/lib/interfaces';\nimport { checkForInput } from 'bip174/src/lib/utils';\n\n/**\n *  Participant key value object.\n */\nexport interface PsbtMusig2ParticipantsKeyValueData {\n  tapOutputKey: Buffer;\n  tapInternalKey: Buffer;\n  participantPubKeys: Tuple<Buffer>;\n}\n\n/**\n *  Nonce key value object.\n */\nexport interface PsbtMusig2NoncesKeyValueData {\n  participantPubKey: Buffer;\n  tapOutputKey: Buffer;\n  pubNonces: Buffer;\n}\n\n/**\n * Psbt proprietary key val util function for participants pub keys. SubType is 0x01\n * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n * @return x-only tapOutputKey||tapInternalKey as sub keydata, plain sigining participant keys as valuedata\n */\nexport function encodePsbtMusig2ParticipantsKeyValData(\n  participantsKeyValData: PsbtMusig2ParticipantsKeyValueData\n): ProprietaryKeyValueData {\n  const keydata = [participantsKeyValData.tapOutputKey, participantsKeyValData.tapInternalKey].map((pubkey) =>\n    checkXOnlyPublicKey(pubkey)\n  );\n  const value = participantsKeyValData.participantPubKeys.map((pubkey) => checkPlainPublicKey(pubkey));\n  const key = {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n    keydata: Buffer.concat(keydata),\n  };\n  return { key, value: Buffer.concat(value) };\n}\n\n/**\n * Psbt proprietary key val util function for pub nonce. SubType is 0x02\n * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n * @return plain-participantPubKey||x-only-tapOutputKey as sub keydata, 66 bytes of 2 pub nonces as valuedata\n */\nexport function encodePsbtMusig2PubNonceKeyValData(\n  noncesKeyValueData: PsbtMusig2NoncesKeyValueData\n): ProprietaryKeyValueData {\n  if (noncesKeyValueData.pubNonces.length !== 66) {\n    throw new Error(`Invalid pubNonces length ${noncesKeyValueData.pubNonces.length}`);\n  }\n  const keydata = Buffer.concat([\n    checkPlainPublicKey(noncesKeyValueData.participantPubKey),\n    checkXOnlyPublicKey(noncesKeyValueData.tapOutputKey),\n  ]);\n  const key = {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n    keydata,\n  };\n  return { key, value: noncesKeyValueData.pubNonces };\n}\n\n/**\n * Decodes proprietary key value data for participant pub keys\n * @param kv\n */\nexport function decodePsbtMusig2ParticipantsKeyValData(\n  kv: ProprietaryKeyValueData\n): PsbtMusig2ParticipantsKeyValueData {\n  if (\n    kv.key.identifier !== PSBT_PROPRIETARY_IDENTIFIER ||\n    kv.key.subtype !== ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS\n  ) {\n    throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`);\n  }\n\n  const key = kv.key.keydata;\n  if (key.length !== 64) {\n    throw new Error(`Invalid keydata size ${key.length} for participant pub keys`);\n  }\n\n  const value = kv.value;\n  if (value.length !== 66) {\n    throw new Error(`Invalid valuedata size ${value.length} for participant pub keys`);\n  }\n  const participantPubKeys: Tuple<Buffer> = [value.subarray(0, 33), value.subarray(33)];\n  if (participantPubKeys[0].equals(participantPubKeys[1])) {\n    throw new Error(`Duplicate participant pub keys found`);\n  }\n\n  return { tapOutputKey: key.subarray(0, 32), tapInternalKey: key.subarray(32), participantPubKeys };\n}\n\nexport function createTapInternalKey(plainPubKeys: Buffer[]): Buffer {\n  plainPubKeys.forEach((pubKey) => checkPlainPublicKey(pubKey));\n  return Buffer.from(musig.getXOnlyPubkey(musig.keyAgg(musig.keySort(plainPubKeys))));\n}\n\nexport function createTapOutputKey(internalPubKey: Buffer, tapTreeRoot: Buffer): Buffer {\n  if (tapTreeRoot.length !== 32) {\n    throw new Error(`Invalid tapTreeRoot size ${tapTreeRoot.length}`);\n  }\n  return Buffer.from(tapTweakPubkey(ecc, toXOnlyPublicKey(internalPubKey), tapTreeRoot).xOnlyPubkey);\n}\n\nfunction deriveWalletPubKey(tapBip32Derivations: TapBip32Derivation[], rootWalletKey: BIP32Interface): Buffer {\n  const myDerivations = tapBip32Derivations.filter((bipDv) => {\n    return bipDv.masterFingerprint.equals(rootWalletKey.fingerprint);\n  });\n\n  if (!myDerivations.length) {\n    throw new Error('Need one tapBip32Derivation masterFingerprint to match the rootWalletKey fingerprint');\n  }\n\n  const myDerivation = myDerivations.filter((bipDv) => {\n    const publicKey = rootWalletKey.derivePath(bipDv.path).publicKey;\n    return bipDv.pubkey.equals(toXOnlyPublicKey(publicKey));\n  });\n\n  if (myDerivation.length !== 1) {\n    throw new Error('root wallet key should derive one tapBip32Derivation');\n  }\n  return rootWalletKey.derivePath(myDerivation[0].path).publicKey;\n}\n\nfunction getMusig2NonceKeyValueData(\n  psbt: UtxoPsbt,\n  inputIndex: number,\n  rootWalletKey: BIP32Interface,\n  sessionId?: Buffer\n): ProprietaryKeyValueData | undefined {\n  const input = checkForInput(psbt.data.inputs, inputIndex);\n  if (!input.tapInternalKey) {\n    return;\n  }\n\n  if (!input.tapMerkleRoot) {\n    throw new Error('tapMerkleRoot is required to generate nonce');\n  }\n\n  if (!input.tapBip32Derivation?.length) {\n    throw new Error('tapBip32Derivation is required to generate nonce');\n  }\n\n  const participantsKeyVals = psbt.getProprietaryKeyVals(inputIndex, {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n  });\n\n  if (participantsKeyVals.length !== 1) {\n    throw new Error(`Found ${participantsKeyVals.length} matching participant key value instead of 1`);\n  }\n\n  const participantKeyValData = decodePsbtMusig2ParticipantsKeyValData(participantsKeyVals[0]);\n  const participantPubKeys = participantKeyValData.participantPubKeys;\n\n  const tapInternalKey = createTapInternalKey(participantPubKeys);\n  if (!tapInternalKey.equals(participantKeyValData.tapInternalKey)) {\n    throw new Error('Invalid participants keyata tapInternalKey');\n  }\n\n  const tapOutputKey = createTapOutputKey(tapInternalKey, input.tapMerkleRoot);\n  if (!tapOutputKey.equals(participantKeyValData.tapOutputKey)) {\n    throw new Error('Invalid participants keyata tapOutputKey');\n  }\n\n  if (!tapInternalKey.equals(input.tapInternalKey)) {\n    throw new Error('tapInternalKey and aggregated participant pub keys does not match');\n  }\n\n  const derivedPubKey = deriveWalletPubKey(input.tapBip32Derivation, rootWalletKey);\n  const participantPubKey = participantPubKeys.find((pubKey) => pubKey.equals(derivedPubKey));\n\n  if (!Buffer.isBuffer(participantPubKey)) {\n    throw new Error('participant plain pub key should match one tapBip32Derivation plain pub key');\n  }\n\n  const { hash } = psbt.getTaprootHashForSigChecked(inputIndex);\n\n  const nonceGenArgs = {\n    sessionId,\n    publicKey: participantPubKey,\n    xOnlyPublicKey: tapOutputKey,\n    msg: hash,\n    secretKey: rootWalletKey.privateKey,\n  };\n\n  const pubNonces = Buffer.from(musig.nonceGen(nonceGenArgs));\n\n  return encodePsbtMusig2PubNonceKeyValData({\n    participantPubKey,\n    tapOutputKey,\n    pubNonces,\n  });\n}\n\n/**\n * Generates and sets Musig2 nonces to p2trMusig2 key path spending inputs.\n * tapInternalkey, tapMerkleRoot, tapBip32Derivation for rootWalletKey are required per p2trMusig2 key path input.\n * Also participant keys are required from psbt proprietary key values.\n * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n * @param psbt\n * @param rootWalletKey\n * @param sessionId If provided it must either be a counter unique to this secret key,\n * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n */\nexport function setMusig2Nonces(psbt: UtxoPsbt, rootWalletKey: BIP32Interface, sessionId?: Buffer): void {\n  if (rootWalletKey.isNeutered()) {\n    throw new Error('private key is required to generate nonce');\n  }\n  if (Buffer.isBuffer(sessionId) && sessionId.length !== 32) {\n    throw new Error(`Invalid sessionId size ${sessionId.length}`);\n  }\n  psbt.data.inputs.forEach((input, inputIndex) => {\n    const noncesKeyValueData = getMusig2NonceKeyValueData(psbt, inputIndex, rootWalletKey, sessionId);\n    if (noncesKeyValueData) {\n      psbt.addProprietaryKeyValToInput(inputIndex, noncesKeyValueData);\n    }\n  });\n}\n"]}
|
314
|
+
exports.assertPsbtMusig2Nonces = assertPsbtMusig2Nonces;
|
315
|
+
/**
|
316
|
+
* @returns Input object but sig hash type data is taken out from partialSig field.
|
317
|
+
* If sig hash type is not common for all sigs, error out, otherwise returns the modified object and single hash type.
|
318
|
+
*/
|
319
|
+
function getSigHashTypeFromSigs(partialSigs) {
|
320
|
+
if (!partialSigs.length) {
|
321
|
+
throw new Error('partialSigs array can not be empty');
|
322
|
+
}
|
323
|
+
const pSigsWithHashType = partialSigs.map((kv) => {
|
324
|
+
const { partialSig, participantPubKey, tapOutputKey } = kv;
|
325
|
+
return partialSig.length === 33
|
326
|
+
? { pSig: { partialSig: partialSig.slice(0, 32), participantPubKey, tapOutputKey }, sigHashType: partialSig[32] }
|
327
|
+
: { pSig: { partialSig, participantPubKey, tapOutputKey }, sigHashType: index_1.Transaction.SIGHASH_DEFAULT };
|
328
|
+
});
|
329
|
+
const sigHashType = pSigsWithHashType[0].sigHashType;
|
330
|
+
if (!pSigsWithHashType.every((pSig) => pSig.sigHashType === sigHashType)) {
|
331
|
+
throw new Error('signatures must use same sig hash type');
|
332
|
+
}
|
333
|
+
return { partialSigs: pSigsWithHashType.map(({ pSig }) => pSig), sigHashType };
|
334
|
+
}
|
335
|
+
exports.getSigHashTypeFromSigs = getSigHashTypeFromSigs;
|
336
|
+
function createMusig2DeterministicNonce(params) {
|
337
|
+
return Buffer.from(noble_ecc_1.musig.deterministicNonceGen({
|
338
|
+
secretKey: params.privateKey,
|
339
|
+
aggOtherNonce: noble_ecc_1.musig.nonceAgg([params.otherNonce]),
|
340
|
+
publicKeys: params.publicKeys,
|
341
|
+
tweaks: [{ tweak: createTapTweak(params.internalPubKey, params.tapTreeRoot), xOnly: true }],
|
342
|
+
msg: params.hash,
|
343
|
+
}).publicNonce);
|
344
|
+
}
|
345
|
+
exports.createMusig2DeterministicNonce = createMusig2DeterministicNonce;
|
346
|
+
function musig2DeterministicSign(params) {
|
347
|
+
const { sig, sessionKey, publicNonce } = noble_ecc_1.musig.deterministicSign({
|
348
|
+
secretKey: params.privateKey,
|
349
|
+
aggOtherNonce: noble_ecc_1.musig.nonceAgg([params.otherNonce]),
|
350
|
+
publicKeys: params.publicKeys,
|
351
|
+
tweaks: [{ tweak: createTapTweak(params.internalPubKey, params.tapTreeRoot), xOnly: true }],
|
352
|
+
msg: params.hash,
|
353
|
+
});
|
354
|
+
return { sig: Buffer.from(sig), sessionKey, publicNonce: Buffer.from(publicNonce) };
|
355
|
+
}
|
356
|
+
exports.musig2DeterministicSign = musig2DeterministicSign;
|
357
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Musig2.js","sourceRoot":"","sources":["../../../src/bitgo/Musig2.ts"],"names":[],"mappings":"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","sourcesContent":["import { SessionKey } from '@brandonblack/musig';\n\nimport {\n  checkPlainPublicKey,\n  checkTapMerkleRoot,\n  checkTxHash,\n  checkXOnlyPublicKey,\n  toXOnlyPublicKey,\n} from './outputScripts';\nimport { ecc, musig } from '../noble_ecc';\nimport { Tuple } from './types';\nimport { calculateTapTweak, tapTweakPubkey } from '../taproot';\nimport { Transaction } from '../index';\nimport { PsbtInput } from 'bip174/src/lib/interfaces';\nimport {\n  getPsbtInputProprietaryKeyVals,\n  ProprietaryKeySubtype,\n  ProprietaryKeyValue,\n  PSBT_PROPRIETARY_IDENTIFIER,\n} from './PsbtUtil';\n\n/**\n *  Participant key value object.\n */\nexport interface PsbtMusig2Participants {\n  tapOutputKey: Buffer;\n  tapInternalKey: Buffer;\n  participantPubKeys: Tuple<Buffer>;\n}\n\nexport interface PsbtMusig2DeterministicParams {\n  privateKey: Buffer;\n  otherNonce: Buffer;\n  publicKeys: Tuple<Buffer>;\n  internalPubKey: Buffer;\n  tapTreeRoot: Buffer;\n  hash: Buffer;\n}\n\n/**\n *  Nonce key value object.\n */\nexport interface PsbtMusig2PubNonce {\n  participantPubKey: Buffer;\n  tapOutputKey: Buffer;\n  pubNonce: Buffer;\n}\n\n/**\n *  Partial signature key value object.\n */\nexport interface PsbtMusig2PartialSig {\n  participantPubKey: Buffer;\n  tapOutputKey: Buffer;\n  partialSig: Buffer;\n}\n\n/**\n * Because musig uses reference-equal buffers to cache nonces, we wrap it here to allow using\n * nonces that are byte-equal but not reference-equal.\n */\nexport class Musig2NonceStore {\n  private nonces: Uint8Array[] = [];\n\n  /**\n   * Get original Buffer instance for nonce (which may be a copy).\n   * @return byte-equal buffer that is reference-equal to what was stored earlier in createMusig2Nonce\n   */\n  getRef(nonce: Uint8Array): Uint8Array {\n    for (const b of this.nonces) {\n      if (Buffer.from(b).equals(nonce)) {\n        return b;\n      }\n    }\n    throw new Error(`unknown nonce`);\n  }\n\n  /**\n   * Creates musig2 nonce and stores buffer reference.\n   * tapInternalkey, tapMerkleRoot, tapBip32Derivation for rootWalletKey are required per p2trMusig2 key path input.\n   * Also participant keys are required from psbt proprietary key values.\n   * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n   * @param privateKey - signer private key\n   * @param publicKey - signer xy public key\n   * @param xOnlyPublicKey - tweaked aggregated key (tapOutputKey)\n   * @param sessionId Additional entropy. If provided it must either be a counter unique to this secret key,\n   * (converted to an array of 32 bytes), or 32 uniformly random bytes.\n   */\n  createMusig2Nonce(\n    privateKey: Uint8Array,\n    publicKey: Uint8Array,\n    xOnlyPublicKey: Uint8Array,\n    txHash: Uint8Array,\n    sessionId?: Buffer\n  ): Uint8Array {\n    if (txHash.length != 32) {\n      throw new Error(`Invalid txHash size ${txHash}`);\n    }\n    const buf = musig.nonceGen({ secretKey: privateKey, publicKey, xOnlyPublicKey, msg: txHash, sessionId });\n    this.nonces.push(buf);\n    return buf;\n  }\n}\n\n/**\n * Psbt proprietary key val util function for participants pub keys. SubType is 0x01\n * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n * @return x-only tapOutputKey||tapInternalKey as sub keydata, plain sigining participant keys as valuedata\n */\nexport function encodePsbtMusig2Participants(participants: PsbtMusig2Participants): ProprietaryKeyValue {\n  const keydata = [participants.tapOutputKey, participants.tapInternalKey].map((pubkey) => checkXOnlyPublicKey(pubkey));\n  const value = participants.participantPubKeys.map((pubkey) => checkPlainPublicKey(pubkey));\n  const key = {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n    keydata: Buffer.concat(keydata),\n  };\n  return { key, value: Buffer.concat(value) };\n}\n\n/**\n * Psbt proprietary key val util function for pub nonce. SubType is 0x02\n * Ref: https://gist.github.com/sanket1729/4b525c6049f4d9e034d27368c49f28a6\n * @return plain-participantPubKey||x-only-tapOutputKey as sub keydata, 66 bytes of 2 pub nonces as valuedata\n */\nexport function encodePsbtMusig2PubNonce(nonce: PsbtMusig2PubNonce): ProprietaryKeyValue {\n  if (nonce.pubNonce.length !== 66) {\n    throw new Error(`Invalid pubNonces length ${nonce.pubNonce.length}`);\n  }\n  const keydata = Buffer.concat([\n    checkPlainPublicKey(nonce.participantPubKey),\n    checkXOnlyPublicKey(nonce.tapOutputKey),\n  ]);\n  const key = {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n    keydata,\n  };\n  return { key, value: nonce.pubNonce };\n}\n\nexport function encodePsbtMusig2PartialSig(partialSig: PsbtMusig2PartialSig): ProprietaryKeyValue {\n  if (partialSig.partialSig.length !== 32 && partialSig.partialSig.length !== 33) {\n    throw new Error(`Invalid partialSig length ${partialSig.partialSig.length}`);\n  }\n  const keydata = Buffer.concat([\n    checkPlainPublicKey(partialSig.participantPubKey),\n    checkXOnlyPublicKey(partialSig.tapOutputKey),\n  ]);\n  const key = {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n    keydata,\n  };\n  return { key, value: partialSig.partialSig };\n}\n\n/**\n * Decodes proprietary key value data for participant pub keys\n * @param kv\n */\nexport function decodePsbtMusig2Participants(kv: ProprietaryKeyValue): PsbtMusig2Participants {\n  if (\n    kv.key.identifier !== PSBT_PROPRIETARY_IDENTIFIER ||\n    kv.key.subtype !== ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS\n  ) {\n    throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for participants pub keys`);\n  }\n\n  const key = kv.key.keydata;\n  if (key.length !== 64) {\n    throw new Error(`Invalid keydata size ${key.length} for participant pub keys`);\n  }\n\n  const value = kv.value;\n  if (value.length !== 66) {\n    throw new Error(`Invalid valuedata size ${value.length} for participant pub keys`);\n  }\n  const participantPubKeys: Tuple<Buffer> = [value.subarray(0, 33), value.subarray(33)];\n  if (participantPubKeys[0].equals(participantPubKeys[1])) {\n    throw new Error(`Duplicate participant pub keys found`);\n  }\n\n  return { tapOutputKey: key.subarray(0, 32), tapInternalKey: key.subarray(32), participantPubKeys };\n}\n\n/**\n * Decodes proprietary key value data for musig2 nonce\n * @param kv\n */\nexport function decodePsbtMusig2Nonce(kv: ProprietaryKeyValue): PsbtMusig2PubNonce {\n  if (kv.key.identifier !== PSBT_PROPRIETARY_IDENTIFIER || kv.key.subtype !== ProprietaryKeySubtype.MUSIG2_PUB_NONCE) {\n    throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for nonce`);\n  }\n\n  const key = kv.key.keydata;\n  if (key.length !== 65) {\n    throw new Error(`Invalid keydata size ${key.length} for nonce`);\n  }\n\n  const value = kv.value;\n  if (value.length !== 66) {\n    throw new Error(`Invalid valuedata size ${value.length} for nonce`);\n  }\n\n  return { participantPubKey: key.subarray(0, 33), tapOutputKey: key.subarray(33), pubNonce: value };\n}\n\n/**\n * Decodes proprietary key value data for musig2 partial sig\n * @param kv\n */\nexport function decodePsbtMusig2PartialSig(kv: ProprietaryKeyValue): PsbtMusig2PartialSig {\n  if (\n    kv.key.identifier !== PSBT_PROPRIETARY_IDENTIFIER ||\n    kv.key.subtype !== ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG\n  ) {\n    throw new Error(`Invalid identifier ${kv.key.identifier} or subtype ${kv.key.subtype} for partial sig`);\n  }\n\n  const key = kv.key.keydata;\n  if (key.length !== 65) {\n    throw new Error(`Invalid keydata size ${key.length} for partial sig`);\n  }\n\n  const value = kv.value;\n  if (value.length !== 32 && value.length !== 33) {\n    throw new Error(`Invalid valuedata size ${value.length} for partial sig`);\n  }\n\n  return { participantPubKey: key.subarray(0, 33), tapOutputKey: key.subarray(33), partialSig: value };\n}\n\nexport function createTapInternalKey(plainPubKeys: Buffer[]): Buffer {\n  return Buffer.from(musig.getXOnlyPubkey(musig.keyAgg(plainPubKeys)));\n}\n\nexport function createTapOutputKey(internalPubKey: Buffer, tapTreeRoot: Buffer): Buffer {\n  return Buffer.from(\n    tapTweakPubkey(ecc, toXOnlyPublicKey(internalPubKey), checkTapMerkleRoot(tapTreeRoot)).xOnlyPubkey\n  );\n}\n\nexport function createAggregateNonce(pubNonces: Tuple<Buffer>): Buffer {\n  return Buffer.from(musig.nonceAgg(pubNonces));\n}\n\nexport function createTapTweak(tapInternalKey: Buffer, tapMerkleRoot: Buffer): Buffer {\n  return Buffer.from(calculateTapTweak(checkXOnlyPublicKey(tapInternalKey), checkTapMerkleRoot(tapMerkleRoot)));\n}\n\nfunction startMusig2SigningSession(\n  aggNonce: Buffer,\n  hash: Buffer,\n  publicKeys: Tuple<Buffer>,\n  tweak: Buffer\n): SessionKey {\n  return musig.startSigningSession(aggNonce, hash, publicKeys, { tweak, xOnly: true });\n}\n\nexport function musig2PartialSign(\n  privateKey: Buffer,\n  publicNonce: Uint8Array,\n  sessionKey: SessionKey,\n  nonceStore: Musig2NonceStore\n): Buffer {\n  checkTxHash(Buffer.from(sessionKey.msg));\n  return Buffer.from(\n    musig.partialSign({\n      secretKey: privateKey,\n      publicNonce: nonceStore.getRef(publicNonce),\n      sessionKey,\n    })\n  );\n}\n\nexport function musig2PartialSigVerify(\n  sig: Buffer,\n  publicKey: Buffer,\n  publicNonce: Buffer,\n  sessionKey: SessionKey\n): boolean {\n  checkTxHash(Buffer.from(sessionKey.msg));\n  return musig.partialVerify({ sig, publicKey, publicNonce, sessionKey });\n}\n\nexport function musig2AggregateSigs(sigs: Buffer[], sessionKey: SessionKey): Buffer {\n  return Buffer.from(musig.signAgg(sigs, sessionKey));\n}\n\n/** @return session key that can be used to reference the session later */\nexport function createMusig2SigningSession(sessionArgs: {\n  pubNonces: Tuple<Buffer>;\n  txHash: Buffer;\n  pubKeys: Tuple<Buffer>;\n  internalPubKey: Buffer;\n  tapTreeRoot: Buffer;\n}): SessionKey {\n  checkTxHash(sessionArgs.txHash);\n  const aggNonce = createAggregateNonce(sessionArgs.pubNonces);\n  const tweak = createTapTweak(sessionArgs.internalPubKey, sessionArgs.tapTreeRoot);\n  return startMusig2SigningSession(aggNonce, sessionArgs.txHash, sessionArgs.pubKeys, tweak);\n}\n\n/**\n * @returns psbt proprietary key for musig2 participant key value data\n * If no key value exists, undefined is returned.\n */\nexport function parsePsbtMusig2Participants(input: PsbtInput): PsbtMusig2Participants | undefined {\n  const participantsKeyVals = getPsbtInputProprietaryKeyVals(input, {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTICIPANT_PUB_KEYS,\n  });\n\n  if (!participantsKeyVals.length) {\n    return undefined;\n  }\n\n  if (participantsKeyVals.length > 1) {\n    throw new Error(`Found ${participantsKeyVals.length} matching participant key value instead of 1`);\n  }\n\n  return decodePsbtMusig2Participants(participantsKeyVals[0]);\n}\n\n/**\n * @returns psbt proprietary key for musig2 public nonce key value data\n * If no key value exists, undefined is returned.\n */\nexport function parsePsbtMusig2Nonces(input: PsbtInput): PsbtMusig2PubNonce[] | undefined {\n  const nonceKeyVals = getPsbtInputProprietaryKeyVals(input, {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n  });\n\n  if (!nonceKeyVals.length) {\n    return undefined;\n  }\n\n  if (nonceKeyVals.length > 2) {\n    throw new Error(`Found ${nonceKeyVals.length} matching nonce key value instead of 1 or 2`);\n  }\n\n  return nonceKeyVals.map((kv) => decodePsbtMusig2Nonce(kv));\n}\n\n/**\n * @returns psbt proprietary key for musig2 partial sig key value data\n * If no key value exists, undefined is returned.\n */\nexport function parsePsbtMusig2PartialSigs(input: PsbtInput): PsbtMusig2PartialSig[] | undefined {\n  const sigKeyVals = getPsbtInputProprietaryKeyVals(input, {\n    identifier: PSBT_PROPRIETARY_IDENTIFIER,\n    subtype: ProprietaryKeySubtype.MUSIG2_PARTIAL_SIG,\n  });\n\n  if (!sigKeyVals.length) {\n    return undefined;\n  }\n\n  if (sigKeyVals.length > 2) {\n    throw new Error(`Found ${sigKeyVals.length} matching partial signature key value instead of 1 or 2`);\n  }\n\n  return sigKeyVals.map((kv) => decodePsbtMusig2PartialSig(kv));\n}\n\n/**\n * Assert musig2 participant key value data with tapInternalKey and tapMerkleRoot.\n * <tapOutputKey><tapInputKey> => <participantKey1><participantKey2>\n * Using tapMerkleRoot and 2 participant keys, the tapInputKey is validated and using tapMerkleRoot and tapInputKey,\n * the tapOutputKey is validated.\n */\nexport function assertPsbtMusig2Participants(\n  participantKeyValData: PsbtMusig2Participants,\n  tapInternalKey: Buffer,\n  tapMerkleRoot: Buffer\n): void {\n  checkXOnlyPublicKey(tapInternalKey);\n  checkTapMerkleRoot(tapMerkleRoot);\n\n  const participantPubKeys = participantKeyValData.participantPubKeys;\n\n  const internalKey = createTapInternalKey(participantPubKeys);\n  if (!internalKey.equals(participantKeyValData.tapInternalKey)) {\n    throw new Error('Invalid participants keydata tapInternalKey');\n  }\n\n  const outputKey = createTapOutputKey(internalKey, tapMerkleRoot);\n  if (!outputKey.equals(participantKeyValData.tapOutputKey)) {\n    throw new Error('Invalid participants keydata tapOutputKey');\n  }\n\n  if (!internalKey.equals(tapInternalKey)) {\n    throw new Error('tapInternalKey and aggregated participant pub keys does not match');\n  }\n}\n\n/**\n * Assert musig2 public nonce key value data with participant key value data\n * (refer assertPsbtMusig2ParticipantsKeyValData).\n * <participantKey1><tapOutputKey> => <pubNonce1>\n * <participantKey2><tapOutputKey> => <pubNonce2>\n * Checks against participant keys and tapOutputKey\n */\nexport function assertPsbtMusig2Nonces(\n  noncesKeyValData: PsbtMusig2PubNonce[],\n  participantKeyValData: PsbtMusig2Participants\n): void {\n  checkXOnlyPublicKey(participantKeyValData.tapOutputKey);\n  participantKeyValData.participantPubKeys.forEach((kv) => checkPlainPublicKey(kv));\n  if (participantKeyValData.participantPubKeys[0].equals(participantKeyValData.participantPubKeys[1])) {\n    throw new Error(`Duplicate participant pub keys found`);\n  }\n\n  if (noncesKeyValData.length > 2) {\n    throw new Error(`Invalid nonce key value count ${noncesKeyValData.length}`);\n  }\n\n  noncesKeyValData.forEach((nonceKv) => {\n    const index = participantKeyValData.participantPubKeys.findIndex((pubKey) =>\n      nonceKv.participantPubKey.equals(pubKey)\n    );\n    if (index < 0) {\n      throw new Error('Invalid nonce keydata participant pub key');\n    }\n\n    if (!nonceKv.tapOutputKey.equals(participantKeyValData.tapOutputKey)) {\n      throw new Error('Invalid nonce keydata tapOutputKey');\n    }\n  });\n}\n\n/**\n * @returns Input object but sig hash type data is taken out from partialSig field.\n * If sig hash type is not common for all sigs, error out, otherwise returns the modified object and single hash type.\n */\nexport function getSigHashTypeFromSigs(partialSigs: PsbtMusig2PartialSig[]): {\n  partialSigs: PsbtMusig2PartialSig[];\n  sigHashType: number;\n} {\n  if (!partialSigs.length) {\n    throw new Error('partialSigs array can not be empty');\n  }\n  const pSigsWithHashType = partialSigs.map((kv) => {\n    const { partialSig, participantPubKey, tapOutputKey } = kv;\n    return partialSig.length === 33\n      ? { pSig: { partialSig: partialSig.slice(0, 32), participantPubKey, tapOutputKey }, sigHashType: partialSig[32] }\n      : { pSig: { partialSig, participantPubKey, tapOutputKey }, sigHashType: Transaction.SIGHASH_DEFAULT };\n  });\n\n  const sigHashType = pSigsWithHashType[0].sigHashType;\n  if (!pSigsWithHashType.every((pSig) => pSig.sigHashType === sigHashType)) {\n    throw new Error('signatures must use same sig hash type');\n  }\n\n  return { partialSigs: pSigsWithHashType.map(({ pSig }) => pSig), sigHashType };\n}\n\nexport function createMusig2DeterministicNonce(params: PsbtMusig2DeterministicParams): Buffer {\n  return Buffer.from(\n    musig.deterministicNonceGen({\n      secretKey: params.privateKey,\n      aggOtherNonce: musig.nonceAgg([params.otherNonce]),\n      publicKeys: params.publicKeys,\n      tweaks: [{ tweak: createTapTweak(params.internalPubKey, params.tapTreeRoot), xOnly: true }],\n      msg: params.hash,\n    }).publicNonce\n  );\n}\n\nexport function musig2DeterministicSign(params: PsbtMusig2DeterministicParams): {\n  sig: Buffer;\n  sessionKey: SessionKey;\n  publicNonce: Buffer;\n} {\n  const { sig, sessionKey, publicNonce } = musig.deterministicSign({\n    secretKey: params.privateKey,\n    aggOtherNonce: musig.nonceAgg([params.otherNonce]),\n    publicKeys: params.publicKeys,\n    tweaks: [{ tweak: createTapTweak(params.internalPubKey, params.tapTreeRoot), xOnly: true }],\n    msg: params.hash,\n  });\n  return { sig: Buffer.from(sig), sessionKey, publicNonce: Buffer.from(publicNonce) };\n}\n"]}
|
@@ -0,0 +1,59 @@
|
|
1
|
+
/// <reference types="node" />
|
2
|
+
import { ProprietaryKey } from 'bip174/src/lib/proprietaryKeyVal';
|
3
|
+
import { PsbtInput } from 'bip174/src/lib/interfaces';
|
4
|
+
import { Psbt } from 'bitcoinjs-lib/src/psbt';
|
5
|
+
/**
|
6
|
+
* bitgo proprietary key identifier
|
7
|
+
*/
|
8
|
+
export declare const PSBT_PROPRIETARY_IDENTIFIER = "BITGO";
|
9
|
+
/**
|
10
|
+
* subtype for proprietary keys that bitgo uses
|
11
|
+
*/
|
12
|
+
export declare enum ProprietaryKeySubtype {
|
13
|
+
ZEC_CONSENSUS_BRANCH_ID = 0,
|
14
|
+
MUSIG2_PARTICIPANT_PUB_KEYS = 1,
|
15
|
+
MUSIG2_PUB_NONCE = 2,
|
16
|
+
MUSIG2_PARTIAL_SIG = 3
|
17
|
+
}
|
18
|
+
/**
|
19
|
+
* Psbt proprietary keydata object.
|
20
|
+
* <compact size uint identifier length> <bytes identifier> <compact size uint subtype> <bytes subkeydata>
|
21
|
+
* => <bytes valuedata>
|
22
|
+
*/
|
23
|
+
export interface ProprietaryKeyValue {
|
24
|
+
key: ProprietaryKey;
|
25
|
+
value: Buffer;
|
26
|
+
}
|
27
|
+
/**
|
28
|
+
* Psbt proprietary keydata object search fields.
|
29
|
+
* <compact size uint identifier length> <bytes identifier> <compact size uint subtype> <bytes subkeydata>
|
30
|
+
*/
|
31
|
+
export interface ProprietaryKeySearch {
|
32
|
+
identifier: string;
|
33
|
+
subtype?: number;
|
34
|
+
keydata?: Buffer;
|
35
|
+
identifierEncoding?: BufferEncoding;
|
36
|
+
}
|
37
|
+
/**
|
38
|
+
* Search any data from psbt proprietary key value against keydata.
|
39
|
+
* Default identifierEncoding is utf-8 for identifier.
|
40
|
+
*/
|
41
|
+
export declare function getPsbtInputProprietaryKeyVals(input: PsbtInput, keySearch?: ProprietaryKeySearch): ProprietaryKeyValue[];
|
42
|
+
/**
|
43
|
+
* @return partialSig/tapScriptSig/MUSIG2_PARTIAL_SIG count iff input is not finalized
|
44
|
+
*/
|
45
|
+
export declare function getPsbtInputSignatureCount(input: PsbtInput): number;
|
46
|
+
/**
|
47
|
+
* @return true iff PSBT input is finalized
|
48
|
+
*/
|
49
|
+
export declare function isPsbtInputFinalized(input: PsbtInput): boolean;
|
50
|
+
/**
|
51
|
+
* @return true iff data starts with magic PSBT byte sequence
|
52
|
+
* @param data byte array or hex string
|
53
|
+
* */
|
54
|
+
export declare function isPsbt(data: Buffer | string): boolean;
|
55
|
+
/**
|
56
|
+
* This function allows signing or validating a psbt with non-segwit inputs those do not contain nonWitnessUtxo.
|
57
|
+
*/
|
58
|
+
export declare function withUnsafeNonSegwit<T>(psbt: Psbt, fn: () => T, unsafe?: boolean): T;
|
59
|
+
//# sourceMappingURL=PsbtUtil.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"PsbtUtil.d.ts","sourceRoot":"","sources":["../../../src/bitgo/PsbtUtil.ts"],"names":[],"mappings":";AAAA,OAAO,EAAwB,cAAc,EAAE,MAAM,kCAAkC,CAAC;AACxF,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAE,IAAI,EAAE,MAAM,wBAAwB,CAAC;AAE9C;;GAEG;AACH,eAAO,MAAM,2BAA2B,UAAU,CAAC;AAEnD;;GAEG;AACH,oBAAY,qBAAqB;IAC/B,uBAAuB,IAAO;IAC9B,2BAA2B,IAAO;IAClC,gBAAgB,IAAO;IACvB,kBAAkB,IAAO;CAC1B;AAED;;;;GAIG;AACH,MAAM,WAAW,mBAAmB;IAClC,GAAG,EAAE,cAAc,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;CACf;AAED;;;GAGG;AACH,MAAM,WAAW,oBAAoB;IACnC,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,kBAAkB,CAAC,EAAE,cAAc,CAAC;CACrC;AAED;;;GAGG;AACH,wBAAgB,8BAA8B,CAC5C,KAAK,EAAE,SAAS,EAChB,SAAS,CAAC,EAAE,oBAAoB,GAC/B,mBAAmB,EAAE,CAmBvB;AAED;;GAEG;AACH,wBAAgB,0BAA0B,CAAC,KAAK,EAAE,SAAS,GAAG,MAAM,CAYnE;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,SAAS,GAAG,OAAO,CAE9D;AAED;;;KAGK;AACL,wBAAgB,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,CAUrD;AAED;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE,MAAM,CAAC,EAAE,MAAM,UAAO,GAAG,CAAC,CAOhF"}
|