0
Fork 0
mirror of https://github.com/verdaccio/verdaccio.git synced 2025-04-01 02:42:23 -05:00

Restore config refactoring

This commit is contained in:
Juan Picado @jotadeveloper 2017-06-10 21:40:58 +02:00
parent 75c3cbafd0
commit e799500893
No known key found for this signature in database
GPG key ID: 18AC54485952D158

View file

@ -2,62 +2,45 @@
/* eslint prefer-spread: "off" */
'use strict';
const _ = require('lodash');
const assert = require('assert');
const Crypto = require('crypto');
const Error = require('http-errors');
const minimatch = require('minimatch');
const Path = require('path');
const LocalData = require('./local-data');
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;
const integrityPackages = ['users', 'uplinks', 'packages'];
const integrityProxy = ['http_proxy', 'https_proxy', 'no_proxy'];
/**
* [[a, [b, c]], d] -> [a, b, c, d]
* @param {*} array
* @return {Array}
*/
function flatten(array) {
const result = [];
for (let i = 0; i < array.length; i++) {
if (_.isArray(array[i])) {
result.push.apply(result, flatten(array[i]));
} else {
result.push(array[i]);
}
}
return result;
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;
}
const parse_interval_table = {
'': 1000,
'ms': 1,
's': 1000,
'm': 60*1000,
'h': 60*60*1000,
'd': 86400000,
'w': 7*86400000,
'M': 30*86400000,
'y': 365*86400000,
};
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;
'': 1000,
'ms': 1,
's': 1000,
'm': 60*1000,
'h': 60*60*1000,
'd': 86400000,
'w': 7*86400000,
'M': 30*86400000,
'y': 365*86400000,
};
/**
@ -66,46 +49,23 @@ const check_user_or_uplink = function(arg) {
* @return {Number}
*/
function parse_interval(interval) {
if (typeof(interval) === 'number') {
return interval * 1000;
}
let result = 0;
let last_suffix = Infinity;
interval.split(/\s+/).forEach(function(x) {
if (!x) return;
let m = x.match(/^((0|[1-9][0-9]*)(\.[0-9]+)?)(ms|s|m|h|d|w|M|y|)$/);
if (!m
|| parse_interval_table[m[4]] >= last_suffix
|| (m[4] === '' && last_suffix !== Infinity)) {
throw Error('invalid interval: ' + interval);
}
last_suffix = parse_interval_table[m[4]];
result += Number(m[1]) * parse_interval_table[m[4]];
});
return result;
}
/**
* Normalise user list.
* @return {Array}
*/
function normalize_userlist() {
const result = [];
for (let i = 0; i<arguments.length; i++) {
if (_.isNil(arguments[i])) {
continue;
}
// if it's a string, split it to array
if (_.isString(arguments[i])) {
result.push(arguments[i].split(/\s+/));
} else if (_.isArray(arguments[i])) {
result.push(arguments[i]);
} else {
throw Error('CONFIG: bad package acl (array or string expected): ' + JSON.stringify(arguments[i]));
}
if (typeof(interval) === 'number') {
return interval * 1000;
}
return flatten(result);
let result = 0;
let last_suffix = Infinity;
interval.split(/\s+/).forEach(function(x) {
if (!x) return;
let m = x.match(/^((0|[1-9][0-9]*)(\.[0-9]+)?)(ms|s|m|h|d|w|M|y|)$/);
if (!m
|| parse_interval_table[m[4]] >= last_suffix
|| (m[4] === '' && last_suffix !== Infinity)) {
throw Error('invalid interval: ' + interval);
}
last_suffix = parse_interval_table[m[4]];
result += Number(m[1]) * parse_interval_table[m[4]];
});
return result;
}
/**
@ -113,188 +73,190 @@ function normalize_userlist() {
*/
class Config {
/**
* Constructor
* @param {*} defaultConfig config the content
*/
constructor(defaultConfig) {
const self = this;
/**
* Constructor
* @param {*} config config the content
*/
constructor(config) {
const self = this;
for (let i in config) {
if (self[i] == null) {
self[i] = config[i];
}
}
if (!self.user_agent) {
self.user_agent = `${pkgName}/${pkgVersion}`;
}
// some weird shell scripts are valid yaml files parsed as string
assert.equal(_.isObject(defaultConfig), true, 'CONFIG: it doesn\'t look like a valid config file');
assert.equal(typeof(config), 'object', 'CONFIG: it doesn\'t look like a valid config file');
this._mixConfigProperties(defaultConfig);
this._setUserAgent();
assert(self.storage, 'CONFIG: storage path not defined');
// sanity check for strategic config properties
integrityPackages.forEach(function(x) {
if (_.isNil(self[x])) {
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(/\/$/, '');
}
}
this._setDefaultPackage();
this._safetyCheckPackages();
this._loadEnvironmentProxies();
this._generateServerId();
}
/**
* Mix the external configuration file.
* @param {object} defaultConfig
* @private
*/
_mixConfigProperties(defaultConfig) {
for (let i in defaultConfig) {
if (_.isNil(this[i])) {
this[i] = defaultConfig[i];
assert(self.storage, 'CONFIG: storage path not defined');
// local data handler is linked with the configuration handler
self.localList = new LocalData(
Path.join(
Path.resolve(Path.dirname(self.self_path || ''), self.storage),
// FUTURE: the database might be parameterizable from config.yaml
'.sinopia-db.json'
)
);
// it generates a secret key
// FUTURE: this might be an external secret key, perhaps whitin config file?
if (!self.secret) {
self.secret = self.localList.data.secret;
if (!self.secret) {
self.secret = Crypto.pseudoRandomBytes(32).toString('hex');
self.localList.data.secret = self.secret;
self.localList.sync();
}
}
}
/**
* Set the user agent.
* @private
*/
_setUserAgent() {
if (!this.user_agent) {
this.user_agent = `${pkgName}/${pkgVersion}`;
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(/\/$/, '');
}
}
/**
* 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);
}
}
/**
*
* @private
*/
_setDefaultPackage() {
// add a default rule for all packages to make writing plugins easier
if (_.isNil(this.packages['**'])) {
this.packages['**'] = {};
if (self.packages['**'] == null) {
self.packages['**'] = {};
}
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)');
self.packages[i].access = normalize_userlist(
self.packages[i].allow_access,
self.packages[i].access
);
delete self.packages[i].allow_access;
self.packages[i].publish = normalize_userlist(
self.packages[i].allow_publish,
self.packages[i].publish
);
delete self.packages[i].allow_publish;
self.packages[i].proxy = normalize_userlist(
self.packages[i].proxy_access,
self.packages[i].proxy
);
delete self.packages[i].proxy_access;
}
}
}
/**
*
* @private
*/
_loadEnvironmentProxies() {
// loading these from ENV if aren't in config
integrityProxy.forEach(((v) => {
if (!(v in this)) {
this[v] = process.env[v] || process.env[v.toUpperCase()];
['http_proxy', 'https_proxy', 'no_proxy'].forEach((function(v) {
if (!(v in self)) {
self[v] = process.env[v] || process.env[v.toUpperCase()];
}
}));
}
/**
* unique identifier of self server (or a cluster), used to avoid loops
* @private
*/
_generateServerId() {
if (!this.server_id) {
this.server_id = Crypto.pseudoRandomBytes(6).toString('hex');
// unique identifier of self server (or a cluster), used to avoid loops
if (!self.server_id) {
self.server_id = Crypto.pseudoRandomBytes(6).toString('hex');
}
}
/**
*
* @private
* Check whether an uplink can proxy
* @param {*} pkg
* @param {*} uplink
* @return {Boolean}
*/
_safetyCheckPackages() {
can_proxy_to(pkg, uplink) {
return (this.get_package_spec(pkg).proxy || []).reduce(function(prev, curr) {
if (uplink === curr) return true;
return prev;
}, false);
}
/**
* Check for package spec
* @param {*} pkg
* @return {Object}
*/
get_package_spec(pkg) {
for (let i in this.packages) {
if (Object.prototype.hasOwnProperty.call(this.packages, i)) {
// validate integrity packages
assert(_.isObject(this.packages[i]) && _.isArray(this.packages[i]) === false, 'CONFIG: bad "'+i+'" package description (object expected)');
this.packages[i].access = normalize_userlist(
this.packages[i].allow_access,
this.packages[i].access
);
delete this.packages[i].allow_access;
this.packages[i].publish = normalize_userlist(
this.packages[i].allow_publish,
this.packages[i].publish
);
delete this.packages[i].allow_publish;
this.packages[i].proxy = normalize_userlist(
this.packages[i].proxy_access,
this.packages[i].proxy
);
delete this.packages[i].proxy_access;
if (minimatch.makeRe(i).exec(pkg)) {
return this.packages[i];
}
}
return {};
}
/**
* Check whether an uplink can proxy
* @param {*} pkg
* @param {*} uplink
* @return {Boolean}
*/
can_proxy_to(pkg, uplink) {
const compatibleProxies = this.get_package_spec(pkg).proxy || [];
return (compatibleProxies).reduce(function(prev, curr) {
return uplink === curr ? true : prev;
}, false);
}
/**
* Check for package spec.
* @param {String} pkgName
* @return {Object}
*/
get_package_spec(pkgName) {
for (let pkg in this.packages) {
if (minimatch.makeRe(pkg).exec(pkgName)) {
return this.packages[pkg];
}
}
return {};
}
}
module.exports = Config;