1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-01 00:08:27 +01:00
unleash.unleash/lib/routes/admin-api/feature.test.js

361 lines
9.9 KiB
JavaScript
Raw Normal View History

2016-08-23 01:29:47 +02:00
'use strict';
2018-12-17 09:24:49 +01:00
const test = require('ava');
const supertest = require('supertest');
const { EventEmitter } = require('events');
const store = require('../../../test/fixtures/store');
const permissions = require('../../../test/fixtures/permissions');
const getLogger = require('../../../test/fixtures/no-logger');
const getApp = require('../../app');
const { UPDATE_FEATURE, CREATE_FEATURE } = require('../../permissions');
2016-11-11 15:46:59 +01:00
const eventBus = new EventEmitter();
function getSetup() {
2016-11-13 15:41:35 +01:00
const base = `/random${Math.round(Math.random() * 1000)}`;
const stores = store.createStores();
const perms = permissions();
const app = getApp({
2016-11-13 15:41:35 +01:00
baseUriPath: base,
stores,
eventBus,
extendedPermissions: true,
preRouterHook: perms.hook,
getLogger,
2016-11-13 15:41:35 +01:00
});
2016-08-23 01:29:47 +02:00
2016-11-13 15:41:35 +01:00
return {
base,
perms,
2016-11-13 15:41:35 +01:00
featureToggleStore: stores.featureToggleStore,
request: supertest(app),
};
}
test('should get empty getFeatures via admin', t => {
t.plan(1);
2016-11-13 15:41:35 +01:00
const { request, base } = getSetup();
return request
.get(`${base}/api/admin/features`)
2016-11-13 15:41:35 +01:00
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
2016-11-13 15:41:35 +01:00
t.true(res.body.features.length === 0);
2016-08-23 01:29:47 +02:00
});
2016-11-13 15:41:35 +01:00
});
2016-08-23 01:29:47 +02:00
2016-11-13 15:41:35 +01:00
test('should get one getFeature', t => {
t.plan(1);
2016-11-13 15:41:35 +01:00
const { request, featureToggleStore, base } = getSetup();
featureToggleStore.addFeature({
name: 'test_',
strategies: [{ name: 'default_' }],
});
2016-08-23 01:29:47 +02:00
2016-11-13 15:41:35 +01:00
return request
.get(`${base}/api/admin/features`)
2016-11-13 15:41:35 +01:00
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
2016-11-13 15:41:35 +01:00
t.true(res.body.features.length === 1);
});
});
2016-08-23 01:29:47 +02:00
2016-11-13 15:41:35 +01:00
test('should add version numbers for /features', t => {
t.plan(1);
2016-11-13 15:41:35 +01:00
const { request, featureToggleStore, base } = getSetup();
featureToggleStore.addFeature({
name: 'test2',
strategies: [{ name: 'default' }],
});
2016-11-13 15:41:35 +01:00
return request
.get(`${base}/api/admin/features`)
2016-11-13 15:41:35 +01:00
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
2016-11-13 15:41:35 +01:00
t.true(res.body.version === 1);
});
2016-08-23 01:29:47 +02:00
});
2016-11-13 15:41:35 +01:00
test('should require at least one strategy when creating a feature toggle', t => {
t.plan(0);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
return request
.post(`${base}/api/admin/features`)
.send({ name: 'sample.missing.strategy' })
.set('Content-Type', 'application/json')
.expect(400);
});
test('should be allowed to use new toggle name', t => {
t.plan(0);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
return request
.post(`${base}/api/admin/features/validate`)
.send({ name: 'new.name' })
.set('Content-Type', 'application/json')
.expect(201);
});
test('should be allowed to have variants="null"', t => {
t.plan(0);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
return request
.post(`${base}/api/admin/features`)
.send({
name: 'new.name.null',
enabled: false,
strategies: [{ name: 'default' }],
variants: null,
})
.set('Content-Type', 'application/json')
.expect(201);
});
test('should not be allowed to reuse active toggle name', t => {
t.plan(1);
const { request, featureToggleStore, base } = getSetup();
featureToggleStore.addFeature({
name: 'ts',
strategies: [{ name: 'default' }],
});
return request
.post(`${base}/api/admin/features/validate`)
.send({ name: 'ts' })
.set('Content-Type', 'application/json')
2020-09-25 22:31:35 +02:00
.expect(409)
.expect(res => {
t.true(
res.body.details[0].message ===
'A toggle with that name already exist',
);
});
});
test('should not be allowed to reuse archived toggle name', t => {
t.plan(1);
const { request, featureToggleStore, base } = getSetup();
featureToggleStore.addArchivedFeature({
name: 'ts.archived',
strategies: [{ name: 'default' }],
});
return request
.post(`${base}/api/admin/features/validate`)
.send({ name: 'ts.archived' })
.set('Content-Type', 'application/json')
2020-09-25 22:31:35 +02:00
.expect(409)
.expect(res => {
t.true(
res.body.details[0].message ===
'An archived toggle with that name already exist',
);
});
});
test('should require at least one strategy when updating a feature toggle', t => {
t.plan(0);
const { request, featureToggleStore, base, perms } = getSetup();
perms.withPermissions(UPDATE_FEATURE);
featureToggleStore.addFeature({
name: 'ts',
strategies: [{ name: 'default' }],
});
return request
.put(`${base}/api/admin/features/ts`)
.send({ name: 'ts' })
.set('Content-Type', 'application/json')
.expect(400);
});
2017-10-23 17:13:26 +02:00
2017-11-02 10:30:14 +01:00
test('valid feature names should pass validation', t => {
2017-10-23 17:13:26 +02:00
t.plan(0);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
2017-10-23 17:13:26 +02:00
const validNames = [
'com.example',
'com.exampleFeature',
'com.example-company.feature',
'com.example-company.exampleFeature',
'123',
'com.example-company.someFeature.123',
];
return Promise.all(
validNames.map(name =>
request
.post(`${base}/api/admin/features`)
.send({
name,
enabled: false,
strategies: [{ name: 'default' }],
})
.set('Content-Type', 'application/json')
.expect(201),
),
2017-10-23 17:13:26 +02:00
);
});
2017-11-02 10:30:14 +01:00
test('invalid feature names should not pass validation', t => {
t.plan(0);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
2017-11-02 10:30:14 +01:00
const invalidNames = [
'some example',
'some$example',
'me&me',
' ',
'o2%ae',
];
return Promise.all(
invalidNames.map(name =>
request
.post(`${base}/api/admin/features`)
.send({
name,
enabled: false,
strategies: [{ name: 'default' }],
})
.set('Content-Type', 'application/json')
.expect(400),
),
2017-11-02 10:30:14 +01:00
);
});
// Make sure current UI works. Should align on joi errors in future.
test('invalid feature names should have error msg', t => {
t.plan(1);
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
const name = 'ØÆ`';
return request
.post(`${base}/api/admin/features`)
.send({
name,
enabled: false,
strategies: [{ name: 'default' }],
})
.set('Content-Type', 'application/json')
.expect(400)
.expect(res => {
t.true(
res.body.details[0].message === '"name" must be URL friendly',
);
});
});
2019-01-24 08:39:21 +01:00
test('should not allow variants with same name when creating feature flag', t => {
t.plan(0);
2019-01-24 11:26:07 +01:00
const { request, base, perms } = getSetup();
perms.withPermissions(CREATE_FEATURE);
2019-01-24 08:39:21 +01:00
return request
.post(`${base}/api/admin/features`)
.send({
2019-01-24 11:26:07 +01:00
name: 't.variant',
enabled: true,
2019-01-24 08:39:21 +01:00
strategies: [{ name: 'default' }],
2019-01-24 11:26:07 +01:00
variants: [
{ name: 'variant1', weight: 50 },
{ name: 'variant1', weight: 50 },
],
2019-01-24 08:39:21 +01:00
})
.set('Content-Type', 'application/json')
2019-01-24 11:26:07 +01:00
.expect(400);
2019-01-24 08:39:21 +01:00
});
test('should not allow variants with same name when updating feature flag', t => {
t.plan(0);
2019-01-24 11:26:07 +01:00
const { request, featureToggleStore, base, perms } = getSetup();
perms.withPermissions(UPDATE_FEATURE);
2019-01-24 08:39:21 +01:00
featureToggleStore.addFeature({
name: 'ts',
strategies: [{ name: 'default' }],
});
return request
.put(`${base}/api/admin/features/ts`)
.send({
name: 'ts',
strategies: [{ name: 'default' }],
variants: [{ name: 'variant1' }, { name: 'variant1' }],
})
.set('Content-Type', 'application/json')
2019-01-24 11:26:07 +01:00
.expect(400);
2019-01-24 08:39:21 +01:00
});
test('should toggle on', t => {
t.plan(1);
const { request, featureToggleStore, base, perms } = getSetup();
perms.withPermissions(UPDATE_FEATURE);
featureToggleStore.addFeature({
name: 'toggle.disabled',
enabled: false,
strategies: [{ name: 'default' }],
});
return request
.post(`${base}/api/admin/features/toggle.disabled/toggle/on`)
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
t.true(res.body.enabled === true);
});
});
test('should toggle off', t => {
t.plan(1);
const { request, featureToggleStore, base, perms } = getSetup();
perms.withPermissions(UPDATE_FEATURE);
featureToggleStore.addFeature({
name: 'toggle.enabled',
enabled: true,
strategies: [{ name: 'default' }],
});
return request
.post(`${base}/api/admin/features/toggle.enabled/toggle/off`)
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
t.true(res.body.enabled === false);
});
});
test('should toggle', t => {
t.plan(1);
const { request, featureToggleStore, base, perms } = getSetup();
perms.withPermissions(UPDATE_FEATURE);
featureToggleStore.addFeature({
name: 'toggle.disabled',
enabled: false,
strategies: [{ name: 'default' }],
});
return request
.post(`${base}/api/admin/features/toggle.disabled/toggle`)
.expect('Content-Type', /json/)
.expect(200)
.expect(res => {
t.true(res.body.enabled === true);
});
});