From 99f65d3c7c8d529484ca1ff412fb54a797106a4e Mon Sep 17 00:00:00 2001 From: Yuriy Vasiyarov Date: Mon, 17 Jun 2019 18:24:00 +0300 Subject: [PATCH] feat: implement GC metrics collection without native(C++) modules --- CHANGELOG.md | 2 ++ example/server.js | 13 +++++++ lib/defaultMetrics.js | 4 ++- lib/metrics/gc.js | 78 ++++++++++++++++++++++++++++++++++++++++++ test/metrics/gcTest.js | 49 ++++++++++++++++++++++++++ 5 files changed, 145 insertions(+), 1 deletion(-) create mode 100644 lib/metrics/gc.js create mode 100644 test/metrics/gcTest.js diff --git a/CHANGELOG.md b/CHANGELOG.md index 1fc8df70..e6e3cc64 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,6 +12,8 @@ project adheres to [Semantic Versioning](http://semver.org/). ### Changed ### Added +- `nodejs_gc_runs` metric to the `collectDefaultMetrics()`. It counts number of GC runs with split by GC type. +- `nodejs_gc_duration_summary` metric to the `collectDefaultMetrics()`. It counts 0.5, 0.75, 0.9, 0.99 percentiles of GC duration (in seconds). ## [11.5.3] - 2019-06-27 diff --git a/example/server.js b/example/server.js index 890be2a1..28036672 100644 --- a/example/server.js +++ b/example/server.js @@ -56,6 +56,19 @@ if (cluster.isWorker) { }, 2000); } +// Generate some garbage +const t = []; +setInterval(() => { + for (let i = 0; i < 100; i++) { + t.push(new Date()); + } +}, 10); +setInterval(() => { + while (t.length > 0) { + t.pop(); + } +}); + server.get('/metrics', (req, res) => { res.set('Content-Type', register.contentType); res.end(register.metrics()); diff --git a/lib/defaultMetrics.js b/lib/defaultMetrics.js index 1972b3c8..553ed24c 100644 --- a/lib/defaultMetrics.js +++ b/lib/defaultMetrics.js @@ -11,6 +11,7 @@ const processRequests = require('./metrics/processRequests'); const heapSizeAndUsed = require('./metrics/heapSizeAndUsed'); const heapSpacesSizeAndUsed = require('./metrics/heapSpacesSizeAndUsed'); const version = require('./metrics/version'); +const gc = require('./metrics/gc'); const { globalRegistry } = require('./registry'); const { printDeprecationCollectDefaultMetricsNumber } = require('./util'); @@ -25,7 +26,8 @@ const metrics = { processRequests, heapSizeAndUsed, heapSpacesSizeAndUsed, - version + version, + gc }; const metricsList = Object.keys(metrics); diff --git a/lib/metrics/gc.js b/lib/metrics/gc.js new file mode 100644 index 00000000..b7cd71bd --- /dev/null +++ b/lib/metrics/gc.js @@ -0,0 +1,78 @@ +'use strict'; +const Counter = require('../counter'); +const Summary = require('../summary'); + +let perf_hooks; + +try { + // eslint-disable-next-line + perf_hooks = require('perf_hooks'); +} catch (e) { + // node version is too old +} + +const NODEJS_GC_RUNS = 'nodejs_gc_runs'; +const NODEJS_GC_DURATION_SUMMARY = 'nodejs_gc_duration_summary'; + +function gcKindToString(gcKind) { + let gcKindName = ''; + switch (gcKind) { + case perf_hooks.constants.NODE_PERFORMANCE_GC_MAJOR: + gcKindName = 'major'; + break; + case perf_hooks.constants.NODE_PERFORMANCE_GC_MINOR: + gcKindName = 'minor'; + break; + case perf_hooks.constants.NODE_PERFORMANCE_GC_INCREMENTAL: + gcKindName = 'incremental'; + break; + case perf_hooks.constants.NODE_PERFORMANCE_GC_WEAKCB: + gcKindName = 'weakcb'; + break; + default: + gcKindName = 'unknown'; + break; + } + return gcKindName; +} + +module.exports = (registry, config = {}) => { + if (!perf_hooks) { + return () => {}; + } + + const namePrefix = config.prefix ? config.prefix : ''; + const gcCount = new Counter({ + name: namePrefix + NODEJS_GC_RUNS, + help: + 'Count of garbage collections. gc_type label is one of major, minor, incremental or weakcb.', + labelNames: ['gc_type'], + registers: registry ? [registry] : undefined + }); + const gcSummary = new Summary({ + name: namePrefix + NODEJS_GC_DURATION_SUMMARY, + help: + 'Summary of garbage collections. gc_type label is one of major, minor, incremental or weakcb.', + labelNames: ['gc_type'], + maxAgeSeconds: 600, + ageBuckets: 5, + percentiles: [0.5, 0.75, 0.9, 0.99], + registers: registry ? [registry] : undefined + }); + + const obs = new perf_hooks.PerformanceObserver(list => { + const entry = list.getEntries()[0]; + const labels = { gc_type: gcKindToString(entry.kind) }; + + gcCount.inc(labels, 1); + // Convert duration from milliseconds to seconds + gcSummary.observe(labels, entry.duration / 1000); + }); + + // We do not expect too many gc events per second, so we do not use buffering + obs.observe({ entryTypes: ['gc'], buffered: false }); + + return () => {}; +}; + +module.exports.metricNames = [NODEJS_GC_RUNS, NODEJS_GC_DURATION_SUMMARY]; diff --git a/test/metrics/gcTest.js b/test/metrics/gcTest.js new file mode 100644 index 00000000..d0bd4a77 --- /dev/null +++ b/test/metrics/gcTest.js @@ -0,0 +1,49 @@ +'use strict'; + +describe('gc', () => { + const register = require('../../index').register; + const processHandles = require('../../lib/metrics/gc'); + + beforeAll(() => { + register.clear(); + }); + + afterEach(() => { + register.clear(); + }); + + it('should add metric to the registry', () => { + expect(register.getMetricsAsJSON()).toHaveLength(0); + + processHandles()(); + + const metrics = register.getMetricsAsJSON(); + + // Check if perf_hooks module is available + let perf_hooks; + try { + // eslint-disable-next-line + perf_hooks = require('perf_hooks'); + } catch (e) { + // node version is too old + } + + if (perf_hooks) { + expect(metrics).toHaveLength(2); + + expect(metrics[0].help).toEqual( + 'Count of garbage collections. gc_type label is one of major, minor, incremental or weakcb.' + ); + expect(metrics[0].type).toEqual('counter'); + expect(metrics[0].name).toEqual('nodejs_gc_runs'); + + expect(metrics[1].help).toEqual( + 'Summary of garbage collections. gc_type label is one of major, minor, incremental or weakcb.' + ); + expect(metrics[1].type).toEqual('summary'); + expect(metrics[1].name).toEqual('nodejs_gc_duration_summary'); + } else { + expect(metrics).toHaveLength(0); + } + }); +});