PHP WebShell

Текущая директория: /opt/BitGoJS/modules/express/dist/src/lightning

Просмотр файла: lightningSignerRoutes.js

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.handleUnlockLightningWallet = exports.handleGetLightningWalletState = exports.handleCreateSignerMacaroon = exports.handleInitLightningWallet = void 0;
const net_1 = require("net");
const sdk_core_1 = require("@bitgo/sdk-core");
const abstract_lightning_1 = require("@bitgo/abstract-lightning");
const utxolib = require("@bitgo/utxo-lib");
const buffer_1 = require("buffer");
const codecs_1 = require("./codecs");
const lndSignerClient_1 = require("./lndSignerClient");
const errors_1 = require("../errors");
async function createSignerMacaroon(lndSignerClient, header, watchOnlyIp) {
    const { macaroon } = await lndSignerClient.bakeMacaroon({ permissions: abstract_lightning_1.signerMacaroonPermissions }, header);
    const macaroonBase64 = watchOnlyIp
        ? (0, abstract_lightning_1.addIPCaveatToMacaroon)(buffer_1.Buffer.from(macaroon, 'hex').toString('base64'), watchOnlyIp)
        : undefined;
    return macaroonBase64 ? buffer_1.Buffer.from(macaroonBase64, 'base64').toString('hex') : macaroon;
}
function getSignerRootKey(passphrase, userMainnetEncryptedPrv, network, decrypt) {
    const userMainnetPrv = decrypt({
        password: passphrase,
        input: userMainnetEncryptedPrv,
    });
    return utxolib.bitgo.keyutil.convertExtendedKeyNetwork(userMainnetPrv, utxolib.networks.bitcoin, network);
}
function getMacaroonRootKey(passphrase, nodeAuthEncryptedPrv, decrypt) {
    const hdNode = utxolib.bip32.fromBase58(decrypt({ password: passphrase, input: nodeAuthEncryptedPrv }));
    if (!hdNode.privateKey) {
        throw new Error('nodeAuthEncryptedPrv is not a private key');
    }
    return hdNode.privateKey.toString('base64');
}
/**
 * Handle the request to initialise remote signer LND for a wallet.
 */
async function handleInitLightningWallet(req) {
    const bitgo = req.bitgo;
    const coinName = req.params.coin;
    if (!(0, abstract_lightning_1.isLightningCoinName)(coinName)) {
        throw new errors_1.ApiResponseError(`Invalid coin ${coinName}. This is not a lightning coin.`, 400);
    }
    const coin = bitgo.coin(coinName);
    const walletId = req.params.id;
    if (typeof walletId !== 'string') {
        throw new errors_1.ApiResponseError(`Invalid wallet id: ${walletId}`, 400);
    }
    const { passphrase, expressHost } = (0, sdk_core_1.decodeOrElse)(codecs_1.InitLightningWalletRequest.name, codecs_1.InitLightningWalletRequest, req.body, (_) => {
        // DON'T throw errors from decodeOrElse. It could leak sensitive information.
        throw new errors_1.ApiResponseError('Invalid request body to initialize lightning wallet', 400);
    });
    const wallet = await coin.wallets().get({ id: walletId, includeBalance: false });
    if (wallet.subType() !== 'lightningSelfCustody') {
        throw new errors_1.ApiResponseError(`not a self custodial lighting wallet ${walletId}`, 400);
    }
    const lndSignerClient = await lndSignerClient_1.LndSignerClient.create(walletId, req.config);
    const userKey = await (0, abstract_lightning_1.getLightningKeychain)(wallet);
    const userKeyEncryptedPrv = userKey.encryptedPrv;
    if (!userKeyEncryptedPrv) {
        throw new errors_1.ApiResponseError('Missing encryptedPrv in user keychain', 400);
    }
    const { nodeAuthKey } = await (0, abstract_lightning_1.getLightningAuthKeychains)(wallet);
    const nodeAuthKeyEncryptedPrv = nodeAuthKey.encryptedPrv;
    if (!nodeAuthKeyEncryptedPrv) {
        throw new errors_1.ApiResponseError('Missing encryptedPrv in node auth keychain', 400);
    }
    const network = (0, abstract_lightning_1.getUtxolibNetwork)(coin.getChain());
    const signerRootKey = getSignerRootKey(passphrase, userKeyEncryptedPrv, network, bitgo.decrypt);
    const macaroonRootKey = getMacaroonRootKey(passphrase, nodeAuthKeyEncryptedPrv, bitgo.decrypt);
    const { admin_macaroon: adminMacaroon } = await lndSignerClient.initWallet({
        // The passphrase at LND can only accommodate a base64 character set
        // For more information, see BTC-1851
        wallet_password: buffer_1.Buffer.from(passphrase).toString('base64'),
        extended_master_key: signerRootKey,
        macaroon_root_key: macaroonRootKey,
    });
    return await (0, abstract_lightning_1.updateWalletCoinSpecific)(wallet, {
        signerAdminMacaroon: expressHost && !!(0, net_1.isIP)(expressHost) ? (0, abstract_lightning_1.addIPCaveatToMacaroon)(adminMacaroon, expressHost) : adminMacaroon,
        watchOnlyAccounts: (0, abstract_lightning_1.createWatchOnly)(signerRootKey, network),
        passphrase,
    });
}
exports.handleInitLightningWallet = handleInitLightningWallet;
/**
 * Handle the request to create a signer macaroon from remote signer LND for a wallet.
 */
