2013-10-09 10:47:55 -05:00
|
|
|
#!/usr/bin/env node
|
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
/* eslint no-sync:0*/
|
|
|
|
'use strict';
|
2014-02-23 12:20:50 -05:00
|
|
|
|
2014-01-23 22:36:03 -05:00
|
|
|
if (process.getuid && process.getuid() === 0) {
|
2017-04-23 13:02:26 -05:00
|
|
|
global.console.error('Verdaccio doesn\'t need superuser privileges. Don\'t run it under root.');
|
2013-12-17 11:25:17 -05:00
|
|
|
}
|
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
process.title = 'verdaccio';
|
2014-09-24 20:13:03 -05:00
|
|
|
|
2013-12-28 19:52:23 -05:00
|
|
|
try {
|
2014-11-12 06:14:37 -05:00
|
|
|
// for debugging memory leaks
|
|
|
|
// totally optional
|
2017-04-23 13:02:26 -05:00
|
|
|
require('heapdump');
|
2014-11-12 06:14:37 -05:00
|
|
|
} catch(err) {}
|
2013-12-28 19:52:23 -05:00
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
let logger = require('./logger');
|
|
|
|
logger.setup(); // default setup
|
|
|
|
|
|
|
|
let commander = require('commander');
|
|
|
|
let constants = require('constants');
|
|
|
|
let fs = require('fs');
|
|
|
|
let http = require('http');
|
|
|
|
let https = require('https');
|
|
|
|
let YAML = require('js-yaml');
|
|
|
|
let Path = require('path');
|
|
|
|
let URL = require('url');
|
|
|
|
let server = require('./index');
|
|
|
|
let Utils = require('./utils');
|
|
|
|
let pkginfo = require('pkginfo')(module); // eslint-disable-line no-unused-vars
|
|
|
|
let pkgVersion = module.exports.version;
|
|
|
|
let pkgName = module.exports.name;
|
2013-10-09 10:47:55 -05:00
|
|
|
|
|
|
|
commander
|
2014-11-12 06:14:37 -05:00
|
|
|
.option('-l, --listen <[host:]port>', 'host:port number to listen on (default: localhost:4873)')
|
|
|
|
.option('-c, --config <config.yaml>', 'use this configuration file (default: ./config.yaml)')
|
2016-04-22 07:36:29 -05:00
|
|
|
.version(pkgVersion)
|
2017-04-23 13:02:26 -05:00
|
|
|
.parse(process.argv);
|
2013-10-09 10:47:55 -05:00
|
|
|
|
|
|
|
if (commander.args.length == 1 && !commander.config) {
|
2016-11-07 12:15:38 -05:00
|
|
|
// handling "verdaccio [config]" case if "-c" is missing in commandline
|
2017-04-23 13:02:26 -05:00
|
|
|
commander.config = commander.args.pop();
|
2013-10-09 10:47:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (commander.args.length != 0) {
|
2017-04-23 13:02:26 -05:00
|
|
|
commander.help();
|
2013-10-09 10:47:55 -05:00
|
|
|
}
|
|
|
|
|
2017-05-04 14:05:47 -05:00
|
|
|
let config;
|
|
|
|
let config_path;
|
2013-10-09 10:47:55 -05:00
|
|
|
try {
|
2014-11-12 06:14:37 -05:00
|
|
|
if (commander.config) {
|
2017-04-23 13:02:26 -05:00
|
|
|
config_path = Path.resolve(commander.config);
|
2014-11-12 06:14:37 -05:00
|
|
|
} else {
|
2017-04-23 13:02:26 -05:00
|
|
|
config_path = require('./config-path')();
|
2014-11-12 06:14:37 -05:00
|
|
|
}
|
2017-04-23 13:02:26 -05:00
|
|
|
config = YAML.safeLoad(fs.readFileSync(config_path, 'utf8'));
|
|
|
|
logger.logger.warn({file: config_path}, 'config file - @{file}');
|
2014-11-12 06:14:37 -05:00
|
|
|
} catch (err) {
|
2017-04-23 13:02:26 -05:00
|
|
|
logger.logger.fatal({file: config_path, err: err}, 'cannot open config file @{file}: @{!err.message}');
|
|
|
|
process.exit(1);
|
2013-10-09 10:47:55 -05:00
|
|
|
}
|
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
afterConfigLoad();
|
2013-10-09 11:22:29 -05:00
|
|
|
|
2017-05-04 14:05:47 -05:00
|
|
|
/**
|
|
|
|
* Retrieve all addresses defined in the config file.
|
|
|
|
* Verdaccio is able to listen multiple ports
|
|
|
|
* eg:
|
|
|
|
* listen:
|
|
|
|
- localhost:5555
|
|
|
|
- localhost:5557
|
|
|
|
*/
|
2015-03-28 09:20:58 -05:00
|
|
|
function get_listen_addresses() {
|
2014-11-12 06:14:37 -05:00
|
|
|
// command line || config file || default
|
2017-04-23 13:02:26 -05:00
|
|
|
let addresses;
|
2015-03-28 09:20:58 -05:00
|
|
|
if (commander.listen) {
|
2017-04-23 13:02:26 -05:00
|
|
|
addresses = [commander.listen];
|
2015-03-28 09:20:58 -05:00
|
|
|
} else if (Array.isArray(config.listen)) {
|
2017-04-23 13:02:26 -05:00
|
|
|
addresses = config.listen;
|
2015-03-28 09:20:58 -05:00
|
|
|
} else if (config.listen) {
|
2017-04-23 13:02:26 -05:00
|
|
|
addresses = [config.listen];
|
2015-03-28 09:20:58 -05:00
|
|
|
} else {
|
2017-04-23 13:02:26 -05:00
|
|
|
addresses = ['4873'];
|
2014-11-12 06:14:37 -05:00
|
|
|
}
|
2015-03-28 09:20:58 -05:00
|
|
|
addresses = addresses.map(function(addr) {
|
2017-04-23 13:02:26 -05:00
|
|
|
let parsed_addr = Utils.parse_address(addr);
|
2015-03-28 10:03:36 -05:00
|
|
|
|
2015-04-21 18:36:57 -05:00
|
|
|
if (!parsed_addr) {
|
2017-04-23 13:02:26 -05:00
|
|
|
logger.logger.warn({addr: addr},
|
2015-03-28 09:20:58 -05:00
|
|
|
'invalid address - @{addr}, we expect a port (e.g. "4873"),'
|
|
|
|
+ ' host:port (e.g. "localhost:4873") or full url'
|
2017-04-23 13:02:26 -05:00
|
|
|
+ ' (e.g. "http://localhost:4873/")');
|
2015-03-28 09:20:58 -05:00
|
|
|
}
|
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
return parsed_addr;
|
|
|
|
}).filter(Boolean);
|
2015-03-28 10:03:36 -05:00
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
return addresses;
|
2013-10-09 10:47:55 -05:00
|
|
|
}
|
2013-10-09 11:22:29 -05:00
|
|
|
|
2017-05-04 14:05:47 -05:00
|
|
|
/**
|
|
|
|
* Trigger the server after configuration has been loaded.
|
|
|
|
*/
|
2013-10-09 11:22:29 -05:00
|
|
|
function afterConfigLoad() {
|
2017-05-04 14:05:47 -05:00
|
|
|
if (!config.self_path) {
|
|
|
|
config.self_path = Path.resolve(config_path);
|
|
|
|
}
|
|
|
|
if (!config.https) {
|
|
|
|
config.https = {enable: false};
|
|
|
|
}
|
|
|
|
const app = server(config);
|
2015-03-28 09:20:58 -05:00
|
|
|
get_listen_addresses().forEach(function(addr) {
|
2017-04-23 13:02:26 -05:00
|
|
|
let webServer;
|
2015-03-28 09:20:58 -05:00
|
|
|
|
|
|
|
if (addr.proto === 'https') { // https
|
|
|
|
if (!config.https || !config.https.key || !config.https.cert) {
|
2017-04-23 13:02:26 -05:00
|
|
|
let conf_path = function(file) {
|
|
|
|
if (!file) return config_path;
|
|
|
|
return Path.resolve(Path.dirname(config_path), file);
|
|
|
|
};
|
2015-03-28 09:20:58 -05:00
|
|
|
|
|
|
|
logger.logger.fatal([
|
|
|
|
'You need to specify "https.key" and "https.cert" to run https server',
|
|
|
|
'',
|
|
|
|
// commands are borrowed from node.js docs
|
|
|
|
'To quickly create self-signed certificate, use:',
|
2016-11-07 12:15:38 -05:00
|
|
|
' $ openssl genrsa -out ' + conf_path('verdaccio-key.pem') + ' 2048',
|
|
|
|
' $ openssl req -new -sha256 -key ' + conf_path('verdaccio-key.pem') + ' -out ' + conf_path('verdaccio-csr.pem'),
|
|
|
|
' $ openssl x509 -req -in ' + conf_path('verdaccio-csr.pem') + ' -signkey ' + conf_path('verdaccio-key.pem') + ' -out ' + conf_path('verdaccio-cert.pem'),
|
2015-03-28 09:20:58 -05:00
|
|
|
'',
|
|
|
|
'And then add to config file (' + conf_path() + '):',
|
|
|
|
' https:',
|
2016-11-07 12:15:38 -05:00
|
|
|
' key: verdaccio-key.pem',
|
|
|
|
' cert: verdaccio-cert.pem',
|
2017-04-23 13:02:26 -05:00
|
|
|
].join('\n'));
|
|
|
|
process.exit(2);
|
2015-03-28 09:20:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
webServer = https.createServer({
|
2014-11-10 14:41:38 -05:00
|
|
|
secureProtocol: 'SSLv23_method', // disable insecure SSLv2 and SSLv3
|
|
|
|
secureOptions: constants.SSL_OP_NO_SSLv2 | constants.SSL_OP_NO_SSLv3,
|
|
|
|
key: fs.readFileSync(config.https.key),
|
2016-03-05 14:44:32 -05:00
|
|
|
cert: fs.readFileSync(config.https.cert),
|
2017-04-23 13:02:26 -05:00
|
|
|
ca: fs.readFileSync(config.https.ca),
|
|
|
|
}, app);
|
2015-03-28 09:20:58 -05:00
|
|
|
} catch (err) { // catch errors related to certificate loading
|
2017-04-23 13:02:26 -05:00
|
|
|
logger.logger.fatal({err: err}, 'cannot create server: @{err.message}');
|
|
|
|
process.exit(2);
|
2015-03-28 09:20:58 -05:00
|
|
|
}
|
|
|
|
} else { // http
|
|
|
|
webServer = http.createServer(app);
|
2014-11-10 14:41:38 -05:00
|
|
|
}
|
|
|
|
|
2015-03-28 09:20:58 -05:00
|
|
|
webServer
|
2015-04-21 18:36:57 -05:00
|
|
|
.listen(addr.port || addr.path, addr.host)
|
2015-03-28 09:20:58 -05:00
|
|
|
.on('error', function(err) {
|
2017-04-23 13:02:26 -05:00
|
|
|
logger.logger.fatal({err: err}, 'cannot create server: @{err.message}');
|
|
|
|
process.exit(2);
|
|
|
|
});
|
2015-03-28 09:20:58 -05:00
|
|
|
|
|
|
|
logger.logger.warn({
|
2015-04-21 18:36:57 -05:00
|
|
|
addr: ( addr.path
|
|
|
|
? URL.format({
|
|
|
|
protocol: 'unix',
|
|
|
|
pathname: addr.path,
|
|
|
|
})
|
|
|
|
: URL.format({
|
|
|
|
protocol: addr.proto,
|
|
|
|
hostname: addr.host,
|
|
|
|
port: addr.port,
|
|
|
|
pathname: '/',
|
|
|
|
})
|
|
|
|
),
|
2016-04-22 07:36:29 -05:00
|
|
|
version: pkgName + '/' + pkgVersion,
|
2017-04-23 13:02:26 -05:00
|
|
|
}, 'http address - @{addr}');
|
|
|
|
});
|
2013-10-09 11:22:29 -05:00
|
|
|
|
2014-11-12 06:14:37 -05:00
|
|
|
// undocumented stuff for tests
|
|
|
|
if (typeof(process.send) === 'function') {
|
2017-04-23 13:02:26 -05:00
|
|
|
process.send({verdaccio_started: true});
|
2014-11-12 06:14:37 -05:00
|
|
|
}
|
2013-10-09 10:47:55 -05:00
|
|
|
}
|
|
|
|
|
2013-12-10 05:29:46 -05:00
|
|
|
process.on('uncaughtException', function(err) {
|
2017-05-04 14:05:47 -05:00
|
|
|
logger.logger.fatal( {err: err},
|
|
|
|
'uncaught exception, please report this\n@{err.stack}' );
|
2017-04-23 13:02:26 -05:00
|
|
|
process.exit(255);
|
|
|
|
});
|