From 7c2cc63afcf9f42ad14c45258bdfda1813e246b2 Mon Sep 17 00:00:00 2001 From: Dan Shappir Date: Tue, 6 Feb 2024 12:40:29 +0200 Subject: [PATCH 1/5] perf: significantly improve the memory usage of histogram --- lib/histogram.js | 56 +++++++++++++++++++++--------------------------- 1 file changed, 24 insertions(+), 32 deletions(-) diff --git a/lib/histogram.js b/lib/histogram.js index 22b65340..101827d3 100644 --- a/lib/histogram.js +++ b/lib/histogram.js @@ -24,7 +24,7 @@ class Histogram extends Metric { this.type = 'histogram'; this.defaultLabels = {}; this.defaultExemplarLabelSet = {}; - this.enableExemplars = false; + this.enableExemplars = config.enableExemplars; for (const label of this.labelNames) { if (label === 'le') { @@ -38,19 +38,10 @@ class Histogram extends Metric { return acc; }, {}); - if (config.enableExemplars) { - this.enableExemplars = true; - this.bucketExemplars = this.upperBounds.reduce((acc, upperBound) => { - acc[upperBound] = null; - return acc; - }, {}); - Object.freeze(this.bucketExemplars); - this.observe = this.observeWithExemplar; - } else { - this.observe = this.observeWithoutExemplar; - } + this.observe = this.enableExemplars + ? this.observeWithExemplar + : this.observeWithoutExemplar; - Object.freeze(this.bucketValues); Object.freeze(this.upperBounds); if (this.labelNames.length === 0) { @@ -58,7 +49,7 @@ class Histogram extends Metric { [hashObject({})]: createBaseValues( {}, this.bucketValues, - this.bucketExemplars, + this.enableExemplars, ), }; } @@ -79,14 +70,15 @@ class Histogram extends Metric { value, exemplarLabels = this.defaultExemplarLabelSet, } = {}) { - observe.call(this, labels === 0 ? 0 : labels || {})(value); - this.updateExemplar(labels, value, exemplarLabels); + const valueFromMap = observe.call( + this, + labels === 0 ? 0 : labels || {}, + )(value); + this.updateExemplar(valueFromMap, value, exemplarLabels); } - updateExemplar(labels, value, exemplarLabels) { - const hash = hashObject(labels, this.sortedLabelNames); + updateExemplar({ bucketExemplars }, value, exemplarLabels) { const bound = findBound(this.upperBounds, value); - const { bucketExemplars } = this.hashMap[hash]; let exemplar = bucketExemplars[bound]; if (!isObject(exemplar)) { exemplar = new Exemplar(); @@ -137,7 +129,7 @@ class Histogram extends Metric { this.hashMap[hash] = createBaseValues( labels, this.bucketValues, - this.bucketExemplars, + this.enableExemplars, ); } @@ -244,8 +236,9 @@ function observe(labels) { valueFromMap = createBaseValues( labelValuePair.labels, this.bucketValues, - this.bucketExemplars, + this.enableExemplars, ); + this.hashMap[hash] = valueFromMap; } const b = findBound(this.upperBounds, labelValuePair.value); @@ -253,23 +246,23 @@ function observe(labels) { valueFromMap.sum += labelValuePair.value; valueFromMap.count += 1; - if (Object.prototype.hasOwnProperty.call(valueFromMap.bucketValues, b)) { + if (typeof valueFromMap.bucketValues[b] === 'number') { valueFromMap.bucketValues[b] += 1; } - this.hashMap[hash] = valueFromMap; + return valueFromMap; }; } -function createBaseValues(labels, bucketValues, bucketExemplars) { +function createBaseValues(labels, bucketValues, enableExemplars) { const result = { labels, - bucketValues: { ...bucketValues }, + bucketValues: Object.create(bucketValues), sum: 0, count: 0, }; - if (bucketExemplars) { - result.bucketExemplars = { ...bucketExemplars }; + if (enableExemplars) { + result.bucketExemplars = {}; } return result; } @@ -290,16 +283,15 @@ function extractBucketValuesForExport(histogram) { const name = `${histogram.name}_bucket`; return bucketData => { let acc = 0; + const { labels, bucketValues, bucketExemplars } = bucketData; const buckets = histogram.upperBounds.map(upperBound => { - acc += bucketData.bucketValues[upperBound]; + acc += bucketValues[upperBound]; return setValuePair( { le: upperBound }, acc, name, - bucketData.bucketExemplars - ? bucketData.bucketExemplars[upperBound] - : null, - bucketData.labels, + (bucketExemplars && bucketExemplars[upperBound]) || null, + labels, ); }); return { buckets, data: bucketData }; From 25a29cd602adeb085103a690ab9c601957b768ae Mon Sep 17 00:00:00 2001 From: Dan Shappir Date: Tue, 6 Feb 2024 14:51:14 +0200 Subject: [PATCH 2/5] perf: significantly improve the memory usage of histogram (changelog) --- CHANGELOG.md | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1617de59..8c301c41 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,21 @@ project adheres to [Semantic Versioning](http://semver.org/). ### Changed +- Improve the memory usage of histograms by delaying allocation of bucket counters in `bucketValues` until their value is > 0 + (Use a prototype of the initial counters) +- Improve the memory usage of histograms by allocating empty `enableExemplars` instead of pre-filling with `null` +- Minor performance improvements: + - Uvoid unneeded hash table updates + - Avoid hash table lookup for examplars + +### Added + +## [Unreleased] + +### Breaking + +### Changed + - Improve the memory usage of histograms when the `enableExemplars` option is disabled ### Added From 2881fab8503003cb934863d047eb05b2b70812e8 Mon Sep 17 00:00:00 2001 From: Simen Bekkhus Date: Mon, 12 Feb 2024 08:56:24 +0100 Subject: [PATCH 3/5] Update CHANGELOG.md --- CHANGELOG.md | 9 --------- 1 file changed, 9 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8c301c41..712a299d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,15 +17,6 @@ project adheres to [Semantic Versioning](http://semver.org/). - Minor performance improvements: - Uvoid unneeded hash table updates - Avoid hash table lookup for examplars - -### Added - -## [Unreleased] - -### Breaking - -### Changed - - Improve the memory usage of histograms when the `enableExemplars` option is disabled ### Added From 81b54400df80577cc1de7a48369c22205afa768f Mon Sep 17 00:00:00 2001 From: Dan Shappir Date: Mon, 12 Feb 2024 11:52:22 +0200 Subject: [PATCH 4/5] Update lib/histogram.js Co-authored-by: Simen Bekkhus --- lib/histogram.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/histogram.js b/lib/histogram.js index 101827d3..e5b2c611 100644 --- a/lib/histogram.js +++ b/lib/histogram.js @@ -290,7 +290,7 @@ function extractBucketValuesForExport(histogram) { { le: upperBound }, acc, name, - (bucketExemplars && bucketExemplars[upperBound]) || null, + bucketExemplars?.[upperBound] ?? null, labels, ); }); From 99281be6805d161d2556f3bb2d3f3a9c1581c4db Mon Sep 17 00:00:00 2001 From: Dan Shappir Date: Mon, 12 Feb 2024 11:57:53 +0200 Subject: [PATCH 5/5] pref: use optional chaining --- .eslintrc | 2 +- lib/histogram.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.eslintrc b/.eslintrc index 067ab963..5055dbf3 100644 --- a/.eslintrc +++ b/.eslintrc @@ -10,7 +10,7 @@ "es6": true }, "parserOptions": { - "ecmaVersion": 2019 + "ecmaVersion": 2020 }, "rules": { "no-underscore-dangle": "off", diff --git a/lib/histogram.js b/lib/histogram.js index 101827d3..e5b2c611 100644 --- a/lib/histogram.js +++ b/lib/histogram.js @@ -290,7 +290,7 @@ function extractBucketValuesForExport(histogram) { { le: upperBound }, acc, name, - (bucketExemplars && bucketExemplars[upperBound]) || null, + bucketExemplars?.[upperBound] ?? null, labels, ); });