PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/abstract-lightning/dist/src/lightning
Просмотр файла: lightningUtils.js
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
Object.defineProperty(exports, "__esModule", { value: true });
exports.lightningNetworkName = exports.signerMacaroonPermissions = void 0;
exports.isLightningCoinName = isLightningCoinName;
exports.getLightningNetwork = getLightningNetwork;
exports.getLightningCoinName = getLightningCoinName;
exports.isValidLightningNetworkName = isValidLightningNetworkName;
exports.isValidLightningNetwork = isValidLightningNetwork;
exports.getStaticsLightningNetwork = getStaticsLightningNetwork;
exports.getUtxolibNetwork = getUtxolibNetwork;
exports.unwrapLightningCoinSpecific = unwrapLightningCoinSpecific;
exports.addIPCaveatToMacaroon = addIPCaveatToMacaroon;
exports.createWatchOnly = createWatchOnly;
exports.deriveLightningServiceSharedSecret = deriveLightningServiceSharedSecret;
exports.deriveMiddlewareSharedSecret = deriveMiddlewareSharedSecret;
exports.deriveTatSharedSecret = deriveTatSharedSecret;
exports.computeBip32DerivationIndexFromSeed = computeBip32DerivationIndexFromSeed;
const statics = __importStar(require("@bitgo/statics"));
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const crypto_1 = require("crypto");
const macaroon_1 = require("macaroon");
const bs58check = __importStar(require("bs58check"));
const sdkcore = __importStar(require("@bitgo/sdk-core"));
// https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#the-signer-node
exports.signerMacaroonPermissions = [
{
entity: 'message',
action: 'write',
},
{
entity: 'signer',
action: 'generate',
},
{
entity: 'address',
action: 'read',
},
{
entity: 'onchain',
action: 'write',
},
];
exports.lightningNetworkName = ['bitcoin', 'testnet'];
/**
* Checks if the coin name is a lightning coin name.
*/
function isLightningCoinName(coinName) {
return coinName === 'lnbtc' || coinName === 'tlnbtc';
}
/**
* Get the utxolib network for a lightning network.
*/
function getLightningNetwork(networkName) {
return utxolib.networks[networkName];
}
/**
* Get the lightning coin name for a utxolib network.
*/
function getLightningCoinName(network) {
return network === utxolib.networks.bitcoin ? 'lnbtc' : 'tlnbtc';
}
/**
* Checks if the network name is a valid lightning network name.
*/
function isValidLightningNetworkName(networkName) {
return exports.lightningNetworkName.includes(networkName);
}
/**
* Checks if the network is a valid lightning network.
*/
function isValidLightningNetwork(network) {
return utxolib.isValidNetwork(network) && isValidLightningNetworkName(utxolib.getNetworkName(network));
}
/**
* Returns the statics network data for a lightning coin.
*/
function getStaticsLightningNetwork(coinName) {
if (!isLightningCoinName(coinName)) {
throw new Error(`${coinName} is not a lightning coin`);
}
const coin = statics.coins.get(coinName);
if (!(coin instanceof statics.LightningCoin)) {
throw new Error('coin is not a lightning coin');
}
return coin.network;
}
/**
* Returns the utxolib network for a lightning coin.
*/
function getUtxolibNetwork(coinName) {
const networkName = getStaticsLightningNetwork(coinName).utxolibName;
if (!isValidLightningNetworkName(networkName)) {
throw new Error('invalid lightning network');
}
return getLightningNetwork(networkName);
}
/**
* Returns coin specific data for a lightning coin.
*/
function unwrapLightningCoinSpecific(obj, coinSpecificPath) {
if (coinSpecificPath !== 'lnbtc' && coinSpecificPath !== 'tlnbtc') {
throw new Error(`invalid coinSpecificPath ${coinSpecificPath} for lightning coin`);
}
if (coinSpecificPath === 'lnbtc' && 'lnbtc' in obj) {
return obj.lnbtc;
}
if (coinSpecificPath === 'tlnbtc' && 'tlnbtc' in obj) {
return obj.tlnbtc;
}
throw new Error('invalid lightning coin specific');
}
/**
* Adds an IP caveat to a macaroon and returns the modified macaroon as a Base64 string.
*/
function addIPCaveatToMacaroon(macaroonBase64, ip) {
const macaroon = (0, macaroon_1.importMacaroon)(macaroonBase64);
macaroon.addFirstPartyCaveat(`ipaddr ${ip}`);
return (0, macaroon_1.bytesToBase64)(macaroon.exportBinary());
}
const PURPOSE_WRAPPED_P2WKH = 49;
const PURPOSE_P2WKH = 84;
const PURPOSE_P2TR = 86;
const PURPOSE_ALL_OTHERS = 1017;
/**
* Converts an extended public key (xpub) to the appropriate prefix (ypub, vpub, etc.) based on its purpose and network.
*/
function convertXpubPrefix(xpub, purpose, isMainnet) {
if (purpose === PURPOSE_P2TR || purpose === PURPOSE_ALL_OTHERS) {
return xpub;
}
const data = bs58check.decode(xpub);
let versionBytes;
switch (purpose) {
case PURPOSE_WRAPPED_P2WKH:
versionBytes = isMainnet ? Buffer.from([0x04, 0x9d, 0x7c, 0xb2]) : Buffer.from([0x04, 0x4a, 0x52, 0x62]); // ypub/upub for p2sh-p2wpkh
break;
case PURPOSE_P2WKH:
versionBytes = isMainnet ? Buffer.from([0x04, 0xb2, 0x47, 0x46]) : Buffer.from([0x04, 0x5f, 0x1c, 0xf6]); // zpub/vpub for p2wpkh
break;
default:
throw new Error('Unsupported purpose');
}
versionBytes.copy(data, 0, 0, 4);
return bs58check.encode(data);
}
/**
* Derives watch-only accounts from the master HD node for the given purposes and network.
*/
function deriveWatchOnlyAccounts(masterHDNode, isMainnet) {
// https://github.com/lightningnetwork/lnd/blob/master/docs/remote-signing.md#required-accounts
if (masterHDNode.isNeutered()) {
throw new Error('masterHDNode must not be neutered');
}
const purposes = [PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR, PURPOSE_ALL_OTHERS];
return purposes.flatMap((purpose) => {
const maxAccount = purpose === PURPOSE_ALL_OTHERS ? 255 : 0;
const coinType = purpose !== PURPOSE_ALL_OTHERS || isMainnet ? 0 : 1;
return Array.from({ length: maxAccount + 1 }, (_, account) => {
const path = `m/${purpose}'/${coinType}'/${account}'`;
const derivedNode = masterHDNode.derivePath(path);
// Ensure the node is neutered (i.e., converted to public key only)
const neuteredNode = derivedNode.neutered();
const xpub = convertXpubPrefix(neuteredNode.toBase58(), purpose, isMainnet);
return {
purpose,
coin_type: coinType,
account,
xpub,
};
});
});
}
/**
* Creates a watch-only wallet init data from the provided signer root key and network.
*/
function createWatchOnly(signerRootKey, network) {
const masterHDNode = utxolib.bip32.fromBase58(signerRootKey, network);
const getCurrentUnixTimestamp = () => {
return Math.floor(Date.now() / 1000);
};
const master_key_birthday_timestamp = getCurrentUnixTimestamp().toString();
const master_key_fingerprint = masterHDNode.fingerprint.toString('hex');
const accounts = deriveWatchOnlyAccounts(masterHDNode, utxolib.isMainnet(network));
return { master_key_birthday_timestamp, master_key_fingerprint, accounts };
}
/**
* Derives the shared Elliptic Curve Diffie-Hellman (ECDH) secret between the user's auth extended private key
* and the Lightning service's public key for secure communication.
*/
function deriveLightningServiceSharedSecret(coinName, userAuthXprv) {
const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).lightningServicePubKey, 'hex');
const userAuthHdNode = utxolib.bip32.fromBase58(userAuthXprv);
return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
}
/**
* Derives the shared secret for the middleware using a private key and the middleware's public key.
*/
function deriveMiddlewareSharedSecret(coinName, xprv) {
const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).middlewarePubKey, 'hex');
const userAuthHdNode = utxolib.bip32.fromBase58(xprv);
return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
}
/**
* Derives the shared secret for TAT service using ta private key and the TAT public key.
*/
function deriveTatSharedSecret(coinName, xprv) {
const publicKey = Buffer.from(getStaticsLightningNetwork(coinName).tatPubKey, 'hex');
const userAuthHdNode = utxolib.bip32.fromBase58(xprv);
return sdkcore.getSharedSecret(userAuthHdNode, publicKey);
}
/**
* Given a seed, compute a BIP32 derivation index.
* 0 <= index < 2147483648 (largest 31 bit number). This needs to be 2^31 - 1 so that the bip32 library
* can derive the hardened key.
* @param seed (optional) If nothing provided, we will generate one randomly
*/
function computeBip32DerivationIndexFromSeed(seed) {
return ((Buffer.from(utxolib.crypto.sha256(Buffer.from(seed ?? (0, crypto_1.randomBytes)(32).toString('hex'), 'utf8'))).readUint32BE(0) %
Math.pow(2, 31)) -
1);
}
//# 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/statics';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport { randomBytes } from 'crypto';\nimport { importMacaroon, bytesToBase64 } from 'macaroon';\nimport * as bs58check from 'bs58check';\nimport * as sdkcore from '@bitgo/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 */\nfunction deriveWatchOnlyAccounts(masterHDNode: utxolib.BIP32Interface, isMainnet: boolean): 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\n  const purposes = [PURPOSE_WRAPPED_P2WKH, PURPOSE_P2WKH, PURPOSE_P2TR, PURPOSE_ALL_OTHERS] as const;\n\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"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!