2016-06-18 21:53:18 +02:00
|
|
|
'use strict';
|
2016-10-26 10:43:11 +02:00
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
const {
|
|
|
|
FEATURE_CREATED,
|
|
|
|
FEATURE_UPDATED,
|
|
|
|
FEATURE_ARCHIVED,
|
|
|
|
FEATURE_REVIVED,
|
2019-03-13 19:10:13 +01:00
|
|
|
FEATURE_IMPORT,
|
|
|
|
DROP_FEATURES,
|
2017-06-28 10:17:14 +02:00
|
|
|
} = require('../event-type');
|
2016-10-27 20:51:21 +02:00
|
|
|
const NotFoundError = require('../error/notfound-error');
|
2020-04-14 22:29:11 +02:00
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
const FEATURE_COLUMNS = [
|
|
|
|
'name',
|
|
|
|
'description',
|
2020-08-06 11:18:52 +02:00
|
|
|
'type',
|
2017-06-28 10:17:14 +02:00
|
|
|
'enabled',
|
2020-08-07 10:46:35 +02:00
|
|
|
'stale',
|
2017-06-28 10:17:14 +02:00
|
|
|
'strategies',
|
2018-01-30 15:57:12 +01:00
|
|
|
'variants',
|
2017-06-28 10:17:14 +02:00
|
|
|
'created_at',
|
|
|
|
];
|
2016-12-09 14:50:30 +01:00
|
|
|
const TABLE = 'features';
|
2014-10-23 09:50:23 +02:00
|
|
|
|
2016-11-05 10:16:48 +01:00
|
|
|
class FeatureToggleStore {
|
2019-04-30 21:14:23 +02:00
|
|
|
constructor(db, eventStore, getLogger) {
|
2016-11-05 10:16:48 +01:00
|
|
|
this.db = db;
|
2019-04-30 21:14:23 +02:00
|
|
|
this.getLogger = getLogger('client-toggle-store.js');
|
2017-06-28 10:17:14 +02:00
|
|
|
eventStore.on(FEATURE_CREATED, event =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this._createFeature(event.data),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
|
|
|
eventStore.on(FEATURE_UPDATED, event =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this._updateFeature(event.data),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
|
|
|
eventStore.on(FEATURE_ARCHIVED, event =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this._archiveFeature(event.data),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
|
|
|
eventStore.on(FEATURE_REVIVED, event =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this._reviveFeature(event.data),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
2019-03-13 19:10:13 +01:00
|
|
|
eventStore.on(FEATURE_IMPORT, event => this._importFeature(event.data));
|
|
|
|
eventStore.on(DROP_FEATURES, () => this._dropFeatures());
|
2016-11-05 10:16:48 +01:00
|
|
|
}
|
2016-05-01 22:53:09 +02:00
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
getFeatures() {
|
2016-11-05 10:16:48 +01:00
|
|
|
return this.db
|
2016-05-01 22:53:09 +02:00
|
|
|
.select(FEATURE_COLUMNS)
|
2016-12-09 14:50:30 +01:00
|
|
|
.from(TABLE)
|
2016-05-01 22:53:09 +02:00
|
|
|
.where({ archived: 0 })
|
|
|
|
.orderBy('name', 'asc')
|
2016-11-05 10:16:48 +01:00
|
|
|
.map(this.rowToFeature);
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2020-02-28 14:50:32 +01:00
|
|
|
count() {
|
|
|
|
return this.db
|
|
|
|
.count('*')
|
|
|
|
.from(TABLE)
|
|
|
|
.where({ archived: 0 })
|
|
|
|
.then(res => Number(res[0].count));
|
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
getFeature(name) {
|
2016-11-05 10:16:48 +01:00
|
|
|
return this.db
|
2016-05-01 22:53:09 +02:00
|
|
|
.first(FEATURE_COLUMNS)
|
2016-12-09 14:50:30 +01:00
|
|
|
.from(TABLE)
|
2017-06-28 17:44:14 +02:00
|
|
|
.where({ name, archived: 0 })
|
2016-11-05 10:16:48 +01:00
|
|
|
.then(this.rowToFeature);
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2018-01-20 13:28:04 +01:00
|
|
|
hasFeature(name) {
|
2017-12-18 09:08:04 +01:00
|
|
|
return this.db
|
2018-01-20 13:28:04 +01:00
|
|
|
.first('name', 'archived')
|
2017-12-18 09:08:04 +01:00
|
|
|
.from(TABLE)
|
|
|
|
.where({ name })
|
2018-01-20 13:28:04 +01:00
|
|
|
.then(row => {
|
|
|
|
if (!row) {
|
|
|
|
throw new NotFoundError('No feature toggle found');
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
name: row.name,
|
|
|
|
archived: row.archived === 1,
|
|
|
|
};
|
|
|
|
});
|
2017-12-18 09:08:04 +01:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
getArchivedFeatures() {
|
2016-11-05 10:16:48 +01:00
|
|
|
return this.db
|
2016-05-01 22:53:09 +02:00
|
|
|
.select(FEATURE_COLUMNS)
|
2016-12-09 14:50:30 +01:00
|
|
|
.from(TABLE)
|
2016-05-01 22:53:09 +02:00
|
|
|
.where({ archived: 1 })
|
|
|
|
.orderBy('name', 'asc')
|
2016-11-05 10:16:48 +01:00
|
|
|
.map(this.rowToFeature);
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
rowToFeature(row) {
|
2016-05-01 22:53:09 +02:00
|
|
|
if (!row) {
|
|
|
|
throw new NotFoundError('No feature toggle found');
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
name: row.name,
|
|
|
|
description: row.description,
|
2020-08-06 11:18:52 +02:00
|
|
|
type: row.type,
|
2016-05-01 22:53:09 +02:00
|
|
|
enabled: row.enabled > 0,
|
2020-08-07 10:46:35 +02:00
|
|
|
stale: row.stale,
|
2016-07-02 16:25:30 +02:00
|
|
|
strategies: row.strategies,
|
2018-01-30 15:57:12 +01:00
|
|
|
variants: row.variants,
|
2016-11-05 13:01:17 +01:00
|
|
|
createdAt: row.created_at,
|
2016-05-01 22:53:09 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
eventDataToRow(data) {
|
2016-05-01 22:53:09 +02:00
|
|
|
return {
|
|
|
|
name: data.name,
|
|
|
|
description: data.description,
|
2020-08-06 11:18:52 +02:00
|
|
|
type: data.type,
|
2016-05-01 22:53:09 +02:00
|
|
|
enabled: data.enabled ? 1 : 0,
|
2020-08-07 10:46:35 +02:00
|
|
|
stale: data.stale,
|
2016-07-02 11:54:50 +02:00
|
|
|
archived: data.archived ? 1 : 0,
|
2016-07-02 16:25:30 +02:00
|
|
|
strategies: JSON.stringify(data.strategies),
|
2018-01-30 15:57:12 +01:00
|
|
|
variants: data.variants ? JSON.stringify(data.variants) : null,
|
2018-02-07 21:58:53 +01:00
|
|
|
created_at: data.createdAt, // eslint-disable-line
|
2016-05-01 22:53:09 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
_createFeature(data) {
|
2016-12-09 14:50:30 +01:00
|
|
|
return this.db(TABLE)
|
2016-11-05 10:16:48 +01:00
|
|
|
.insert(this.eventDataToRow(data))
|
2017-06-28 10:17:14 +02:00
|
|
|
.catch(err =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this.logger.error('Could not insert feature, error: ', err),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
_updateFeature(data) {
|
2016-12-09 14:50:30 +01:00
|
|
|
return this.db(TABLE)
|
2016-05-01 22:53:09 +02:00
|
|
|
.where({ name: data.name })
|
2016-11-05 10:16:48 +01:00
|
|
|
.update(this.eventDataToRow(data))
|
2017-06-28 10:17:14 +02:00
|
|
|
.catch(err =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this.logger.error('Could not update feature, error: ', err),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
_archiveFeature({ name }) {
|
2016-12-09 14:54:18 +01:00
|
|
|
return this.db(TABLE)
|
|
|
|
.where({ name })
|
2017-06-28 17:44:14 +02:00
|
|
|
.update({ archived: 1, enabled: 0 })
|
|
|
|
.catch(err => {
|
2019-04-30 21:14:23 +02:00
|
|
|
this.logger.error('Could not archive feature, error: ', err);
|
2017-06-28 17:44:14 +02:00
|
|
|
});
|
2016-05-01 22:53:09 +02:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:17:14 +02:00
|
|
|
_reviveFeature({ name }) {
|
2016-12-09 14:54:18 +01:00
|
|
|
return this.db(TABLE)
|
|
|
|
.where({ name })
|
2016-05-01 22:53:09 +02:00
|
|
|
.update({ archived: 0, enabled: 0 })
|
2017-06-28 10:17:14 +02:00
|
|
|
.catch(err =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this.logger.error('Could not archive feature, error: ', err),
|
2017-06-28 10:17:14 +02:00
|
|
|
);
|
2014-11-14 16:58:05 +01:00
|
|
|
}
|
2019-03-13 19:10:13 +01:00
|
|
|
|
|
|
|
_importFeature(data) {
|
2019-03-14 17:56:02 +01:00
|
|
|
const rowData = this.eventDataToRow(data);
|
|
|
|
return this.db(TABLE)
|
|
|
|
.where({ name: rowData.name })
|
|
|
|
.update(rowData)
|
|
|
|
.then(result =>
|
2020-04-14 22:29:11 +02:00
|
|
|
result === 0 ? this.db(TABLE).insert(rowData) : result,
|
2019-03-14 17:56:02 +01:00
|
|
|
)
|
2019-03-13 19:10:13 +01:00
|
|
|
.catch(err =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this.logger.error('Could not import feature, error: ', err),
|
2019-03-13 19:10:13 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
_dropFeatures() {
|
|
|
|
return this.db(TABLE)
|
|
|
|
.delete()
|
|
|
|
.catch(err =>
|
2020-04-14 22:29:11 +02:00
|
|
|
this.logger.error('Could not drop features, error: ', err),
|
2019-03-13 19:10:13 +01:00
|
|
|
);
|
|
|
|
}
|
2017-06-28 10:17:14 +02:00
|
|
|
}
|
2016-11-05 10:16:48 +01:00
|
|
|
|
|
|
|
module.exports = FeatureToggleStore;
|