config
Version:
Configuration control for production node deployments
92 lines (85 loc) • 3.18 kB
JavaScript
const asyncSymbol = Symbol('asyncSymbol');
const { deferConfig } = require('./defer');
/** @typedef {import('./lib/config').Config} Config */
/** @typedef {import('./defer').DeferredConfig} DeferredConfig */
/**
* @template T
* @overload
* @param {Promise<T>} promiseOrFunc
* @returns {Promise<T>}
*/
/**
* @template T
* @overload
* @param {(config: Config, original: any) => Promise<T>} promiseOrFunc
* @returns {DeferredConfig}
*/
/**
* @template T
* @param {Promise<T> | ((config: Config, original: any) => Promise<T>)} promiseOrFunc
* the promise will determine a property's value once resolved
* can also be a function to defer which resolves to a promise
* @returns {Promise<T> | DeferredConfig} a marked promise to be resolve later using `resolveAsyncConfigs`
* @deprecated please use async functions with defer
*/
function asyncConfig(promiseOrFunc) {
const { Util } = require('./lib/util.js');
Util.errorOnce("ASYNC_CONFIG", 'config/async.js is deprecated. Please use async functions with the new defer functionality');
if (typeof promiseOrFunc === 'function') { // also acts as deferConfig
return deferConfig(function (config, original) {
var release;
function registerRelease(resolve) { release = resolve; }
function callFunc() { return promiseOrFunc.call(config, config, original); }
var promise = asyncConfig(new Promise(registerRelease).then(callFunc));
promise.release = release;
return promise;
});
}
var promise = promiseOrFunc;
promise.async = asyncSymbol;
promise.prepare = function(config, prop, property) {
if (promise.release) {
promise.release();
}
return function() {
return promise.then(function(value) {
Object.defineProperty(prop, property, {value: value});
});
};
};
return promise;
}
/**
* Do not use `config.get` before executing this method, it will freeze the config object.
* @param {Config} config the main config object, returned from require('config')
* @returns {Promise<Config>} once all promises are resolved, return the original config object
* @deprecated please use async functions with defer and Util.resolveAsyncConfigs
*/
function resolveAsyncConfigs(config) {
var promises = [];
var resolvers = [];
(function iterate(prop) {
if (prop.constructor === String) {
return;
}
var propsToSort = Object.keys(prop).filter((property) => prop[property] != null);
propsToSort.sort().forEach(function(property) {
if (prop[property].constructor === Object) {
iterate(prop[property]);
}
else if (prop[property].constructor === Array) {
prop[property].forEach(iterate);
}
else if (prop[property].async === asyncSymbol) {
resolvers.push(prop[property].prepare(config, prop, property));
promises.push(prop[property]);
}
});
})(config);
return Promise.all(promises).then(function() {
resolvers.forEach(function(resolve) { resolve(); });
return config;
});
}
module.exports.asyncConfig = asyncConfig;
module.exports.resolveAsyncConfigs = resolveAsyncConfigs;