1063 lines
19 KiB
JavaScript
1063 lines
19 KiB
JavaScript
/*!
|
|
* config.js - configuration parsing for bcoin
|
|
* Copyright (c) 2016-2017, Christopher Jeffrey (MIT License).
|
|
* https://github.com/bcoin-org/bcoin
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
const assert = require('assert');
|
|
const Path = require('path');
|
|
const os = require('os');
|
|
const fs = require('../utils/fs');
|
|
const util = require('../utils/util');
|
|
const HOME = os.homedir ? os.homedir() : '/';
|
|
|
|
/**
|
|
* Config Parser
|
|
* @alias module:node.Config
|
|
* @constructor
|
|
* @param {String} module - Module name (e.g. `bcoin`).
|
|
*/
|
|
|
|
function Config(module) {
|
|
if (!(this instanceof Config))
|
|
return new Config(module);
|
|
|
|
assert(typeof module === 'string');
|
|
assert(module.length > 0);
|
|
|
|
this.module = module;
|
|
this.network = 'main';
|
|
this.prefix = Path.join(HOME, `.${module}`);
|
|
|
|
this.options = Object.create(null);
|
|
this.data = Object.create(null);
|
|
this.env = Object.create(null);
|
|
this.args = Object.create(null);
|
|
this.argv = [];
|
|
this.pass = [];
|
|
this.query = Object.create(null);
|
|
this.hash = Object.create(null);
|
|
}
|
|
|
|
/**
|
|
* Option name aliases.
|
|
* @const {Object}
|
|
*/
|
|
|
|
Config.alias = {
|
|
'seed': 'seeds',
|
|
'node': 'nodes',
|
|
'n': 'network'
|
|
};
|
|
|
|
/**
|
|
* Inject options.
|
|
* @param {Object} options
|
|
*/
|
|
|
|
Config.prototype.inject = function inject(options) {
|
|
for (const key of Object.keys(options)) {
|
|
const value = options[key];
|
|
|
|
switch (key) {
|
|
case 'hash':
|
|
case 'query':
|
|
case 'env':
|
|
case 'argv':
|
|
case 'config':
|
|
continue;
|
|
}
|
|
|
|
this.set(key, value);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Load options from hash, query, env, or args.
|
|
* @param {Object} options
|
|
*/
|
|
|
|
Config.prototype.load = function load(options) {
|
|
if (options.hash)
|
|
this.parseHash(options.hash);
|
|
|
|
if (options.query)
|
|
this.parseQuery(options.query);
|
|
|
|
if (options.env)
|
|
this.parseEnv(options.env);
|
|
|
|
if (options.argv)
|
|
this.parseArg(options.argv);
|
|
|
|
this.network = this.getNetwork();
|
|
this.prefix = this.getPrefix();
|
|
};
|
|
|
|
/**
|
|
* Open a config file.
|
|
* @param {String} file - e.g. `bcoin.conf`.
|
|
* @throws on IO error
|
|
*/
|
|
|
|
Config.prototype.open = function open(file) {
|
|
if (fs.unsupported)
|
|
return;
|
|
|
|
const path = this.getFile(file);
|
|
|
|
let text;
|
|
try {
|
|
text = fs.readFileSync(path, 'utf8');
|
|
} catch (e) {
|
|
if (e.code === 'ENOENT')
|
|
return;
|
|
throw e;
|
|
}
|
|
|
|
this.parseConfig(text);
|
|
|
|
this.network = this.getNetwork();
|
|
this.prefix = this.getPrefix();
|
|
};
|
|
|
|
/**
|
|
* Set default option.
|
|
* @param {String} key
|
|
* @param {Object} value
|
|
*/
|
|
|
|
Config.prototype.set = function set(key, value) {
|
|
assert(typeof key === 'string', 'Key must be a string.');
|
|
|
|
if (value == null)
|
|
return;
|
|
|
|
key = key.toLowerCase().replace(/-/g, '');
|
|
|
|
this.options[key] = value;
|
|
};
|
|
|
|
/**
|
|
* Test whether a config option is present.
|
|
* @param {String} key
|
|
* @returns {Boolean}
|
|
*/
|
|
|
|
Config.prototype.has = function has(key) {
|
|
if (typeof key === 'number') {
|
|
assert(key >= 0, 'Index must be positive.');
|
|
if (key >= this.argv.length)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
assert(typeof key === 'string', 'Key must be a string.');
|
|
|
|
key = key.toLowerCase().replace(/-/g, '');
|
|
|
|
if (this.hash[key] != null)
|
|
return true;
|
|
|
|
if (this.query[key] != null)
|
|
return true;
|
|
|
|
if (this.args[key] != null)
|
|
return true;
|
|
|
|
if (this.env[key] != null)
|
|
return true;
|
|
|
|
if (this.data[key] != null)
|
|
return true;
|
|
|
|
if (this.options[key] != null)
|
|
return true;
|
|
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Get a config option.
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Object|null}
|
|
*/
|
|
|
|
Config.prototype.get = function get(key, fallback) {
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (Array.isArray(key)) {
|
|
const keys = key;
|
|
for (const key of keys) {
|
|
const value = this.get(key);
|
|
if (value !== null)
|
|
return value;
|
|
}
|
|
return fallback;
|
|
}
|
|
|
|
if (typeof key === 'number') {
|
|
assert(key >= 0, 'Index must be positive.');
|
|
|
|
if (key >= this.argv.length)
|
|
return fallback;
|
|
|
|
if (this.argv[key] != null)
|
|
return this.argv[key];
|
|
|
|
return fallback;
|
|
}
|
|
|
|
assert(typeof key === 'string', 'Key must be a string.');
|
|
|
|
key = key.toLowerCase().replace(/-/g, '');
|
|
|
|
if (this.hash[key] != null)
|
|
return this.hash[key];
|
|
|
|
if (this.query[key] != null)
|
|
return this.query[key];
|
|
|
|
if (this.args[key] != null)
|
|
return this.args[key];
|
|
|
|
if (this.env[key] != null)
|
|
return this.env[key];
|
|
|
|
if (this.data[key] != null)
|
|
return this.data[key];
|
|
|
|
if (this.options[key] != null)
|
|
return this.options[key];
|
|
|
|
return fallback;
|
|
};
|
|
|
|
/**
|
|
* Get a value's type.
|
|
* @param {String} key
|
|
* @returns {String}
|
|
*/
|
|
|
|
Config.prototype.typeOf = function typeOf(key) {
|
|
const value = this.get(key);
|
|
|
|
if (value === null)
|
|
return 'null';
|
|
|
|
return typeof value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a string).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {String|null}
|
|
*/
|
|
|
|
Config.prototype.str = function str(key, fallback) {
|
|
const value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'string')
|
|
throw new Error(`${fmt(key)} must be a string.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as an integer).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.int = function int(key, fallback) {
|
|
let value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'string') {
|
|
if (typeof value !== 'number')
|
|
throw new Error(`${fmt(key)} must be an int.`);
|
|
|
|
if (!Number.isSafeInteger(value))
|
|
throw new Error(`${fmt(key)} must be an int.`);
|
|
|
|
return value;
|
|
}
|
|
|
|
if (!/^\-?\d+$/.test(value))
|
|
throw new Error(`${fmt(key)} must be an int.`);
|
|
|
|
value = parseInt(value, 10);
|
|
|
|
if (!Number.isSafeInteger(value))
|
|
throw new Error(`${fmt(key)} must be an int.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a unsigned integer).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.uint = function uint(key, fallback) {
|
|
const value = this.int(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (value < 0)
|
|
throw new Error(`${fmt(key)} must be a uint.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a float).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.float = function float(key, fallback) {
|
|
let value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'string') {
|
|
if (typeof value !== 'number')
|
|
throw new Error(`${fmt(key)} must be a float.`);
|
|
|
|
if (!isFinite(value))
|
|
throw new Error(`${fmt(key)} must be a float.`);
|
|
|
|
return value;
|
|
}
|
|
|
|
if (!/^\-?\d*(?:\.\d*)?$/.test(value))
|
|
throw new Error(`${fmt(key)} must be a float.`);
|
|
|
|
if (!/\d/.test(value))
|
|
throw new Error(`${fmt(key)} must be a float.`);
|
|
|
|
value = parseFloat(value);
|
|
|
|
if (!isFinite(value))
|
|
throw new Error(`${fmt(key)} must be a float.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a positive float).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.ufloat = function ufloat(key, fallback) {
|
|
const value = this.float(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (value < 0)
|
|
throw new Error(`${fmt(key)} must be a positive float.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a value (as a fixed number).
|
|
* @param {String} key
|
|
* @param {Number?} exp
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.fixed = function fixed(key, exp, fallback) {
|
|
const value = this.float(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
try {
|
|
return util.fromFixed(value.toString(10), exp || 0);
|
|
} catch (e) {
|
|
throw new Error(`${fmt(key)} must be a fixed number.`);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Get a value (as a positive fixed number).
|
|
* @param {String} key
|
|
* @param {Number?} exp
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.ufixed = function ufixed(key, exp, fallback) {
|
|
const value = this.fixed(key, exp);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (value < 0)
|
|
throw new Error(`${fmt(key)} must be a positive fixed number.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a boolean).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Boolean|null}
|
|
*/
|
|
|
|
Config.prototype.bool = function bool(key, fallback) {
|
|
const value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
// Bitcoin Core compat.
|
|
if (typeof value === 'number') {
|
|
if (value === 1)
|
|
return true;
|
|
|
|
if (value === 0)
|
|
return false;
|
|
}
|
|
|
|
if (typeof value !== 'string') {
|
|
if (typeof value !== 'boolean')
|
|
throw new Error(`${fmt(key)} must be a boolean.`);
|
|
return value;
|
|
}
|
|
|
|
if (value === 'true' || value === '1')
|
|
return true;
|
|
|
|
if (value === 'false' || value === '0')
|
|
return false;
|
|
|
|
throw new Error(`${fmt(key)} must be a boolean.`);
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a buffer).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Buffer|null}
|
|
*/
|
|
|
|
Config.prototype.buf = function buf(key, fallback, enc) {
|
|
const value = this.get(key);
|
|
|
|
if (!enc)
|
|
enc = 'hex';
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'string') {
|
|
if (!Buffer.isBuffer(value))
|
|
throw new Error(`${fmt(key)} must be a buffer.`);
|
|
return value;
|
|
}
|
|
|
|
const data = Buffer.from(value, enc);
|
|
|
|
if (data.length !== Buffer.byteLength(value, enc))
|
|
throw new Error(`${fmt(key)} must be a ${enc} string.`);
|
|
|
|
return data;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as an array of strings).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {String[]|null}
|
|
*/
|
|
|
|
Config.prototype.array = function array(key, fallback) {
|
|
const value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'string') {
|
|
if (!Array.isArray(value))
|
|
throw new Error(`${fmt(key)} must be an array.`);
|
|
return value;
|
|
}
|
|
|
|
const parts = value.trim().split(/\s*,\s*/);
|
|
const result = [];
|
|
|
|
for (const part of parts) {
|
|
if (part.length === 0)
|
|
continue;
|
|
|
|
result.push(part);
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as an object).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Object|null}
|
|
*/
|
|
|
|
Config.prototype.obj = function obj(key, fallback) {
|
|
const value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'object')
|
|
throw new Error(`${fmt(key)} must be an object.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a function).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Function|null}
|
|
*/
|
|
|
|
Config.prototype.func = function func(key, fallback) {
|
|
const value = this.get(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
if (typeof value !== 'function')
|
|
throw new Error(`${fmt(key)} must be a function.`);
|
|
|
|
return value;
|
|
};
|
|
|
|
/**
|
|
* Get a config option (as a string).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {String|null}
|
|
*/
|
|
|
|
Config.prototype.path = function path(key, fallback) {
|
|
let value = this.str(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
switch (value[0]) {
|
|
case '~': // home dir
|
|
value = Path.join(HOME, value.substring(1));
|
|
break;
|
|
case '@': // prefix
|
|
value = Path.join(this.prefix, value.substring(1));
|
|
break;
|
|
default: // cwd
|
|
break;
|
|
}
|
|
|
|
return Path.normalize(value);
|
|
};
|
|
|
|
/**
|
|
* Get a config option (in MB).
|
|
* @param {String} key
|
|
* @param {Object?} fallback
|
|
* @returns {Number|null}
|
|
*/
|
|
|
|
Config.prototype.mb = function mb(key, fallback) {
|
|
const value = this.uint(key);
|
|
|
|
if (fallback === undefined)
|
|
fallback = null;
|
|
|
|
if (value === null)
|
|
return fallback;
|
|
|
|
return value * 1024 * 1024;
|
|
};
|
|
|
|
/**
|
|
* Grab network type from config data.
|
|
* @private
|
|
* @returns {String}
|
|
*/
|
|
|
|
Config.prototype.getNetwork = function getNetwork() {
|
|
let network = this.str('network');
|
|
|
|
if (!network)
|
|
network = 'main';
|
|
|
|
assert(isAlpha(network), 'Bad network.');
|
|
|
|
return network;
|
|
};
|
|
|
|
/**
|
|
* Grab prefix from config data.
|
|
* @private
|
|
* @returns {String}
|
|
*/
|
|
|
|
Config.prototype.getPrefix = function getPrefix() {
|
|
let prefix = this.str('prefix');
|
|
|
|
if (prefix) {
|
|
if (prefix[0] === '~')
|
|
prefix = Path.join(HOME, prefix.substring(1));
|
|
return prefix;
|
|
}
|
|
|
|
prefix = Path.join(HOME, `.${this.module}`);
|
|
|
|
const network = this.str('network');
|
|
|
|
if (network) {
|
|
assert(isAlpha(network), 'Bad network.');
|
|
if (network !== 'main')
|
|
prefix = Path.join(prefix, network);
|
|
}
|
|
|
|
return Path.normalize(prefix);
|
|
};
|
|
|
|
/**
|
|
* Grab config filename from config data.
|
|
* @private
|
|
* @param {String} file
|
|
* @returns {String}
|
|
*/
|
|
|
|
Config.prototype.getFile = function getFile(file) {
|
|
const name = this.str('config');
|
|
|
|
if (name)
|
|
return name;
|
|
|
|
return Path.join(this.prefix, file);
|
|
};
|
|
|
|
/**
|
|
* Ensure prefix.
|
|
* @returns {Promise}
|
|
*/
|
|
|
|
Config.prototype.ensure = function ensure() {
|
|
if (fs.unsupported)
|
|
return Promise.resolve();
|
|
|
|
return fs.mkdirp(this.prefix);
|
|
};
|
|
|
|
/**
|
|
* Create a file path using `prefix`.
|
|
* @param {String} file
|
|
* @returns {String}
|
|
*/
|
|
|
|
Config.prototype.location = function location(file) {
|
|
return Path.join(this.prefix, file);
|
|
};
|
|
|
|
/**
|
|
* Parse config text.
|
|
* @private
|
|
* @param {String} text
|
|
*/
|
|
|
|
Config.prototype.parseConfig = function parseConfig(text) {
|
|
assert(typeof text === 'string', 'Config must be text.');
|
|
|
|
text = text.trim();
|
|
|
|
const parts = text.split(/\n+/);
|
|
|
|
for (let line of parts) {
|
|
line = line.trim();
|
|
|
|
if (line.length === 0)
|
|
continue;
|
|
|
|
if (/^\s*#/.test(line))
|
|
continue;
|
|
|
|
let eq = line.indexOf('=');
|
|
const col = line.indexOf(':');
|
|
|
|
if (col !== -1 && (col < eq || eq === -1))
|
|
eq = col;
|
|
|
|
let key, value;
|
|
if (eq === -1) {
|
|
key = line.trim();
|
|
value = '';
|
|
} else {
|
|
key = line.substring(0, eq).trim();
|
|
value = line.substring(eq + 1).trim();
|
|
}
|
|
|
|
key = key.replace(/\-/g, '').toLowerCase();
|
|
|
|
const alias = Config.alias[key];
|
|
|
|
if (alias)
|
|
key = alias;
|
|
|
|
if (key.length === 0)
|
|
continue;
|
|
|
|
if (value.length === 0)
|
|
continue;
|
|
|
|
this.data[key] = value;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Parse arguments.
|
|
* @private
|
|
* @param {Array?} argv
|
|
*/
|
|
|
|
Config.prototype.parseArg = function parseArg(argv) {
|
|
if (!argv || typeof argv !== 'object')
|
|
argv = process.argv;
|
|
|
|
assert(Array.isArray(argv));
|
|
|
|
let last = null;
|
|
let pass = false;
|
|
|
|
for (let i = 2; i < argv.length; i++) {
|
|
const arg = argv[i];
|
|
|
|
assert(typeof arg === 'string');
|
|
|
|
if (arg === '--') {
|
|
pass = true;
|
|
continue;
|
|
}
|
|
|
|
if (pass) {
|
|
this.pass.push(arg);
|
|
continue;
|
|
}
|
|
|
|
if (arg.length === 0) {
|
|
last = null;
|
|
continue;
|
|
}
|
|
|
|
if (arg.indexOf('--') === 0) {
|
|
// e.g. --opt
|
|
const parts = arg.split('=');
|
|
|
|
let key = parts[0];
|
|
let value = null;
|
|
let empty = false;
|
|
|
|
if (parts.length > 1) {
|
|
// e.g. --opt=val
|
|
value = parts.slice(1).join('=').trim();
|
|
last = null;
|
|
empty = false;
|
|
} else {
|
|
value = 'true';
|
|
last = null;
|
|
empty = true;
|
|
}
|
|
|
|
key = key.replace(/\-/g, '');
|
|
|
|
if (key.length === 0)
|
|
continue;
|
|
|
|
if (value.length === 0)
|
|
continue;
|
|
|
|
// Do not allow one-letter aliases.
|
|
if (key.length > 1) {
|
|
const alias = Config.alias[key];
|
|
if (alias)
|
|
key = alias;
|
|
}
|
|
|
|
this.args[key] = value;
|
|
|
|
if (empty)
|
|
last = key;
|
|
|
|
continue;
|
|
}
|
|
|
|
if (arg[0] === '-') {
|
|
// e.g. -abc
|
|
last = null;
|
|
for (let j = 1; j < arg.length; j++) {
|
|
let key = arg[j];
|
|
const alias = Config.alias[key];
|
|
if (alias)
|
|
key = alias;
|
|
this.args[key] = 'true';
|
|
last = key;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// e.g. foo
|
|
const value = arg.trim();
|
|
|
|
if (value.length === 0) {
|
|
last = null;
|
|
continue;
|
|
}
|
|
|
|
if (last) {
|
|
this.args[last] = value;
|
|
last = null;
|
|
} else {
|
|
this.argv.push(value);
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Parse environment variables.
|
|
* @private
|
|
* @param {Object?} env
|
|
* @returns {Object}
|
|
*/
|
|
|
|
Config.prototype.parseEnv = function parseEnv(env) {
|
|
let prefix = this.module;
|
|
|
|
prefix = prefix.toUpperCase();
|
|
prefix = prefix.replace(/-/g, '_');
|
|
prefix += '_';
|
|
|
|
if (!env || typeof env !== 'object')
|
|
env = process.env;
|
|
|
|
assert(env && typeof env === 'object');
|
|
|
|
for (let key of Object.keys(env)) {
|
|
let value = env[key];
|
|
|
|
assert(typeof value === 'string');
|
|
|
|
if (key.indexOf(prefix) !== 0)
|
|
continue;
|
|
|
|
key = key.substring(prefix.length);
|
|
key = key.replace(/_/g, '').toLowerCase();
|
|
|
|
if (key.length === 0)
|
|
continue;
|
|
|
|
value = value.trim();
|
|
|
|
if (value.length === 0)
|
|
continue;
|
|
|
|
// Do not allow one-letter aliases.
|
|
if (key.length > 1) {
|
|
const alias = Config.alias[key];
|
|
if (alias)
|
|
key = alias;
|
|
}
|
|
|
|
this.env[key] = value;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Parse uri querystring variables.
|
|
* @private
|
|
* @param {String} query
|
|
*/
|
|
|
|
Config.prototype.parseQuery = function parseQuery(query) {
|
|
if (typeof query !== 'string') {
|
|
if (!global.location)
|
|
return {};
|
|
|
|
query = global.location.search;
|
|
|
|
if (typeof query !== 'string')
|
|
return {};
|
|
}
|
|
|
|
return this.parseForm(query, this.query);
|
|
};
|
|
|
|
/**
|
|
* Parse uri hash variables.
|
|
* @private
|
|
* @param {String} hash
|
|
*/
|
|
|
|
Config.prototype.parseHash = function parseHash(hash) {
|
|
if (typeof hash !== 'string') {
|
|
if (!global.location)
|
|
return {};
|
|
|
|
hash = global.location.hash;
|
|
|
|
if (typeof hash !== 'string')
|
|
return {};
|
|
}
|
|
|
|
return this.parseForm(hash, this.hash);
|
|
};
|
|
|
|
/**
|
|
* Parse form-urlencoded variables.
|
|
* @private
|
|
* @param {String} query
|
|
*/
|
|
|
|
Config.prototype.parseForm = function parseForm(query, map) {
|
|
assert(typeof query === 'string');
|
|
|
|
if (query.length === 0)
|
|
return;
|
|
|
|
if (query[0] === '?' || query[0] === '#')
|
|
query = query.substring(1);
|
|
|
|
const parts = query.split('&');
|
|
|
|
for (const pair of parts) {
|
|
const index = pair.indexOf('=');
|
|
|
|
let key, value;
|
|
if (index === -1) {
|
|
key = pair;
|
|
value = '';
|
|
} else {
|
|
key = pair.substring(0, index);
|
|
value = pair.substring(index + 1);
|
|
}
|
|
|
|
key = unescape(key);
|
|
key = key.replace(/\-/g, '').toLowerCase();
|
|
|
|
if (key.length === 0)
|
|
continue;
|
|
|
|
value = unescape(value);
|
|
|
|
if (value.length === 0)
|
|
continue;
|
|
|
|
const alias = Config.alias[key];
|
|
|
|
if (alias)
|
|
key = alias;
|
|
|
|
map[key] = value;
|
|
}
|
|
};
|
|
|
|
/*
|
|
* Helpers
|
|
*/
|
|
|
|
function fmt(key) {
|
|
if (Array.isArray(key))
|
|
key = key[0];
|
|
|
|
if (typeof key === 'number')
|
|
return `Argument #${key}`;
|
|
|
|
return key;
|
|
}
|
|
|
|
function unescape(str) {
|
|
try {
|
|
str = decodeURIComponent(str);
|
|
str = str.replace(/\+/g, ' ');
|
|
} catch (e) {
|
|
;
|
|
}
|
|
str = str.replace(/\0/g, '');
|
|
return str;
|
|
}
|
|
|
|
function isAlpha(str) {
|
|
return /^[a-z0-9]+$/.test(str);
|
|
}
|
|
|
|
/*
|
|
* Expose
|
|
*/
|
|
|
|
module.exports = Config;
|