PHP WebShell
Текущая директория: /opt/BitGoJS/modules/express/dist/src
Просмотр файла: expressApp.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.init = exports.prepareIpc = exports.app = exports.setupRoutes = exports.createBaseUri = exports.createServer = exports.startup = void 0;
/**
* @prettier
*/
const express = require("express");
const path = require("path");
const _ = require("lodash");
const debugLib = require("debug");
const https = require("https");
const http = require("http");
const morgan = require("morgan");
const fs = require("fs");
const timeout = require("connect-timeout");
const config_1 = require("./config");
const debug = debugLib('bitgo:express');
const constants_1 = require("constants");
const errors_1 = require("./errors");
const bitgo_1 = require("bitgo");
const clientRoutes = require("./clientRoutes");
/**
* Set up the logging middleware provided by morgan
*
* @param app
* @param config
*/
function setupLogging(app, config) {
// Set up morgan for logging, with optional logging into a file
let middleware;
if (config.logFile) {
// create a write stream (in append mode)
const accessLogPath = path.resolve(config.logFile);
const accessLogStream = fs.createWriteStream(accessLogPath, { flags: 'a' });
/* eslint-disable-next-line no-console */
console.log('Log location: ' + accessLogPath);
// setup the logger
middleware = morgan('combined', { stream: accessLogStream });
}
else {
middleware = morgan('combined');
}
app.use(middleware);
morgan.token('remote-user', function (req) {
return req.isProxy ? 'proxy' : 'local_express';
});
}
/**
* If we're running in a custom env, set the appropriate environment URI and network properties
*
* @param config
*/
function configureEnvironment(config) {
const { customRootUri, customBitcoinNetwork } = config;
if (customRootUri) {
bitgo_1.Environments['custom'].uri = customRootUri;
}
if (customBitcoinNetwork) {
bitgo_1.Environments['custom'].network = customBitcoinNetwork;
}
}
/**
* Create an HTTP server configured for accepting HTTPS connections
*
* @param config application configuration
* @param app
* @return {Server}
*/
async function createHttpsServer(app, config) {
const { keyPath, crtPath, sslKey, sslCert } = config;
let key;
let cert;
if (sslKey && sslCert) {
key = sslKey;
cert = sslCert;
}
else if (keyPath && crtPath) {
const privateKeyPromise = fs.promises.readFile(keyPath, 'utf8');
const certificatePromise = fs.promises.readFile(crtPath, 'utf8');
[key, cert] = await Promise.all([privateKeyPromise, certificatePromise]);
}
else {
throw new Error('Failed to get ssl key and certificate');
}
return https.createServer({ secureOptions: constants_1.SSL_OP_NO_TLSv1, key, cert }, app);
}
/**
* Create an HTTP server configured for accepting plain old HTTP connections
*
* @param app
* @return {Server}
*/
function createHttpServer(app) {
return http.createServer(app);
}
/**
* Create a startup function which will be run upon server initialization
*
* @param config
* @param baseUri
* @return {Function}
*/
function startup(config, baseUri) {
return function () {
const { env, ipc, customRootUri, customBitcoinNetwork, signerMode, lightningSignerFileSystemPath } = config;
/* eslint-disable no-console */
console.log('BitGo-Express running');
console.log(`Environment: ${env}`);
if (ipc) {
console.log(`IPC path: ${ipc}`);
}
else {
console.log(`Base URI: ${baseUri}`);
}
if (customRootUri) {
console.log(`Custom root URI: ${customRootUri}`);
}
if (customBitcoinNetwork) {
console.log(`Custom bitcoin network: ${customBitcoinNetwork}`);
}
if (signerMode) {
console.log(`External signer mode: ${signerMode}`);
}
if (lightningSignerFileSystemPath) {
console.log(`Lightning signer file system path: ${lightningSignerFileSystemPath}`);
}
/* eslint-enable no-console */
};
}
exports.startup = startup;
/**
* helper function to determine whether we should run the server over TLS or not
*/
function isTLS(config) {
const { keyPath, crtPath, sslKey, sslCert } = config;
return Boolean((keyPath && crtPath) || (sslKey && sslCert));
}
/**
* Create either a HTTP or HTTPS server
*/
async function createServer(config, app) {
const server = isTLS(config) ? await createHttpsServer(app, config) : createHttpServer(app);
// Set keepAliveTimeout and headersTimeout if specified in config
if (config.keepAliveTimeout !== undefined) {
server.keepAliveTimeout = config.keepAliveTimeout;
}
if (config.headersTimeout !== undefined) {
server.headersTimeout = config.headersTimeout;
}
return server;
}
exports.createServer = createServer;
/**
* Create the base URI where the BitGoExpress server will be available once started
* @return {string}
*/
function createBaseUri(config) {
const { bind, port } = config;
const tls = isTLS(config);
const isStandardPort = (port === 80 && !tls) || (port === 443 && tls);
return `http${tls ? 's' : ''}://${bind}${!isStandardPort ? ':' + port : ''}`;
}
exports.createBaseUri = createBaseUri;
/**
* Check the that the json file exists
* @param path
*/
function checkJsonFilePath(path) {
try {
const jsonFile = fs.readFileSync(path, { encoding: 'utf8' });
JSON.parse(jsonFile);
}
catch (e) {
throw new Error(`Failed to parse ${path} - ${e.message}`);
}
}
/**
* Check environment and other preconditions to ensure bitgo-express can start safely
* @param config
*/
function checkPreconditions(config) {
const { env, disableEnvCheck, bind, ipc, disableSSL, keyPath, crtPath, sslKey, sslCert, customRootUri, customBitcoinNetwork, externalSignerUrl, signerMode, signerFileSystemPath, lightningSignerFileSystemPath, } = config;
// warn or throw if the NODE_ENV is not production when BITGO_ENV is production - this can leak system info from express
if (env === 'prod' && process.env.NODE_ENV !== 'production') {
if (!disableEnvCheck) {
throw new errors_1.NodeEnvironmentError('NODE_ENV should be set to production when running against prod environment. Use --disableenvcheck if you really want to run in a non-production node configuration.');
}
else {
console.warn(`warning: unsafe NODE_ENV '${process.env.NODE_ENV}'. NODE_ENV must be set to 'production' when running against BitGo production environment.`);
}
}
const needsTLS = !ipc && env === 'prod' && bind !== 'localhost' && !disableSSL;
// make sure keyPath and crtPath are set when running over TLS
if (needsTLS && !(keyPath && crtPath) && !(sslKey && sslCert)) {
throw new errors_1.TlsConfigurationError('Must enable TLS when running against prod and listening on external interfaces!');
}
if (Boolean(keyPath) !== Boolean(crtPath) || Boolean(sslKey) !== Boolean(sslCert)) {
throw new errors_1.TlsConfigurationError('Must provide both keypath and crtpath when running in TLS mode!');
}
if ((customRootUri || customBitcoinNetwork) && env !== 'custom') {
console.warn(`customRootUri or customBitcoinNetwork is set, but env is '${env}'. Setting env to 'custom'.`);
config.env = 'custom';
}
if (externalSignerUrl !== undefined && (signerMode !== undefined || signerFileSystemPath !== undefined)) {
throw new errors_1.ExternalSignerConfigError('signerMode or signerFileSystemPath is set, but externalSignerUrl is also set.');
}
if ((signerMode !== undefined || signerFileSystemPath !== undefined) && !(signerMode && signerFileSystemPath)) {
throw new errors_1.ExternalSignerConfigError('signerMode and signerFileSystemPath must both be set in order to run in external signing mode.');
}
if (signerMode !== undefined && lightningSignerFileSystemPath !== undefined) {
throw new errors_1.LightningSignerConfigError('signerMode and lightningSignerFileSystemPath cannot be set at the same time.');
}
if (signerFileSystemPath !== undefined) {
checkJsonFilePath(signerFileSystemPath);
}
if (lightningSignerFileSystemPath !== undefined) {
checkJsonFilePath(lightningSignerFileSystemPath);
}
}
function setupRoutes(app, config) {
if (config.signerMode) {
clientRoutes.setupSigningRoutes(app, config);
}
else {
if (config.lightningSignerFileSystemPath) {
clientRoutes.setupLightningSignerNodeRoutes(app, config);
}
clientRoutes.setupAPIRoutes(app, config);
}
}
exports.setupRoutes = setupRoutes;
function app(cfg) {
debug('app is initializing');
const app = express();
setupLogging(app, cfg);
debug('logging setup');
const { debugNamespace } = cfg;
// enable specified debug namespaces
if (_.isArray(debugNamespace)) {
for (const ns of debugNamespace) {
if (ns && !debugLib.enabled(ns)) {
debugLib.enable(ns);
}
}
}
checkPreconditions(cfg);
debug('preconditions satisfied');
// Be more robust about accepting URLs with double slashes
app.use(function replaceUrlSlashes(req, res, next) {
req.url = req.url.replace(/\/\//g, '/');
next();
});
app.use(timeout(cfg.timeout));
// Decorate the client routes
setupRoutes(app, cfg);
configureEnvironment(cfg);
return app;
}
exports.app = app;
/**
* Prepare to listen on an IPC (unix domain) socket instead of a normal TCP port.
* @param ipcSocketFilePath path to file where IPC socket should be created
*/
async function prepareIpc(ipcSocketFilePath) {
if (process.platform === 'win32') {
throw new errors_1.IpcError(`IPC option is not supported on platform ${process.platform}`);
}
try {
const stat = fs.statSync(ipcSocketFilePath);
if (!stat.isSocket()) {
throw new errors_1.IpcError('IPC socket is not actually a socket');
}
// ipc socket does exist and is indeed a socket. However, the socket cannot already exist prior
// to being bound since it will be created by express internally when binding. If there's a stale
// socket from the last run, clean it up before attempting to bind to it again. Arguably, it would
// be better to do this before exiting, but that gets a bit more complicated when all exit paths
// need to clean up the socket file correctly.
fs.unlinkSync(ipcSocketFilePath);
}
catch (e) {
if (e.code !== 'ENOENT') {
throw e;
}
}
}
exports.prepareIpc = prepareIpc;
async function init() {
const cfg = (0, config_1.config)();
const expressApp = app(cfg);
const server = await createServer(cfg, expressApp);
const { port, bind, ipc } = cfg;
const baseUri = createBaseUri(cfg);
if (ipc) {
await prepareIpc(ipc);
server.listen(ipc, startup(cfg, baseUri));
}
else {
server.listen(port, bind, startup(cfg, baseUri));
}
}
exports.init = init;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expressApp.js","sourceRoot":"","sources":["../../src/expressApp.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as express from 'express';\nimport * as path from 'path';\nimport * as _ from 'lodash';\nimport * as debugLib from 'debug';\nimport * as https from 'https';\nimport * as http from 'http';\nimport * as morgan from 'morgan';\nimport * as fs from 'fs';\nimport type { Request as StaticRequest } from 'express-serve-static-core';\nimport * as timeout from 'connect-timeout';\n\nimport { Config, config } from './config';\n\nconst debug = debugLib('bitgo:express');\n\nimport { SSL_OP_NO_TLSv1 } from 'constants';\nimport {\n  IpcError,\n  NodeEnvironmentError,\n  TlsConfigurationError,\n  ExternalSignerConfigError,\n  LightningSignerConfigError,\n} from './errors';\n\nimport { Environments } from 'bitgo';\nimport * as clientRoutes from './clientRoutes';\n\n/**\n * Set up the logging middleware provided by morgan\n *\n * @param app\n * @param config\n */\nfunction setupLogging(app: express.Application, config: Config): void {\n  // Set up morgan for logging, with optional logging into a file\n  let middleware;\n  if (config.logFile) {\n    // create a write stream (in append mode)\n    const accessLogPath = path.resolve(config.logFile);\n    const accessLogStream = fs.createWriteStream(accessLogPath, { flags: 'a' });\n    /* eslint-disable-next-line no-console */\n    console.log('Log location: ' + accessLogPath);\n    // setup the logger\n    middleware = morgan('combined', { stream: accessLogStream });\n  } else {\n    middleware = morgan('combined');\n  }\n\n  app.use(middleware);\n  morgan.token('remote-user', function (req: StaticRequest) {\n    return req.isProxy ? 'proxy' : 'local_express';\n  });\n}\n\n/**\n * If we're running in a custom env, set the appropriate environment URI and network properties\n *\n * @param config\n */\nfunction configureEnvironment(config: Config): void {\n  const { customRootUri, customBitcoinNetwork } = config;\n  if (customRootUri) {\n    Environments['custom'].uri = customRootUri;\n  }\n\n  if (customBitcoinNetwork) {\n    Environments['custom'].network = customBitcoinNetwork;\n  }\n}\n\n/**\n * Create an HTTP server configured for accepting HTTPS connections\n *\n * @param config application configuration\n * @param app\n * @return {Server}\n */\nasync function createHttpsServer(app: express.Application, config: Config): Promise<https.Server> {\n  const { keyPath, crtPath, sslKey, sslCert } = config;\n  let key: string;\n  let cert: string;\n  if (sslKey && sslCert) {\n    key = sslKey;\n    cert = sslCert;\n  } else if (keyPath && crtPath) {\n    const privateKeyPromise = fs.promises.readFile(keyPath, 'utf8');\n    const certificatePromise = fs.promises.readFile(crtPath, 'utf8');\n\n    [key, cert] = await Promise.all([privateKeyPromise, certificatePromise]);\n  } else {\n    throw new Error('Failed to get ssl key and certificate');\n  }\n\n  return https.createServer({ secureOptions: SSL_OP_NO_TLSv1, key, cert }, app);\n}\n\n/**\n * Create an HTTP server configured for accepting plain old HTTP connections\n *\n * @param app\n * @return {Server}\n */\nfunction createHttpServer(app: express.Application): http.Server {\n  return http.createServer(app);\n}\n\n/**\n * Create a startup function which will be run upon server initialization\n *\n * @param config\n * @param baseUri\n * @return {Function}\n */\nexport function startup(config: Config, baseUri: string): () => void {\n  return function () {\n    const { env, ipc, customRootUri, customBitcoinNetwork, signerMode, lightningSignerFileSystemPath } = config;\n    /* eslint-disable no-console */\n    console.log('BitGo-Express running');\n    console.log(`Environment: ${env}`);\n    if (ipc) {\n      console.log(`IPC path: ${ipc}`);\n    } else {\n      console.log(`Base URI: ${baseUri}`);\n    }\n    if (customRootUri) {\n      console.log(`Custom root URI: ${customRootUri}`);\n    }\n    if (customBitcoinNetwork) {\n      console.log(`Custom bitcoin network: ${customBitcoinNetwork}`);\n    }\n    if (signerMode) {\n      console.log(`External signer mode: ${signerMode}`);\n    }\n    if (lightningSignerFileSystemPath) {\n      console.log(`Lightning signer file system path: ${lightningSignerFileSystemPath}`);\n    }\n    /* eslint-enable no-console */\n  };\n}\n\n/**\n * helper function to determine whether we should run the server over TLS or not\n */\nfunction isTLS(config: Config): config is Config & { keyPath: string; crtPath: string } {\n  const { keyPath, crtPath, sslKey, sslCert } = config;\n  return Boolean((keyPath && crtPath) || (sslKey && sslCert));\n}\n\n/**\n * Create either a HTTP or HTTPS server\n */\nexport async function createServer(config: Config, app: express.Application): Promise<https.Server | http.Server> {\n  const server = isTLS(config) ? await createHttpsServer(app, config) : createHttpServer(app);\n\n  // Set keepAliveTimeout and headersTimeout if specified in config\n  if (config.keepAliveTimeout !== undefined) {\n    server.keepAliveTimeout = config.keepAliveTimeout;\n  }\n  if (config.headersTimeout !== undefined) {\n    server.headersTimeout = config.headersTimeout;\n  }\n  return server;\n}\n\n/**\n * Create the base URI where the BitGoExpress server will be available once started\n * @return {string}\n */\nexport function createBaseUri(config: Config): string {\n  const { bind, port } = config;\n  const tls = isTLS(config);\n  const isStandardPort = (port === 80 && !tls) || (port === 443 && tls);\n  return `http${tls ? 's' : ''}://${bind}${!isStandardPort ? ':' + port : ''}`;\n}\n\n/**\n * Check the that the json file exists\n * @param path\n */\nfunction checkJsonFilePath(path: string) {\n  try {\n    const jsonFile = fs.readFileSync(path, { encoding: 'utf8' });\n    JSON.parse(jsonFile);\n  } catch (e) {\n    throw new Error(`Failed to parse ${path} - ${e.message}`);\n  }\n}\n\n/**\n * Check environment and other preconditions to ensure bitgo-express can start safely\n * @param config\n */\nfunction checkPreconditions(config: Config) {\n  const {\n    env,\n    disableEnvCheck,\n    bind,\n    ipc,\n    disableSSL,\n    keyPath,\n    crtPath,\n    sslKey,\n    sslCert,\n    customRootUri,\n    customBitcoinNetwork,\n    externalSignerUrl,\n    signerMode,\n    signerFileSystemPath,\n    lightningSignerFileSystemPath,\n  } = config;\n\n  // warn or throw if the NODE_ENV is not production when BITGO_ENV is production - this can leak system info from express\n  if (env === 'prod' && process.env.NODE_ENV !== 'production') {\n    if (!disableEnvCheck) {\n      throw new NodeEnvironmentError(\n        'NODE_ENV should be set to production when running against prod environment. Use --disableenvcheck if you really want to run in a non-production node configuration.'\n      );\n    } else {\n      console.warn(\n        `warning: unsafe NODE_ENV '${process.env.NODE_ENV}'. NODE_ENV must be set to 'production' when running against BitGo production environment.`\n      );\n    }\n  }\n\n  const needsTLS = !ipc && env === 'prod' && bind !== 'localhost' && !disableSSL;\n\n  // make sure keyPath and crtPath are set when running over TLS\n  if (needsTLS && !(keyPath && crtPath) && !(sslKey && sslCert)) {\n    throw new TlsConfigurationError('Must enable TLS when running against prod and listening on external interfaces!');\n  }\n\n  if (Boolean(keyPath) !== Boolean(crtPath) || Boolean(sslKey) !== Boolean(sslCert)) {\n    throw new TlsConfigurationError('Must provide both keypath and crtpath when running in TLS mode!');\n  }\n\n  if ((customRootUri || customBitcoinNetwork) && env !== 'custom') {\n    console.warn(`customRootUri or customBitcoinNetwork is set, but env is '${env}'. Setting env to 'custom'.`);\n    config.env = 'custom';\n  }\n\n  if (externalSignerUrl !== undefined && (signerMode !== undefined || signerFileSystemPath !== undefined)) {\n    throw new ExternalSignerConfigError(\n      'signerMode or signerFileSystemPath is set, but externalSignerUrl is also set.'\n    );\n  }\n\n  if ((signerMode !== undefined || signerFileSystemPath !== undefined) && !(signerMode && signerFileSystemPath)) {\n    throw new ExternalSignerConfigError(\n      'signerMode and signerFileSystemPath must both be set in order to run in external signing mode.'\n    );\n  }\n\n  if (signerMode !== undefined && lightningSignerFileSystemPath !== undefined) {\n    throw new LightningSignerConfigError(\n      'signerMode and lightningSignerFileSystemPath cannot be set at the same time.'\n    );\n  }\n\n  if (signerFileSystemPath !== undefined) {\n    checkJsonFilePath(signerFileSystemPath);\n  }\n\n  if (lightningSignerFileSystemPath !== undefined) {\n    checkJsonFilePath(lightningSignerFileSystemPath);\n  }\n}\n\nexport function setupRoutes(app: express.Application, config: Config): void {\n  if (config.signerMode) {\n    clientRoutes.setupSigningRoutes(app, config);\n  } else {\n    if (config.lightningSignerFileSystemPath) {\n      clientRoutes.setupLightningSignerNodeRoutes(app, config);\n    }\n    clientRoutes.setupAPIRoutes(app, config);\n  }\n}\n\nexport function app(cfg: Config): express.Application {\n  debug('app is initializing');\n\n  const app = express();\n\n  setupLogging(app, cfg);\n  debug('logging setup');\n\n  const { debugNamespace } = cfg;\n\n  // enable specified debug namespaces\n  if (_.isArray(debugNamespace)) {\n    for (const ns of debugNamespace) {\n      if (ns && !debugLib.enabled(ns)) {\n        debugLib.enable(ns);\n      }\n    }\n  }\n\n  checkPreconditions(cfg);\n  debug('preconditions satisfied');\n\n  // Be more robust about accepting URLs with double slashes\n  app.use(function replaceUrlSlashes(req, res, next) {\n    req.url = req.url.replace(/\\/\\//g, '/');\n    next();\n  });\n\n  app.use(timeout(cfg.timeout));\n\n  // Decorate the client routes\n  setupRoutes(app, cfg);\n\n  configureEnvironment(cfg);\n\n  return app;\n}\n\n/**\n * Prepare to listen on an IPC (unix domain) socket instead of a normal TCP port.\n * @param ipcSocketFilePath path to file where IPC socket should be created\n */\nexport async function prepareIpc(ipcSocketFilePath: string) {\n  if (process.platform === 'win32') {\n    throw new IpcError(`IPC option is not supported on platform ${process.platform}`);\n  }\n\n  try {\n    const stat = fs.statSync(ipcSocketFilePath);\n    if (!stat.isSocket()) {\n      throw new IpcError('IPC socket is not actually a socket');\n    }\n    // ipc socket does exist and is indeed a socket. However, the socket cannot already exist prior\n    // to being bound since it will be created by express internally when binding. If there's a stale\n    // socket from the last run, clean it up before attempting to bind to it again. Arguably, it would\n    // be better to do this before exiting, but that gets a bit more complicated when all exit paths\n    // need to clean up the socket file correctly.\n    fs.unlinkSync(ipcSocketFilePath);\n  } catch (e) {\n    if (e.code !== 'ENOENT') {\n      throw e;\n    }\n  }\n}\n\nexport async function init(): Promise<void> {\n  const cfg = config();\n  const expressApp = app(cfg);\n\n  const server = await createServer(cfg, expressApp);\n\n  const { port, bind, ipc } = cfg;\n  const baseUri = createBaseUri(cfg);\n\n  if (ipc) {\n    await prepareIpc(ipc);\n    server.listen(ipc, startup(cfg, baseUri));\n  } else {\n    server.listen(port, bind, startup(cfg, baseUri));\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!