@bitgo-beta/abstract-utxo 1.6.1-alpha.456 → 1.6.1-alpha.457

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.
Files changed (156) hide show
  1. package/dist/cjs/src/abstractUtxoCoin.d.ts +4 -1
  2. package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
  3. package/dist/cjs/src/abstractUtxoCoin.js +14 -11
  4. package/dist/cjs/src/address/fixedScript.d.ts +4 -4
  5. package/dist/cjs/src/address/fixedScript.d.ts.map +1 -1
  6. package/dist/cjs/src/address/fixedScript.js +13 -11
  7. package/dist/cjs/src/descriptor/assertDescriptorWalletAddress.d.ts +2 -2
  8. package/dist/cjs/src/descriptor/assertDescriptorWalletAddress.d.ts.map +1 -1
  9. package/dist/cjs/src/descriptor/assertDescriptorWalletAddress.js +4 -2
  10. package/dist/cjs/src/names.d.ts +2 -0
  11. package/dist/cjs/src/names.d.ts.map +1 -1
  12. package/dist/cjs/src/names.js +9 -1
  13. package/dist/cjs/src/offlineVault/OfflineVaultHalfSigned.d.ts +2 -1
  14. package/dist/cjs/src/offlineVault/OfflineVaultHalfSigned.d.ts.map +1 -1
  15. package/dist/cjs/src/offlineVault/OfflineVaultHalfSigned.js +3 -5
  16. package/dist/cjs/src/offlineVault/TransactionExplanation.d.ts +2 -1
  17. package/dist/cjs/src/offlineVault/TransactionExplanation.d.ts.map +1 -1
  18. package/dist/cjs/src/offlineVault/TransactionExplanation.js +3 -4
  19. package/dist/cjs/src/offlineVault/descriptor/transaction.d.ts +3 -2
  20. package/dist/cjs/src/offlineVault/descriptor/transaction.d.ts.map +1 -1
  21. package/dist/cjs/src/offlineVault/descriptor/transaction.js +6 -3
  22. package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
  23. package/dist/cjs/src/recovery/backupKeyRecovery.js +8 -7
  24. package/dist/cjs/src/recovery/crossChainRecovery.d.ts +3 -2
  25. package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
  26. package/dist/cjs/src/recovery/crossChainRecovery.js +7 -6
  27. package/dist/cjs/src/recovery/psbt.d.ts +3 -2
  28. package/dist/cjs/src/recovery/psbt.d.ts.map +1 -1
  29. package/dist/cjs/src/recovery/psbt.js +5 -4
  30. package/dist/cjs/src/transaction/bip322.d.ts.map +1 -1
  31. package/dist/cjs/src/transaction/bip322.js +26 -11
  32. package/dist/cjs/src/transaction/decode.d.ts +4 -3
  33. package/dist/cjs/src/transaction/decode.d.ts.map +1 -1
  34. package/dist/cjs/src/transaction/decode.js +8 -5
  35. package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts.map +1 -1
  36. package/dist/cjs/src/transaction/descriptor/explainPsbt.js +7 -5
  37. package/dist/cjs/src/transaction/descriptor/parse.d.ts +2 -1
  38. package/dist/cjs/src/transaction/descriptor/parse.d.ts.map +1 -1
  39. package/dist/cjs/src/transaction/descriptor/parse.js +14 -14
  40. package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts +2 -1
  41. package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  42. package/dist/cjs/src/transaction/descriptor/verifyTransaction.js +4 -4
  43. package/dist/cjs/src/transaction/explainTransaction.d.ts +2 -1
  44. package/dist/cjs/src/transaction/explainTransaction.d.ts.map +1 -1
  45. package/dist/cjs/src/transaction/explainTransaction.js +5 -5
  46. package/dist/cjs/src/transaction/fetchInputs.d.ts +2 -1
  47. package/dist/cjs/src/transaction/fetchInputs.d.ts.map +1 -1
  48. package/dist/cjs/src/transaction/fetchInputs.js +4 -2
  49. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts +4 -4
  50. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  51. package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +24 -19
  52. package/dist/cjs/src/transaction/fixedScript/parseTransaction.js +3 -3
  53. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts +6 -4
  54. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts.map +1 -1
  55. package/dist/cjs/src/transaction/fixedScript/replayProtection.js +17 -49
  56. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts +4 -2
  57. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -1
  58. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.js +6 -5
  59. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts +2 -1
  60. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  61. package/dist/cjs/src/transaction/fixedScript/signTransaction.js +4 -4
  62. package/dist/cjs/src/transaction/fixedScript/verifyTransaction.js +2 -2
  63. package/dist/cjs/src/transaction/getPayGoVerificationPubkey.d.ts +3 -3
  64. package/dist/cjs/src/transaction/getPayGoVerificationPubkey.d.ts.map +1 -1
  65. package/dist/cjs/src/transaction/getPayGoVerificationPubkey.js +6 -39
  66. package/dist/cjs/src/transaction/recipient.d.ts +4 -4
  67. package/dist/cjs/src/transaction/recipient.d.ts.map +1 -1
  68. package/dist/cjs/src/transaction/recipient.js +7 -4
  69. package/dist/cjs/src/transaction/signTransaction.js +2 -2
  70. package/dist/cjs/test/unit/address.js +9 -9
  71. package/dist/cjs/test/unit/bip322.js +319 -3
  72. package/dist/cjs/test/unit/customChangeWallet.js +2 -2
  73. package/dist/cjs/test/unit/prebuildAndSign.js +2 -2
  74. package/dist/cjs/test/unit/recovery/crossChainRecovery.js +6 -7
  75. package/dist/cjs/test/unit/transaction/descriptor/parse.js +2 -2
  76. package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +7 -10
  77. package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +4 -5
  78. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.js +3 -2
  79. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.js +5 -6
  80. package/dist/cjs/test/unit/transaction.js +23 -24
  81. package/dist/cjs/test/unit/util/unspents.d.ts.map +1 -1
  82. package/dist/cjs/test/unit/util/unspents.js +4 -2
  83. package/dist/cjs/tsconfig.tsbuildinfo +1 -1
  84. package/dist/esm/abstractUtxoCoin.d.ts +4 -1
  85. package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
  86. package/dist/esm/abstractUtxoCoin.js +16 -13
  87. package/dist/esm/address/fixedScript.d.ts +4 -4
  88. package/dist/esm/address/fixedScript.d.ts.map +1 -1
  89. package/dist/esm/address/fixedScript.js +13 -11
  90. package/dist/esm/descriptor/assertDescriptorWalletAddress.d.ts +2 -2
  91. package/dist/esm/descriptor/assertDescriptorWalletAddress.d.ts.map +1 -1
  92. package/dist/esm/descriptor/assertDescriptorWalletAddress.js +4 -2
  93. package/dist/esm/names.d.ts +2 -0
  94. package/dist/esm/names.d.ts.map +1 -1
  95. package/dist/esm/names.js +7 -1
  96. package/dist/esm/offlineVault/OfflineVaultHalfSigned.d.ts +2 -1
  97. package/dist/esm/offlineVault/OfflineVaultHalfSigned.d.ts.map +1 -1
  98. package/dist/esm/offlineVault/OfflineVaultHalfSigned.js +3 -5
  99. package/dist/esm/offlineVault/TransactionExplanation.d.ts +2 -1
  100. package/dist/esm/offlineVault/TransactionExplanation.d.ts.map +1 -1
  101. package/dist/esm/offlineVault/TransactionExplanation.js +3 -4
  102. package/dist/esm/offlineVault/descriptor/transaction.d.ts +3 -2
  103. package/dist/esm/offlineVault/descriptor/transaction.d.ts.map +1 -1
  104. package/dist/esm/offlineVault/descriptor/transaction.js +6 -3
  105. package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
  106. package/dist/esm/recovery/backupKeyRecovery.js +8 -7
  107. package/dist/esm/recovery/crossChainRecovery.d.ts +3 -2
  108. package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
  109. package/dist/esm/recovery/crossChainRecovery.js +8 -7
  110. package/dist/esm/recovery/psbt.d.ts +3 -2
  111. package/dist/esm/recovery/psbt.d.ts.map +1 -1
  112. package/dist/esm/recovery/psbt.js +5 -4
  113. package/dist/esm/transaction/bip322.d.ts.map +1 -1
  114. package/dist/esm/transaction/bip322.js +27 -12
  115. package/dist/esm/transaction/decode.d.ts +4 -3
  116. package/dist/esm/transaction/decode.d.ts.map +1 -1
  117. package/dist/esm/transaction/decode.js +8 -5
  118. package/dist/esm/transaction/descriptor/explainPsbt.d.ts.map +1 -1
  119. package/dist/esm/transaction/descriptor/explainPsbt.js +7 -5
  120. package/dist/esm/transaction/descriptor/parse.d.ts +2 -1
  121. package/dist/esm/transaction/descriptor/parse.d.ts.map +1 -1
  122. package/dist/esm/transaction/descriptor/parse.js +14 -14
  123. package/dist/esm/transaction/descriptor/verifyTransaction.d.ts +2 -1
  124. package/dist/esm/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  125. package/dist/esm/transaction/descriptor/verifyTransaction.js +4 -4
  126. package/dist/esm/transaction/explainTransaction.d.ts +2 -1
  127. package/dist/esm/transaction/explainTransaction.d.ts.map +1 -1
  128. package/dist/esm/transaction/explainTransaction.js +5 -5
  129. package/dist/esm/transaction/fetchInputs.d.ts +2 -1
  130. package/dist/esm/transaction/fetchInputs.d.ts.map +1 -1
  131. package/dist/esm/transaction/fetchInputs.js +4 -2
  132. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts +4 -4
  133. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  134. package/dist/esm/transaction/fixedScript/explainTransaction.js +24 -19
  135. package/dist/esm/transaction/fixedScript/parseTransaction.js +3 -3
  136. package/dist/esm/transaction/fixedScript/replayProtection.d.ts +6 -4
  137. package/dist/esm/transaction/fixedScript/replayProtection.d.ts.map +1 -1
  138. package/dist/esm/transaction/fixedScript/replayProtection.js +17 -16
  139. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts +4 -2
  140. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -1
  141. package/dist/esm/transaction/fixedScript/signLegacyTransaction.js +6 -5
  142. package/dist/esm/transaction/fixedScript/signTransaction.d.ts +2 -1
  143. package/dist/esm/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  144. package/dist/esm/transaction/fixedScript/signTransaction.js +4 -4
  145. package/dist/esm/transaction/fixedScript/verifyTransaction.js +2 -2
  146. package/dist/esm/transaction/getPayGoVerificationPubkey.d.ts +3 -3
  147. package/dist/esm/transaction/getPayGoVerificationPubkey.d.ts.map +1 -1
  148. package/dist/esm/transaction/getPayGoVerificationPubkey.js +6 -6
  149. package/dist/esm/transaction/recipient.d.ts +4 -4
  150. package/dist/esm/transaction/recipient.d.ts.map +1 -1
  151. package/dist/esm/transaction/recipient.js +7 -4
  152. package/dist/esm/transaction/signTransaction.js +2 -2
  153. package/package.json +12 -12
  154. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts +0 -3
  155. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +0 -1
  156. package/dist/cjs/test/unit/transaction/fixedScript/util.js +0 -41
