@bitgo-beta/abstract-lightning 1.0.1-beta.82 → 1.0.1-beta.820

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 (69) hide show
  1. package/.mocharc.yml +1 -1
  2. package/CHANGELOG.md +385 -0
  3. package/dist/src/abstractLightningCoin.d.ts +3 -2
  4. package/dist/src/abstractLightningCoin.d.ts.map +1 -1
  5. package/dist/src/abstractLightningCoin.js +7 -2
  6. package/dist/src/codecs/api/backup.d.ts +15 -0
  7. package/dist/src/codecs/api/backup.d.ts.map +1 -0
  8. package/dist/src/codecs/api/backup.js +47 -0
  9. package/dist/src/codecs/api/balance.d.ts +132 -0
  10. package/dist/src/codecs/api/balance.d.ts.map +1 -0
  11. package/dist/src/codecs/api/balance.js +101 -0
  12. package/dist/src/codecs/api/index.d.ts +8 -0
  13. package/dist/src/codecs/api/index.d.ts.map +1 -0
  14. package/dist/src/codecs/api/index.js +24 -0
  15. package/dist/src/codecs/api/invoice.d.ts +66 -0
  16. package/dist/src/codecs/api/invoice.d.ts.map +1 -0
  17. package/dist/src/codecs/api/invoice.js +102 -0
  18. package/dist/src/codecs/api/payment.d.ts +107 -0
  19. package/dist/src/codecs/api/payment.d.ts.map +1 -0
  20. package/dist/src/codecs/api/payment.js +153 -0
  21. package/dist/src/codecs/api/transaction.d.ts +82 -0
  22. package/dist/src/codecs/api/transaction.d.ts.map +1 -0
  23. package/dist/src/codecs/api/transaction.js +104 -0
  24. package/dist/src/codecs/api/wallet.d.ts +99 -0
  25. package/dist/src/codecs/api/wallet.d.ts.map +1 -0
  26. package/dist/src/codecs/api/wallet.js +103 -0
  27. package/dist/src/codecs/api/withdraw.d.ts +97 -0
  28. package/dist/src/codecs/api/withdraw.d.ts.map +1 -0
  29. package/dist/src/codecs/api/withdraw.js +92 -0
  30. package/dist/src/codecs/index.d.ts +3 -0
  31. package/dist/src/codecs/index.d.ts.map +1 -0
  32. package/dist/src/codecs/index.js +19 -0
  33. package/dist/src/codecs/shared.d.ts +7 -0
  34. package/dist/src/codecs/shared.d.ts.map +1 -0
  35. package/dist/src/codecs/shared.js +42 -0
  36. package/dist/src/index.d.ts +3 -0
  37. package/dist/src/index.d.ts.map +1 -1
  38. package/dist/src/index.js +4 -1
  39. package/dist/src/lightning/index.d.ts +5 -0
  40. package/dist/src/lightning/index.d.ts.map +1 -0
  41. package/dist/src/lightning/index.js +21 -0
  42. package/dist/src/lightning/lightningUtils.d.ts +80 -0
  43. package/dist/src/lightning/lightningUtils.d.ts.map +1 -0
  44. package/dist/src/lightning/lightningUtils.js +258 -0
  45. package/dist/src/lightning/parseWithdrawPsbt.d.ts +8 -0
  46. package/dist/src/lightning/parseWithdrawPsbt.d.ts.map +1 -0
  47. package/dist/src/lightning/parseWithdrawPsbt.js +149 -0
  48. package/dist/src/lightning/signableJson.d.ts +17 -0
  49. package/dist/src/lightning/signableJson.d.ts.map +1 -0
  50. package/dist/src/lightning/signableJson.js +29 -0
  51. package/dist/src/lightning/signature.d.ts +22 -0
  52. package/dist/src/lightning/signature.d.ts.map +1 -0
  53. package/dist/src/lightning/signature.js +69 -0
  54. package/dist/src/wallet/custodialLightning.d.ts +7 -0
  55. package/dist/src/wallet/custodialLightning.d.ts.map +1 -0
  56. package/dist/src/wallet/custodialLightning.js +14 -0
  57. package/dist/src/wallet/index.d.ts +5 -0
  58. package/dist/src/wallet/index.d.ts.map +1 -0
  59. package/dist/src/wallet/index.js +21 -0
  60. package/dist/src/wallet/lightning.d.ts +146 -0
  61. package/dist/src/wallet/lightning.d.ts.map +1 -0
  62. package/dist/src/wallet/lightning.js +307 -0
  63. package/dist/src/wallet/selfCustodialLightning.d.ts +32 -0
  64. package/dist/src/wallet/selfCustodialLightning.d.ts.map +1 -0
  65. package/dist/src/wallet/selfCustodialLightning.js +131 -0
  66. package/dist/src/wallet/wallet.d.ts +7 -0
  67. package/dist/src/wallet/wallet.d.ts.map +1 -0
  68. package/dist/src/wallet/wallet.js +22 -0
  69. package/package.json +15 -5