async function handleCreateSignerMacaroon(req) {
    const bitgo = req.bitgo;
    const coinName = req.params.coin;
    if (!(0, abstract_lightning_1.isLightningCoinName)(coinName)) {
        throw new errors_1.ApiResponseError(`Invalid coin to create signer macaroon: ${coinName}. Must be a lightning coin.`, 400);
    }
    const coin = bitgo.coin(coinName);
    const walletId = req.params.id;
    if (typeof walletId !== 'string') {
        throw new errors_1.ApiResponseError(`Invalid wallet id: ${walletId}`, 400);
    }
    const { passphrase, addIpCaveatToMacaroon } = (0, sdk_core_1.decodeOrElse)(codecs_1.CreateSignerMacaroonRequest.name, codecs_1.CreateSignerMacaroonRequest, req.body, (_) => {
        // DON'T throw errors from decodeOrElse. It could leak sensitive information.
        throw new errors_1.ApiResponseError('Invalid request body to create signer macaroon', 400);
    });
    const wallet = await coin.wallets().get({ id: walletId, includeBalance: false });
    if (wallet.subType() !== 'lightningSelfCustody') {
        throw new errors_1.ApiResponseError(`not a self custodial lighting wallet ${walletId}`, 400);
    }
    const watchOnlyIp = wallet.coinSpecific()?.watchOnlyExternalIp;
    if (!watchOnlyIp && addIpCaveatToMacaroon) {
        throw new errors_1.ApiResponseError('Cannot create signer macaroon because the external IP is not set. This can take some time. Contact support@bitgo.com if longer than 24 hours.', 400);
    }
    if (watchOnlyIp && !(0, net_1.isIP)(watchOnlyIp)) {
        throw new errors_1.ApiResponseError(`Invalid IP address: ${watchOnlyIp}. Contact support@bitgo.com`, 500);
    }
    const lndSignerClient = await lndSignerClient_1.LndSignerClient.create(walletId, req.config);
    const encryptedSignerAdminMacaroon = wallet.coinSpecific()?.encryptedSignerAdminMacaroon;
    if (!encryptedSignerAdminMacaroon) {
        throw new errors_1.ApiResponseError('Missing encryptedSignerAdminMacaroon in wallet', 400);
    }
    const adminMacaroon = bitgo.decrypt({
        password: passphrase,
        input: encryptedSignerAdminMacaroon,
    });
    const signerMacaroon = await createSignerMacaroon(lndSignerClient, { adminMacaroonHex: buffer_1.Buffer.from(adminMacaroon, 'base64').toString('hex') }, addIpCaveatToMacaroon ? watchOnlyIp : null);
    return await (0, abstract_lightning_1.updateWalletCoinSpecific)(wallet, {
        signerMacaroon,
        passphrase,
    });
}
exports.handleCreateSignerMacaroon = handleCreateSignerMacaroon;
/**
 * Handle the request to get the state of a wallet from the signer.
 */
