PHP WebShell
Текущая директория: /opt/BitGoJS/modules/express/dist/test/unit
Просмотр файла: bitgoExpress.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const should = require("should");
require("should-http");
require("should-sinon");
require("../lib/asserts");
const nock = require("nock");
const sinon = require("sinon");
const fs = require("fs");
const http = require("http");
const https = require("https");
const debugLib = require("debug");
const path = require("path");
const bitgo_1 = require("bitgo");
const constants_1 = require("constants");
const errors_1 = require("../../src/errors");
nock.disableNetConnect();
const expressApp_1 = require("../../src/expressApp");
const clientRoutes = require("../../src/clientRoutes");
describe('Bitgo Express', function () {
describe('server initialization', function () {
const validPrvJSON = '{"61f039aad587c2000745c687373e0fa9":"xprv9s21ZrQH143K3EuPWCBuqnWxydaQV6et9htQige4EswvcHKEzNmkVmwTwKoadyHzJYppuADB7Us7AbaNLToNvoFoSxuWqndQRYtnNy5DUY2"}';
const validLightningSignerConfigJSON = '{"fakeid":{"url": "https://127.0.0.1:8080","tlsCert":"dummy"}}';
it('should require NODE_ENV to be production when running against prod env', function () {
const envStub = sinon.stub(process, 'env').value({ NODE_ENV: 'production' });
try {
(() => (0, expressApp_1.app)({
env: 'prod',
bind: 'localhost',
})).should.not.throw();
process.env.NODE_ENV = 'dev';
(() => (0, expressApp_1.app)({
env: 'prod',
bind: 'localhost',
})).should.throw(errors_1.NodeEnvironmentError);
}
finally {
envStub.restore();
}
});
it('should disable NODE_ENV check if disableenvcheck argument is given', function () {
const envStub = sinon.stub(process, 'env').value({ NODE_ENV: 'dev' });
try {
(() => (0, expressApp_1.app)({
env: 'prod',
bind: 'localhost',
disableEnvCheck: true,
})).should.not.throw();
}
finally {
envStub.restore();
}
});
it('should require TLS for prod env when listening on external interfaces', function () {
const args = {
env: 'prod',
bind: '1',
disableEnvCheck: true,
disableSSL: false,
crtPath: undefined,
keyPath: undefined,
};
(() => (0, expressApp_1.app)(args)).should.throw(errors_1.TlsConfigurationError);
args.bind = 'localhost';
(() => (0, expressApp_1.app)(args)).should.not.throw();
args.bind = '1';
args.env = 'test';
(() => (0, expressApp_1.app)(args)).should.not.throw();
args.disableSSL = true;
args.env = 'prod';
(() => (0, expressApp_1.app)(args)).should.not.throw();
delete args.disableSSL;
args.crtPath = '/tmp/cert.pem';
(() => (0, expressApp_1.app)(args)).should.throw(errors_1.TlsConfigurationError);
delete args.crtPath;
args.keyPath = '/tmp/key.pem';
(() => (0, expressApp_1.app)(args)).should.throw(errors_1.TlsConfigurationError);
});
it('should require both keypath and crtpath when using TLS, but TLS is not required', function () {
const args = {
env: 'test',
bind: '1',
keyPath: '/tmp/key.pem',
crtPath: undefined,
};
(() => (0, expressApp_1.app)(args)).should.throw(errors_1.TlsConfigurationError);
delete args.keyPath;
args.crtPath = '/tmp/cert.pem';
(() => (0, expressApp_1.app)(args)).should.throw(errors_1.TlsConfigurationError);
});
it('should create an http server when not using TLS', async function () {
const createServerStub = sinon.stub(http, 'createServer');
const args = {
env: 'prod',
bind: 'localhost',
};
(0, expressApp_1.createServer)(args, null);
createServerStub.should.be.calledOnce();
createServerStub.restore();
});
it('should create an https server when using TLS', async function () {
const createServerStub = sinon.stub(https, 'createServer');
const readFileAsyncStub = sinon
.stub(fs.promises, 'readFile')
.onFirstCall()
.resolves('key')
.onSecondCall()
.resolves('cert');
const args = {
env: 'prod',
bind: '1.2.3.4',
crtPath: '/tmp/crt.pem',
keyPath: '/tmp/key.pem',
};
await (0, expressApp_1.createServer)(args, null);
https.createServer.should.be.calledOnce();
https.createServer.should.be.calledWith({ secureOptions: constants_1.SSL_OP_NO_TLSv1, key: 'key', cert: 'cert' });
createServerStub.restore();
readFileAsyncStub.restore();
});
it('should create https server with sslkey and sslcert', async () => {
const createServerStub = sinon.stub(https, 'createServer');
const args = {
env: 'test',
bind: '1',
sslKey: 'ssl-key',
sslCert: 'ssl-cert',
};
try {
await (0, expressApp_1.createServer)(args, null);
https.createServer.should.be.calledOnce();
https.createServer.should.be.calledWith({ secureOptions: constants_1.SSL_OP_NO_TLSv1, key: 'ssl-key', cert: 'ssl-cert' });
}
finally {
createServerStub.restore();
}
});
it('should output basic information upon server startup', () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.callCount(3);
logStub.should.have.been.calledWith('BitGo-Express running');
logStub.should.have.been.calledWith(`Environment: ${args.env}`);
logStub.should.have.been.calledWith('Base URI: base');
logStub.restore();
});
it('should output custom root uri information upon server startup', () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
customRootUri: 'customuri',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.callCount(4);
logStub.should.have.been.calledWith('BitGo-Express running');
logStub.should.have.been.calledWith(`Environment: ${args.env}`);
logStub.should.have.been.calledWith('Base URI: base');
logStub.should.have.been.calledWith(`Custom root URI: ${args.customRootUri}`);
logStub.restore();
});
it('should output custom bitcoin network information upon server startup', () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
customBitcoinNetwork: 'customnetwork',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.callCount(4);
logStub.should.have.been.calledWith('BitGo-Express running');
logStub.should.have.been.calledWith(`Environment: ${args.env}`);
logStub.should.have.been.calledWith('Base URI: base');
logStub.should.have.been.calledWith(`Custom bitcoin network: ${args.customBitcoinNetwork}`);
logStub.restore();
});
it('should output signer mode upon server startup', () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
signerMode: 'signerMode',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.callCount(4);
logStub.should.have.been.calledWith('BitGo-Express running');
logStub.should.have.been.calledWith(`Environment: ${args.env}`);
logStub.should.have.been.calledWith('Base URI: base');
logStub.should.have.been.calledWith(`External signer mode: ${args.signerMode}`);
logStub.restore();
});
it('should output lightning signer file system path upon server startup', () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.callCount(4);
logStub.should.have.been.calledWith('BitGo-Express running');
logStub.should.have.been.calledWith(`Environment: ${args.env}`);
logStub.should.have.been.calledWith('Base URI: base');
logStub.should.have.been.calledWith(`Lightning signer file system path: ${args.lightningSignerFileSystemPath}`);
logStub.restore();
});
it('should create http base URIs', () => {
const args = {
bind: '1',
port: 2,
};
(0, expressApp_1.createBaseUri)(args).should.equal(`http://${args.bind}:${args.port}`);
args.port = 80;
(0, expressApp_1.createBaseUri)(args).should.equal(`http://${args.bind}`);
args.port = 443;
(0, expressApp_1.createBaseUri)(args).should.equal(`http://${args.bind}:443`);
});
it('should create https base URIs', () => {
const args = {
bind: '6',
port: 8,
keyPath: '3',
crtPath: '4',
};
(0, expressApp_1.createBaseUri)(args).should.equal(`https://${args.bind}:${args.port}`);
args.port = 80;
(0, expressApp_1.createBaseUri)(args).should.equal(`https://${args.bind}:80`);
args.port = 443;
(0, expressApp_1.createBaseUri)(args).should.equal(`https://${args.bind}`);
});
it('should set up logging with a logfile', () => {
const resolveSpy = sinon.spy(path, 'resolve');
const createWriteStreamSpy = sinon.spy(fs, 'createWriteStream');
const logStub = sinon.stub(console, 'log');
const args = {
logFile: '/dev/null',
disableProxy: true,
};
(0, expressApp_1.app)(args);
path.resolve.should.have.been.calledWith(args.logFile);
fs.createWriteStream.should.have.been.calledOnceWith(args.logFile, { flags: 'a' });
logStub.should.have.been.calledOnceWith(`Log location: ${args.logFile}`);
resolveSpy.restore();
createWriteStreamSpy.restore();
logStub.restore();
});
it('should enable specified debug namespaces', () => {
const enableStub = sinon.stub(debugLib, 'enable');
const args = {
debugNamespace: ['a', 'b'],
disableProxy: true,
};
(0, expressApp_1.app)(args);
enableStub.should.have.been.calledTwice();
enableStub.should.have.been.calledWith(args.debugNamespace[0]);
enableStub.should.have.been.calledWith(args.debugNamespace[1]);
enableStub.restore();
});
it('should configure a custom root URI', () => {
const args = {
customRootUri: 'customroot',
env: undefined,
};
(0, expressApp_1.app)(args);
should(args.env).equal('custom');
bitgo_1.Environments.custom.uri.should.equal(args.customRootUri);
});
it('should configure a custom bitcoin network', () => {
const args = {
customBitcoinNetwork: 'custombitcoinnetwork',
env: undefined,
};
(0, expressApp_1.app)(args);
should(args.env).equal('custom');
bitgo_1.Environments.custom.network.should.equal(args.customBitcoinNetwork);
});
it('should fail if IPC option is used on windows', async () => {
const platformStub = sinon.stub(process, 'platform').value('win32');
await (0, expressApp_1.prepareIpc)('testipc').should.be.rejectedWith(/^IPC option is not supported on platform/);
platformStub.restore();
});
it("should not remove the IPC socket if it doesn't exist", async () => {
const statStub = sinon.stub(fs, 'statSync').throws({ code: 'ENOENT' });
const unlinkStub = sinon.stub(fs, 'unlinkSync');
await (0, expressApp_1.prepareIpc)('testipc').should.be.resolved();
unlinkStub.notCalled.should.be.true();
statStub.restore();
unlinkStub.restore();
});
it('should remove the socket before binding if IPC socket exists and is a socket', async () => {
const statStub = sinon.stub(fs, 'statSync').returns({ isSocket: () => true });
const unlinkStub = sinon.stub(fs, 'unlinkSync');
await (0, expressApp_1.prepareIpc)('testipc').should.be.resolved();
unlinkStub.calledWithExactly('testipc').should.be.true();
unlinkStub.calledOnce.should.be.true();
statStub.restore();
unlinkStub.restore();
});
it('should fail if IPC socket is not actually a socket', async () => {
const statStub = sinon.stub(fs, 'statSync').returns({ isSocket: () => false });
const unlinkStub = sinon.stub(fs, 'unlinkSync');
await (0, expressApp_1.prepareIpc)('testipc').should.be.rejectedWith(/IPC socket is not actually a socket/);
unlinkStub.notCalled.should.be.true();
statStub.restore();
unlinkStub.restore();
});
it('should print the IPC socket path on startup', async () => {
const logStub = sinon.stub(console, 'log');
const args = {
env: 'test',
customRootUri: 'customuri',
ipc: 'expressIPC',
};
(0, expressApp_1.startup)(args, 'base')();
logStub.should.have.been.calledWith('IPC path: expressIPC');
logStub.restore();
});
it('should only call setupAPIRoutes when running in regular mode', () => {
const args = {
env: 'test',
signerMode: undefined,
};
const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');
const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');
(0, expressApp_1.app)(args);
apiStub.should.have.been.calledOnce();
signerStub.called.should.be.false();
apiStub.restore();
signerStub.restore();
});
it('should only call setupLightningSignerNodeRoutes when running with lightningSignerFileSystemPath', () => {
const args = {
env: 'test',
lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',
};
const readValidStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);
const lightningSignerStub = sinon.stub(clientRoutes, 'setupLightningSignerNodeRoutes');
const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');
const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');
(0, expressApp_1.app)(args);
lightningSignerStub.should.have.been.calledOnce();
apiStub.should.have.been.calledOnce();
signerStub.called.should.be.false();
apiStub.restore();
signerStub.restore();
readValidStub.restore();
});
it('should only call setupSigningRoutes when running in signer mode', () => {
const args = {
env: 'test',
signerMode: 'signerMode',
signerFileSystemPath: 'signerFileSystemPath',
};
const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');
const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');
const readFileStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);
(0, expressApp_1.app)(args);
signerStub.should.have.been.calledOnce();
apiStub.called.should.be.false();
apiStub.restore();
signerStub.restore();
readFileStub.restore();
});
it('should require a signerFileSystemPath and signerMode are both set when running in signer mode', function () {
const args = {
env: 'test',
signerMode: 'signerMode',
signerFileSystemPath: undefined,
};
(() => (0, expressApp_1.app)(args)).should.throw({
name: 'ExternalSignerConfigError',
message: 'signerMode and signerFileSystemPath must both be set in order to run in external signing mode.',
});
args.signerMode = undefined;
args.signerFileSystemPath = 'signerFileSystemPath';
(() => (0, expressApp_1.app)(args)).should.throw({
name: 'ExternalSignerConfigError',
message: 'signerMode and signerFileSystemPath must both be set in order to run in external signing mode.',
});
const readFileStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);
args.signerMode = 'signerMode';
(() => (0, expressApp_1.app)(args)).should.not.throw();
readFileStub.restore();
});
it('should require that an externalSignerUrl and signerMode are not both set', function () {
const args = {
env: 'test',
signerMode: 'signerMode',
externalSignerUrl: 'externalSignerUrl',
};
(() => (0, expressApp_1.app)(args)).should.throw({
name: 'ExternalSignerConfigError',
message: 'signerMode or signerFileSystemPath is set, but externalSignerUrl is also set.',
});
args.signerMode = undefined;
(() => (0, expressApp_1.app)(args)).should.not.throw();
});
it('should require that signerMode and lightningSignerFileSystemPath not coexist', function () {
const args = {
env: 'test',
signerMode: 'signerMode',
signerFileSystemPath: 'signerFileSystemPath',
lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',
};
(() => (0, expressApp_1.app)(args)).should.throw({
name: 'LightningSignerConfigError',
message: 'signerMode and lightningSignerFileSystemPath cannot be set at the same time.',
});
const readFileStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);
args.signerMode = undefined;
args.signerFileSystemPath = undefined;
(() => (0, expressApp_1.app)(args)).should.not.throw();
readFileStub.restore();
});
it('should require that an signerFileSystemPath contains a parsable json', function () {
const args = {
env: 'test',
signerMode: 'signerMode',
signerFileSystemPath: 'invalidSignerFileSystemPath',
};
(() => (0, expressApp_1.app)(args)).should.throw();
const invalidPrv = '{"invalid json"}';
const readInvalidStub = sinon.stub(fs, 'readFileSync').returns(invalidPrv);
(() => (0, expressApp_1.app)(args)).should.throw();
readInvalidStub.restore();
const readValidStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);
(() => (0, expressApp_1.app)(args)).should.not.throw();
readValidStub.restore();
});
it('should require that an lightningSignerFileSystemPath contains a parsable json', function () {
const args = {
env: 'test',
lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',
};
(() => (0, expressApp_1.app)(args)).should.throw();
const invalidPrv = '{"invalid json"}';
const readInvalidStub = sinon.stub(fs, 'readFileSync').returns(invalidPrv);
(() => (0, expressApp_1.app)(args)).should.throw();
readInvalidStub.restore();
const readValidStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);
(() => (0, expressApp_1.app)(args)).should.not.throw();
readValidStub.restore();
});
it('should set keepAliveTimeout and headersTimeout if specified in config for HTTP server', async function () {
const createServerStub = sinon.stub(http, 'createServer').callsFake(() => {
return { listen: sinon.stub(), setTimeout: sinon.stub() };
});
const args = {
env: 'test',
bind: 'localhost',
keepAliveTimeout: 5000,
headersTimeout: 10000,
};
const server = await (0, expressApp_1.createServer)(args, null);
server.keepAliveTimeout.should.equal(args.keepAliveTimeout);
server.headersTimeout.should.equal(args.headersTimeout);
createServerStub.restore();
});
it('should set keepAliveTimeout and headersTimeout if specified in config for HTTPS server', async function () {
const createServerStub = sinon.stub(https, 'createServer').callsFake(() => {
return { listen: sinon.stub(), setTimeout: sinon.stub() };
});
const args = {
env: 'test',
bind: 'localhost',
sslKey: 'ssl-key',
sslCert: 'ssl-cert',
keepAliveTimeout: 5000,
headersTimeout: 10000,
};
const server = await (0, expressApp_1.createServer)(args, null);
server.keepAliveTimeout.should.equal(args.keepAliveTimeout);
server.headersTimeout.should.equal(args.headersTimeout);
createServerStub.restore();
});
it('should not set keepAliveTimeout and headersTimeout if not specified in config for HTTP server', async function () {
const createServerStub = sinon.stub(http, 'createServer').callsFake(() => {
return { listen: sinon.stub(), setTimeout: sinon.stub() };
});
const args = {
env: 'test',
bind: 'localhost',
// keepAliveTimeout and headersTimeout are not specified
};
const server = await (0, expressApp_1.createServer)(args, null);
should(server.keepAliveTimeout).be.undefined();
should(server.headersTimeout).be.undefined();
createServerStub.restore();
});
it('should not set keepAliveTimeout and headersTimeout if not specified in config for HTTPS server', async function () {
const createServerStub = sinon.stub(https, 'createServer').callsFake(() => {
return { listen: sinon.stub(), setTimeout: sinon.stub() };
});
const args = {
env: 'test',
bind: 'localhost',
sslKey: 'ssl-key',
sslCert: 'ssl-cert',
// keepAliveTimeout and headersTimeout are not specified
};
const server = await (0, expressApp_1.createServer)(args, null);
should(server.keepAliveTimeout).be.undefined();
should(server.headersTimeout).be.undefined();
createServerStub.restore();
});
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bitgoExpress.js","sourceRoot":"","sources":["../../../test/unit/bitgoExpress.ts"],"names":[],"mappings":"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","sourcesContent":["import * as should from 'should';\nimport 'should-http';\nimport 'should-sinon';\nimport '../lib/asserts';\n\nimport * as nock from 'nock';\nimport * as sinon from 'sinon';\n\nimport * as fs from 'fs';\nimport * as http from 'http';\nimport * as https from 'https';\nimport * as debugLib from 'debug';\nimport * as path from 'path';\nimport { Environments } from 'bitgo';\n\nimport { SSL_OP_NO_TLSv1 } from 'constants';\nimport { TlsConfigurationError, NodeEnvironmentError } from '../../src/errors';\n\nnock.disableNetConnect();\n\nimport { app as expressApp, startup, createServer, createBaseUri, prepareIpc } from '../../src/expressApp';\nimport * as clientRoutes from '../../src/clientRoutes';\n\ndescribe('Bitgo Express', function () {\n  describe('server initialization', function () {\n    const validPrvJSON =\n      '{\"61f039aad587c2000745c687373e0fa9\":\"xprv9s21ZrQH143K3EuPWCBuqnWxydaQV6et9htQige4EswvcHKEzNmkVmwTwKoadyHzJYppuADB7Us7AbaNLToNvoFoSxuWqndQRYtnNy5DUY2\"}';\n    const validLightningSignerConfigJSON = '{\"fakeid\":{\"url\": \"https://127.0.0.1:8080\",\"tlsCert\":\"dummy\"}}';\n\n    it('should require NODE_ENV to be production when running against prod env', function () {\n      const envStub = sinon.stub(process, 'env').value({ NODE_ENV: 'production' });\n\n      try {\n        (() =>\n          expressApp({\n            env: 'prod',\n            bind: 'localhost',\n          } as any)).should.not.throw();\n\n        process.env.NODE_ENV = 'dev';\n        (() =>\n          expressApp({\n            env: 'prod',\n            bind: 'localhost',\n          } as any)).should.throw(NodeEnvironmentError);\n      } finally {\n        envStub.restore();\n      }\n    });\n\n    it('should disable NODE_ENV check if disableenvcheck argument is given', function () {\n      const envStub = sinon.stub(process, 'env').value({ NODE_ENV: 'dev' });\n\n      try {\n        (() =>\n          expressApp({\n            env: 'prod',\n            bind: 'localhost',\n            disableEnvCheck: true,\n          } as any)).should.not.throw();\n      } finally {\n        envStub.restore();\n      }\n    });\n\n    it('should require TLS for prod env when listening on external interfaces', function () {\n      const args: any = {\n        env: 'prod',\n        bind: '1',\n        disableEnvCheck: true,\n        disableSSL: false,\n        crtPath: undefined as string | undefined,\n        keyPath: undefined as string | undefined,\n      };\n\n      (() => expressApp(args)).should.throw(TlsConfigurationError);\n\n      args.bind = 'localhost';\n      (() => expressApp(args)).should.not.throw();\n\n      args.bind = '1';\n      args.env = 'test';\n      (() => expressApp(args)).should.not.throw();\n\n      args.disableSSL = true;\n      args.env = 'prod';\n      (() => expressApp(args)).should.not.throw();\n\n      delete args.disableSSL;\n      args.crtPath = '/tmp/cert.pem';\n      (() => expressApp(args)).should.throw(TlsConfigurationError);\n\n      delete args.crtPath;\n      args.keyPath = '/tmp/key.pem';\n      (() => expressApp(args)).should.throw(TlsConfigurationError);\n    });\n\n    it('should require both keypath and crtpath when using TLS, but TLS is not required', function () {\n      const args: any = {\n        env: 'test',\n        bind: '1',\n        keyPath: '/tmp/key.pem',\n        crtPath: undefined as string | undefined,\n      };\n\n      (() => expressApp(args)).should.throw(TlsConfigurationError);\n\n      delete args.keyPath;\n      args.crtPath = '/tmp/cert.pem';\n\n      (() => expressApp(args)).should.throw(TlsConfigurationError);\n    });\n\n    it('should create an http server when not using TLS', async function () {\n      const createServerStub = sinon.stub(http, 'createServer');\n\n      const args: any = {\n        env: 'prod',\n        bind: 'localhost',\n      };\n\n      createServer(args, null as any);\n\n      createServerStub.should.be.calledOnce();\n      createServerStub.restore();\n    });\n\n    it('should create an https server when using TLS', async function () {\n      const createServerStub = sinon.stub(https, 'createServer');\n      const readFileAsyncStub = sinon\n        .stub(fs.promises, 'readFile' as any)\n        .onFirstCall()\n        .resolves('key')\n        .onSecondCall()\n        .resolves('cert');\n\n      const args: any = {\n        env: 'prod',\n        bind: '1.2.3.4',\n        crtPath: '/tmp/crt.pem',\n        keyPath: '/tmp/key.pem',\n      };\n\n      await createServer(args, null as any);\n\n      https.createServer.should.be.calledOnce();\n      https.createServer.should.be.calledWith({ secureOptions: SSL_OP_NO_TLSv1, key: 'key', cert: 'cert' });\n\n      createServerStub.restore();\n      readFileAsyncStub.restore();\n    });\n\n    it('should create https server with sslkey and sslcert', async () => {\n      const createServerStub = sinon.stub(https, 'createServer');\n      const args: any = {\n        env: 'test',\n        bind: '1',\n        sslKey: 'ssl-key',\n        sslCert: 'ssl-cert',\n      };\n\n      try {\n        await createServer(args, null as any);\n        https.createServer.should.be.calledOnce();\n        https.createServer.should.be.calledWith({ secureOptions: SSL_OP_NO_TLSv1, key: 'ssl-key', cert: 'ssl-cert' });\n      } finally {\n        createServerStub.restore();\n      }\n    });\n\n    it('should output basic information upon server startup', () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n      };\n\n      startup(args, 'base')();\n\n      logStub.should.have.callCount(3);\n      logStub.should.have.been.calledWith('BitGo-Express running');\n      logStub.should.have.been.calledWith(`Environment: ${args.env}`);\n      logStub.should.have.been.calledWith('Base URI: base');\n\n      logStub.restore();\n    });\n\n    it('should output custom root uri information upon server startup', () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n        customRootUri: 'customuri',\n      };\n\n      startup(args, 'base')();\n\n      logStub.should.have.callCount(4);\n      logStub.should.have.been.calledWith('BitGo-Express running');\n      logStub.should.have.been.calledWith(`Environment: ${args.env}`);\n      logStub.should.have.been.calledWith('Base URI: base');\n      logStub.should.have.been.calledWith(`Custom root URI: ${args.customRootUri}`);\n\n      logStub.restore();\n    });\n\n    it('should output custom bitcoin network information upon server startup', () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n        customBitcoinNetwork: 'customnetwork',\n      };\n\n      startup(args, 'base')();\n\n      logStub.should.have.callCount(4);\n      logStub.should.have.been.calledWith('BitGo-Express running');\n      logStub.should.have.been.calledWith(`Environment: ${args.env}`);\n      logStub.should.have.been.calledWith('Base URI: base');\n      logStub.should.have.been.calledWith(`Custom bitcoin network: ${args.customBitcoinNetwork}`);\n\n      logStub.restore();\n    });\n\n    it('should output signer mode upon server startup', () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n      };\n\n      startup(args, 'base')();\n\n      logStub.should.have.callCount(4);\n      logStub.should.have.been.calledWith('BitGo-Express running');\n      logStub.should.have.been.calledWith(`Environment: ${args.env}`);\n      logStub.should.have.been.calledWith('Base URI: base');\n      logStub.should.have.been.calledWith(`External signer mode: ${args.signerMode}`);\n\n      logStub.restore();\n    });\n\n    it('should output lightning signer file system path upon server startup', () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n        lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',\n      };\n\n      startup(args, 'base')();\n\n      logStub.should.have.callCount(4);\n      logStub.should.have.been.calledWith('BitGo-Express running');\n      logStub.should.have.been.calledWith(`Environment: ${args.env}`);\n      logStub.should.have.been.calledWith('Base URI: base');\n      logStub.should.have.been.calledWith(`Lightning signer file system path: ${args.lightningSignerFileSystemPath}`);\n\n      logStub.restore();\n    });\n\n    it('should create http base URIs', () => {\n      const args: any = {\n        bind: '1',\n        port: 2,\n      };\n\n      createBaseUri(args).should.equal(`http://${args.bind}:${args.port}`);\n\n      args.port = 80;\n      createBaseUri(args).should.equal(`http://${args.bind}`);\n\n      args.port = 443;\n      createBaseUri(args).should.equal(`http://${args.bind}:443`);\n    });\n\n    it('should create https base URIs', () => {\n      const args: any = {\n        bind: '6',\n        port: 8,\n        keyPath: '3',\n        crtPath: '4',\n      };\n\n      createBaseUri(args).should.equal(`https://${args.bind}:${args.port}`);\n\n      args.port = 80;\n      createBaseUri(args).should.equal(`https://${args.bind}:80`);\n\n      args.port = 443;\n      createBaseUri(args).should.equal(`https://${args.bind}`);\n    });\n\n    it('should set up logging with a logfile', () => {\n      const resolveSpy = sinon.spy(path, 'resolve');\n      const createWriteStreamSpy = sinon.spy(fs, 'createWriteStream');\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        logFile: '/dev/null',\n        disableProxy: true,\n      };\n\n      expressApp(args);\n\n      path.resolve.should.have.been.calledWith(args.logFile);\n      fs.createWriteStream.should.have.been.calledOnceWith(args.logFile, { flags: 'a' });\n      logStub.should.have.been.calledOnceWith(`Log location: ${args.logFile}`);\n\n      resolveSpy.restore();\n      createWriteStreamSpy.restore();\n      logStub.restore();\n    });\n\n    it('should enable specified debug namespaces', () => {\n      const enableStub = sinon.stub(debugLib, 'enable');\n\n      const args: any = {\n        debugNamespace: ['a', 'b'],\n        disableProxy: true,\n      };\n\n      expressApp(args);\n\n      enableStub.should.have.been.calledTwice();\n      enableStub.should.have.been.calledWith(args.debugNamespace[0]);\n      enableStub.should.have.been.calledWith(args.debugNamespace[1]);\n\n      enableStub.restore();\n    });\n\n    it('should configure a custom root URI', () => {\n      const args: any = {\n        customRootUri: 'customroot',\n        env: undefined as string | undefined,\n      };\n\n      expressApp(args);\n\n      should(args.env).equal('custom');\n      Environments.custom.uri.should.equal(args.customRootUri);\n    });\n\n    it('should configure a custom bitcoin network', () => {\n      const args: any = {\n        customBitcoinNetwork: 'custombitcoinnetwork',\n        env: undefined as string | undefined,\n      };\n\n      expressApp(args);\n\n      should(args.env).equal('custom');\n      Environments.custom.network.should.equal(args.customBitcoinNetwork);\n    });\n\n    it('should fail if IPC option is used on windows', async () => {\n      const platformStub = sinon.stub(process, 'platform').value('win32');\n      await prepareIpc('testipc').should.be.rejectedWith(/^IPC option is not supported on platform/);\n      platformStub.restore();\n    });\n\n    it(\"should not remove the IPC socket if it doesn't exist\", async () => {\n      const statStub = sinon.stub(fs, 'statSync').throws({ code: 'ENOENT' });\n      const unlinkStub = sinon.stub(fs, 'unlinkSync');\n      await prepareIpc('testipc').should.be.resolved();\n      unlinkStub.notCalled.should.be.true();\n      statStub.restore();\n      unlinkStub.restore();\n    });\n\n    it('should remove the socket before binding if IPC socket exists and is a socket', async () => {\n      const statStub = sinon.stub(fs, 'statSync').returns({ isSocket: () => true } as unknown as fs.Stats);\n      const unlinkStub = sinon.stub(fs, 'unlinkSync');\n      await prepareIpc('testipc').should.be.resolved();\n      unlinkStub.calledWithExactly('testipc').should.be.true();\n      unlinkStub.calledOnce.should.be.true();\n      statStub.restore();\n      unlinkStub.restore();\n    });\n\n    it('should fail if IPC socket is not actually a socket', async () => {\n      const statStub = sinon.stub(fs, 'statSync').returns({ isSocket: () => false } as unknown as fs.Stats);\n      const unlinkStub = sinon.stub(fs, 'unlinkSync');\n      await prepareIpc('testipc').should.be.rejectedWith(/IPC socket is not actually a socket/);\n      unlinkStub.notCalled.should.be.true();\n      statStub.restore();\n      unlinkStub.restore();\n    });\n\n    it('should print the IPC socket path on startup', async () => {\n      const logStub = sinon.stub(console, 'log');\n\n      const args: any = {\n        env: 'test',\n        customRootUri: 'customuri',\n        ipc: 'expressIPC',\n      };\n\n      startup(args, 'base')();\n      logStub.should.have.been.calledWith('IPC path: expressIPC');\n      logStub.restore();\n    });\n\n    it('should only call setupAPIRoutes when running in regular mode', () => {\n      const args: any = {\n        env: 'test',\n        signerMode: undefined,\n      };\n\n      const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');\n      const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');\n\n      expressApp(args);\n      apiStub.should.have.been.calledOnce();\n      signerStub.called.should.be.false();\n      apiStub.restore();\n      signerStub.restore();\n    });\n\n    it('should only call setupLightningSignerNodeRoutes when running with lightningSignerFileSystemPath', () => {\n      const args: any = {\n        env: 'test',\n        lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',\n      };\n\n      const readValidStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);\n      const lightningSignerStub = sinon.stub(clientRoutes, 'setupLightningSignerNodeRoutes');\n      const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');\n      const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');\n\n      expressApp(args);\n      lightningSignerStub.should.have.been.calledOnce();\n      apiStub.should.have.been.calledOnce();\n      signerStub.called.should.be.false();\n      apiStub.restore();\n      signerStub.restore();\n      readValidStub.restore();\n    });\n\n    it('should only call setupSigningRoutes when running in signer mode', () => {\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n        signerFileSystemPath: 'signerFileSystemPath',\n      };\n\n      const apiStub = sinon.stub(clientRoutes, 'setupAPIRoutes');\n      const signerStub = sinon.stub(clientRoutes, 'setupSigningRoutes');\n      const readFileStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);\n\n      expressApp(args);\n      signerStub.should.have.been.calledOnce();\n      apiStub.called.should.be.false();\n      apiStub.restore();\n      signerStub.restore();\n      readFileStub.restore();\n    });\n\n    it('should require a signerFileSystemPath and signerMode are both set when running in signer mode', function () {\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n        signerFileSystemPath: undefined,\n      };\n\n      (() => expressApp(args)).should.throw({\n        name: 'ExternalSignerConfigError',\n        message: 'signerMode and signerFileSystemPath must both be set in order to run in external signing mode.',\n      });\n\n      args.signerMode = undefined;\n      args.signerFileSystemPath = 'signerFileSystemPath';\n      (() => expressApp(args)).should.throw({\n        name: 'ExternalSignerConfigError',\n        message: 'signerMode and signerFileSystemPath must both be set in order to run in external signing mode.',\n      });\n\n      const readFileStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);\n      args.signerMode = 'signerMode';\n      (() => expressApp(args)).should.not.throw();\n\n      readFileStub.restore();\n    });\n\n    it('should require that an externalSignerUrl and signerMode are not both set', function () {\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n        externalSignerUrl: 'externalSignerUrl',\n      };\n      (() => expressApp(args)).should.throw({\n        name: 'ExternalSignerConfigError',\n        message: 'signerMode or signerFileSystemPath is set, but externalSignerUrl is also set.',\n      });\n\n      args.signerMode = undefined;\n      (() => expressApp(args)).should.not.throw();\n    });\n\n    it('should require that signerMode and lightningSignerFileSystemPath not coexist', function () {\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n        signerFileSystemPath: 'signerFileSystemPath',\n        lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',\n      };\n      (() => expressApp(args)).should.throw({\n        name: 'LightningSignerConfigError',\n        message: 'signerMode and lightningSignerFileSystemPath cannot be set at the same time.',\n      });\n\n      const readFileStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);\n      args.signerMode = undefined;\n      args.signerFileSystemPath = undefined;\n      (() => expressApp(args)).should.not.throw();\n      readFileStub.restore();\n    });\n\n    it('should require that an signerFileSystemPath contains a parsable json', function () {\n      const args: any = {\n        env: 'test',\n        signerMode: 'signerMode',\n        signerFileSystemPath: 'invalidSignerFileSystemPath',\n      };\n      (() => expressApp(args)).should.throw();\n\n      const invalidPrv = '{\"invalid json\"}';\n      const readInvalidStub = sinon.stub(fs, 'readFileSync').returns(invalidPrv);\n      (() => expressApp(args)).should.throw();\n      readInvalidStub.restore();\n\n      const readValidStub = sinon.stub(fs, 'readFileSync').returns(validPrvJSON);\n      (() => expressApp(args)).should.not.throw();\n      readValidStub.restore();\n    });\n\n    it('should require that an lightningSignerFileSystemPath contains a parsable json', function () {\n      const args: any = {\n        env: 'test',\n        lightningSignerFileSystemPath: 'lightningSignerFileSystemPath',\n      };\n      (() => expressApp(args)).should.throw();\n\n      const invalidPrv = '{\"invalid json\"}';\n      const readInvalidStub = sinon.stub(fs, 'readFileSync').returns(invalidPrv);\n      (() => expressApp(args)).should.throw();\n      readInvalidStub.restore();\n\n      const readValidStub = sinon.stub(fs, 'readFileSync').returns(validLightningSignerConfigJSON);\n      (() => expressApp(args)).should.not.throw();\n      readValidStub.restore();\n    });\n\n    it('should set keepAliveTimeout and headersTimeout if specified in config for HTTP server', async function () {\n      const createServerStub = sinon.stub(http, 'createServer').callsFake(() => {\n        return { listen: sinon.stub(), setTimeout: sinon.stub() } as unknown as http.Server;\n      });\n\n      const args: any = {\n        env: 'test',\n        bind: 'localhost',\n        keepAliveTimeout: 5000,\n        headersTimeout: 10000,\n      };\n\n      const server = await createServer(args, null as any);\n\n      server.keepAliveTimeout.should.equal(args.keepAliveTimeout);\n      server.headersTimeout.should.equal(args.headersTimeout);\n\n      createServerStub.restore();\n    });\n\n    it('should set keepAliveTimeout and headersTimeout if specified in config for HTTPS server', async function () {\n      const createServerStub = sinon.stub(https, 'createServer').callsFake(() => {\n        return { listen: sinon.stub(), setTimeout: sinon.stub() } as unknown as https.Server;\n      });\n\n      const args: any = {\n        env: 'test',\n        bind: 'localhost',\n        sslKey: 'ssl-key',\n        sslCert: 'ssl-cert',\n        keepAliveTimeout: 5000,\n        headersTimeout: 10000,\n      };\n\n      const server = await createServer(args, null as any);\n\n      server.keepAliveTimeout.should.equal(args.keepAliveTimeout);\n      server.headersTimeout.should.equal(args.headersTimeout);\n\n      createServerStub.restore();\n    });\n\n    it('should not set keepAliveTimeout and headersTimeout if not specified in config for HTTP server', async function () {\n      const createServerStub = sinon.stub(http, 'createServer').callsFake(() => {\n        return { listen: sinon.stub(), setTimeout: sinon.stub() } as unknown as http.Server;\n      });\n\n      const args: any = {\n        env: 'test',\n        bind: 'localhost',\n        // keepAliveTimeout and headersTimeout are not specified\n      };\n\n      const server = await createServer(args, null as any);\n\n      should(server.keepAliveTimeout).be.undefined();\n      should(server.headersTimeout).be.undefined();\n\n      createServerStub.restore();\n    });\n\n    it('should not set keepAliveTimeout and headersTimeout if not specified in config for HTTPS server', async function () {\n      const createServerStub = sinon.stub(https, 'createServer').callsFake(() => {\n        return { listen: sinon.stub(), setTimeout: sinon.stub() } as unknown as https.Server;\n      });\n\n      const args: any = {\n        env: 'test',\n        bind: 'localhost',\n        sslKey: 'ssl-key',\n        sslCert: 'ssl-cert',\n        // keepAliveTimeout and headersTimeout are not specified\n      };\n\n      const server = await createServer(args, null as any);\n\n      should(server.keepAliveTimeout).be.undefined();\n      should(server.headersTimeout).be.undefined();\n\n      createServerStub.restore();\n    });\n  });\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!