2016-06-18 21:53:18 +02:00
|
|
|
'use strict';
|
2016-10-26 10:43:11 +02:00
|
|
|
|
2017-01-08 20:40:50 +01:00
|
|
|
const joi = require('joi');
|
2016-06-20 10:27:19 +02:00
|
|
|
const logger = require('../logger');
|
2016-12-09 14:50:30 +01:00
|
|
|
const { FEATURE_CREATED, FEATURE_UPDATED, FEATURE_ARCHIVED } = require('../event-type');
|
2016-10-27 20:51:21 +02:00
|
|
|
const NameExistsError = require('../error/name-exists-error');
|
|
|
|
const NotFoundError = require('../error/notfound-error');
|
|
|
|
const ValidationError = require('../error/validation-error.js');
|
|
|
|
const validateRequest = require('../error/validate-request');
|
|
|
|
const extractUser = require('../extract-user');
|
2014-10-20 15:12:30 +02:00
|
|
|
|
2016-11-13 20:33:23 +01:00
|
|
|
const legacyFeatureMapper = require('../data-helper/legacy-feature-mapper');
|
2016-09-05 16:50:52 +02:00
|
|
|
const version = 1;
|
2016-07-02 16:25:30 +02:00
|
|
|
|
2016-11-05 15:08:00 +01:00
|
|
|
const handleErrors = (req, res, error) => {
|
2017-06-21 11:24:36 +02:00
|
|
|
logger.warn('Error creating or updating feature', error);
|
2016-11-05 15:08:00 +01:00
|
|
|
switch (error.constructor) {
|
|
|
|
case NotFoundError:
|
|
|
|
return res
|
|
|
|
.status(404)
|
|
|
|
.end();
|
|
|
|
case NameExistsError:
|
|
|
|
return res
|
|
|
|
.status(403)
|
2017-01-28 17:29:22 +01:00
|
|
|
.json([{ msg: 'A feature with this name already exists. Try re-activating it from the archive.' }])
|
2016-11-05 15:08:00 +01:00
|
|
|
.end();
|
|
|
|
case ValidationError:
|
|
|
|
return res
|
|
|
|
.status(400)
|
|
|
|
.json(req.validationErrors())
|
|
|
|
.end();
|
|
|
|
default:
|
|
|
|
logger.error('Server failed executing request', error);
|
|
|
|
return res
|
|
|
|
.status(500)
|
|
|
|
.end();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-05-01 22:53:09 +02:00
|
|
|
module.exports = function (app, config) {
|
2016-11-05 14:08:47 +01:00
|
|
|
const { featureToggleStore, eventStore } = config.stores;
|
2016-05-01 22:53:09 +02:00
|
|
|
|
2016-06-18 21:53:18 +02:00
|
|
|
app.get('/features', (req, res) => {
|
2016-11-05 10:16:48 +01:00
|
|
|
featureToggleStore.getFeatures()
|
2016-07-02 16:25:30 +02:00
|
|
|
.then((features) => features.map(legacyFeatureMapper.addOldFields))
|
2016-09-05 16:50:52 +02:00
|
|
|
.then(features => res.json({ version, features }));
|
2014-10-20 15:12:30 +02:00
|
|
|
});
|
|
|
|
|
2016-06-18 21:53:18 +02:00
|
|
|
app.get('/features/:featureName', (req, res) => {
|
2016-11-05 10:16:48 +01:00
|
|
|
featureToggleStore.getFeature(req.params.featureName)
|
2016-07-02 16:25:30 +02:00
|
|
|
.then(legacyFeatureMapper.addOldFields)
|
|
|
|
.then(feature => res.json(feature).end())
|
|
|
|
.catch(() => res.status(404).json({ error: 'Could not find feature' }));
|
2014-10-20 15:12:30 +02:00
|
|
|
});
|
|
|
|
|
2016-11-05 11:37:59 +01:00
|
|
|
app.post('/features-validate', (req, res) => {
|
|
|
|
req.checkBody('name', 'Name is required').notEmpty();
|
|
|
|
req.checkBody('name', 'Name must match format ^[0-9a-zA-Z\\.\\-]+$').matches(/^[0-9a-zA-Z\\.\\-]+$/i);
|
|
|
|
|
|
|
|
validateRequest(req)
|
|
|
|
.then(validateFormat)
|
|
|
|
.then(validateUniqueName)
|
|
|
|
.then(() => res.status(201).end())
|
2016-11-05 15:08:00 +01:00
|
|
|
.catch(error => handleErrors(req, res, error));
|
2016-11-05 11:37:59 +01:00
|
|
|
});
|
|
|
|
|
2016-06-18 21:53:18 +02:00
|
|
|
app.post('/features', (req, res) => {
|
2014-11-01 11:47:21 +01:00
|
|
|
req.checkBody('name', 'Name is required').notEmpty();
|
2016-08-22 15:16:50 +02:00
|
|
|
req.checkBody('name', 'Name must match format ^[0-9a-zA-Z\\.\\-]+$').matches(/^[0-9a-zA-Z\\.\\-]+$/i);
|
2016-12-27 21:03:50 +01:00
|
|
|
const userName = extractUser(req);
|
2014-11-01 11:47:21 +01:00
|
|
|
|
2014-12-10 19:11:52 +01:00
|
|
|
validateRequest(req)
|
2016-08-23 00:04:11 +02:00
|
|
|
.then(validateFormat)
|
2014-12-10 18:49:08 +01:00
|
|
|
.then(validateUniqueName)
|
2016-12-04 14:09:37 +01:00
|
|
|
.then((_req) => legacyFeatureMapper.toNewFormat(_req.body))
|
2016-11-15 21:27:03 +01:00
|
|
|
.then(validateStrategy)
|
|
|
|
.then((featureToggle) => eventStore.store({
|
2016-12-09 14:50:30 +01:00
|
|
|
type: FEATURE_CREATED,
|
2016-11-15 21:27:03 +01:00
|
|
|
createdBy: userName,
|
|
|
|
data: featureToggle,
|
2016-06-18 21:55:46 +02:00
|
|
|
}))
|
2016-07-02 16:25:30 +02:00
|
|
|
.then(() => res.status(201).end())
|
2016-11-05 15:08:00 +01:00
|
|
|
.catch(error => handleErrors(req, res, error));
|
2016-04-24 22:41:37 +02:00
|
|
|
});
|
2014-10-20 15:12:30 +02:00
|
|
|
|
2016-06-18 21:53:18 +02:00
|
|
|
app.put('/features/:featureName', (req, res) => {
|
2016-11-15 21:27:03 +01:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
2016-07-02 16:25:30 +02:00
|
|
|
const updatedFeature = legacyFeatureMapper.toNewFormat(req.body);
|
2014-11-14 12:56:23 +01:00
|
|
|
|
|
|
|
updatedFeature.name = featureName;
|
|
|
|
|
2016-11-05 10:16:48 +01:00
|
|
|
featureToggleStore.getFeature(featureName)
|
2016-11-15 21:27:03 +01:00
|
|
|
.then(() => validateStrategy(updatedFeature))
|
2016-11-05 10:16:48 +01:00
|
|
|
.then(() => eventStore.store({
|
2016-12-09 14:50:30 +01:00
|
|
|
type: FEATURE_UPDATED,
|
2016-06-18 21:55:46 +02:00
|
|
|
createdBy: userName,
|
|
|
|
data: updatedFeature,
|
|
|
|
}))
|
2016-07-02 16:25:30 +02:00
|
|
|
.then(() => res.status(200).end())
|
2016-11-05 15:08:00 +01:00
|
|
|
.catch(error => handleErrors(req, res, error));
|
2014-10-21 15:28:10 +02:00
|
|
|
});
|
2014-12-10 18:49:08 +01:00
|
|
|
|
2017-01-09 11:06:10 +01:00
|
|
|
app.post('/features/:featureName/toggle', (req, res) => {
|
2017-01-08 20:04:46 +01:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
|
|
|
|
|
|
|
featureToggleStore.getFeature(featureName)
|
|
|
|
.then((feature) => {
|
|
|
|
feature.enabled = !feature.enabled;
|
|
|
|
return eventStore.store({
|
|
|
|
type: FEATURE_UPDATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: feature,
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.then(() => res.status(200).end())
|
|
|
|
.catch(error => handleErrors(req, res, error));
|
|
|
|
});
|
|
|
|
|
2016-06-18 21:53:18 +02:00
|
|
|
app.delete('/features/:featureName', (req, res) => {
|
2016-11-15 21:27:03 +01:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
2014-12-15 22:40:07 +01:00
|
|
|
|
2016-11-05 10:16:48 +01:00
|
|
|
featureToggleStore.getFeature(featureName)
|
|
|
|
.then(() => eventStore.store({
|
2016-12-09 14:50:30 +01:00
|
|
|
type: FEATURE_ARCHIVED,
|
2016-06-18 21:55:46 +02:00
|
|
|
createdBy: userName,
|
|
|
|
data: {
|
|
|
|
name: featureName,
|
|
|
|
},
|
|
|
|
}))
|
2016-07-02 16:25:30 +02:00
|
|
|
.then(() => res.status(200).end())
|
2016-11-05 15:08:00 +01:00
|
|
|
.catch(error => handleErrors(req, res, error));
|
2014-12-15 22:40:07 +01:00
|
|
|
});
|
|
|
|
|
2016-07-02 11:54:50 +02:00
|
|
|
function validateUniqueName (req) {
|
2016-11-05 15:08:00 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2016-11-05 10:16:48 +01:00
|
|
|
featureToggleStore.getFeature(req.body.name)
|
2016-11-05 15:08:00 +01:00
|
|
|
.then(() => reject(new NameExistsError('Feature name already exist')))
|
|
|
|
.catch(() => resolve(req));
|
2014-12-10 18:49:08 +01:00
|
|
|
});
|
|
|
|
}
|
2016-08-23 00:04:11 +02:00
|
|
|
|
|
|
|
function validateFormat (req) {
|
|
|
|
if (req.body.strategy && req.body.strategies) {
|
2016-11-05 15:08:00 +01:00
|
|
|
return Promise.reject(new ValidationError('Cannot use both "strategy" and "strategies".'));
|
2016-08-23 00:04:11 +02:00
|
|
|
}
|
|
|
|
|
2016-11-05 15:08:00 +01:00
|
|
|
return Promise.resolve(req);
|
2016-08-23 00:04:11 +02:00
|
|
|
}
|
2016-11-15 21:27:03 +01:00
|
|
|
|
|
|
|
|
2017-01-08 20:40:50 +01:00
|
|
|
const strategiesSchema = joi.object().keys({
|
|
|
|
name: joi.string()
|
|
|
|
.regex(/^[a-zA-Z0-9\\.\\-]{3,100}$/)
|
|
|
|
.required(),
|
|
|
|
parameters: joi.object(),
|
|
|
|
});
|
|
|
|
|
|
|
|
function validateStrategy (featureToggle) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
if (!featureToggle.strategies || featureToggle.strategies.length === 0) {
|
|
|
|
return reject(new ValidationError('You must define at least one strategy'));
|
|
|
|
}
|
|
|
|
|
|
|
|
featureToggle.strategies = featureToggle.strategies.map((strategyConfig) => {
|
|
|
|
const result = joi.validate(strategyConfig, strategiesSchema);
|
|
|
|
if (result.error) {
|
|
|
|
throw result.error;
|
|
|
|
}
|
|
|
|
return result.value;
|
|
|
|
});
|
|
|
|
|
|
|
|
return resolve(featureToggle);
|
|
|
|
});
|
2016-11-15 21:27:03 +01:00
|
|
|
}
|
2014-10-20 15:12:30 +02:00
|
|
|
};
|