async function handleGetLightningWalletState(req) {
    const coinName = req.params.coin;
    if (!(0, abstract_lightning_1.isLightningCoinName)(coinName)) {
        throw new errors_1.ApiResponseError(`Invalid coin to get lightning wallet state: ${coinName}`, 400);
    }
    const walletId = req.params.id;
    if (typeof walletId !== 'string') {
        throw new errors_1.ApiResponseError(`Invalid wallet id: ${walletId}`, 400);
    }
    const lndSignerClient = await lndSignerClient_1.LndSignerClient.create(walletId, req.config);
    return await lndSignerClient.getWalletState();
}
exports.handleGetLightningWalletState = handleGetLightningWalletState;
/**
 * Handle the request to unlock a wallet in the signer.
 */
async function handleUnlockLightningWallet(req) {
    const coinName = req.params.coin;
    if (!(0, abstract_lightning_1.isLightningCoinName)(coinName)) {
        throw new errors_1.ApiResponseError(`Invalid coin to unlock lightning wallet: ${coinName}`, 400);
    }
    const walletId = req.params.id;
    if (typeof walletId !== 'string') {
        throw new errors_1.ApiResponseError(`Invalid wallet id: ${walletId}`, 400);
    }
    const { passphrase } = (0, sdk_core_1.decodeOrElse)(codecs_1.UnlockLightningWalletRequest.name, codecs_1.UnlockLightningWalletRequest, req.body, (_) => {
        // DON'T throw errors from decodeOrElse. It could leak sensitive information.
        throw new errors_1.ApiResponseError('Invalid request body to unlock lightning wallet', 400);
    });
    const lndSignerClient = await lndSignerClient_1.LndSignerClient.create(walletId, req.config);
    // The passphrase at LND can only accommodate a base64 character set
    // For more information, see BTC-1851
    await lndSignerClient.unlockWallet({
        wallet_password: buffer_1.Buffer.from(passphrase).toString('base64'),
    });
    return { message: 'ok' };
}
exports.handleUnlockLightningWallet = handleUnlockLightningWallet;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lightningSignerRoutes.js","sourceRoot":"","sources":["../../../src/lightning/lightningSignerRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["import { isIP } from 'net';\nimport * as express from 'express';\nimport { decodeOrElse } from '@bitgo/sdk-core';\nimport {\n  getUtxolibNetwork,\n  signerMacaroonPermissions,\n  createWatchOnly,\n  addIPCaveatToMacaroon,\n  isLightningCoinName,\n  getLightningKeychain,\n  getLightningAuthKeychains,\n  updateWalletCoinSpecific,\n} from '@bitgo/abstract-lightning';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport { Buffer } from 'buffer';\n\nimport {\n  CreateSignerMacaroonRequest,\n  GetWalletStateResponse,\n  InitLightningWalletRequest,\n  UnlockLightningWalletRequest,\n} from './codecs';\nimport { LndSignerClient } from './lndSignerClient';\nimport { ApiResponseError } from '../errors';\n\ntype Decrypt = (params: { input: string; password: string }) => string;\n\nasync function createSignerMacaroon(\n  lndSignerClient: LndSignerClient,\n  header: { adminMacaroonHex: string },\n  watchOnlyIp: string | undefined | null\n): Promise<string> {\n  const { macaroon } = await lndSignerClient.bakeMacaroon({ permissions: signerMacaroonPermissions }, header);\n  const macaroonBase64 = watchOnlyIp\n    ? addIPCaveatToMacaroon(Buffer.from(macaroon, 'hex').toString('base64'), watchOnlyIp)\n    : undefined;\n  return macaroonBase64 ? Buffer.from(macaroonBase64, 'base64').toString('hex') : macaroon;\n}\n\nfunction getSignerRootKey(\n  passphrase: string,\n  userMainnetEncryptedPrv: string,\n  network: utxolib.Network,\n  decrypt: Decrypt\n) {\n  const userMainnetPrv = decrypt({\n    password: passphrase,\n    input: userMainnetEncryptedPrv,\n  });\n  return utxolib.bitgo.keyutil.convertExtendedKeyNetwork(userMainnetPrv, utxolib.networks.bitcoin, network);\n}\n\nfunction getMacaroonRootKey(passphrase: string, nodeAuthEncryptedPrv: string, decrypt: Decrypt) {\n  const hdNode = utxolib.bip32.fromBase58(decrypt({ password: passphrase, input: nodeAuthEncryptedPrv }));\n  if (!hdNode.privateKey) {\n    throw new Error('nodeAuthEncryptedPrv is not a private key');\n  }\n  return hdNode.privateKey.toString('base64');\n}\n\n/**\n * Handle the request to initialise remote signer LND for a wallet.\n */\nexport async function handleInitLightningWallet(req: express.Request): Promise<unknown> {\n  const bitgo = req.bitgo;\n  const coinName = req.params.coin;\n  if (!isLightningCoinName(coinName)) {\n    throw new ApiResponseError(`Invalid coin ${coinName}. This is not a lightning coin.`, 400);\n  }\n  const coin = bitgo.coin(coinName);\n\n  const walletId = req.params.id;\n  if (typeof walletId !== 'string') {\n    throw new ApiResponseError(`Invalid wallet id: ${walletId}`, 400);\n  }\n\n  const { passphrase, expressHost } = decodeOrElse(\n    InitLightningWalletRequest.name,\n    InitLightningWalletRequest,\n    req.body,\n    (_) => {\n      // DON'T throw errors from decodeOrElse. It could leak sensitive information.\n      throw new ApiResponseError('Invalid request body to initialize lightning wallet', 400);\n    }\n  );\n\n  const wallet = await coin.wallets().get({ id: walletId, includeBalance: false });\n  if (wallet.subType() !== 'lightningSelfCustody') {\n    throw new ApiResponseError(`not a self custodial lighting wallet ${walletId}`, 400);\n  }\n  const lndSignerClient = await LndSignerClient.create(walletId, req.config);\n\n  const userKey = await getLightningKeychain(wallet);\n  const userKeyEncryptedPrv = userKey.encryptedPrv;\n  if (!userKeyEncryptedPrv) {\n    throw new ApiResponseError('Missing encryptedPrv in user keychain', 400);\n  }\n  const { nodeAuthKey } = await getLightningAuthKeychains(wallet);\n  const nodeAuthKeyEncryptedPrv = nodeAuthKey.encryptedPrv;\n  if (!nodeAuthKeyEncryptedPrv) {\n    throw new ApiResponseError('Missing encryptedPrv in node auth keychain', 400);\n  }\n  const network = getUtxolibNetwork(coin.getChain());\n  const signerRootKey = getSignerRootKey(passphrase, userKeyEncryptedPrv, network, bitgo.decrypt);\n  const macaroonRootKey = getMacaroonRootKey(passphrase, nodeAuthKeyEncryptedPrv, bitgo.decrypt);\n\n  const { admin_macaroon: adminMacaroon } = await lndSignerClient.initWallet({\n    // The passphrase at LND can only accommodate a base64 character set\n    // For more information, see BTC-1851\n    wallet_password: Buffer.from(passphrase).toString('base64'),\n    extended_master_key: signerRootKey,\n    macaroon_root_key: macaroonRootKey,\n  });\n\n  return await updateWalletCoinSpecific(wallet, {\n    signerAdminMacaroon:\n      expressHost && !!isIP(expressHost) ? addIPCaveatToMacaroon(adminMacaroon, expressHost) : adminMacaroon,\n    watchOnlyAccounts: createWatchOnly(signerRootKey, network),\n    passphrase,\n  });\n}\n\n/**\n * Handle the request to create a signer macaroon from remote signer LND for a wallet.\n */\nexport async function handleCreateSignerMacaroon(req: express.Request): Promise<unknown> {\n  const bitgo = req.bitgo;\n  const coinName = req.params.coin;\n  if (!isLightningCoinName(coinName)) {\n    throw new ApiResponseError(`Invalid coin to create signer macaroon: ${coinName}. Must be a lightning coin.`, 400);\n  }\n  const coin = bitgo.coin(coinName);\n  const walletId = req.params.id;\n  if (typeof walletId !== 'string') {\n    throw new ApiResponseError(`Invalid wallet id: ${walletId}`, 400);\n  }\n\n  const { passphrase, addIpCaveatToMacaroon } = decodeOrElse(\n    CreateSignerMacaroonRequest.name,\n    CreateSignerMacaroonRequest,\n    req.body,\n    (_) => {\n      // DON'T throw errors from decodeOrElse. It could leak sensitive information.\n      throw new ApiResponseError('Invalid request body to create signer macaroon', 400);\n    }\n  );\n\n  const wallet = await coin.wallets().get({ id: walletId, includeBalance: false });\n  if (wallet.subType() !== 'lightningSelfCustody') {\n    throw new ApiResponseError(`not a self custodial lighting wallet ${walletId}`, 400);\n  }\n  const watchOnlyIp = wallet.coinSpecific()?.watchOnlyExternalIp;\n  if (!watchOnlyIp && addIpCaveatToMacaroon) {\n    throw new ApiResponseError(\n      'Cannot create signer macaroon because the external IP is not set. This can take some time. Contact support@bitgo.com if longer than 24 hours.',\n      400\n    );\n  }\n\n  if (watchOnlyIp && !isIP(watchOnlyIp)) {\n    throw new ApiResponseError(`Invalid IP address: ${watchOnlyIp}. Contact support@bitgo.com`, 500);\n  }\n\n  const lndSignerClient = await LndSignerClient.create(walletId, req.config);\n\n  const encryptedSignerAdminMacaroon = wallet.coinSpecific()?.encryptedSignerAdminMacaroon;\n  if (!encryptedSignerAdminMacaroon) {\n    throw new ApiResponseError('Missing encryptedSignerAdminMacaroon in wallet', 400);\n  }\n  const adminMacaroon = bitgo.decrypt({\n    password: passphrase,\n    input: encryptedSignerAdminMacaroon,\n  });\n\n  const signerMacaroon = await createSignerMacaroon(\n    lndSignerClient,\n    { adminMacaroonHex: Buffer.from(adminMacaroon, 'base64').toString('hex') },\n    addIpCaveatToMacaroon ? watchOnlyIp : null\n  );\n\n  return await updateWalletCoinSpecific(wallet, {\n    signerMacaroon,\n    passphrase,\n  });\n}\n\n/**\n * Handle the request to get the state of a wallet from the signer.\n */\nexport async function handleGetLightningWalletState(req: express.Request): Promise<GetWalletStateResponse> {\n  const coinName = req.params.coin;\n  if (!isLightningCoinName(coinName)) {\n    throw new ApiResponseError(`Invalid coin to get lightning wallet state: ${coinName}`, 400);\n  }\n  const walletId = req.params.id;\n  if (typeof walletId !== 'string') {\n    throw new ApiResponseError(`Invalid wallet id: ${walletId}`, 400);\n  }\n\n  const lndSignerClient = await LndSignerClient.create(walletId, req.config);\n  return await lndSignerClient.getWalletState();\n}\n\n/**\n * Handle the request to unlock a wallet in the signer.\n */\nexport async function handleUnlockLightningWallet(req: express.Request): Promise<{ message: string }> {\n  const coinName = req.params.coin;\n  if (!isLightningCoinName(coinName)) {\n    throw new ApiResponseError(`Invalid coin to unlock lightning wallet: ${coinName}`, 400);\n  }\n  const walletId = req.params.id;\n  if (typeof walletId !== 'string') {\n    throw new ApiResponseError(`Invalid wallet id: ${walletId}`, 400);\n  }\n\n  const { passphrase } = decodeOrElse(\n    UnlockLightningWalletRequest.name,\n    UnlockLightningWalletRequest,\n    req.body,\n    (_) => {\n      // DON'T throw errors from decodeOrElse. It could leak sensitive information.\n      throw new ApiResponseError('Invalid request body to unlock lightning wallet', 400);\n    }\n  );\n\n  const lndSignerClient = await LndSignerClient.create(walletId, req.config);\n  // The passphrase at LND can only accommodate a base64 character set\n  // For more information, see BTC-1851\n  await lndSignerClient.unlockWallet({\n    wallet_password: Buffer.from(passphrase).toString('base64'),\n  });\n  return { message: 'ok' };\n}\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!