1
0
mirror of https://github.com/Unleash/unleash.git synced 2024-10-18 20:09:08 +02:00
unleash.unleash/src/lib/metrics.ts

184 lines
6.0 KiB
TypeScript
Raw Normal View History

import client from 'prom-client';
import * as events from './events';
import {
FEATURE_CREATED,
FEATURE_UPDATED,
FEATURE_ARCHIVED,
FEATURE_REVIVED,
DB_POOL_UPDATE,
} from'./event-type';
const THREE_HOURS = 3 * 60 * 60 * 1000;
const ONE_MINUTE = 60 * 1000;
class MetricsMonitor {
timer: any;
constructor() {
this.timer = null;
}
startMonitoring({ serverMetrics, eventBus, stores, version }) {
if (!serverMetrics) {
return;
}
const { eventStore, clientMetricsStore, featureToggleStore } = stores;
2017-06-28 14:21:05 +02:00
client.collectDefaultMetrics();
const requestDuration = new client.Summary({
name: 'http_request_duration_milliseconds',
help: 'App response time',
labelNames: ['path', 'method', 'status'],
percentiles: [0.1, 0.5, 0.9, 0.99],
});
const dbDuration = new client.Summary({
name: 'db_query_duration_seconds',
help: 'DB query duration time',
labelNames: ['store', 'action'],
percentiles: [0.1, 0.5, 0.9, 0.99],
});
const featureToggleUpdateTotal = new client.Counter({
name: 'feature_toggle_update_total',
help: 'Number of times a toggle has been updated',
labelNames: ['toggle'],
});
const featureToggleUsageTotal = new client.Counter({
name: 'feature_toggle_usage_total',
help: 'Number of times a feature toggle has been used',
labelNames: ['toggle', 'active', 'appName'],
});
const featureTogglesTotal = new client.Gauge({
name: 'feature_toggles_total',
help: 'Number of feature toggles',
labelNames: ['version'],
});
async function collectFeatureToggleMetrics() {
featureTogglesTotal.reset();
let togglesCount;
try {
togglesCount = await featureToggleStore.count();
// eslint-disable-next-line no-empty
} catch (e) {}
togglesCount = togglesCount || 0;
featureTogglesTotal.labels(version).set(togglesCount);
}
collectFeatureToggleMetrics();
this.timer = setInterval(
() => collectFeatureToggleMetrics(),
THREE_HOURS,
).unref();
eventBus.on(
events.REQUEST_TIME,
({ path, method, time, statusCode }) => {
requestDuration.labels(path, method, statusCode).observe(time);
},
);
eventBus.on(events.DB_TIME, ({ store, action, time }) => {
dbDuration.labels(store, action).observe(time);
});
eventStore.on(FEATURE_CREATED, ({ data }) => {
featureToggleUpdateTotal.labels(data.name).inc();
});
eventStore.on(FEATURE_UPDATED, ({ data }) => {
featureToggleUpdateTotal.labels(data.name).inc();
});
eventStore.on(FEATURE_ARCHIVED, ({ data }) => {
featureToggleUpdateTotal.labels(data.name).inc();
});
eventStore.on(FEATURE_REVIVED, ({ data }) => {
featureToggleUpdateTotal.labels(data.name).inc();
});
clientMetricsStore.on('metrics', m => {
for (const entry of Object.entries(
m.bucket.toggles,
)) {
featureToggleUsageTotal
.labels(entry[0], 'true', m.appName)
.inc(entry[1]['yes']);
featureToggleUsageTotal
.labels(entry[0], 'false', m.appName)
.inc(entry[1]['no']);
}
});
this.configureDbMetrics(stores, eventBus);
}
stopMonitoring() {
clearInterval(this.timer);
}
configureDbMetrics(stores, eventBus) {
if (stores.db && stores.db.client) {
const dbPoolMin = new client.Gauge({
name: 'db_pool_min',
help: 'Minimum DB pool size',
});
dbPoolMin.set(stores.db.client.pool.min);
const dbPoolMax = new client.Gauge({
name: 'db_pool_max',
help: 'Maximum DB pool size',
});
dbPoolMax.set(stores.db.client.pool.max);
const dbPoolFree = new client.Gauge({
name: 'db_pool_free',
help: 'Current free connections in DB pool',
});
const dbPoolUsed = new client.Gauge({
name: 'db_pool_used',
help: 'Current connections in use in DB pool',
});
const dbPoolPendingCreates = new client.Gauge({
name: 'db_pool_pending_creates',
help:
'how many asynchronous create calls are running in DB pool',
});
const dbPoolPendingAcquires = new client.Gauge({
name: 'db_pool_pending_acquires',
help:
'how many acquires are waiting for a resource to be released in DB pool',
});
eventBus.on(DB_POOL_UPDATE, data => {
dbPoolFree.set(data.free);
dbPoolUsed.set(data.used);
dbPoolPendingCreates.set(data.pendingCreates);
dbPoolPendingAcquires.set(data.pendingAcquires);
});
this.registerPoolMetrics(stores.db.client.pool, eventBus);
setInterval(
() => this.registerPoolMetrics(stores.db.client.pool, eventBus),
ONE_MINUTE,
);
}
}
registerPoolMetrics(pool, eventBus) {
try {
eventBus.emit(DB_POOL_UPDATE, {
used: pool.numUsed(),
free: pool.numFree(),
pendingCreates: pool.numPendingCreates(),
pendingAcquires: pool.numPendingAcquires(),
});
// eslint-disable-next-line no-empty
} catch (e) {}
}
}
module.exports = {
createMetricsMonitor() {
return new MetricsMonitor();
},
2016-12-04 14:09:37 +01:00
};