@@ -0,0 +1,258 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.lightningNetworkName = exports.signerMacaroonPermissions = void 0;
37
+ exports.isLightningCoinName = isLightningCoinName;
38
+ exports.getLightningNetwork = getLightningNetwork;
39
+ exports.getLightningCoinName = getLightningCoinName;
40
+ exports.isValidLightningNetworkName = isValidLightningNetworkName;
41
+ exports.isValidLightningNetwork = isValidLightningNetwork;
42
+ exports.getStaticsLightningNetwork = getStaticsLightningNetwork;
43
+ exports.getUtxolibNetwork = getUtxolibNetwork;
44
+ exports.unwrapLightningCoinSpecific = unwrapLightningCoinSpecific;
45
+ exports.addIPCaveatToMacaroon = addIPCaveatToMacaroon;
46
+ exports.deriveWatchOnlyAccounts = deriveWatchOnlyAccounts;
47
+ exports.createWatchOnly = createWatchOnly;
48
+ exports.deriveLightningServiceSharedSecret = deriveLightningServiceSharedSecret;
49
+ exports.deriveMiddlewareSharedSecret = deriveMiddlewareSharedSecret;
50
+ exports.deriveTatSharedSecret = deriveTatSharedSecret;
51
+ exports.computeBip32DerivationIndexFromSeed = computeBip32DerivationIndexFromSeed;
52
+ const statics = __importStar(require("@bitgo-beta/statics"));
53
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
54
+ const crypto_1 = require("crypto");
55
+ const macaroon_1 = require("macaroon");
56
+ const bs58check = __importStar(require("bs58check"));
57
+ const sdkcore = __importStar(require("@bitgo-beta/sdk-core"));
58
+ // https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#the-signer-node
59
+ exports.signerMacaroonPermissions = [
60
+ {
61
+ entity: 'message',
62
+ action: 'write',
63
+ },
64
+ {
65
+ entity: 'signer',
66
+ action: 'generate',
67
+ },
68
+ {
69
+ entity: 'address',
70
+ action: 'read',
71
+ },
72
+ {
73
+ entity: 'onchain',
74
+ action: 'write',
75
+ },
76
+ ];
77
+ exports.lightningNetworkName = ['bitcoin', 'testnet'];
78
+ /**
79
+ * Checks if the coin name is a lightning coin name.
80
+ */
81
+ function isLightningCoinName(coinName) {
82
+ return coinName === 'lnbtc' || coinName === 'tlnbtc';
83
+ }
84
+ /**
85
+ * Get the utxolib network for a lightning network.
86
+ */
87
+ function getLightningNetwork(networkName) {
88
+ return utxolib.networks[networkName];
89
+ }
90
+ /**
91
+ * Get the lightning coin name for a utxolib network.
92
+ */
93
+ function getLightningCoinName(network) {
94
+ return network === utxolib.networks.bitcoin ? 'lnbtc' : 'tlnbtc';
95
+ }
96
+ /**
97
+ * Checks if the network name is a valid lightning network name.
98
+ */
99
+ function isValidLightningNetworkName(networkName) {
100
+ return exports.lightningNetworkName.includes(networkName);
101
+ }
102
+ /**
103
+ * Checks if the network is a valid lightning network.
104
+ */
105
+ function isValidLightningNetwork(network) {
106
+ return utxolib.isValidNetwork(network) && isValidLightningNetworkName(utxolib.getNetworkName(network));
107
+ }
108
+ /**
109
+ * Returns the statics network data for a lightning coin.
110
+ */
111
+ function getStaticsLightningNetwork(coinName) {
112
+ if (!isLightningCoinName(coinName)) {
113
+ throw new Error(`${coinName} is not a lightning coin`);
114
+ }
115
+ const coin = statics.coins.get(coinName);
116
+ if (!(coin instanceof statics.LightningCoin)) {
117
+ throw new Error('coin is not a lightning coin');
118
+ }
119
+ return coin.network;
120
+ }
121
+ /**
122
+ * Returns the utxolib network for a lightning coin.
123
+ */
124
+ function getUtxolibNetwork(coinName) {
125
+ const networkName = getStaticsLightningNetwork(coinName).utxolibName;
126
+ if (!isValidLightningNetworkName(networkName)) {
127
+ throw new Error('invalid lightning network');
128
+ }
129
+ return getLightningNetwork(networkName);
130
+ }
131
+ /**
132
+ * Returns coin specific data for a lightning coin.
133
+ */
134
+ function unwrapLightningCoinSpecific(obj, coinSpecificPath) {
135
+ if (coinSpecificPath !== 'lnbtc' && coinSpecificPath !== 'tlnbtc') {
136
+ throw new Error(`invalid coinSpecificPath ${coinSpecificPath} for lightning coin`);
137
+ }
138
+ if (coinSpecificPath === 'lnbtc' && 'lnbtc' in obj) {
139
+ return obj.lnbtc;
140
+ }
141
+ if (coinSpecificPath === 'tlnbtc' && 'tlnbtc' in obj) {
142
+ return obj.tlnbtc;
143
+ }
144
+ throw new Error('invalid lightning coin specific');
145
+ }
146
+ /**
147
+ * Adds an IP caveat to a macaroon and returns the modified macaroon as a Base64 string.
148
+ */
149
+ function addIPCaveatToMacaroon(macaroonBase64, ip) {
150
+ const macaroon = (0, macaroon_1.importMacaroon)(macaroonBase64);
151
+ macaroon.addFirstPartyCaveat(`ipaddr ${ip}`);
152
+ return (0, macaroon_1.bytesToBase64)(macaroon.exportBinary());
153
+ }
154
+ const PURPOSE_WRAPPED_P2WKH = 49;
155
+ const PURPOSE_P2WKH = 84;
156
+ const PURPOSE_P2TR = 86;
157
+ const PURPOSE_ALL_OTHERS = 1017;
158
+ /**
159
+ * Converts an extended public key (xpub) to the appropriate prefix (ypub, vpub, etc.) based on its purpose and network.
160
+ */
161
+ function convertXpubPrefix(xpub, purpose, isMainnet) {
162
+ if (purpose === PURPOSE_P2TR || purpose === PURPOSE_ALL_OTHERS) {
163
+ return xpub;
164
+ }
165
+ const data = bs58check.decode(xpub);
166
+ let versionBytes;
167
+ switch (purpose) {
168
+ case PURPOSE_WRAPPED_P2WKH:
169
+ versionBytes = isMainnet ? Buffer.from([0x04, 0x9d, 0x7c, 0xb2]) : Buffer.from([0x04, 0x4a, 0x52, 0x62]); // ypub/upub for p2sh-p2wpkh
170
+ break;
171
+ case PURPOSE_P2WKH:
172
+ versionBytes = isMainnet ? Buffer.from([0x04, 0xb2, 0x47, 0x46]) : Buffer.from([0x04, 0x5f, 0x1c, 0xf6]); // zpub/vpub for p2wpkh
173
+ break;
174
+ default:
175
+ throw new Error('Unsupported purpose');
176
+ }
177
+ versionBytes.copy(data, 0, 0, 4);
178
+ return bs58check.encode(data);
179
+ }
180
+ /**
181
+ * Derives watch-only accounts from the master HD node for the given purposes and network.
182
+ */
183
+ function deriveWatchOnlyAccounts(masterHDNode, isMainnet, params = { onlyAddressCreationAccounts: false }) {
184
+ // https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#required-accounts
185
+ if (masterHDNode.isNeutered()) {
186
+ throw new Error('masterHDNode must not be neutered');
187
+ }
188
+ const purposes = params.onlyAddressCreationAccounts
189
+ ? [PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR]
190
+ : [PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR, PURPOSE_ALL_OTHERS];
191
+ return purposes.flatMap((purpose) => {
192
+ const maxAccount = purpose === PURPOSE_ALL_OTHERS ? 255 : 0;
193
+ const coinType = purpose !== PURPOSE_ALL_OTHERS || isMainnet ? 0 : 1;
194
+ return Array.from({ length: maxAccount + 1 }, (_, account) => {
195
+ const path = `m/${purpose}'/${coinType}'/${account}'`;
196
+ const derivedNode = masterHDNode.derivePath(path);
197
+ // Ensure the node is neutered (i.e., converted to public key only)
198
+ const neuteredNode = derivedNode.neutered();
199
+ const xpub = convertXpubPrefix(neuteredNode.toBase58(), purpose, isMainnet);
200
+ return {
201
+ purpose,
202
+ coin_type: coinType,
203
+ account,
204
+ xpub,
205
+ };
206
+ });
207
+ });
208
+ }
209
+ /**
210
+ * Creates a watch-only wallet init data from the provided signer root key and network.
211
+ */
212
+ function createWatchOnly(signerRootKey, network) {
213
+ const masterHDNode = utxolib.bip32.fromBase58(signerRootKey, network);
214
+ const getCurrentUnixTimestamp = () => {
215
+ return Math.floor(Date.now() / 1000);
216
+ };
217
+ const master_key_birthday_timestamp = getCurrentUnixTimestamp().toString();
218
+ const master_key_fingerprint = masterHDNode.fingerprint.toString('hex');
219
+ const accounts = deriveWatchOnlyAccounts(masterHDNode, utxolib.isMainnet(network));
220
+ return { master_key_birthday_timestamp, master_key_fingerprint, accounts };
221
+ }
222
+ /**
223
+ * Derives the shared Elliptic Curve Diffie-Hellman (ECDH) secret between the user's auth extended private key
224
+ * and the Lightning service's public key for secure communication.
225
+ */
226
+ function deriveLightningServiceSharedSecret(coinName, userAuthXprv) {
227
+ const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).lightningServicePubKey, 'hex');
228
+ const userAuthHdNode = utxolib.bip32.fromBase58(userAuthXprv);
229
+ return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
230
+ }
231
+ /**
232
+ * Derives the shared secret for the middleware using a private key and the middleware's public key.
233
+ */
234
+ function deriveMiddlewareSharedSecret(coinName, xprv) {
235
+ const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).middlewarePubKey, 'hex');
236
+ const userAuthHdNode = utxolib.bip32.fromBase58(xprv);
237
+ return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
238
+ }
239
+ /**
240
+ * Derives the shared secret for TAT service using ta private key and the TAT public key.
241
+ */
242
+ function deriveTatSharedSecret(coinName, xprv) {
243
+ const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).tatPubKey, 'hex');
244
+ const userAuthHdNode = utxolib.bip32.fromBase58(xprv);
245
+ return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
246
+ }
247
+ /**
248
+ * Given a seed, compute a BIP32 derivation index.
249
+ * 0 <= index < 2147483648 (largest 31 bit number). This needs to be 2^31 - 1 so that the bip32 library
250
+ * can derive the hardened key.
251
+ * @param seed (optional) If nothing provided, we will generate one randomly
252
+ */
253
+ function computeBip32DerivationIndexFromSeed(seed) {
254
+ return ((Buffer.from(utxolib.crypto.sha256(Buffer.from(seed ?? (0, crypto_1.randomBytes)(32).toString('hex'), 'utf8'))).readUint32BE(0) %
255
+ Math.pow(2, 31)) -
256
+ 1);
257
+ }
258
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lightningUtils.js","sourceRoot":"","sources":["../../../src/lightning/lightningUtils.ts"],"names":[],"mappings":"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","sourcesContent":["import * as statics from '@bitgo-beta/statics';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { randomBytes } from 'crypto';\nimport { importMacaroon, bytesToBase64 } from 'macaroon';\nimport * as bs58check from 'bs58check';\nimport * as sdkcore from '@bitgo-beta/sdk-core';\nimport { WatchOnly, WatchOnlyAccount } from '../codecs';\n\n// https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#the-signer-node\nexport const signerMacaroonPermissions = [\n  {\n    entity: 'message',\n    action: 'write',\n  },\n  {\n    entity: 'signer',\n    action: 'generate',\n  },\n  {\n    entity: 'address',\n    action: 'read',\n  },\n  {\n    entity: 'onchain',\n    action: 'write',\n  },\n];\n\nexport const lightningNetworkName = ['bitcoin', 'testnet'] as const;\nexport type LightningNetworkName = (typeof lightningNetworkName)[number];\n\n/**\n * Checks if the coin name is a lightning coin name.\n */\nexport function isLightningCoinName(coinName: unknown): coinName is 'lnbtc' | 'tlnbtc' {\n  return coinName === 'lnbtc' || coinName === 'tlnbtc';\n}\n\n/**\n * Get the utxolib network for a lightning network.\n */\nexport function getLightningNetwork(networkName: LightningNetworkName): utxolib.Network {\n  return utxolib.networks[networkName];\n}\n\n/**\n * Get the lightning coin name for a utxolib network.\n */\nexport function getLightningCoinName(network: utxolib.Network): string {\n  return network === utxolib.networks.bitcoin ? 'lnbtc' : 'tlnbtc';\n}\n\n/**\n * Checks if the network name is a valid lightning network name.\n */\nexport function isValidLightningNetworkName(networkName: unknown): networkName is LightningNetworkName {\n  return lightningNetworkName.includes(networkName as LightningNetworkName);\n}\n\n/**\n * Checks if the network is a valid lightning network.\n */\nexport function isValidLightningNetwork(network: unknown): network is utxolib.Network {\n  return utxolib.isValidNetwork(network) && isValidLightningNetworkName(utxolib.getNetworkName(network));\n}\n\n/**\n * Returns the statics network data for a lightning coin.\n */\nexport function getStaticsLightningNetwork(coinName: string): statics.LightningNetwork {\n  if (!isLightningCoinName(coinName)) {\n    throw new Error(`${coinName} is not a lightning coin`);\n  }\n  const coin = statics.coins.get(coinName);\n  if (!(coin instanceof statics.LightningCoin)) {\n    throw new Error('coin is not a lightning coin');\n  }\n  return coin.network;\n}\n\n/**\n * Returns the utxolib network for a lightning coin.\n */\nexport function getUtxolibNetwork(coinName: string): utxolib.Network {\n  const networkName = getStaticsLightningNetwork(coinName).utxolibName;\n  if (!isValidLightningNetworkName(networkName)) {\n    throw new Error('invalid lightning network');\n  }\n  return getLightningNetwork(networkName);\n}\n\n/**\n * Returns coin specific data for a lightning coin.\n */\nexport function unwrapLightningCoinSpecific<V>(obj: { lnbtc: V } | { tlnbtc: V }, coinSpecificPath: string): V {\n  if (coinSpecificPath !== 'lnbtc' && coinSpecificPath !== 'tlnbtc') {\n    throw new Error(`invalid coinSpecificPath ${coinSpecificPath} for lightning coin`);\n  }\n  if (coinSpecificPath === 'lnbtc' && 'lnbtc' in obj) {\n    return obj.lnbtc;\n  }\n  if (coinSpecificPath === 'tlnbtc' && 'tlnbtc' in obj) {\n    return obj.tlnbtc;\n  }\n  throw new Error('invalid lightning coin specific');\n}\n\n/**\n * Adds an IP caveat to a macaroon and returns the modified macaroon as a Base64 string.\n */\nexport function addIPCaveatToMacaroon(macaroonBase64: string, ip: string): string {\n  const macaroon = importMacaroon(macaroonBase64);\n  macaroon.addFirstPartyCaveat(`ipaddr ${ip}`);\n  return bytesToBase64(macaroon.exportBinary());\n}\n\nconst PURPOSE_WRAPPED_P2WKH = 49;\nconst PURPOSE_P2WKH = 84;\nconst PURPOSE_P2TR = 86;\nconst PURPOSE_ALL_OTHERS = 1017;\n\ntype ExtendedKeyPurpose =\n  | typeof PURPOSE_WRAPPED_P2WKH\n  | typeof PURPOSE_P2WKH\n  | typeof PURPOSE_P2TR\n  | typeof PURPOSE_ALL_OTHERS;\n\n/**\n * Converts an extended public key (xpub) to the appropriate prefix (ypub, vpub, etc.) based on its purpose and network.\n */\nfunction convertXpubPrefix(xpub: string, purpose: ExtendedKeyPurpose, isMainnet: boolean): string {\n  if (purpose === PURPOSE_P2TR || purpose === PURPOSE_ALL_OTHERS) {\n    return xpub;\n  }\n  const data = bs58check.decode(xpub);\n\n  let versionBytes: Buffer;\n\n  switch (purpose) {\n    case PURPOSE_WRAPPED_P2WKH:\n      versionBytes = isMainnet ? Buffer.from([0x04, 0x9d, 0x7c, 0xb2]) : Buffer.from([0x04, 0x4a, 0x52, 0x62]); // ypub/upub for p2sh-p2wpkh\n      break;\n    case PURPOSE_P2WKH:\n      versionBytes = isMainnet ? Buffer.from([0x04, 0xb2, 0x47, 0x46]) : Buffer.from([0x04, 0x5f, 0x1c, 0xf6]); // zpub/vpub for p2wpkh\n      break;\n    default:\n      throw new Error('Unsupported purpose');\n  }\n\n  versionBytes.copy(data, 0, 0, 4);\n  return bs58check.encode(data);\n}\n\n/**\n * Derives watch-only accounts from the master HD node for the given purposes and network.\n */\nexport function deriveWatchOnlyAccounts(\n  masterHDNode: utxolib.BIP32Interface,\n  isMainnet: boolean,\n  params: { onlyAddressCreationAccounts?: boolean } = { onlyAddressCreationAccounts: false }\n): WatchOnlyAccount[] {\n  // https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#required-accounts\n  if (masterHDNode.isNeutered()) {\n    throw new Error('masterHDNode must not be neutered');\n  }\n  const purposes = params.onlyAddressCreationAccounts\n    ? ([PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR] as const)\n    : ([PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR, PURPOSE_ALL_OTHERS] as const);\n  return purposes.flatMap((purpose) => {\n    const maxAccount = purpose === PURPOSE_ALL_OTHERS ? 255 : 0;\n    const coinType = purpose !== PURPOSE_ALL_OTHERS || isMainnet ? 0 : 1;\n\n    return Array.from({ length: maxAccount + 1 }, (_, account) => {\n      const path = `m/${purpose}'/${coinType}'/${account}'`;\n      const derivedNode = masterHDNode.derivePath(path);\n\n      // Ensure the node is neutered (i.e., converted to public key only)\n      const neuteredNode = derivedNode.neutered();\n      const xpub = convertXpubPrefix(neuteredNode.toBase58(), purpose, isMainnet);\n\n      return {\n        purpose,\n        coin_type: coinType,\n        account,\n        xpub,\n      };\n    });\n  });\n}\n\n/**\n * Creates a watch-only wallet init data from the provided signer root key and network.\n */\nexport function createWatchOnly(signerRootKey: string, network: utxolib.Network): WatchOnly {\n  const masterHDNode = utxolib.bip32.fromBase58(signerRootKey, network);\n  const getCurrentUnixTimestamp = () => {\n    return Math.floor(Date.now() / 1000);\n  };\n  const master_key_birthday_timestamp = getCurrentUnixTimestamp().toString();\n  const master_key_fingerprint = masterHDNode.fingerprint.toString('hex');\n  const accounts = deriveWatchOnlyAccounts(masterHDNode, utxolib.isMainnet(network));\n  return { master_key_birthday_timestamp, master_key_fingerprint, accounts };\n}\n\n/**\n * Derives the shared Elliptic Curve Diffie-Hellman (ECDH) secret between the user's auth extended private key\n * and the Lightning service's public key for secure communication.\n */\nexport function deriveLightningServiceSharedSecret(coinName: 'lnbtc' | 'tlnbtc', userAuthXprv: string): Buffer {\n  const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).lightningServicePubKey, 'hex');\n  const userAuthHdNode = utxolib.bip32.fromBase58(userAuthXprv);\n  return sdkcore.getSharedSecret(userAuthHdNode, publicKey);\n}\n\n/**\n * Derives the shared secret for the middleware using a private key and the middleware's public key.\n */\nexport function deriveMiddlewareSharedSecret(coinName: 'lnbtc' | 'tlnbtc', xprv: string): Buffer {\n  const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).middlewarePubKey, 'hex');\n  const userAuthHdNode = utxolib.bip32.fromBase58(xprv);\n  return sdkcore.getSharedSecret(userAuthHdNode, publicKey);\n}\n\n/**\n * Derives the shared secret for TAT service using ta private key and the TAT public key.\n */\nexport function deriveTatSharedSecret(coinName: 'lnbtc' | 'tlnbtc', xprv: string): Buffer {\n  const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).tatPubKey, 'hex');\n  const userAuthHdNode = utxolib.bip32.fromBase58(xprv);\n  return sdkcore.getSharedSecret(userAuthHdNode, publicKey);\n}\n\n/**\n * Given a seed, compute a BIP32 derivation index.\n * 0 <= index < 2147483648 (largest 31 bit number). This needs to be 2^31 - 1 so that the bip32 library\n * can derive the hardened key.\n * @param seed (optional) If nothing provided, we will generate one randomly\n */\nexport function computeBip32DerivationIndexFromSeed(seed?: string): number {\n  return (\n    (Buffer.from(utxolib.crypto.sha256(Buffer.from(seed ?? randomBytes(32).toString('hex'), 'utf8'))).readUint32BE(0) %\n      Math.pow(2, 31)) -\n    1\n  );\n}\n"]}
@@ -0,0 +1,8 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { WatchOnlyAccount } from '../codecs';
3
+ import { LightningOnchainRecipient } from '@bitgo/public-types';
4
+ /**
5
+ * Validates the funded psbt before creating the signatures for withdraw.
6
+ */
7
+ export declare function validatePsbtForWithdraw(psbtHex: string, network: utxolib.Network, recipients: LightningOnchainRecipient[], accounts: WatchOnlyAccount[]): void;
8
+ //# sourceMappingURL=parseWithdrawPsbt.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parseWithdrawPsbt.d.ts","sourceRoot":"","sources":["../../../src/lightning/parseWithdrawPsbt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAE,gBAAgB,EAA0B,MAAM,WAAW,CAAC;AACrE,OAAO,EAAE,yBAAyB,EAAE,MAAM,qBAAqB,CAAC;AA+GhE;;GAEG;AACH,wBAAgB,uBAAuB,CACrC,OAAO,EAAE,MAAM,EACf,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,UAAU,EAAE,yBAAyB,EAAE,EACvC,QAAQ,EAAE,gBAAgB,EAAE,GAC3B,IAAI,CAsBN"}
@@ -0,0 +1,149 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.validatePsbtForWithdraw = validatePsbtForWithdraw;
37
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
38
+ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
39
+ function parseDerivationPath(derivationPath) {
40
+ const pathSegments = derivationPath.split('/');
41
+ const purpose = Number(pathSegments[1].replace(/'/g, ''));
42
+ const change = Number(pathSegments[pathSegments.length - 2]);
43
+ const addressIndex = Number(pathSegments[pathSegments.length - 1]);
44
+ return { purpose, change, addressIndex };
45
+ }
46
+ function parsePsbtOutputs(psbt, network) {
47
+ const parsedOutputs = [];
48
+ let bip32Derivation;
49
+ for (let i = 0; i < psbt.data.outputs.length; i++) {
50
+ const output = psbt.data.outputs[i];
51
+ const txOutput = psbt.txOutputs[i];
52
+ let address = '';
53
+ const value = txOutput.value;
54
+ let isChange = false;
55
+ if (output.bip32Derivation && output.bip32Derivation.length > 0) {
56
+ isChange = true;
57
+ bip32Derivation = output.bip32Derivation[0];
58
+ }
59
+ if (txOutput.script) {
60
+ address = utxolib.address.fromOutputScript(txOutput.script, network);
61
+ }
62
+ const valueBigInt = BigInt(value);
63
+ parsedOutputs.push({
64
+ address,
65
+ value: valueBigInt,
66
+ change: isChange,
67
+ bip32Derivation,
68
+ });
69
+ }
70
+ return parsedOutputs;
71
+ }
72
+ function verifyChangeAddress(output, accounts, network) {
73
+ if (!output.bip32Derivation || !output.bip32Derivation.path) {
74
+ throw new Error(`bip32Derivation path not found for change address`);
75
+ }
76
+ // derivation path example: m/84'/0'/0'/1/0
77
+ const { purpose, change, addressIndex } = parseDerivationPath(output.bip32Derivation.path);
78
+ // Find the corresponding account using the purpose
79
+ const account = accounts.find((acc) => acc.purpose === purpose);
80
+ if (!account) {
81
+ throw new Error(`Account not found for purpose: ${purpose}`);
82
+ }
83
+ // Create a BIP32 node from the xpub
84
+ const xpubNode = utxolib.bip32.fromBase58(account.xpub, network);
85
+ // Derive the public key from the xpub using the change and address index
86
+ const derivedPubkey = xpubNode.derive(change).derive(addressIndex).publicKey;
87
+ if (derivedPubkey.toString('hex') !== output.bip32Derivation.pubkey.toString('hex')) {
88
+ throw new Error(`Derived pubkey does not match for address: ${output.address}, derived: ${derivedPubkey.toString('hex')}, expected: ${output.bip32Derivation.pubkey.toString('hex')}`);
89
+ }
90
+ // Determine the correct payment type based on the purpose
91
+ let derivedAddress;
92
+ switch (purpose) {
93
+ case 49: // P2SH-P2WPKH (Nested SegWit)
94
+ derivedAddress = utxolib.payments.p2sh({
95
+ redeem: utxolib.payments.p2wpkh({
96
+ pubkey: derivedPubkey,
97
+ network,
98
+ }),
99
+ network,
100
+ }).address;
101
+ break;
102
+ case 84: // P2WPKH (Native SegWit)
103
+ derivedAddress = utxolib.payments.p2wpkh({
104
+ pubkey: derivedPubkey,
105
+ network,
106
+ }).address;
107
+ break;
108
+ case 86: // P2TR (Taproot)
109
+ derivedAddress = utxolib.payments.p2tr({
110
+ pubkey: derivedPubkey,
111
+ network,
112
+ }).address;
113
+ break;
114
+ default:
115
+ throw new Error(`Unsupported purpose: ${purpose}`);
116
+ }
117
+ if (derivedAddress !== output.address) {
118
+ throw new Error(`invalid change address: expected ${derivedAddress}, got ${output.address}`);
119
+ }
120
+ }
121
+ /**
122
+ * Validates the funded psbt before creating the signatures for withdraw.
123
+ */
124
+ function validatePsbtForWithdraw(psbtHex, network, recipients, accounts) {
125
+ const parsedPsbt = utxo_lib_1.Psbt.fromHex(psbtHex, { network: network });
126
+ const outputs = parsePsbtOutputs(parsedPsbt, network);
127
+ outputs.forEach((output) => {
128
+ if (output.change) {
129
+ try {
130
+ verifyChangeAddress(output, accounts, network);
131
+ }
132
+ catch (e) {
133
+ throw new Error(`Unable to verify change address: ${e}`);
134
+ }
135
+ }
136
+ else {
137
+ let match = false;
138
+ recipients.forEach((recipient) => {
139
+ if (recipient.address === output.address && BigInt(recipient.amountSat) === output.value) {
140
+ match = true;
141
+ }
142
+ });
143
+ if (!match) {
144
+ throw new Error(`PSBT output ${output.address} with value ${output.value} does not match any recipient`);
145
+ }
146
+ }
147
+ });
148
+ }
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseWithdrawPsbt.js","sourceRoot":"","sources":["../../../src/lightning/parseWithdrawPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Psbt } from '@bitgo-beta/utxo-lib';\nimport { WatchOnlyAccount, WithdrawBaseOutputUTXO } from '../codecs';\nimport { LightningOnchainRecipient } from '@bitgo/public-types';\nimport { Bip32Derivation } from 'bip174/src/lib/interfaces';\n\nfunction parseDerivationPath(derivationPath: string): {\n  purpose: number;\n  change: number;\n  addressIndex: number;\n} {\n  const pathSegments = derivationPath.split('/');\n  const purpose = Number(pathSegments[1].replace(/'/g, ''));\n  const change = Number(pathSegments[pathSegments.length - 2]);\n  const addressIndex = Number(pathSegments[pathSegments.length - 1]);\n  return { purpose, change, addressIndex };\n}\n\nfunction parsePsbtOutputs(psbt: Psbt, network: utxolib.Network): WithdrawBaseOutputUTXO<bigint>[] {\n  const parsedOutputs: WithdrawBaseOutputUTXO<bigint>[] = [];\n  let bip32Derivation: Bip32Derivation | undefined;\n\n  for (let i = 0; i < psbt.data.outputs.length; i++) {\n    const output = psbt.data.outputs[i];\n    const txOutput = psbt.txOutputs[i];\n\n    let address = '';\n    const value = txOutput.value;\n    let isChange = false;\n\n    if (output.bip32Derivation && output.bip32Derivation.length > 0) {\n      isChange = true;\n      bip32Derivation = output.bip32Derivation[0];\n    }\n    if (txOutput.script) {\n      address = utxolib.address.fromOutputScript(txOutput.script, network);\n    }\n    const valueBigInt = BigInt(value);\n\n    parsedOutputs.push({\n      address,\n      value: valueBigInt,\n      change: isChange,\n      bip32Derivation,\n    });\n  }\n\n  return parsedOutputs;\n}\n\nfunction verifyChangeAddress(\n  output: WithdrawBaseOutputUTXO<bigint>,\n  accounts: WatchOnlyAccount[],\n  network: utxolib.Network\n): void {\n  if (!output.bip32Derivation || !output.bip32Derivation.path) {\n    throw new Error(`bip32Derivation path not found for change address`);\n  }\n  // derivation path example: m/84'/0'/0'/1/0\n  const { purpose, change, addressIndex } = parseDerivationPath(output.bip32Derivation.path);\n\n  // Find the corresponding account using the purpose\n  const account = accounts.find((acc) => acc.purpose === purpose);\n  if (!account) {\n    throw new Error(`Account not found for purpose: ${purpose}`);\n  }\n\n  // Create a BIP32 node from the xpub\n  const xpubNode = utxolib.bip32.fromBase58(account.xpub, network);\n\n  // Derive the public key from the xpub using the change and address index\n  const derivedPubkey = xpubNode.derive(change).derive(addressIndex).publicKey;\n\n  if (derivedPubkey.toString('hex') !== output.bip32Derivation.pubkey.toString('hex')) {\n    throw new Error(\n      `Derived pubkey does not match for address: ${output.address}, derived: ${derivedPubkey.toString(\n        'hex'\n      )}, expected: ${output.bip32Derivation.pubkey.toString('hex')}`\n    );\n  }\n\n  // Determine the correct payment type based on the purpose\n  let derivedAddress: string | undefined;\n  switch (purpose) {\n    case 49: // P2SH-P2WPKH (Nested SegWit)\n      derivedAddress = utxolib.payments.p2sh({\n        redeem: utxolib.payments.p2wpkh({\n          pubkey: derivedPubkey,\n          network,\n        }),\n        network,\n      }).address;\n      break;\n    case 84: // P2WPKH (Native SegWit)\n      derivedAddress = utxolib.payments.p2wpkh({\n        pubkey: derivedPubkey,\n        network,\n      }).address;\n      break;\n    case 86: // P2TR (Taproot)\n      derivedAddress = utxolib.payments.p2tr({\n        pubkey: derivedPubkey,\n        network,\n      }).address;\n      break;\n    default:\n      throw new Error(`Unsupported purpose: ${purpose}`);\n  }\n\n  if (derivedAddress !== output.address) {\n    throw new Error(`invalid change address: expected ${derivedAddress}, got ${output.address}`);\n  }\n}\n\n/**\n * Validates the funded psbt before creating the signatures for withdraw.\n */\nexport function validatePsbtForWithdraw(\n  psbtHex: string,\n  network: utxolib.Network,\n  recipients: LightningOnchainRecipient[],\n  accounts: WatchOnlyAccount[]\n): void {\n  const parsedPsbt = Psbt.fromHex(psbtHex, { network: network });\n  const outputs = parsePsbtOutputs(parsedPsbt, network);\n  outputs.forEach((output) => {\n    if (output.change) {\n      try {\n        verifyChangeAddress(output, accounts, network);\n      } catch (e: any) {\n        throw new Error(`Unable to verify change address: ${e}`);\n      }\n    } else {\n      let match = false;\n      recipients.forEach((recipient) => {\n        if (recipient.address === output.address && BigInt(recipient.amountSat) === output.value) {\n          match = true;\n        }\n      });\n      if (!match) {\n        throw new Error(`PSBT output ${output.address} with value ${output.value} does not match any recipient`);\n      }\n    }\n  });\n}\n"]}
@@ -0,0 +1,17 @@
1
+ export type Signable = boolean | number | string | SignableRecord | SignableArray;
2
+ export interface SignableRecord {
3
+ [key: string]: Signable;
4
+ }
5
+ export interface SignableArray {
6
+ [key: number]: Signable;
7
+ }
8
+ /**
9
+ * Recursively canonicalizes an object by sorting its keys.
10
+ *
11
+ * @param obj - The object to be canonicalized. It can be a boolean, number, string,
12
+ * a record of signable values, or an array of signable values.
13
+ * @returns The canonicalized object with sorted keys.
14
+ * @throws Will throw an error if the object type is invalid.
15
+ */
16
+ export declare function canonicalizeObject(obj: Signable): Signable;
17
+ //# sourceMappingURL=signableJson.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signableJson.d.ts","sourceRoot":"","sources":["../../../src/lightning/signableJson.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,QAAQ,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,cAAc,GAAG,aAAa,CAAC;AAElF,MAAM,WAAW,cAAc;IAC7B,CAAC,GAAG,EAAE,MAAM,GAAG,QAAQ,CAAC;CACzB;AAED,MAAM,WAAW,aAAa;IAC5B,CAAC,GAAG,EAAE,MAAM,GAAG,QAAQ,CAAC;CACzB;AAED;;;;;;;GAOG;AACH,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,QAAQ,GAAG,QAAQ,CAmB1D"}
@@ -0,0 +1,29 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.canonicalizeObject = canonicalizeObject;
4
+ /**
5
+ * Recursively canonicalizes an object by sorting its keys.
6
+ *
7
+ * @param obj - The object to be canonicalized. It can be a boolean, number, string,
8
+ * a record of signable values, or an array of signable values.
9
+ * @returns The canonicalized object with sorted keys.
10
+ * @throws Will throw an error if the object type is invalid.
11
+ */
12
+ function canonicalizeObject(obj) {
13
+ if (typeof obj === 'boolean' || typeof obj === 'number' || typeof obj === 'string') {
14
+ return obj;
15
+ }
16
+ if (Array.isArray(obj)) {
17
+ return obj.map(canonicalizeObject);
18
+ }
19
+ if (obj !== null && typeof obj === 'object') {
20
+ return Object.keys(obj)
21
+ .sort()
22
+ .reduce((result, key) => {
23
+ result[key] = canonicalizeObject(obj[key]);
24
+ return result;
25
+ }, {});
26
+ }
27
+ throw new Error('Invalid object type');
28
+ }
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2lnbmFibGVKc29uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpZ2h0bmluZy9zaWduYWJsZUpzb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFrQkEsZ0RBbUJDO0FBM0JEOzs7Ozs7O0dBT0c7QUFDSCxTQUFnQixrQkFBa0IsQ0FBQyxHQUFhO0lBQzlDLElBQUksT0FBTyxHQUFHLEtBQUssU0FBUyxJQUFJLE9BQU8sR0FBRyxLQUFLLFFBQVEsSUFBSSxPQUFPLEdBQUcsS0FBSyxRQUFRLEVBQUUsQ0FBQztRQUNuRixPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7SUFFRCxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztRQUN2QixPQUFPLEdBQUcsQ0FBQyxHQUFHLENBQUMsa0JBQWtCLENBQUMsQ0FBQztJQUNyQyxDQUFDO0lBRUQsSUFBSSxHQUFHLEtBQUssSUFBSSxJQUFJLE9BQU8sR0FBRyxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQzVDLE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUM7YUFDcEIsSUFBSSxFQUFFO2FBQ04sTUFBTSxDQUFDLENBQUMsTUFBZ0MsRUFBRSxHQUFHLEVBQUUsRUFBRTtZQUNoRCxNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsa0JBQWtCLENBQUUsR0FBbUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO1lBQzVFLE9BQU8sTUFBTSxDQUFDO1FBQ2hCLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUNYLENBQUM7SUFFRCxNQUFNLElBQUksS0FBSyxDQUFDLHFCQUFxQixDQUFDLENBQUM7QUFDekMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIFNpZ25hYmxlID0gYm9vbGVhbiB8IG51bWJlciB8IHN0cmluZyB8IFNpZ25hYmxlUmVjb3JkIHwgU2lnbmFibGVBcnJheTtcblxuZXhwb3J0IGludGVyZmFjZSBTaWduYWJsZVJlY29yZCB7XG4gIFtrZXk6IHN0cmluZ106IFNpZ25hYmxlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNpZ25hYmxlQXJyYXkge1xuICBba2V5OiBudW1iZXJdOiBTaWduYWJsZTtcbn1cblxuLyoqXG4gKiBSZWN1cnNpdmVseSBjYW5vbmljYWxpemVzIGFuIG9iamVjdCBieSBzb3J0aW5nIGl0cyBrZXlzLlxuICpcbiAqIEBwYXJhbSBvYmogLSBUaGUgb2JqZWN0IHRvIGJlIGNhbm9uaWNhbGl6ZWQuIEl0IGNhbiBiZSBhIGJvb2xlYW4sIG51bWJlciwgc3RyaW5nLFxuICogICAgICAgICAgICAgIGEgcmVjb3JkIG9mIHNpZ25hYmxlIHZhbHVlcywgb3IgYW4gYXJyYXkgb2Ygc2lnbmFibGUgdmFsdWVzLlxuICogQHJldHVybnMgVGhlIGNhbm9uaWNhbGl6ZWQgb2JqZWN0IHdpdGggc29ydGVkIGtleXMuXG4gKiBAdGhyb3dzIFdpbGwgdGhyb3cgYW4gZXJyb3IgaWYgdGhlIG9iamVjdCB0eXBlIGlzIGludmFsaWQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjYW5vbmljYWxpemVPYmplY3Qob2JqOiBTaWduYWJsZSk6IFNpZ25hYmxlIHtcbiAgaWYgKHR5cGVvZiBvYmogPT09ICdib29sZWFuJyB8fCB0eXBlb2Ygb2JqID09PSAnbnVtYmVyJyB8fCB0eXBlb2Ygb2JqID09PSAnc3RyaW5nJykge1xuICAgIHJldHVybiBvYmo7XG4gIH1cblxuICBpZiAoQXJyYXkuaXNBcnJheShvYmopKSB7XG4gICAgcmV0dXJuIG9iai5tYXAoY2Fub25pY2FsaXplT2JqZWN0KTtcbiAgfVxuXG4gIGlmIChvYmogIT09IG51bGwgJiYgdHlwZW9mIG9iaiA9PT0gJ29iamVjdCcpIHtcbiAgICByZXR1cm4gT2JqZWN0LmtleXMob2JqKVxuICAgICAgLnNvcnQoKVxuICAgICAgLnJlZHVjZSgocmVzdWx0OiBSZWNvcmQ8c3RyaW5nLCBTaWduYWJsZT4sIGtleSkgPT4ge1xuICAgICAgICByZXN1bHRba2V5XSA9IGNhbm9uaWNhbGl6ZU9iamVjdCgob2JqIGFzIHsgW2tleTogc3RyaW5nXTogU2lnbmFibGUgfSlba2V5XSk7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9LCB7fSk7XG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoJ0ludmFsaWQgb2JqZWN0IHR5cGUnKTtcbn1cbiJdfQ==
@@ -0,0 +1,22 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { Signable } from './signableJson';
3
+ /**
4
+ * Verifies a signature for a given message.
5
+ *
6
+ * @param {Signable} message - The message to verify.
7
+ * @param {string} signature - The signature to verify, in hexadecimal format.
8
+ * @param {string} pub - The public key in BIP32 format.
9
+ * @param {utxolib.Network} network - The network to use for verification.
10
+ * @returns {boolean} - Returns true if the signature is valid, false otherwise.
11
+ */
12
+ export declare function verifyMessageSignature(message: Signable, signature: string, pub: string, network?: utxolib.Network): boolean;
13
+ /**
14
+ * Creates a signature for a given message.
15
+ *
16
+ * @param {Signable} message - The message to sign.
17
+ * @param {string} prv - The private key in BIP32 format.
18
+ * @param {utxolib.Network} network - The network to use for signing.
19
+ * @returns {string} - Returns the signature in hexadecimal format.
20
+ */
21
+ export declare function createMessageSignature(message: Signable, xprv: string, network?: utxolib.Network): string;
22
+ //# sourceMappingURL=signature.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signature.d.ts","sourceRoot":"","sources":["../../../src/lightning/signature.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAsB,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE9D;;;;;;;;GAQG;AACH,wBAAgB,sBAAsB,CACpC,OAAO,EAAE,QAAQ,EACjB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE,MAAM,EACX,OAAO,GAAE,OAAO,CAAC,OAAkC,GAClD,OAAO,CAKT;AAED;;;;;;;GAOG;AACH,wBAAgB,sBAAsB,CACpC,OAAO,EAAE,QAAQ,EACjB,IAAI,EAAE,MAAM,EACZ,OAAO,GAAE,OAAO,CAAC,OAAkC,GAClD,MAAM,CAIR"}
@@ -0,0 +1,69 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.verifyMessageSignature = verifyMessageSignature;
37
+ exports.createMessageSignature = createMessageSignature;
38
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
39
+ const sdkcore = __importStar(require("@bitgo-beta/sdk-core"));
40
+ const signableJson_1 = require("./signableJson");
41
+ /**
42
+ * Verifies a signature for a given message.
43
+ *
44
+ * @param {Signable} message - The message to verify.
45
+ * @param {string} signature - The signature to verify, in hexadecimal format.
46
+ * @param {string} pub - The public key in BIP32 format.
47
+ * @param {utxolib.Network} network - The network to use for verification.
48
+ * @returns {boolean} - Returns true if the signature is valid, false otherwise.
49
+ */
50
+ function verifyMessageSignature(message, signature, pub, network = utxolib.networks.bitcoin) {
51
+ const messageString = JSON.stringify((0, signableJson_1.canonicalizeObject)(message));
52
+ const pubKey = utxolib.bip32.fromBase58(pub, network).publicKey;
53
+ const signatureBuffer = Buffer.from(signature, 'hex');
54
+ return sdkcore.verifyMessage(messageString, pubKey, signatureBuffer, network);
55
+ }
56
+ /**
57
+ * Creates a signature for a given message.
58
+ *
59
+ * @param {Signable} message - The message to sign.
60
+ * @param {string} prv - The private key in BIP32 format.
61
+ * @param {utxolib.Network} network - The network to use for signing.
62
+ * @returns {string} - Returns the signature in hexadecimal format.
63
+ */
64
+ function createMessageSignature(message, xprv, network = utxolib.networks.bitcoin) {
65
+ const requestString = JSON.stringify((0, signableJson_1.canonicalizeObject)(message));
66
+ const prvKey = utxolib.bip32.fromBase58(xprv, network);
67
+ return sdkcore.signMessage(requestString, prvKey, network).toString('hex');
68
+ }
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2lnbmF0dXJlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpZ2h0bmluZy9zaWduYXR1cmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFhQSx3REFVQztBQVVELHdEQVFDO0FBekNELDhEQUFnRDtBQUNoRCw4REFBZ0Q7QUFDaEQsaURBQThEO0FBRTlEOzs7Ozs7OztHQVFHO0FBQ0gsU0FBZ0Isc0JBQXNCLENBQ3BDLE9BQWlCLEVBQ2pCLFNBQWlCLEVBQ2pCLEdBQVcsRUFDWCxVQUEyQixPQUFPLENBQUMsUUFBUSxDQUFDLE9BQU87SUFFbkQsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFBLGlDQUFrQixFQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7SUFDbEUsTUFBTSxNQUFNLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsR0FBRyxFQUFFLE9BQU8sQ0FBQyxDQUFDLFNBQVMsQ0FBQztJQUNoRSxNQUFNLGVBQWUsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUN0RCxPQUFPLE9BQU8sQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLE1BQU0sRUFBRSxlQUFlLEVBQUUsT0FBTyxDQUFDLENBQUM7QUFDaEYsQ0FBQztBQUVEOzs7Ozs7O0dBT0c7QUFDSCxTQUFnQixzQkFBc0IsQ0FDcEMsT0FBaUIsRUFDakIsSUFBWSxFQUNaLFVBQTJCLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTztJQUVuRCxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUEsaUNBQWtCLEVBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztJQUNsRSxNQUFNLE1BQU0sR0FBRyxPQUFPLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDdkQsT0FBTyxPQUFPLENBQUMsV0FBVyxDQUFDLGFBQWEsRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO0FBQzdFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB1dHhvbGliIGZyb20gJ0BiaXRnby1iZXRhL3V0eG8tbGliJztcbmltcG9ydCAqIGFzIHNka2NvcmUgZnJvbSAnQGJpdGdvLWJldGEvc2RrLWNvcmUnO1xuaW1wb3J0IHsgY2Fub25pY2FsaXplT2JqZWN0LCBTaWduYWJsZSB9IGZyb20gJy4vc2lnbmFibGVKc29uJztcblxuLyoqXG4gKiBWZXJpZmllcyBhIHNpZ25hdHVyZSBmb3IgYSBnaXZlbiBtZXNzYWdlLlxuICpcbiAqIEBwYXJhbSB7U2lnbmFibGV9IG1lc3NhZ2UgLSBUaGUgbWVzc2FnZSB0byB2ZXJpZnkuXG4gKiBAcGFyYW0ge3N0cmluZ30gc2lnbmF0dXJlIC0gVGhlIHNpZ25hdHVyZSB0byB2ZXJpZnksIGluIGhleGFkZWNpbWFsIGZvcm1hdC5cbiAqIEBwYXJhbSB7c3RyaW5nfSBwdWIgLSBUaGUgcHVibGljIGtleSBpbiBCSVAzMiBmb3JtYXQuXG4gKiBAcGFyYW0ge3V0eG9saWIuTmV0d29ya30gbmV0d29yayAtIFRoZSBuZXR3b3JrIHRvIHVzZSBmb3IgdmVyaWZpY2F0aW9uLlxuICogQHJldHVybnMge2Jvb2xlYW59IC0gUmV0dXJucyB0cnVlIGlmIHRoZSBzaWduYXR1cmUgaXMgdmFsaWQsIGZhbHNlIG90aGVyd2lzZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHZlcmlmeU1lc3NhZ2VTaWduYXR1cmUoXG4gIG1lc3NhZ2U6IFNpZ25hYmxlLFxuICBzaWduYXR1cmU6IHN0cmluZyxcbiAgcHViOiBzdHJpbmcsXG4gIG5ldHdvcms6IHV0eG9saWIuTmV0d29yayA9IHV0eG9saWIubmV0d29ya3MuYml0Y29pblxuKTogYm9vbGVhbiB7XG4gIGNvbnN0IG1lc3NhZ2VTdHJpbmcgPSBKU09OLnN0cmluZ2lmeShjYW5vbmljYWxpemVPYmplY3QobWVzc2FnZSkpO1xuICBjb25zdCBwdWJLZXkgPSB1dHhvbGliLmJpcDMyLmZyb21CYXNlNTgocHViLCBuZXR3b3JrKS5wdWJsaWNLZXk7XG4gIGNvbnN0IHNpZ25hdHVyZUJ1ZmZlciA9IEJ1ZmZlci5mcm9tKHNpZ25hdHVyZSwgJ2hleCcpO1xuICByZXR1cm4gc2RrY29yZS52ZXJpZnlNZXNzYWdlKG1lc3NhZ2VTdHJpbmcsIHB1YktleSwgc2lnbmF0dXJlQnVmZmVyLCBuZXR3b3JrKTtcbn1cblxuLyoqXG4gKiBDcmVhdGVzIGEgc2lnbmF0dXJlIGZvciBhIGdpdmVuIG1lc3NhZ2UuXG4gKlxuICogQHBhcmFtIHtTaWduYWJsZX0gbWVzc2FnZSAtIFRoZSBtZXNzYWdlIHRvIHNpZ24uXG4gKiBAcGFyYW0ge3N0cmluZ30gcHJ2IC0gVGhlIHByaXZhdGUga2V5IGluIEJJUDMyIGZvcm1hdC5cbiAqIEBwYXJhbSB7dXR4b2xpYi5OZXR3b3JrfSBuZXR3b3JrIC0gVGhlIG5ldHdvcmsgdG8gdXNlIGZvciBzaWduaW5nLlxuICogQHJldHVybnMge3N0cmluZ30gLSBSZXR1cm5zIHRoZSBzaWduYXR1cmUgaW4gaGV4YWRlY2ltYWwgZm9ybWF0LlxuICovXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlTWVzc2FnZVNpZ25hdHVyZShcbiAgbWVzc2FnZTogU2lnbmFibGUsXG4gIHhwcnY6IHN0cmluZyxcbiAgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrID0gdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luXG4pOiBzdHJpbmcge1xuICBjb25zdCByZXF1ZXN0U3RyaW5nID0gSlNPTi5zdHJpbmdpZnkoY2Fub25pY2FsaXplT2JqZWN0KG1lc3NhZ2UpKTtcbiAgY29uc3QgcHJ2S2V5ID0gdXR4b2xpYi5iaXAzMi5mcm9tQmFzZTU4KHhwcnYsIG5ldHdvcmspO1xuICByZXR1cm4gc2RrY29yZS5zaWduTWVzc2FnZShyZXF1ZXN0U3RyaW5nLCBwcnZLZXksIG5ldHdvcmspLnRvU3RyaW5nKCdoZXgnKTtcbn1cbiJdfQ==
@@ -0,0 +1,7 @@
1
+ import { ILightningWallet, LightningWallet } from './lightning';
2
+ import * as sdkcore from '@bitgo-beta/sdk-core';
3
+ export type ICustodialLightningWallet = ILightningWallet;
4
+ export declare class CustodialLightningWallet extends LightningWallet implements ICustodialLightningWallet {
5
+ constructor(wallet: sdkcore.IWallet);
6
+ }
7
+ //# sourceMappingURL=custodialLightning.d.ts.map