2016-10-27 13:13:51 +02:00
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
const logger = require('../logger');
|
|
|
|
|
const ClientMetrics = require('../client-metrics');
|
2016-11-09 11:16:44 +01:00
|
|
|
|
const joi = require('joi');
|
2016-11-11 15:46:59 +01:00
|
|
|
|
const { clientMetricsSchema, clientRegisterSchema } = require('./metrics-schema');
|
2016-12-01 17:15:55 +01:00
|
|
|
|
const { catchLogAndSendErrorResponse } = require('./route-utils');
|
|
|
|
|
|
2016-10-27 16:55:38 +02:00
|
|
|
|
module.exports = function (app, config) {
|
2016-11-04 23:02:55 +01:00
|
|
|
|
const {
|
2016-11-05 13:36:44 +01:00
|
|
|
|
clientMetricsStore,
|
2016-11-05 14:08:47 +01:00
|
|
|
|
clientInstanceStore,
|
2016-12-06 09:19:15 +01:00
|
|
|
|
clientApplicationsStore,
|
2016-12-13 22:43:24 +01:00
|
|
|
|
strategyStore,
|
|
|
|
|
featureToggleStore,
|
2016-11-05 14:08:47 +01:00
|
|
|
|
} = config.stores;
|
2016-11-30 23:41:57 +01:00
|
|
|
|
|
2016-11-28 17:11:11 +01:00
|
|
|
|
const metrics = new ClientMetrics(clientMetricsStore);
|
2016-12-04 14:09:37 +01:00
|
|
|
|
|
2016-11-28 17:11:11 +01:00
|
|
|
|
app.get('/client/seen-toggles', (req, res) => {
|
2016-12-05 13:26:53 +01:00
|
|
|
|
const seenAppToggles = metrics.getAppsWithToggles();
|
2016-11-28 17:11:11 +01:00
|
|
|
|
res.json(seenAppToggles);
|
2016-11-02 12:49:25 +01:00
|
|
|
|
});
|
2016-10-27 16:55:38 +02:00
|
|
|
|
|
2016-12-09 20:21:24 +01:00
|
|
|
|
app.get('/client/seen-apps', (req, res) => {
|
|
|
|
|
const seenApps = metrics.getSeenAppsPerToggle();
|
|
|
|
|
clientApplicationsStore.getApplications()
|
|
|
|
|
.then(toLookup)
|
|
|
|
|
.then(metaData => {
|
|
|
|
|
Object.keys(seenApps).forEach(key => {
|
|
|
|
|
seenApps[key] = seenApps[key].map(entry => {
|
|
|
|
|
if (metaData[entry.appName]) {
|
2016-12-09 22:03:25 +01:00
|
|
|
|
return Object.assign({}, entry, metaData[entry.appName]);
|
2016-12-09 20:21:24 +01:00
|
|
|
|
}
|
|
|
|
|
return entry;
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
res.json(seenApps);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2016-12-01 17:15:55 +01:00
|
|
|
|
app.get('/client/metrics/feature-toggles', (req, res) => {
|
2016-11-04 16:16:55 +01:00
|
|
|
|
res.json(metrics.getTogglesMetrics());
|
2016-10-27 13:13:51 +02:00
|
|
|
|
});
|
|
|
|
|
|
2016-12-04 18:08:19 +01:00
|
|
|
|
app.get('/client/metrics/feature-toggles/:name', (req, res) => {
|
|
|
|
|
const name = req.params.name;
|
|
|
|
|
const data = metrics.getTogglesMetrics();
|
|
|
|
|
const lastHour = data.lastHour[name] || {};
|
|
|
|
|
const lastMinute = data.lastMinute[name] || {};
|
|
|
|
|
res.json({
|
|
|
|
|
lastHour,
|
|
|
|
|
lastMinute,
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2016-11-02 12:49:25 +01:00
|
|
|
|
app.post('/client/metrics', (req, res) => {
|
2016-11-11 15:46:59 +01:00
|
|
|
|
const data = req.body;
|
2016-11-28 17:11:11 +01:00
|
|
|
|
const clientIp = req.ip;
|
|
|
|
|
|
2016-11-11 15:46:59 +01:00
|
|
|
|
joi.validate(data, clientMetricsSchema, (err, cleaned) => {
|
|
|
|
|
if (err) {
|
2017-06-21 11:24:36 +02:00
|
|
|
|
logger.warn('Invalid metrics posted', err);
|
2016-11-11 15:46:59 +01:00
|
|
|
|
return res.status(400).json(err);
|
2016-11-09 11:16:44 +01:00
|
|
|
|
}
|
2016-11-28 17:11:11 +01:00
|
|
|
|
|
|
|
|
|
clientMetricsStore
|
|
|
|
|
.insert(cleaned)
|
|
|
|
|
.then(() => clientInstanceStore.insert({
|
|
|
|
|
appName: cleaned.appName,
|
|
|
|
|
instanceId: cleaned.instanceId,
|
|
|
|
|
clientIp,
|
|
|
|
|
}))
|
2016-12-02 17:19:59 +01:00
|
|
|
|
.catch(err => logger.error('failed to store metrics', err));
|
2016-12-04 14:09:37 +01:00
|
|
|
|
|
2016-11-11 15:46:59 +01:00
|
|
|
|
res.status(202).end();
|
|
|
|
|
});
|
2016-11-09 11:16:44 +01:00
|
|
|
|
});
|
|
|
|
|
|
2016-11-02 12:49:25 +01:00
|
|
|
|
app.post('/client/register', (req, res) => {
|
2016-11-04 23:02:55 +01:00
|
|
|
|
const data = req.body;
|
2016-11-02 12:49:25 +01:00
|
|
|
|
|
2016-12-09 16:25:18 +01:00
|
|
|
|
joi.validate(data, clientRegisterSchema, (err, clientRegistration) => {
|
2016-11-09 11:16:44 +01:00
|
|
|
|
if (err) {
|
2017-06-21 11:24:36 +02:00
|
|
|
|
logger.warn('Invalid client data posted', err);
|
2016-11-10 22:05:50 +01:00
|
|
|
|
return res.status(400).json(err);
|
2016-11-09 11:16:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-09 16:25:18 +01:00
|
|
|
|
clientRegistration.clientIp = req.ip;
|
|
|
|
|
|
|
|
|
|
clientApplicationsStore
|
|
|
|
|
.upsert(clientRegistration)
|
|
|
|
|
.then(() => clientInstanceStore.insert(clientRegistration))
|
2016-12-04 18:08:19 +01:00
|
|
|
|
.then(() => logger.info(`New client registered with
|
2016-12-09 16:25:18 +01:00
|
|
|
|
appName=${clientRegistration.appName} and instanceId=${clientRegistration.instanceId}`))
|
2016-12-02 17:19:59 +01:00
|
|
|
|
.catch(err => logger.error('failed to register client', err));
|
2016-11-09 11:16:44 +01:00
|
|
|
|
|
2016-11-11 15:46:59 +01:00
|
|
|
|
res.status(202).end();
|
2016-11-09 11:16:44 +01:00
|
|
|
|
});
|
2016-10-27 16:55:38 +02:00
|
|
|
|
});
|
2016-11-02 23:17:28 +01:00
|
|
|
|
|
2016-12-06 09:19:15 +01:00
|
|
|
|
app.post('/client/applications/:appName', (req, res) => {
|
|
|
|
|
const input = Object.assign({}, req.body, {
|
|
|
|
|
appName: req.params.appName,
|
|
|
|
|
});
|
|
|
|
|
clientApplicationsStore
|
|
|
|
|
.upsert(input)
|
|
|
|
|
.then(() => res.status(202).end())
|
|
|
|
|
.catch((e) => {
|
|
|
|
|
logger.error(e);
|
|
|
|
|
res.status(500).end();
|
|
|
|
|
});
|
|
|
|
|
});
|
2016-12-05 22:17:40 +01:00
|
|
|
|
|
2016-12-09 20:21:24 +01:00
|
|
|
|
function toLookup (metaData) {
|
|
|
|
|
return metaData.reduce((result, entry) => {
|
|
|
|
|
result[entry.appName] = entry;
|
|
|
|
|
return result;
|
|
|
|
|
}, {});
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-06 09:19:15 +01:00
|
|
|
|
app.get('/client/applications/', (req, res) => {
|
2016-12-09 17:30:12 +01:00
|
|
|
|
clientApplicationsStore
|
|
|
|
|
.getApplications(req.query)
|
2016-12-09 22:03:25 +01:00
|
|
|
|
.then(applications => res.json({ applications }))
|
2016-12-09 17:30:12 +01:00
|
|
|
|
.catch(err => catchLogAndSendErrorResponse(err, res));
|
2016-11-02 23:17:28 +01:00
|
|
|
|
});
|
2016-11-04 23:02:55 +01:00
|
|
|
|
|
2016-11-28 17:11:11 +01:00
|
|
|
|
app.get('/client/applications/:appName', (req, res) => {
|
|
|
|
|
const appName = req.params.appName;
|
|
|
|
|
const seenToggles = metrics.getSeenTogglesByAppName(appName);
|
2016-12-09 17:30:12 +01:00
|
|
|
|
|
2016-11-28 17:11:11 +01:00
|
|
|
|
Promise.all([
|
2016-12-09 17:30:12 +01:00
|
|
|
|
clientApplicationsStore.getApplication(appName),
|
2016-12-04 14:09:37 +01:00
|
|
|
|
clientInstanceStore.getByAppName(appName),
|
2016-12-13 22:43:24 +01:00
|
|
|
|
strategyStore.getStrategies(),
|
|
|
|
|
featureToggleStore.getFeatures(),
|
2016-12-04 14:09:37 +01:00
|
|
|
|
])
|
2016-12-13 22:43:24 +01:00
|
|
|
|
.then(([application, instances, strategies, features]) => {
|
2016-12-09 17:30:12 +01:00
|
|
|
|
const appDetails = {
|
|
|
|
|
appName: application.appName,
|
|
|
|
|
createdAt: application.createdAt,
|
|
|
|
|
description: application.description,
|
|
|
|
|
url: application.url,
|
|
|
|
|
color: application.color,
|
|
|
|
|
icon: application.icon,
|
2016-12-13 22:43:24 +01:00
|
|
|
|
strategies: application.strategies.map(name => {
|
|
|
|
|
const found = strategies.find((feature) => feature.name === name);
|
|
|
|
|
if (found) {
|
|
|
|
|
return found;
|
|
|
|
|
}
|
2016-12-27 21:03:50 +01:00
|
|
|
|
return { name, notFound: true };
|
2016-12-13 22:43:24 +01:00
|
|
|
|
}),
|
2016-12-09 17:30:12 +01:00
|
|
|
|
instances,
|
2016-12-13 22:43:24 +01:00
|
|
|
|
seenToggles: seenToggles.map(name => {
|
|
|
|
|
const found = features.find((feature) => feature.name === name);
|
|
|
|
|
if (found) {
|
|
|
|
|
return found;
|
|
|
|
|
}
|
|
|
|
|
return { name, notFound: true };
|
|
|
|
|
}),
|
2016-12-09 17:30:12 +01:00
|
|
|
|
links: {
|
|
|
|
|
self: `/api/client/applications/${application.appName}`,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
res.json(appDetails);
|
|
|
|
|
})
|
|
|
|
|
.catch(err => catchLogAndSendErrorResponse(err, res));
|
2016-11-04 23:02:55 +01:00
|
|
|
|
});
|
2016-10-27 13:13:51 +02:00
|
|
|
|
};
|