Ghost/core/shared/labs.js
Rishabh 025eb8bd79 Updated tiers as a beta feature from alpha
refs https://github.com/TryGhost/Team/issues/1026

Tiers is moving up as a beta feature with an early-access opt-in flow. This means site owners can now opt-in for early access to Tiers feature in Ghost, but it's a one way door and its not possible to switch off tiers once enabled. This is to ensure that sites don't break in any unexpected ways once the tiers feature is enabled by switching it off.
2021-09-09 16:57:46 +05:30

107 lines
3.5 KiB
JavaScript

const _ = require('lodash');
const Promise = require('bluebird');
const SafeString = require('express-hbs').SafeString;
const errors = require('@tryghost/errors');
const logging = require('@tryghost/logging');
const tpl = require('@tryghost/tpl');
const settingsCache = require('./settings-cache');
const config = require('./config');
const messages = {
errorMessage: 'The \\{\\{{helperName}\\}\\} helper is not available.',
errorContext: 'The {flagName} flag must be enabled in labs if you wish to use the \\{\\{{helperName}\\}\\} helper.',
errorHelp: 'See {url}'
};
// NOTE: this allowlist is meant to be used to filter out any unexpected
// input for the "labs" setting value
const BETA_FEATURES = [
'activitypub',
'matchHelper',
'multipleProducts'
];
const ALPHA_FEATURES = [
'oauthLogin',
'membersFiltering',
'emailOnlyPosts',
'dashboardTwo'
];
module.exports.WRITABLE_KEYS_ALLOWLIST = [...BETA_FEATURES, ...ALPHA_FEATURES];
module.exports.getAll = () => {
const labs = _.cloneDeep(settingsCache.get('labs')) || {};
ALPHA_FEATURES.forEach((alphaKey) => {
if (labs[alphaKey] && !(config.get('enableDeveloperExperiments') || process.env.NODE_ENV.match(/^testing/))) {
delete labs[alphaKey];
}
});
labs.members = settingsCache.get('members_signup_access') !== 'none';
return labs;
};
/**
* @param {string} flag
* @returns {boolean}
*/
module.exports.isSet = function isSet(flag) {
const labsConfig = module.exports.getAll();
return !!(labsConfig && labsConfig[flag] && labsConfig[flag] === true);
};
/**
*
* @param {object} options
* @param {string} options.flagKey the internal lookup key of the flag e.g. labs.isSet(matchHelper)
* @param {string} options.flagName the user-facing name of the flag e.g. Match helper
* @param {string} options.helperName Name of the helper to be enabled/disabled
* @param {string} [options.errorMessage] Optional replacement error message
* @param {string} [options.errorContext] Optional replacement context message
* @param {string} [options.errorHelp] Optional replacement help message
* @param {string} [options.helpUrl] Url to show in the help message
* @param {string} [options.async] is the helper async?
* @param {function} callback
* @returns {Promise<Handlebars.SafeString>|Handlebars.SafeString}
*/
module.exports.enabledHelper = function enabledHelper(options, callback) {
const errDetails = {};
let errString;
if (module.exports.isSet(options.flagKey) === true) {
// helper is active, use the callback
return callback();
}
// Else, the helper is not active and we need to handle this as an error
errDetails.message = tpl(options.errorMessage || messages.errorMessage, {helperName: options.helperName}),
errDetails.context = tpl(options.errorContext || messages.errorContext, {
helperName: options.helperName,
flagName: options.flagName
});
errDetails.help = tpl(options.errorHelp || messages.errorHelp, {url: options.helpUrl});
logging.error(new errors.DisabledFeatureError(errDetails));
errString = new SafeString(`<script>console.error("${_.values(errDetails).join(' ')}");</script>`);
if (options.async) {
return Promise.resolve(errString);
}
return errString;
};
module.exports.enabledMiddleware = flag => (req, res, next) => {
if (module.exports.isSet(flag) === true) {
return next();
} else {
return next(new errors.NotFoundError());
}
};