@@ -1,10 +1,59 @@
1
1
  "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
2
35
  var __importDefault = (this && this.__importDefault) || function (mod) {
3
36
  return (mod && mod.__esModule) ? mod : { "default": mod };
4
37
  };
5
38
  Object.defineProperty(exports, "__esModule", { value: true });
6
39
  const assert_1 = __importDefault(require("assert"));
40
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
41
+ const wasm_utxo_1 = require("@bitgo/wasm-utxo");
7
42
  const bip322_1 = require("../../src/transaction/bip322");
43
+ function createTestWalletKeys(seed) {
44
+ const keys = utxolib.testutil.getKeyTriple(seed);
45
+ return {
46
+ xpubs: keys.map((k) => k.neutered().toBase58()),
47
+ xprivs: keys.map((k) => k.toBase58()),
48
+ };
49
+ }
50
+ function getDerivedPubkeys(seed, chain, index) {
51
+ const keys = utxolib.testutil.getKeyTriple(seed);
52
+ return keys.map((k) => k.derivePath(`m/0/0/${chain}/${index}`).publicKey.toString('hex'));
53
+ }
54
+ function getAddress(walletKeys, chain, index) {
55
+ return wasm_utxo_1.fixedScriptWallet.address(walletKeys, chain, index, utxolib.networks.bitcoin);
56
+ }
8
57
  describe('BIP322', function () {
9
58
  describe('BIP322MessageBroadcastable', () => {
10
59
  it('should serialize and deserialize correctly', () => {
@@ -14,7 +63,7 @@ describe('BIP322', function () {
14
63
  {
15
64
  address: 'someAddress',
16
65
  message: 'someMessage',
17
- pubkeys: ['pubkey1', 'pubkey2'],
66
+ pubkeys: ['pubkey1', 'pubkey2', 'pubkey3'],
18
67
  scriptType: 'p2sh',
19
68
  },
20
69
  ],
@@ -30,7 +79,7 @@ describe('BIP322', function () {
30
79
  {
31
80
  address: 'someAddress',
32
81
  message: 'someMessage',
33
- pubkeys: ['pubkey1', 'pubkey2'],
82
+ pubkeys: ['pubkey1', 'pubkey2', 'pubkey3'],
34
83
  scriptType: 'unsupported',
35
84
  },
36
85
  ],
@@ -41,5 +90,272 @@ describe('BIP322', function () {
41
90
  });
42
91
  });
43
92
  });
93
+ describe('verifyTransactionFromBroadcastableMessage', function () {
94
+ const seed = 'bip322_verify_test';
95
+ const { xpubs, xprivs } = createTestWalletKeys(seed);
96
+ const walletKeys = wasm_utxo_1.fixedScriptWallet.RootWalletKeys.from(xpubs);
97
+ it('should verify a valid p2shP2wsh PSBT proof', function () {
98
+ const chain = 10;
99
+ const index = 0;
100
+ const messageText = 'Hello, BitGo!';
101
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
102
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
103
+ message: messageText,
104
+ scriptId: { chain, index },
105
+ rootWalletKeys: walletKeys,
106
+ });
107
+ // Sign with user and bitgo keys
108
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
109
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
110
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
111
+ const address = getAddress(walletKeys, chain, index);
112
+ const broadcastable = {
113
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
114
+ messageInfo: [
115
+ {
116
+ address,
117
+ message: messageText,
118
+ pubkeys,
119
+ scriptType: 'p2shP2wsh',
120
+ },
121
+ ],
122
+ };
123
+ assert_1.default.strictEqual((0, bip322_1.verifyTransactionFromBroadcastableMessage)(broadcastable, 'btc'), true);
124
+ });
125
+ it('should verify a valid p2wsh PSBT proof', function () {
126
+ const chain = 20;
127
+ const index = 5;
128
+ const messageText = 'P2WSH proof test';
129
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
130
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
131
+ message: messageText,
132
+ scriptId: { chain, index },
133
+ rootWalletKeys: walletKeys,
134
+ });
135
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
136
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
137
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
138
+ const address = getAddress(walletKeys, chain, index);
139
+ const broadcastable = {
140
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
141
+ messageInfo: [
142
+ {
143
+ address,
144
+ message: messageText,
145
+ pubkeys,
146
+ scriptType: 'p2wsh',
147
+ },
148
+ ],
149
+ };
150
+ assert_1.default.strictEqual((0, bip322_1.verifyTransactionFromBroadcastableMessage)(broadcastable, 'btc'), true);
151
+ });
152
+ it('should verify multiple inputs in a single PSBT', function () {
153
+ const messages = ['Message 1', 'Message 2'];
154
+ const scriptIds = [
155
+ { chain: 10, index: 0 },
156
+ { chain: 20, index: 1 },
157
+ ];
158
+ const scriptTypes = ['p2shP2wsh', 'p2wsh'];
159
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
160
+ for (let i = 0; i < messages.length; i++) {
161
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
162
+ message: messages[i],
163
+ scriptId: scriptIds[i],
164
+ rootWalletKeys: walletKeys,
165
+ });
166
+ }
167
+ // Sign all inputs
168
+ for (let i = 0; i < messages.length; i++) {
169
+ psbt.sign(i, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
170
+ psbt.sign(i, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
171
+ }
172
+ const messageInfo = messages.map((msg, i) => {
173
+ const pubkeys = getDerivedPubkeys(seed, scriptIds[i].chain, scriptIds[i].index);
174
+ return {
175
+ address: getAddress(walletKeys, scriptIds[i].chain, scriptIds[i].index),
176
+ message: msg,
177
+ pubkeys,
178
+ scriptType: scriptTypes[i],
179
+ };
180
+ });
181
+ const broadcastable = {
182
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
183
+ messageInfo,
184
+ };
185
+ assert_1.default.strictEqual((0, bip322_1.verifyTransactionFromBroadcastableMessage)(broadcastable, 'btc'), true);
186
+ });
187
+ it('should return false for wrong message', function () {
188
+ const chain = 10;
189
+ const index = 0;
190
+ const messageText = 'Original message';
191
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
192
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
193
+ message: messageText,
194
+ scriptId: { chain, index },
195
+ rootWalletKeys: walletKeys,
196
+ });
197
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
198
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
199
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
200
+ const address = getAddress(walletKeys, chain, index);
201
+ const broadcastable = {
202
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
203
+ messageInfo: [
204
+ {
205
+ address,
206
+ message: 'Different message', // Wrong message
207
+ pubkeys,
208
+ scriptType: 'p2shP2wsh',
209
+ },
210
+ ],
211
+ };
212
+ assert_1.default.strictEqual((0, bip322_1.verifyTransactionFromBroadcastableMessage)(broadcastable, 'btc'), false);
213
+ });
214
+ it('should throw for unsupported coin', function () {
215
+ const broadcastable = {
216
+ txHex: '00',
217
+ messageInfo: [],
218
+ };
219
+ assert_1.default.throws(() => (0, bip322_1.verifyTransactionFromBroadcastableMessage)(broadcastable, 'ltc'), /Only tbtc4 or btc coinNames are supported/);
220
+ });
221
+ });
222
+ describe('generateBIP322MessageListAndVerifyFromMessageBroadcastable', function () {
223
+ const seed = 'bip322_generate_test';
224
+ const { xpubs, xprivs } = createTestWalletKeys(seed);
225
+ const walletKeys = wasm_utxo_1.fixedScriptWallet.RootWalletKeys.from(xpubs);
226
+ it('should generate message list from valid proofs', function () {
227
+ const chain = 10;
228
+ const index = 0;
229
+ const messageText = 'Proof message';
230
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
231
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
232
+ message: messageText,
233
+ scriptId: { chain, index },
234
+ rootWalletKeys: walletKeys,
235
+ });
236
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
237
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
238
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
239
+ const address = getAddress(walletKeys, chain, index);
240
+ const broadcastables = [
241
+ {
242
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
243
+ messageInfo: [
244
+ {
245
+ address,
246
+ message: messageText,
247
+ pubkeys,
248
+ scriptType: 'p2shP2wsh',
249
+ },
250
+ ],
251
+ },
252
+ ];
253
+ const result = (0, bip322_1.generateBIP322MessageListAndVerifyFromMessageBroadcastable)(broadcastables, 'btc');
254
+ assert_1.default.strictEqual(result.length, 1);
255
+ assert_1.default.strictEqual(result[0].address, address);
256
+ assert_1.default.strictEqual(result[0].message, messageText);
257
+ });
258
+ it('should deduplicate addresses with same message', function () {
259
+ const chain = 10;
260
+ const index = 0;
261
+ const messageText = 'Same message';
262
+ // Create two separate PSBTs for the same address/message
263
+ const psbt1 = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
264
+ wasm_utxo_1.bip322.addBip322Input(psbt1, {
265
+ message: messageText,
266
+ scriptId: { chain, index },
267
+ rootWalletKeys: walletKeys,
268
+ });
269
+ psbt1.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
270
+ psbt1.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
271
+ const psbt2 = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
272
+ wasm_utxo_1.bip322.addBip322Input(psbt2, {
273
+ message: messageText,
274
+ scriptId: { chain, index },
275
+ rootWalletKeys: walletKeys,
276
+ });
277
+ psbt2.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
278
+ psbt2.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
279
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
280
+ const address = getAddress(walletKeys, chain, index);
281
+ const broadcastables = [
282
+ {
283
+ txHex: Buffer.from(psbt1.serialize()).toString('hex'),
284
+ messageInfo: [{ address, message: messageText, pubkeys, scriptType: 'p2shP2wsh' }],
285
+ },
286
+ {
287
+ txHex: Buffer.from(psbt2.serialize()).toString('hex'),
288
+ messageInfo: [{ address, message: messageText, pubkeys, scriptType: 'p2shP2wsh' }],
289
+ },
290
+ ];
291
+ const result = (0, bip322_1.generateBIP322MessageListAndVerifyFromMessageBroadcastable)(broadcastables, 'btc');
292
+ // Should deduplicate to a single entry
293
+ assert_1.default.strictEqual(result.length, 1);
294
+ assert_1.default.strictEqual(result[0].address, address);
295
+ assert_1.default.strictEqual(result[0].message, messageText);
296
+ });
297
+ it('should throw for duplicate address with different message', function () {
298
+ const chain = 10;
299
+ const index = 0;
300
+ const psbt1 = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
301
+ wasm_utxo_1.bip322.addBip322Input(psbt1, {
302
+ message: 'Message 1',
303
+ scriptId: { chain, index },
304
+ rootWalletKeys: walletKeys,
305
+ });
306
+ psbt1.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
307
+ psbt1.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
308
+ const psbt2 = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
309
+ wasm_utxo_1.bip322.addBip322Input(psbt2, {
310
+ message: 'Message 2',
311
+ scriptId: { chain, index },
312
+ rootWalletKeys: walletKeys,
313
+ });
314
+ psbt2.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
315
+ psbt2.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
316
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
317
+ const address = getAddress(walletKeys, chain, index);
318
+ const broadcastables = [
319
+ {
320
+ txHex: Buffer.from(psbt1.serialize()).toString('hex'),
321
+ messageInfo: [{ address, message: 'Message 1', pubkeys, scriptType: 'p2shP2wsh' }],
322
+ },
323
+ {
324
+ txHex: Buffer.from(psbt2.serialize()).toString('hex'),
325
+ messageInfo: [{ address, message: 'Message 2', pubkeys, scriptType: 'p2shP2wsh' }],
326
+ },
327
+ ];
328
+ assert_1.default.throws(() => (0, bip322_1.generateBIP322MessageListAndVerifyFromMessageBroadcastable)(broadcastables, 'btc'), /Duplicate address.*has different messages/);
329
+ });
330
+ it('should throw for invalid proof', function () {
331
+ const chain = 10;
332
+ const index = 0;
333
+ const messageText = 'Valid message';
334
+ const psbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });
335
+ wasm_utxo_1.bip322.addBip322Input(psbt, {
336
+ message: messageText,
337
+ scriptId: { chain, index },
338
+ rootWalletKeys: walletKeys,
339
+ });
340
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[0]));
341
+ psbt.sign(0, wasm_utxo_1.BIP32.fromBase58(xprivs[2]));
342
+ const pubkeys = getDerivedPubkeys(seed, chain, index);
343
+ const address = getAddress(walletKeys, chain, index);
344
+ const broadcastables = [
345
+ {
346
+ txHex: Buffer.from(psbt.serialize()).toString('hex'),
347
+ messageInfo: [
348
+ {
349
+ address,
350
+ message: 'Wrong message', // Doesn't match what was signed
351
+ pubkeys,
352
+ scriptType: 'p2shP2wsh',
353
+ },
354
+ ],
355
+ },
356
+ ];
357
+ assert_1.default.throws(() => (0, bip322_1.generateBIP322MessageListAndVerifyFromMessageBroadcastable)(broadcastables, 'btc'), /did not have a successful validation/);
358
+ });
359
+ });
44
360
  });
