2017-04-26 23:54:15 -05:00
|
|
|
/* eslint prefer-rest-params: "off" */
|
|
|
|
/* eslint prefer-spread: "off" */
|
|
|
|
|
2017-04-23 13:02:26 -05:00
|
|
|
'use strict';
|
2017-06-10 14:40:58 -05:00
|
|
|
|
2017-04-26 23:54:15 -05:00
|
|
|
const assert = require('assert');
|
2017-06-10 16:07:08 -05:00
|
|
|
const _ = require('lodash');
|
2017-04-26 23:54:15 -05:00
|
|
|
const Error = require('http-errors');
|
2017-06-10 16:07:08 -05:00
|
|
|
const Crypto = require('crypto');
|
2017-04-26 23:54:15 -05:00
|
|
|
const minimatch = require('minimatch');
|
2017-06-10 16:41:24 -05:00
|
|
|
|
2017-04-26 23:54:15 -05:00
|
|
|
const Utils = require('./utils');
|
|
|
|
const pkginfo = require('pkginfo')(module); // eslint-disable-line no-unused-vars
|
|
|
|
const pkgVersion = module.exports.version;
|
|
|
|
const pkgName = module.exports.name;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [[a, [b, c]], d] -> [a, b, c, d]
|
|
|
|
* @param {*} array
|
|
|
|
* @return {Array}
|
|
|
|
*/
|
2013-06-07 20:16:28 -05:00
|
|
|
function flatten(array) {
|
2017-06-10 14:40:58 -05:00
|
|
|
let result = [];
|
|
|
|
for (let i=0; i<array.length; i++) {
|
|
|
|
if (Array.isArray(array[i])) {
|
|
|
|
result.push.apply(result, flatten(array[i]));
|
|
|
|
} else {
|
|
|
|
result.push(array[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2013-06-07 20:16:28 -05:00
|
|
|
}
|
|
|
|
|
2017-04-26 23:54:15 -05:00
|
|
|
/**
|
|
|
|
* Coordinates the application configuration
|
|
|
|
*/
|
|
|
|
class Config {
|
2017-04-23 13:02:26 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
/**
|
|
|
|
* @param {*} config config the content
|
|
|
|
*/
|
|
|
|
constructor(config) {
|
|
|
|
const self = this;
|
|
|
|
for (let i in config) {
|
|
|
|
if (self[i] == null) {
|
|
|
|
self[i] = config[i];
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
2017-06-10 14:40:58 -05:00
|
|
|
}
|
2017-06-06 16:07:51 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
if (!self.user_agent) {
|
|
|
|
self.user_agent = `${pkgName}/${pkgVersion}`;
|
|
|
|
}
|
2017-06-06 16:07:51 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
// some weird shell scripts are valid yaml files parsed as string
|
|
|
|
assert.equal(typeof(config), 'object', 'CONFIG: it doesn\'t look like a valid config file');
|
|
|
|
|
|
|
|
assert(self.storage, 'CONFIG: storage path not defined');
|
2017-06-06 16:07:51 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
const users = {
|
|
|
|
'all': true,
|
|
|
|
'anonymous': true,
|
|
|
|
'undefined': true,
|
|
|
|
'owner': true,
|
|
|
|
'none': true,
|
|
|
|
};
|
|
|
|
|
|
|
|
const check_user_or_uplink = function(arg) {
|
|
|
|
assert(arg !== 'all' && arg !== 'owner'
|
|
|
|
&& arg !== 'anonymous' && arg !== 'undefined' && arg !== 'none', 'CONFIG: reserved user/uplink name: ' + arg);
|
|
|
|
assert(!arg.match(/\s/), 'CONFIG: invalid user name: ' + arg);
|
|
|
|
assert(users[arg] == null, 'CONFIG: duplicate user/uplink name: ' + arg);
|
|
|
|
users[arg] = true;
|
|
|
|
}
|
|
|
|
// sanity check for strategic config properties
|
|
|
|
;['users', 'uplinks', 'packages'].forEach(function(x) {
|
|
|
|
if (self[x] == null) self[x] = {};
|
|
|
|
assert(Utils.is_object(self[x]), `CONFIG: bad "${x}" value (object expected)`);
|
|
|
|
});
|
|
|
|
// sanity check for users
|
|
|
|
for (let i in self.users) {
|
|
|
|
if (Object.prototype.hasOwnProperty.call(self.users, i)) {
|
|
|
|
check_user_or_uplink(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// sanity check for uplinks
|
|
|
|
for (let i in self.uplinks) {
|
|
|
|
if (self.uplinks[i].cache == null) {
|
|
|
|
self.uplinks[i].cache = true;
|
|
|
|
}
|
|
|
|
if (Object.prototype.hasOwnProperty.call(self.uplinks, i)) {
|
|
|
|
check_user_or_uplink(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (let i in self.users) {
|
|
|
|
if (Object.prototype.hasOwnProperty.call(self.users, i)) {
|
|
|
|
assert(self.users[i].password, 'CONFIG: no password for user: ' + i);
|
|
|
|
assert(typeof(self.users[i].password) === 'string' &&
|
|
|
|
self.users[i].password.match(/^[a-f0-9]{40}$/)
|
|
|
|
, 'CONFIG: wrong password format for user: ' + i + ', sha1 expected');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (let i in self.uplinks) {
|
|
|
|
if (Object.prototype.hasOwnProperty.call(self.uplinks, i)) {
|
|
|
|
assert(self.uplinks[i].url, 'CONFIG: no url for uplink: ' + i);
|
|
|
|
assert( typeof(self.uplinks[i].url) === 'string'
|
|
|
|
, 'CONFIG: wrong url format for uplink: ' + i);
|
|
|
|
self.uplinks[i].url = self.uplinks[i].url.replace(/\/$/, '');
|
|
|
|
}
|
|
|
|
}
|
2017-06-06 16:07:51 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
/**
|
|
|
|
* Normalise user list.
|
|
|
|
* @return {Array}
|
|
|
|
*/
|
|
|
|
function normalize_userlist() {
|
|
|
|
let result = [];
|
|
|
|
|
|
|
|
for (let i=0; i<arguments.length; i++) {
|
|
|
|
if (arguments[i] == null) continue;
|
|
|
|
|
|
|
|
// if it's a string, split it to array
|
|
|
|
if (typeof(arguments[i]) === 'string') {
|
|
|
|
result.push(arguments[i].split(/\s+/));
|
|
|
|
} else if (Array.isArray(arguments[i])) {
|
|
|
|
result.push(arguments[i]);
|
|
|
|
} else {
|
|
|
|
throw Error('CONFIG: bad package acl (array or string expected): ' + JSON.stringify(arguments[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return flatten(result);
|
|
|
|
}
|
2017-04-26 23:54:15 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
// add a default rule for all packages to make writing plugins easier
|
|
|
|
if (self.packages['**'] == null) {
|
|
|
|
self.packages['**'] = {};
|
|
|
|
}
|
2017-04-26 23:54:15 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
for (let i in self.packages) {
|
|
|
|
if (Object.prototype.hasOwnProperty.call(self.packages, i)) {
|
|
|
|
assert(
|
|
|
|
typeof(self.packages[i]) === 'object' &&
|
|
|
|
!Array.isArray(self.packages[i])
|
|
|
|
, 'CONFIG: bad "'+i+'" package description (object expected)');
|
2017-04-26 23:54:15 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
self.packages[i].access = normalize_userlist(
|
|
|
|
self.packages[i].allow_access,
|
|
|
|
self.packages[i].access
|
|
|
|
);
|
|
|
|
delete self.packages[i].allow_access;
|
2017-04-26 23:54:15 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
self.packages[i].publish = normalize_userlist(
|
|
|
|
self.packages[i].allow_publish,
|
|
|
|
self.packages[i].publish
|
|
|
|
);
|
|
|
|
delete self.packages[i].allow_publish;
|
2017-04-26 23:54:15 -05:00
|
|
|
|
2017-06-10 14:40:58 -05:00
|
|
|
self.packages[i].proxy = normalize_userlist(
|
|
|
|
self.packages[i].proxy_access,
|
|
|
|
self.packages[i].proxy
|
|
|
|
);
|
|
|
|
delete self.packages[i].proxy_access;
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// loading these from ENV if aren't in config
|
2017-06-10 14:40:58 -05:00
|
|
|
['http_proxy', 'https_proxy', 'no_proxy'].forEach((function(v) {
|
|
|
|
if (!(v in self)) {
|
|
|
|
self[v] = process.env[v] || process.env[v.toUpperCase()];
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
|
|
|
}));
|
2017-06-10 14:40:58 -05:00
|
|
|
|
|
|
|
// unique identifier of self server (or a cluster), used to avoid loops
|
|
|
|
if (!self.server_id) {
|
|
|
|
self.server_id = Crypto.pseudoRandomBytes(6).toString('hex');
|
|
|
|
}
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-10 14:40:58 -05:00
|
|
|
* Check whether an uplink can proxy
|
2017-06-10 16:41:24 -05:00
|
|
|
* @param {String} pkg package anem
|
|
|
|
* @param {*} upLink
|
2017-06-10 14:40:58 -05:00
|
|
|
* @return {Boolean}
|
2017-06-06 16:07:51 -05:00
|
|
|
*/
|
2017-06-10 16:41:24 -05:00
|
|
|
hasProxyTo(pkg, upLink) {
|
|
|
|
return (this.getMatchedPackagesSpec(pkg).proxy || []).reduce(function(prev, curr) {
|
|
|
|
if (upLink === curr) {
|
|
|
|
return true;
|
|
|
|
}
|
2017-06-10 14:40:58 -05:00
|
|
|
return prev;
|
|
|
|
}, false);
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-10 14:40:58 -05:00
|
|
|
* Check for package spec
|
2017-06-10 16:41:24 -05:00
|
|
|
* @param {String} pkg package name
|
2017-06-10 14:40:58 -05:00
|
|
|
* @return {Object}
|
2017-06-06 16:07:51 -05:00
|
|
|
*/
|
2017-06-10 16:41:24 -05:00
|
|
|
getMatchedPackagesSpec(pkg) {
|
2017-06-06 16:07:51 -05:00
|
|
|
for (let i in this.packages) {
|
2017-06-10 14:40:58 -05:00
|
|
|
if (minimatch.makeRe(i).exec(pkg)) {
|
|
|
|
return this.packages[i];
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
|
|
|
}
|
2017-06-10 14:40:58 -05:00
|
|
|
return {};
|
2017-06-06 16:07:51 -05:00
|
|
|
}
|
2017-06-10 16:07:08 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Store or create whether recieve a secret key
|
|
|
|
* @param {String} secret
|
|
|
|
* @return {String}
|
|
|
|
*/
|
|
|
|
checkSecretKey(secret) {
|
|
|
|
if (_.isNil(secret) === false) {
|
2017-06-10 16:18:50 -05:00
|
|
|
this.secret = secret;
|
2017-06-10 16:07:08 -05:00
|
|
|
return secret;
|
|
|
|
}
|
|
|
|
// it generates a secret key
|
|
|
|
// FUTURE: this might be an external secret key, perhaps whitin config file?
|
|
|
|
this.secret = Crypto.pseudoRandomBytes(32).toString('hex');
|
|
|
|
return this.secret;
|
|
|
|
}
|
2017-04-26 23:54:15 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = Config;
|