From 183a1d20fc3b5ce2914bd0e97890ff9d81704b83 Mon Sep 17 00:00:00 2001 From: Marc Pichler Date: Thu, 1 Feb 2024 12:48:40 +0100 Subject: [PATCH 1/3] fix(sdk-metrics): allow single bucket histograms --- CHANGELOG.md | 5 + packages/sdk-metrics/src/view/Aggregation.ts | 6 +- packages/sdk-metrics/test/Instruments.test.ts | 14 +++ .../test/aggregator/Histogram.test.ts | 92 +++++++++++++++++++ .../sdk-metrics/test/view/Aggregation.test.ts | 7 ++ 5 files changed, 122 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9b57195513f..7955d35f763 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,8 +11,13 @@ For experimental package changes, see the [experimental CHANGELOG](experimental/ ### :rocket: (Enhancement) +* feat(sdk-metrics): allow single bucket histograms [#4456](https://github.com/open-telemetry/opentelemetry-js/pull/4456) @pichlermarc + ### :bug: (Bug Fix) +* fix(sdk-metrics): allow single bucket histograms [#4456](https://github.com/open-telemetry/opentelemetry-js/pull/4456) @pichlermarc + * fixes a bug where `Meter.createHistogram()` with the advice `explicitBucketBoundaries: []` would throw + ### :books: (Refine Doc) ### :house: (Internal) diff --git a/packages/sdk-metrics/src/view/Aggregation.ts b/packages/sdk-metrics/src/view/Aggregation.ts index 6edf50c495a..e7be4d98191 100644 --- a/packages/sdk-metrics/src/view/Aggregation.ts +++ b/packages/sdk-metrics/src/view/Aggregation.ts @@ -130,8 +130,10 @@ export class ExplicitBucketHistogramAggregation extends Aggregation { private readonly _recordMinMax = true ) { super(); - if (boundaries === undefined || boundaries.length === 0) { - throw new Error('HistogramAggregator should be created with boundaries.'); + if (boundaries === undefined) { + throw new Error( + 'ExplicitBucketHistogramAggregation should be created with explicit boundaries, if a single bucket histogram is required, please pass an empty array' + ); } // Copy the boundaries array for modification. boundaries = boundaries.concat(); diff --git a/packages/sdk-metrics/test/Instruments.test.ts b/packages/sdk-metrics/test/Instruments.test.ts index ab466c295a1..62bb98390a7 100644 --- a/packages/sdk-metrics/test/Instruments.test.ts +++ b/packages/sdk-metrics/test/Instruments.test.ts @@ -406,6 +406,20 @@ describe('Instruments', () => { }); }); + it('should allow metric advice with empty explicit boundaries', function () { + const meter = new MeterProvider({ + readers: [new TestMetricReader()], + }).getMeter('meter'); + + assert.doesNotThrow(() => { + meter.createHistogram('histogram', { + advice: { + explicitBucketBoundaries: [], + }, + }); + }); + }); + it('should collect min and max', async () => { const { meter, deltaReader, cumulativeReader } = setup(); const histogram = meter.createHistogram('test', { diff --git a/packages/sdk-metrics/test/aggregator/Histogram.test.ts b/packages/sdk-metrics/test/aggregator/Histogram.test.ts index 5d03477d3ee..cab38843a61 100644 --- a/packages/sdk-metrics/test/aggregator/Histogram.test.ts +++ b/packages/sdk-metrics/test/aggregator/Histogram.test.ts @@ -81,6 +81,27 @@ describe('HistogramAggregator', () => { sum: -65, }); }); + + it('with single bucket', function () { + const aggregator = new HistogramAggregator([0], true); + const prev = aggregator.createAccumulation([0, 0]); + prev.record(0); + prev.record(1); + + const delta = aggregator.createAccumulation([1, 1]); + delta.record(2); + delta.record(11); + + const expected = aggregator.createAccumulation([0, 0]); + // replay actions on prev + expected.record(0); + expected.record(1); + // replay actions on delta + expected.record(2); + expected.record(11); + + assert.deepStrictEqual(aggregator.merge(prev, delta), expected); + }); }); describe('diff', () => { @@ -112,6 +133,35 @@ describe('HistogramAggregator', () => { assert.deepStrictEqual(aggregator.diff(prev, curr), expected); }); + + it('with single bucket', function () { + const aggregator = new HistogramAggregator([], true); + const prev = aggregator.createAccumulation([0, 0]); + prev.record(0); + prev.record(1); + + const curr = aggregator.createAccumulation([1, 1]); + // replay actions on prev + curr.record(0); + curr.record(1); + // perform new actions + curr.record(2); + curr.record(11); + + const expected = new HistogramAccumulation([1, 1], [], true, { + buckets: { + boundaries: [], + counts: [2], + }, + count: 2, + sum: 13, + hasMinMax: false, + min: Infinity, + max: -Infinity, + }); + + assert.deepStrictEqual(aggregator.diff(prev, curr), expected); + }); }); describe('toMetricData', () => { @@ -199,6 +249,48 @@ describe('HistogramAggregator', () => { ); }); + it('should transform to expected data with empty boundaries', () => { + const aggregator = new HistogramAggregator([], false); + + const startTime: HrTime = [0, 0]; + const endTime: HrTime = [1, 1]; + const accumulation = aggregator.createAccumulation(startTime); + accumulation.record(0); + accumulation.record(1); + + const expected: MetricData = { + descriptor: defaultInstrumentDescriptor, + aggregationTemporality: AggregationTemporality.CUMULATIVE, + dataPointType: DataPointType.HISTOGRAM, + dataPoints: [ + { + attributes: {}, + startTime, + endTime, + value: { + buckets: { + boundaries: [], + counts: [2], + }, + count: 2, + sum: 1, + min: undefined, + max: undefined, + }, + }, + ], + }; + assert.deepStrictEqual( + aggregator.toMetricData( + defaultInstrumentDescriptor, + AggregationTemporality.CUMULATIVE, + [[{}, accumulation]], + endTime + ), + expected + ); + }); + function testSum(instrumentType: InstrumentType, expectSum: boolean) { const aggregator = new HistogramAggregator([1, 10, 100], true); diff --git a/packages/sdk-metrics/test/view/Aggregation.test.ts b/packages/sdk-metrics/test/view/Aggregation.test.ts index 0da2c09bee5..ec3f421ae79 100644 --- a/packages/sdk-metrics/test/view/Aggregation.test.ts +++ b/packages/sdk-metrics/test/view/Aggregation.test.ts @@ -158,6 +158,13 @@ describe('ExplicitBucketHistogramAggregation', () => { } }); + it('construct with empty boundaries', function () { + const boundaries: number[] = []; + const aggregation = new ExplicitBucketHistogramAggregation(boundaries); + assert.ok(aggregation instanceof ExplicitBucketHistogramAggregation); + assert.deepStrictEqual(aggregation['_boundaries'], []); + }); + it('constructor should not modify inputs', () => { const boundaries = [100, 10, 1]; const aggregation = new ExplicitBucketHistogramAggregation(boundaries); From e80b30ec560c1a8deefcab9fcedef3d23d8e84b7 Mon Sep 17 00:00:00 2001 From: Marc Pichler Date: Thu, 1 Feb 2024 13:38:43 +0100 Subject: [PATCH 2/3] test(sdk-metrics): undefined and null inputs for bucket boundaries --- packages/sdk-metrics/src/view/Aggregation.ts | 2 +- packages/sdk-metrics/test/view/Aggregation.test.ts | 12 ++++++++++++ 2 files changed, 13 insertions(+), 1 deletion(-) diff --git a/packages/sdk-metrics/src/view/Aggregation.ts b/packages/sdk-metrics/src/view/Aggregation.ts index e7be4d98191..2f74063ca2b 100644 --- a/packages/sdk-metrics/src/view/Aggregation.ts +++ b/packages/sdk-metrics/src/view/Aggregation.ts @@ -130,7 +130,7 @@ export class ExplicitBucketHistogramAggregation extends Aggregation { private readonly _recordMinMax = true ) { super(); - if (boundaries === undefined) { + if (boundaries == null) { throw new Error( 'ExplicitBucketHistogramAggregation should be created with explicit boundaries, if a single bucket histogram is required, please pass an empty array' ); diff --git a/packages/sdk-metrics/test/view/Aggregation.test.ts b/packages/sdk-metrics/test/view/Aggregation.test.ts index ec3f421ae79..51595aa5907 100644 --- a/packages/sdk-metrics/test/view/Aggregation.test.ts +++ b/packages/sdk-metrics/test/view/Aggregation.test.ts @@ -165,6 +165,18 @@ describe('ExplicitBucketHistogramAggregation', () => { assert.deepStrictEqual(aggregation['_boundaries'], []); }); + it('construct with undefined boundaries should throw', function () { + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-ignore simulate how a JS user could pass undefined + assert.throws(() => new ExplicitBucketHistogramAggregation(undefined)); + }); + + it('construct with null boundaries should throw', function () { + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-ignore simulate how a JS user could pass null + assert.throws(() => new ExplicitBucketHistogramAggregation(null)); + }); + it('constructor should not modify inputs', () => { const boundaries = [100, 10, 1]; const aggregation = new ExplicitBucketHistogramAggregation(boundaries); From 1641bd70d95b2c93c28ccc68d02abbcc42fe8716 Mon Sep 17 00:00:00 2001 From: Marc Pichler Date: Mon, 5 Feb 2024 14:13:13 +0100 Subject: [PATCH 3/3] fixup! test(sdk-metrics): undefined and null inputs for bucket boundaries --- .../test/aggregator/Histogram.test.ts | 21 +++++++++++-------- 1 file changed, 12 insertions(+), 9 deletions(-) diff --git a/packages/sdk-metrics/test/aggregator/Histogram.test.ts b/packages/sdk-metrics/test/aggregator/Histogram.test.ts index cab38843a61..b9facd8b351 100644 --- a/packages/sdk-metrics/test/aggregator/Histogram.test.ts +++ b/packages/sdk-metrics/test/aggregator/Histogram.test.ts @@ -83,7 +83,7 @@ describe('HistogramAggregator', () => { }); it('with single bucket', function () { - const aggregator = new HistogramAggregator([0], true); + const aggregator = new HistogramAggregator([], true); const prev = aggregator.createAccumulation([0, 0]); prev.record(0); prev.record(1); @@ -92,14 +92,17 @@ describe('HistogramAggregator', () => { delta.record(2); delta.record(11); - const expected = aggregator.createAccumulation([0, 0]); - // replay actions on prev - expected.record(0); - expected.record(1); - // replay actions on delta - expected.record(2); - expected.record(11); - + const expected = new HistogramAccumulation([0, 0], [], true, { + buckets: { + boundaries: [], + counts: [4], + }, + count: 4, + sum: 14, + hasMinMax: true, + min: 0, + max: 11, + }); assert.deepStrictEqual(aggregator.merge(prev, delta), expected); }); });