45
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmlwMzIyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdGVzdC91bml0L2JpcDMyMi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLG9EQUE0QjtBQUU1Qix5REFJc0M7QUFFdEMsUUFBUSxDQUFDLFFBQVEsRUFBRTtJQUNqQixRQUFRLENBQUMsNEJBQTRCLEVBQUUsR0FBRyxFQUFFO1FBQzFDLEVBQUUsQ0FBQyw0Q0FBNEMsRUFBRSxHQUFHLEVBQUU7WUFDcEQsTUFBTSxPQUFPLEdBQStCO2dCQUMxQyxLQUFLLEVBQUUsUUFBUTtnQkFDZixXQUFXLEVBQUU7b0JBQ1g7d0JBQ0UsT0FBTyxFQUFFLGFBQWE7d0JBQ3RCLE9BQU8sRUFBRSxhQUFhO3dCQUN0QixPQUFPLEVBQUUsQ0FBQyxTQUFTLEVBQUUsU0FBUyxDQUFDO3dCQUMvQixVQUFVLEVBQUUsTUFBTTtxQkFDbkI7aUJBQ0Y7YUFDRixDQUFDO1lBRUYsTUFBTSxVQUFVLEdBQUcsSUFBQSw0Q0FBbUMsRUFBQyxPQUFPLENBQUMsQ0FBQztZQUNoRSxNQUFNLFlBQVksR0FBRyxJQUFBLDhDQUFxQyxFQUFDLFVBQVUsQ0FBQyxDQUFDO1lBQ3ZFLGdCQUFNLENBQUMsZUFBZSxDQUFDLFlBQVksRUFBRSxPQUFPLENBQUMsQ0FBQztRQUNoRCxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyxvREFBb0QsRUFBRTtZQUN2RCxNQUFNLE9BQU8sR0FBRztnQkFDZCxLQUFLLEVBQUUsUUFBUTtnQkFDZixXQUFXLEVBQUU7b0JBQ1g7d0JBQ0UsT0FBTyxFQUFFLGFBQWE7d0JBQ3RCLE9BQU8sRUFBRSxhQUFhO3dCQUN0QixPQUFPLEVBQUUsQ0FBQyxTQUFTLEVBQUUsU0FBUyxDQUFDO3dCQUMvQixVQUFVLEVBQUUsYUFBYTtxQkFDMUI7aUJBQ0Y7YUFDdUMsQ0FBQztZQUUzQyxNQUFNLFVBQVUsR0FBRyxJQUFBLDRDQUFtQyxFQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQ2hFLGdCQUFNLENBQUMsTUFBTSxDQUFDLEdBQUcsRUFBRTtnQkFDakIsSUFBQSw4Q0FBcUMsRUFBQyxVQUFVLENBQUMsQ0FBQztZQUNwRCxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBhc3NlcnQgZnJvbSAnYXNzZXJ0JztcblxuaW1wb3J0IHtcbiAgQklQMzIyTWVzc2FnZUJyb2FkY2FzdGFibGUsXG4gIGRlc2VyaWFsaXplQklQMzIyQnJvYWRjYXN0YWJsZU1lc3NhZ2UsXG4gIHNlcmlhbGl6ZUJJUDMyMkJyb2FkY2FzdGFibGVNZXNzYWdlLFxufSBmcm9tICcuLi8uLi9zcmMvdHJhbnNhY3Rpb24vYmlwMzIyJztcblxuZGVzY3JpYmUoJ0JJUDMyMicsIGZ1bmN0aW9uICgpIHtcbiAgZGVzY3JpYmUoJ0JJUDMyMk1lc3NhZ2VCcm9hZGNhc3RhYmxlJywgKCkgPT4ge1xuICAgIGl0KCdzaG91bGQgc2VyaWFsaXplIGFuZCBkZXNlcmlhbGl6ZSBjb3JyZWN0bHknLCAoKSA9PiB7XG4gICAgICBjb25zdCBtZXNzYWdlOiBCSVAzMjJNZXNzYWdlQnJvYWRjYXN0YWJsZSA9IHtcbiAgICAgICAgdHhIZXg6ICcwMTAyMDMnLFxuICAgICAgICBtZXNzYWdlSW5mbzogW1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIGFkZHJlc3M6ICdzb21lQWRkcmVzcycsXG4gICAgICAgICAgICBtZXNzYWdlOiAnc29tZU1lc3NhZ2UnLFxuICAgICAgICAgICAgcHVia2V5czogWydwdWJrZXkxJywgJ3B1YmtleTInXSxcbiAgICAgICAgICAgIHNjcmlwdFR5cGU6ICdwMnNoJyxcbiAgICAgICAgICB9LFxuICAgICAgICBdLFxuICAgICAgfTtcblxuICAgICAgY29uc3Qgc2VyaWFsaXplZCA9IHNlcmlhbGl6ZUJJUDMyMkJyb2FkY2FzdGFibGVNZXNzYWdlKG1lc3NhZ2UpO1xuICAgICAgY29uc3QgZGVzZXJpYWxpemVkID0gZGVzZXJpYWxpemVCSVAzMjJCcm9hZGNhc3RhYmxlTWVzc2FnZShzZXJpYWxpemVkKTtcbiAgICAgIGFzc2VydC5kZWVwU3RyaWN0RXF1YWwoZGVzZXJpYWxpemVkLCBtZXNzYWdlKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgZmFpbCBpZiB0aGVyZSBpcyBhbiB1bnN1cHBvcnRlZCBzY3JpcHQgdHlwZScsIGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IG1lc3NhZ2UgPSB7XG4gICAgICAgIHR4SGV4OiAnMDEwMjAzJyxcbiAgICAgICAgbWVzc2FnZUluZm86IFtcbiAgICAgICAgICB7XG4gICAgICAgICAgICBhZGRyZXNzOiAnc29tZUFkZHJlc3MnLFxuICAgICAgICAgICAgbWVzc2FnZTogJ3NvbWVNZXNzYWdlJyxcbiAgICAgICAgICAgIHB1YmtleXM6IFsncHVia2V5MScsICdwdWJrZXkyJ10sXG4gICAgICAgICAgICBzY3JpcHRUeXBlOiAndW5zdXBwb3J0ZWQnLFxuICAgICAgICAgIH0sXG4gICAgICAgIF0sXG4gICAgICB9IGFzIHVua25vd24gYXMgQklQMzIyTWVzc2FnZUJyb2FkY2FzdGFibGU7XG5cbiAgICAgIGNvbnN0IHNlcmlhbGl6ZWQgPSBzZXJpYWxpemVCSVAzMjJCcm9hZGNhc3RhYmxlTWVzc2FnZShtZXNzYWdlKTtcbiAgICAgIGFzc2VydC50aHJvd3MoKCkgPT4ge1xuICAgICAgICBkZXNlcmlhbGl6ZUJJUDMyMkJyb2FkY2FzdGFibGVNZXNzYWdlKHNlcmlhbGl6ZWQpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH0pO1xufSk7XG4iXX0=
361
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bip322.js","sourceRoot":"","sources":["../../../../test/unit/bip322.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip322 as wasmBip322, fixedScriptWallet, BIP32, type Triple } from '@bitgo/wasm-utxo';\n\nimport {\n  BIP322MessageBroadcastable,\n  BIP322MessageInfo,\n  deserializeBIP322BroadcastableMessage,\n  generateBIP322MessageListAndVerifyFromMessageBroadcastable,\n  serializeBIP322BroadcastableMessage,\n  verifyTransactionFromBroadcastableMessage,\n} from '../../src/transaction/bip322';\n\nfunction createTestWalletKeys(seed: string): {\n  xpubs: Triple<string>;\n  xprivs: Triple<string>;\n} {\n  const keys = utxolib.testutil.getKeyTriple(seed);\n  return {\n    xpubs: keys.map((k) => k.neutered().toBase58()) as Triple<string>,\n    xprivs: keys.map((k) => k.toBase58()) as Triple<string>,\n  };\n}\n\nfunction getDerivedPubkeys(seed: string, chain: number, index: number): Triple<string> {\n  const keys = utxolib.testutil.getKeyTriple(seed);\n  return keys.map((k) => k.derivePath(`m/0/0/${chain}/${index}`).publicKey.toString('hex')) as Triple<string>;\n}\n\nfunction getAddress(walletKeys: fixedScriptWallet.RootWalletKeys, chain: number, index: number): string {\n  return fixedScriptWallet.address(walletKeys, chain, index, utxolib.networks.bitcoin);\n}\n\ndescribe('BIP322', function () {\n  describe('BIP322MessageBroadcastable', () => {\n    it('should serialize and deserialize correctly', () => {\n      const message: BIP322MessageBroadcastable = {\n        txHex: '010203',\n        messageInfo: [\n          {\n            address: 'someAddress',\n            message: 'someMessage',\n            pubkeys: ['pubkey1', 'pubkey2', 'pubkey3'],\n            scriptType: 'p2sh',\n          },\n        ],\n      };\n\n      const serialized = serializeBIP322BroadcastableMessage(message);\n      const deserialized = deserializeBIP322BroadcastableMessage(serialized);\n      assert.deepStrictEqual(deserialized, message);\n    });\n\n    it('should fail if there is an unsupported script type', function () {\n      const message = {\n        txHex: '010203',\n        messageInfo: [\n          {\n            address: 'someAddress',\n            message: 'someMessage',\n            pubkeys: ['pubkey1', 'pubkey2', 'pubkey3'],\n            scriptType: 'unsupported',\n          },\n        ],\n      } as unknown as BIP322MessageBroadcastable;\n\n      const serialized = serializeBIP322BroadcastableMessage(message);\n      assert.throws(() => {\n        deserializeBIP322BroadcastableMessage(serialized);\n      });\n    });\n  });\n\n  describe('verifyTransactionFromBroadcastableMessage', function () {\n    const seed = 'bip322_verify_test';\n    const { xpubs, xprivs } = createTestWalletKeys(seed);\n    const walletKeys = fixedScriptWallet.RootWalletKeys.from(xpubs);\n\n    it('should verify a valid p2shP2wsh PSBT proof', function () {\n      const chain = 10;\n      const index = 0;\n      const messageText = 'Hello, BitGo!';\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n\n      // Sign with user and bitgo keys\n      psbt.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastable: BIP322MessageBroadcastable = {\n        txHex: Buffer.from(psbt.serialize()).toString('hex'),\n        messageInfo: [\n          {\n            address,\n            message: messageText,\n            pubkeys,\n            scriptType: 'p2shP2wsh',\n          },\n        ],\n      };\n\n      assert.strictEqual(verifyTransactionFromBroadcastableMessage(broadcastable, 'btc'), true);\n    });\n\n    it('should verify a valid p2wsh PSBT proof', function () {\n      const chain = 20;\n      const index = 5;\n      const messageText = 'P2WSH proof test';\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n\n      psbt.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastable: BIP322MessageBroadcastable = {\n        txHex: Buffer.from(psbt.serialize()).toString('hex'),\n        messageInfo: [\n          {\n            address,\n            message: messageText,\n            pubkeys,\n            scriptType: 'p2wsh',\n          },\n        ],\n      };\n\n      assert.strictEqual(verifyTransactionFromBroadcastableMessage(broadcastable, 'btc'), true);\n    });\n\n    it('should verify multiple inputs in a single PSBT', function () {\n      const messages = ['Message 1', 'Message 2'];\n      const scriptIds = [\n        { chain: 10, index: 0 },\n        { chain: 20, index: 1 },\n      ];\n      const scriptTypes: BIP322MessageInfo['scriptType'][] = ['p2shP2wsh', 'p2wsh'];\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n\n      for (let i = 0; i < messages.length; i++) {\n        wasmBip322.addBip322Input(psbt, {\n          message: messages[i],\n          scriptId: scriptIds[i],\n          rootWalletKeys: walletKeys,\n        });\n      }\n\n      // Sign all inputs\n      for (let i = 0; i < messages.length; i++) {\n        psbt.sign(i, BIP32.fromBase58(xprivs[0]));\n        psbt.sign(i, BIP32.fromBase58(xprivs[2]));\n      }\n\n      const messageInfo: BIP322MessageInfo[] = messages.map((msg, i) => {\n        const pubkeys = getDerivedPubkeys(seed, scriptIds[i].chain, scriptIds[i].index);\n        return {\n          address: getAddress(walletKeys, scriptIds[i].chain, scriptIds[i].index),\n          message: msg,\n          pubkeys,\n          scriptType: scriptTypes[i],\n        };\n      });\n\n      const broadcastable: BIP322MessageBroadcastable = {\n        txHex: Buffer.from(psbt.serialize()).toString('hex'),\n        messageInfo,\n      };\n\n      assert.strictEqual(verifyTransactionFromBroadcastableMessage(broadcastable, 'btc'), true);\n    });\n\n    it('should return false for wrong message', function () {\n      const chain = 10;\n      const index = 0;\n      const messageText = 'Original message';\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n\n      psbt.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastable: BIP322MessageBroadcastable = {\n        txHex: Buffer.from(psbt.serialize()).toString('hex'),\n        messageInfo: [\n          {\n            address,\n            message: 'Different message', // Wrong message\n            pubkeys,\n            scriptType: 'p2shP2wsh',\n          },\n        ],\n      };\n\n      assert.strictEqual(verifyTransactionFromBroadcastableMessage(broadcastable, 'btc'), false);\n    });\n\n    it('should throw for unsupported coin', function () {\n      const broadcastable: BIP322MessageBroadcastable = {\n        txHex: '00',\n        messageInfo: [],\n      };\n\n      assert.throws(\n        () => verifyTransactionFromBroadcastableMessage(broadcastable, 'ltc'),\n        /Only tbtc4 or btc coinNames are supported/\n      );\n    });\n  });\n\n  describe('generateBIP322MessageListAndVerifyFromMessageBroadcastable', function () {\n    const seed = 'bip322_generate_test';\n    const { xpubs, xprivs } = createTestWalletKeys(seed);\n    const walletKeys = fixedScriptWallet.RootWalletKeys.from(xpubs);\n\n    it('should generate message list from valid proofs', function () {\n      const chain = 10;\n      const index = 0;\n      const messageText = 'Proof message';\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n\n      psbt.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastables: BIP322MessageBroadcastable[] = [\n        {\n          txHex: Buffer.from(psbt.serialize()).toString('hex'),\n          messageInfo: [\n            {\n              address,\n              message: messageText,\n              pubkeys,\n              scriptType: 'p2shP2wsh',\n            },\n          ],\n        },\n      ];\n\n      const result = generateBIP322MessageListAndVerifyFromMessageBroadcastable(broadcastables, 'btc');\n\n      assert.strictEqual(result.length, 1);\n      assert.strictEqual(result[0].address, address);\n      assert.strictEqual(result[0].message, messageText);\n    });\n\n    it('should deduplicate addresses with same message', function () {\n      const chain = 10;\n      const index = 0;\n      const messageText = 'Same message';\n\n      // Create two separate PSBTs for the same address/message\n      const psbt1 = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt1, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n      psbt1.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt1.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const psbt2 = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt2, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n      psbt2.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt2.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastables: BIP322MessageBroadcastable[] = [\n        {\n          txHex: Buffer.from(psbt1.serialize()).toString('hex'),\n          messageInfo: [{ address, message: messageText, pubkeys, scriptType: 'p2shP2wsh' }],\n        },\n        {\n          txHex: Buffer.from(psbt2.serialize()).toString('hex'),\n          messageInfo: [{ address, message: messageText, pubkeys, scriptType: 'p2shP2wsh' }],\n        },\n      ];\n\n      const result = generateBIP322MessageListAndVerifyFromMessageBroadcastable(broadcastables, 'btc');\n\n      // Should deduplicate to a single entry\n      assert.strictEqual(result.length, 1);\n      assert.strictEqual(result[0].address, address);\n      assert.strictEqual(result[0].message, messageText);\n    });\n\n    it('should throw for duplicate address with different message', function () {\n      const chain = 10;\n      const index = 0;\n\n      const psbt1 = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt1, {\n        message: 'Message 1',\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n      psbt1.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt1.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const psbt2 = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt2, {\n        message: 'Message 2',\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n      psbt2.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt2.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastables: BIP322MessageBroadcastable[] = [\n        {\n          txHex: Buffer.from(psbt1.serialize()).toString('hex'),\n          messageInfo: [{ address, message: 'Message 1', pubkeys, scriptType: 'p2shP2wsh' }],\n        },\n        {\n          txHex: Buffer.from(psbt2.serialize()).toString('hex'),\n          messageInfo: [{ address, message: 'Message 2', pubkeys, scriptType: 'p2shP2wsh' }],\n        },\n      ];\n\n      assert.throws(\n        () => generateBIP322MessageListAndVerifyFromMessageBroadcastable(broadcastables, 'btc'),\n        /Duplicate address.*has different messages/\n      );\n    });\n\n    it('should throw for invalid proof', function () {\n      const chain = 10;\n      const index = 0;\n      const messageText = 'Valid message';\n\n      const psbt = fixedScriptWallet.BitGoPsbt.createEmpty('btc', walletKeys, { version: 0 });\n      wasmBip322.addBip322Input(psbt, {\n        message: messageText,\n        scriptId: { chain, index },\n        rootWalletKeys: walletKeys,\n      });\n      psbt.sign(0, BIP32.fromBase58(xprivs[0]));\n      psbt.sign(0, BIP32.fromBase58(xprivs[2]));\n\n      const pubkeys = getDerivedPubkeys(seed, chain, index);\n      const address = getAddress(walletKeys, chain, index);\n\n      const broadcastables: BIP322MessageBroadcastable[] = [\n        {\n          txHex: Buffer.from(psbt.serialize()).toString('hex'),\n          messageInfo: [\n            {\n              address,\n              message: 'Wrong message', // Doesn't match what was signed\n              pubkeys,\n              scriptType: 'p2shP2wsh',\n            },\n          ],\n        },\n      ];\n\n      assert.throws(\n        () => generateBIP322MessageListAndVerifyFromMessageBroadcastable(broadcastables, 'btc'),\n        /did not have a successful validation/\n      );\n    });\n  });\n});\n"]}
@@ -69,7 +69,7 @@ describe('Custom Change Wallets', () => {
69
69
  ],
70
70
  threshold: 2,
71
71
  };
72
- const changeAddress = (0, src_1.generateAddress)(coin.network, addressData);
72
+ const changeAddress = (0, src_1.generateAddress)(coin.name, addressData);
73
73
  const changeWalletId = 'changeWalletId';
74
74
  const stubData = {
75
75
  signedSendingWallet: {
@@ -151,4 +151,4 @@ describe('Custom Change Wallets', () => {
151
151
  coin.keychains.restore();
152
152
  });
153
153
  });
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"customChangeWallet.js","sourceRoot":"","sources":["../../../../test/unit/customChangeWallet.ts"],"names":[],"mappings":"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","sourcesContent":["import should = require('should');\nimport * as sinon from 'sinon';\nimport { Wallet } from '@bitgo-beta/sdk-core';\n\nimport { generateAddress } from '../../src';\n\nimport { getUtxoCoin } from './util';\n\ndescribe('Custom Change Wallets', () => {\n  const coin = getUtxoCoin('tbtc');\n\n  const keys = {\n    send: {\n      user: { id: '0', key: coin.keychains().create() },\n      backup: { id: '1', key: coin.keychains().create() },\n      bitgo: { id: '2', key: coin.keychains().create() },\n    },\n    change: {\n      user: { id: '3', key: coin.keychains().create() },\n      backup: { id: '4', key: coin.keychains().create() },\n      bitgo: { id: '5', key: coin.keychains().create() },\n    },\n  };\n\n  const customChangeKeySignatures = {\n    user: '',\n    backup: '',\n    bitgo: '',\n  };\n\n  const addressData = {\n    chain: 11,\n    index: 1,\n    addressType: 'p2shP2wsh' as const,\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    keychains: [\n      { pub: keys.change.user.key.pub! },\n      { pub: keys.change.backup.key.pub! },\n      { pub: keys.change.bitgo.key.pub! },\n    ],\n    threshold: 2,\n  };\n\n  const changeAddress = generateAddress(coin.network, addressData);\n\n  const changeWalletId = 'changeWalletId';\n  const stubData = {\n    signedSendingWallet: {\n      keyIds: sinon.stub().returns([keys.send.user.id, keys.send.backup.id, keys.send.bitgo.id]),\n      coinSpecific: sinon.stub().returns({ customChangeWalletId: changeWalletId }),\n    },\n    changeWallet: {\n      keyIds: sinon.stub().returns([keys.change.user.id, keys.change.backup.id, keys.change.bitgo.id]),\n      createAddress: sinon.stub().resolves(changeAddress),\n    },\n  };\n\n  before(async () => {\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    const sign = async ({ key }) => (await coin.signMessage({ prv: keys.send.user.key.prv }, key.pub!)).toString('hex');\n    customChangeKeySignatures.user = await sign(keys.change.user);\n    customChangeKeySignatures.backup = await sign(keys.change.backup);\n    customChangeKeySignatures.bitgo = await sign(keys.change.bitgo);\n  });\n\n  it('should consider addresses derived from the custom change keys as internal spends', async () => {\n    const signedSendingWallet = sinon.createStubInstance(Wallet, stubData.signedSendingWallet as any);\n    const changeWallet = sinon.createStubInstance(Wallet, stubData.changeWallet as any);\n\n    sinon.stub(coin, 'keychains').returns({\n      get: sinon.stub().callsFake(({ id }) => {\n        switch (id) {\n          case keys.send.user.id:\n            return Promise.resolve({ id, ...keys.send.user.key });\n          case keys.send.backup.id:\n            return Promise.resolve({ id, ...keys.send.backup.key });\n          case keys.send.bitgo.id:\n            return Promise.resolve({ id, ...keys.send.bitgo.key });\n          case keys.change.user.id:\n            return Promise.resolve({ id, ...keys.change.user.key });\n          case keys.change.backup.id:\n            return Promise.resolve({ id, ...keys.change.backup.key });\n          case keys.change.bitgo.id:\n            return Promise.resolve({ id, ...keys.change.bitgo.key });\n        }\n      }),\n    } as any);\n\n    sinon.stub(coin, 'wallets').returns({\n      get: sinon.stub().callsFake(() => Promise.resolve(changeWallet)),\n    } as any);\n\n    const outputAmount = 10000;\n    const recipients = [];\n\n    sinon.stub(coin, 'explainTransaction').resolves({\n      outputs: [],\n      changeOutputs: [\n        {\n          address: changeAddress,\n          amount: outputAmount,\n        },\n      ],\n    } as any);\n\n    signedSendingWallet._wallet = signedSendingWallet._wallet || {\n      customChangeKeySignatures,\n    };\n\n    const parsedTransaction = await coin.parseTransaction({\n      txParams: { changeAddress, recipients },\n      txPrebuild: { txHex: '' },\n      wallet: signedSendingWallet as any,\n      verification: {\n        addresses: {\n          [changeAddress]: {\n            chain: addressData.chain,\n            index: addressData.index,\n          },\n        },\n      },\n    });\n\n    should.exist(parsedTransaction.outputs[0]);\n    parsedTransaction.outputs[0].should.deepEqual({\n      address: changeAddress,\n      amount: outputAmount,\n      external: false,\n      needsCustomChangeKeySignatureVerification: true,\n    });\n\n    (coin.explainTransaction as any).restore();\n    (coin.wallets as any).restore();\n    (coin.keychains as any).restore();\n  });\n});\n"]}
154
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"customChangeWallet.js","sourceRoot":"","sources":["../../../../test/unit/customChangeWallet.ts"],"names":[],"mappings":"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","sourcesContent":["import should = require('should');\nimport * as sinon from 'sinon';\nimport { Wallet } from '@bitgo-beta/sdk-core';\n\nimport { generateAddress } from '../../src';\n\nimport { getUtxoCoin } from './util';\n\ndescribe('Custom Change Wallets', () => {\n  const coin = getUtxoCoin('tbtc');\n\n  const keys = {\n    send: {\n      user: { id: '0', key: coin.keychains().create() },\n      backup: { id: '1', key: coin.keychains().create() },\n      bitgo: { id: '2', key: coin.keychains().create() },\n    },\n    change: {\n      user: { id: '3', key: coin.keychains().create() },\n      backup: { id: '4', key: coin.keychains().create() },\n      bitgo: { id: '5', key: coin.keychains().create() },\n    },\n  };\n\n  const customChangeKeySignatures = {\n    user: '',\n    backup: '',\n    bitgo: '',\n  };\n\n  const addressData = {\n    chain: 11,\n    index: 1,\n    addressType: 'p2shP2wsh' as const,\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    keychains: [\n      { pub: keys.change.user.key.pub! },\n      { pub: keys.change.backup.key.pub! },\n      { pub: keys.change.bitgo.key.pub! },\n    ],\n    threshold: 2,\n  };\n\n  const changeAddress = generateAddress(coin.name, addressData);\n\n  const changeWalletId = 'changeWalletId';\n  const stubData = {\n    signedSendingWallet: {\n      keyIds: sinon.stub().returns([keys.send.user.id, keys.send.backup.id, keys.send.bitgo.id]),\n      coinSpecific: sinon.stub().returns({ customChangeWalletId: changeWalletId }),\n    },\n    changeWallet: {\n      keyIds: sinon.stub().returns([keys.change.user.id, keys.change.backup.id, keys.change.bitgo.id]),\n      createAddress: sinon.stub().resolves(changeAddress),\n    },\n  };\n\n  before(async () => {\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    const sign = async ({ key }) => (await coin.signMessage({ prv: keys.send.user.key.prv }, key.pub!)).toString('hex');\n    customChangeKeySignatures.user = await sign(keys.change.user);\n    customChangeKeySignatures.backup = await sign(keys.change.backup);\n    customChangeKeySignatures.bitgo = await sign(keys.change.bitgo);\n  });\n\n  it('should consider addresses derived from the custom change keys as internal spends', async () => {\n    const signedSendingWallet = sinon.createStubInstance(Wallet, stubData.signedSendingWallet as any);\n    const changeWallet = sinon.createStubInstance(Wallet, stubData.changeWallet as any);\n\n    sinon.stub(coin, 'keychains').returns({\n      get: sinon.stub().callsFake(({ id }) => {\n        switch (id) {\n          case keys.send.user.id:\n            return Promise.resolve({ id, ...keys.send.user.key });\n          case keys.send.backup.id:\n            return Promise.resolve({ id, ...keys.send.backup.key });\n          case keys.send.bitgo.id:\n            return Promise.resolve({ id, ...keys.send.bitgo.key });\n          case keys.change.user.id:\n            return Promise.resolve({ id, ...keys.change.user.key });\n          case keys.change.backup.id:\n            return Promise.resolve({ id, ...keys.change.backup.key });\n          case keys.change.bitgo.id:\n            return Promise.resolve({ id, ...keys.change.bitgo.key });\n        }\n      }),\n    } as any);\n\n    sinon.stub(coin, 'wallets').returns({\n      get: sinon.stub().callsFake(() => Promise.resolve(changeWallet)),\n    } as any);\n\n    const outputAmount = 10000;\n    const recipients = [];\n\n    sinon.stub(coin, 'explainTransaction').resolves({\n      outputs: [],\n      changeOutputs: [\n        {\n          address: changeAddress,\n          amount: outputAmount,\n        },\n      ],\n    } as any);\n\n    signedSendingWallet._wallet = signedSendingWallet._wallet || {\n      customChangeKeySignatures,\n    };\n\n    const parsedTransaction = await coin.parseTransaction({\n      txParams: { changeAddress, recipients },\n      txPrebuild: { txHex: '' },\n      wallet: signedSendingWallet as any,\n      verification: {\n        addresses: {\n          [changeAddress]: {\n            chain: addressData.chain,\n            index: addressData.index,\n          },\n        },\n      },\n    });\n\n    should.exist(parsedTransaction.outputs[0]);\n    parsedTransaction.outputs[0].should.deepEqual({\n      address: changeAddress,\n      amount: outputAmount,\n      external: false,\n      needsCustomChangeKeySignatureVerification: true,\n    });\n\n    (coin.explainTransaction as any).restore();\n    (coin.wallets as any).restore();\n    (coin.keychains as any).restore();\n  });\n});\n"]}