2017-06-28 10:20:22 +02:00
|
|
|
'use strict';
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
const Controller = require('../controller');
|
2017-06-28 10:20:22 +02:00
|
|
|
const joi = require('joi');
|
|
|
|
|
2017-08-04 16:03:15 +02:00
|
|
|
const logger = require('../../logger')('/admin-api/feature.js');
|
2017-06-28 10:20:22 +02:00
|
|
|
const {
|
|
|
|
FEATURE_CREATED,
|
|
|
|
FEATURE_UPDATED,
|
|
|
|
FEATURE_ARCHIVED,
|
|
|
|
} = require('../../event-type');
|
|
|
|
const NameExistsError = require('../../error/name-exists-error');
|
2018-12-01 12:03:47 +01:00
|
|
|
const NameInvalidError = require('../../error/name-invalid-error');
|
|
|
|
const { isUrlFriendlyName } = require('./util');
|
|
|
|
|
2017-06-28 10:20:22 +02:00
|
|
|
const extractUser = require('../../extract-user');
|
|
|
|
|
2018-12-01 12:03:47 +01:00
|
|
|
const { featureShema } = require('./feature-schema');
|
|
|
|
|
2017-06-28 10:20:22 +02:00
|
|
|
const handleErrors = (req, res, error) => {
|
|
|
|
logger.warn('Error creating or updating feature', error);
|
2018-12-01 12:03:47 +01:00
|
|
|
switch (error.name) {
|
|
|
|
case 'NotFoundError':
|
2017-06-28 10:20:22 +02:00
|
|
|
return res.status(404).end();
|
2018-12-01 12:03:47 +01:00
|
|
|
case 'NameInvalidError':
|
|
|
|
return res
|
|
|
|
.status(400)
|
|
|
|
.json([{ msg: error.message }])
|
|
|
|
.end();
|
|
|
|
case 'NameExistsError':
|
2017-06-28 10:20:22 +02:00
|
|
|
return res
|
|
|
|
.status(403)
|
2018-01-20 13:28:04 +01:00
|
|
|
.json([{ msg: error.message }])
|
2017-06-28 10:20:22 +02:00
|
|
|
.end();
|
2018-12-01 12:03:47 +01:00
|
|
|
case 'ValidationError':
|
2017-11-02 09:23:38 +01:00
|
|
|
return res
|
|
|
|
.status(400)
|
2018-12-01 12:03:47 +01:00
|
|
|
.json(error)
|
2017-11-02 09:23:38 +01:00
|
|
|
.end();
|
2017-06-28 10:20:22 +02:00
|
|
|
default:
|
|
|
|
logger.error('Server failed executing request', error);
|
|
|
|
return res.status(500).end();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const version = 1;
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
class FeatureController extends Controller {
|
|
|
|
constructor({ featureToggleStore, eventStore }) {
|
|
|
|
super();
|
|
|
|
this.featureToggleStore = featureToggleStore;
|
|
|
|
this.eventStore = eventStore;
|
|
|
|
|
|
|
|
this.get('/', this.getAllToggles);
|
|
|
|
this.post('/', this.createToggle);
|
|
|
|
this.get('/:featureName', this.getToggle);
|
|
|
|
this.put('/:featureName', this.updateToggle);
|
|
|
|
this.delete('/:featureName', this.deleteToggle);
|
|
|
|
this.post('/validate', this.validate);
|
|
|
|
this.post('/:featureName/toggle', this.toggle);
|
|
|
|
}
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
async getAllToggles(req, res) {
|
|
|
|
const features = await this.featureToggleStore.getFeatures();
|
|
|
|
res.json({ version, features });
|
|
|
|
}
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
async getToggle(req, res) {
|
|
|
|
try {
|
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const feature = await this.featureToggleStore.getFeature(
|
|
|
|
featureName
|
2017-06-28 10:20:22 +02:00
|
|
|
);
|
2018-11-30 11:11:12 +01:00
|
|
|
res.json(feature).end();
|
|
|
|
} catch (err) {
|
|
|
|
res.status(404).json({ error: 'Could not find feature' });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async validate(req, res) {
|
2018-12-01 12:03:47 +01:00
|
|
|
const name = req.body.name;
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
try {
|
2018-12-01 12:03:47 +01:00
|
|
|
await this.validateNameFormat(name);
|
|
|
|
await this.validateUniqueName(name);
|
2018-11-30 11:11:12 +01:00
|
|
|
res.status(201).end();
|
|
|
|
} catch (error) {
|
|
|
|
handleErrors(req, res, error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-01 12:03:47 +01:00
|
|
|
// TODO: cleanup this validation
|
|
|
|
async validateUniqueName(name) {
|
|
|
|
let msg;
|
|
|
|
try {
|
|
|
|
const definition = await this.featureToggleStore.hasFeature(name);
|
|
|
|
msg = definition.archived
|
|
|
|
? 'An archived toggle with that name already exist'
|
|
|
|
: 'A toggle with that name already exist';
|
|
|
|
} catch (error) {
|
|
|
|
// No conflict, everything ok!
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Interntional throw here!
|
|
|
|
throw new NameExistsError(msg);
|
2017-06-28 10:20:22 +02:00
|
|
|
}
|
|
|
|
|
2018-12-01 12:03:47 +01:00
|
|
|
// TODO: this should be part of the schema validation!
|
|
|
|
validateNameFormat(name) {
|
|
|
|
if (!isUrlFriendlyName(name)) {
|
|
|
|
throw new NameInvalidError('Name must be URL friendly');
|
|
|
|
}
|
|
|
|
}
|
2017-11-02 10:30:14 +01:00
|
|
|
|
2018-12-01 12:03:47 +01:00
|
|
|
async createToggle(req, res) {
|
|
|
|
const toggleName = req.body.name;
|
2017-06-28 10:20:22 +02:00
|
|
|
const userName = extractUser(req);
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
try {
|
2018-12-01 12:03:47 +01:00
|
|
|
await this.validateNameFormat(toggleName);
|
|
|
|
await this.validateUniqueName(toggleName);
|
|
|
|
const featureToggle = await joi.validate(req.body, featureShema);
|
2018-11-30 11:11:12 +01:00
|
|
|
await this.eventStore.store({
|
|
|
|
type: FEATURE_CREATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: featureToggle,
|
|
|
|
});
|
|
|
|
res.status(201).end();
|
|
|
|
} catch (error) {
|
|
|
|
handleErrors(req, res, error);
|
|
|
|
}
|
|
|
|
}
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
async updateToggle(req, res) {
|
2017-06-28 10:20:22 +02:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
|
|
|
const updatedFeature = req.body;
|
|
|
|
|
|
|
|
updatedFeature.name = featureName;
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
try {
|
|
|
|
await this.featureToggleStore.getFeature(featureName);
|
2018-12-01 12:03:47 +01:00
|
|
|
await joi.validate(updatedFeature, featureShema);
|
2018-11-30 11:11:12 +01:00
|
|
|
await this.eventStore.store({
|
|
|
|
type: FEATURE_UPDATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: updatedFeature,
|
|
|
|
});
|
|
|
|
res.status(200).end();
|
|
|
|
} catch (error) {
|
|
|
|
handleErrors(req, res, error);
|
|
|
|
}
|
|
|
|
}
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
async toggle(req, res) {
|
2017-06-28 10:20:22 +02:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
try {
|
|
|
|
const feature = await this.featureToggleStore.getFeature(
|
|
|
|
featureName
|
|
|
|
);
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
feature.enabled = !feature.enabled;
|
|
|
|
await this.eventStore.store({
|
|
|
|
type: FEATURE_UPDATED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: feature,
|
|
|
|
});
|
|
|
|
res.status(200).end();
|
|
|
|
} catch (error) {
|
|
|
|
handleErrors(req, res, error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async deleteToggle(req, res) {
|
2017-06-28 10:20:22 +02:00
|
|
|
const featureName = req.params.featureName;
|
|
|
|
const userName = extractUser(req);
|
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
try {
|
|
|
|
await this.featureToggleStore.getFeature(featureName);
|
|
|
|
await this.eventStore.store({
|
|
|
|
type: FEATURE_ARCHIVED,
|
|
|
|
createdBy: userName,
|
|
|
|
data: {
|
|
|
|
name: featureName,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
res.status(200).end();
|
|
|
|
} catch (error) {
|
|
|
|
handleErrors(req, res, error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-06-28 10:20:22 +02:00
|
|
|
|
2018-11-30 11:11:12 +01:00
|
|
|
module.exports = FeatureController;
|