diff --git a/packages/opentelemetry-exporter-prometheus/src/prometheus.ts b/packages/opentelemetry-exporter-prometheus/src/prometheus.ts index 27c4bcf7a19..b3a8aac621f 100644 --- a/packages/opentelemetry-exporter-prometheus/src/prometheus.ts +++ b/packages/opentelemetry-exporter-prometheus/src/prometheus.ts @@ -15,7 +15,7 @@ */ import { ExportResult } from '@opentelemetry/base'; -import { NoopLogger } from '@opentelemetry/core'; +import { NoopLogger, hrTimeToMilliseconds } from '@opentelemetry/core'; import { CounterSumAggregator, LastValue, @@ -124,27 +124,30 @@ export class PrometheusExporter implements MetricExporter { if (!metric) return; const labelKeys = record.descriptor.labelKeys; - const value = record.aggregator.value(); + const point = record.aggregator.toPoint(); if (metric instanceof Counter) { // Prometheus counter saves internal state and increments by given value. // MetricRecord value is the current state, not the delta to be incremented by. // Currently, _registerMetric creates a new counter every time the value changes, // so the increment here behaves as a set value (increment from 0) - metric.inc(this._getLabelValues(labelKeys, record.labels), value as Sum); + metric.inc( + this._getLabelValues(labelKeys, record.labels), + point.value as Sum + ); } if (metric instanceof Gauge) { if (record.aggregator instanceof CounterSumAggregator) { metric.set( this._getLabelValues(labelKeys, record.labels), - value as Sum + point.value as Sum ); } else if (record.aggregator instanceof ObserverAggregator) { metric.set( this._getLabelValues(labelKeys, record.labels), - value as LastValue, - new Date() + point.value as LastValue, + hrTimeToMilliseconds(point.timestamp) ); } } diff --git a/packages/opentelemetry-metrics/src/export/Aggregator.ts b/packages/opentelemetry-metrics/src/export/Aggregator.ts index f9edf8ef939..f7488b49c4a 100644 --- a/packages/opentelemetry-metrics/src/export/Aggregator.ts +++ b/packages/opentelemetry-metrics/src/export/Aggregator.ts @@ -14,37 +14,50 @@ * limitations under the License. */ -import { Aggregator, Distribution, LastValue, Sum } from './types'; +import { Aggregator, Distribution, Point } from './types'; +import { HrTime } from '@opentelemetry/api'; +import { hrTime } from '@opentelemetry/core'; /** Basic aggregator which calculates a Sum from individual measurements. */ export class CounterSumAggregator implements Aggregator { private _current: number = 0; + private _lastUpdateTime: HrTime = [0, 0]; update(value: number): void { this._current += value; + this._lastUpdateTime = hrTime(); } - value(): Sum { - return this._current; + toPoint(): Point { + return { + value: this._current, + timestamp: this._lastUpdateTime, + }; } } /** Basic aggregator for Observer which keeps the last recorded value. */ export class ObserverAggregator implements Aggregator { private _current: number = 0; + private _lastUpdateTime: HrTime = [0, 0]; update(value: number): void { this._current = value; + this._lastUpdateTime = hrTime(); } - value(): LastValue { - return this._current; + toPoint(): Point { + return { + value: this._current, + timestamp: this._lastUpdateTime, + }; } } /** Basic aggregator keeping all raw values (events, sum, max and min). */ export class MeasureExactAggregator implements Aggregator { private _distribution: Distribution; + private _lastUpdateTime: HrTime = [0, 0]; constructor() { this._distribution = { @@ -62,7 +75,10 @@ export class MeasureExactAggregator implements Aggregator { this._distribution.max = Math.max(this._distribution.max, value); } - value(): Distribution { - return this._distribution; + toPoint(): Point { + return { + value: this._distribution, + timestamp: this._lastUpdateTime, + }; } } diff --git a/packages/opentelemetry-metrics/src/export/ConsoleMetricExporter.ts b/packages/opentelemetry-metrics/src/export/ConsoleMetricExporter.ts index 1d848dd9686..ba0858be1cd 100644 --- a/packages/opentelemetry-metrics/src/export/ConsoleMetricExporter.ts +++ b/packages/opentelemetry-metrics/src/export/ConsoleMetricExporter.ts @@ -37,11 +37,12 @@ export class ConsoleMetricExporter implements MetricExporter { console.log(metric.labels.labels); switch (metric.descriptor.metricKind) { case MetricKind.COUNTER: - const sum = metric.aggregator.value() as Sum; + const sum = metric.aggregator.toPoint().value as Sum; console.log('value: ' + sum); break; default: - const distribution = metric.aggregator.value() as Distribution; + const distribution = metric.aggregator.toPoint() + .value as Distribution; console.log( 'min: ' + distribution.min + diff --git a/packages/opentelemetry-metrics/src/export/types.ts b/packages/opentelemetry-metrics/src/export/types.ts index b57ab562b74..32e0c077a57 100644 --- a/packages/opentelemetry-metrics/src/export/types.ts +++ b/packages/opentelemetry-metrics/src/export/types.ts @@ -14,7 +14,7 @@ * limitations under the License. */ -import { ValueType } from '@opentelemetry/api'; +import { ValueType, HrTime } from '@opentelemetry/api'; import { ExportResult } from '@opentelemetry/base'; import { LabelSet } from '../LabelSet'; @@ -76,6 +76,11 @@ export interface Aggregator { /** Updates the current with the new value. */ update(value: number): void; - /** Returns snapshot of the current value. */ - value(): Sum | Distribution; + /** Returns snapshot of the current point (value with timestamp). */ + toPoint(): Point; +} + +export interface Point { + value: Sum | LastValue | Distribution; + timestamp: HrTime; } diff --git a/packages/opentelemetry-metrics/test/Meter.test.ts b/packages/opentelemetry-metrics/test/Meter.test.ts index f0ed7b161df..a17ca889ec3 100644 --- a/packages/opentelemetry-metrics/test/Meter.test.ts +++ b/packages/opentelemetry-metrics/test/Meter.test.ts @@ -84,9 +84,9 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), 10); + assert.strictEqual(record1.aggregator.toPoint().value, 10); counter.add(10, labelSet); - assert.strictEqual(record1.aggregator.value(), 20); + assert.strictEqual(record1.aggregator.toPoint().value, 20); }); it('should return counter with resource', () => { @@ -102,9 +102,9 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), 10); + assert.strictEqual(record1.aggregator.toPoint().value, 10); boundCounter.add(10); - assert.strictEqual(record1.aggregator.value(), 20); + assert.strictEqual(record1.aggregator.toPoint().value, 20); }); it('should return the aggregator', () => { @@ -123,9 +123,9 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), 10); + assert.strictEqual(record1.aggregator.toPoint().value, 10); boundCounter.add(-100); - assert.strictEqual(record1.aggregator.value(), 10); + assert.strictEqual(record1.aggregator.toPoint().value, 10); }); it('should not add the instrument data when disabled', () => { @@ -136,7 +136,7 @@ describe('Meter', () => { boundCounter.add(10); meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), 0); + assert.strictEqual(record1.aggregator.toPoint().value, 0); }); it('should add negative value when monotonic is set to false', () => { @@ -147,7 +147,7 @@ describe('Meter', () => { boundCounter.add(-10); meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), -10); + assert.strictEqual(record1.aggregator.toPoint().value, -10); }); it('should return same instrument on same label values', () => { @@ -159,7 +159,7 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.strictEqual(record1.aggregator.value(), 20); + assert.strictEqual(record1.aggregator.toPoint().value, 20); assert.strictEqual(boundCounter, boundCounter1); }); }); @@ -214,7 +214,7 @@ describe('Meter', () => { unit: '1', valueType: ValueType.DOUBLE, }); - assert.strictEqual(record[0].aggregator.value(), 10); + assert.strictEqual(record[0].aggregator.toPoint().value, 10); }); }); @@ -319,12 +319,15 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.deepStrictEqual(record1.aggregator.value() as Distribution, { - count: 0, - max: -Infinity, - min: Infinity, - sum: 0, - }); + assert.deepStrictEqual( + record1.aggregator.toPoint().value as Distribution, + { + count: 0, + max: -Infinity, + min: Infinity, + sum: 0, + } + ); }); it('should not set the instrument data when disabled', () => { @@ -336,12 +339,15 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.deepStrictEqual(record1.aggregator.value() as Distribution, { - count: 0, - max: -Infinity, - min: Infinity, - sum: 0, - }); + assert.deepStrictEqual( + record1.aggregator.toPoint().value as Distribution, + { + count: 0, + max: -Infinity, + min: Infinity, + sum: 0, + } + ); }); it('should accept negative (and positive) values when absolute is set to false', () => { @@ -354,12 +360,15 @@ describe('Meter', () => { meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.deepStrictEqual(record1.aggregator.value() as Distribution, { - count: 2, - max: 50, - min: -10, - sum: 40, - }); + assert.deepStrictEqual( + record1.aggregator.toPoint().value as Distribution, + { + count: 2, + max: 50, + min: -10, + sum: 40, + } + ); }); it('should return same instrument on same label values', () => { @@ -370,12 +379,15 @@ describe('Meter', () => { boundMeasure2.record(100); meter.collect(); const [record1] = meter.getBatcher().checkPointSet(); - assert.deepStrictEqual(record1.aggregator.value() as Distribution, { - count: 2, - max: 100, - min: 10, - sum: 110, - }); + assert.deepStrictEqual( + record1.aggregator.toPoint().value as Distribution, + { + count: 2, + max: 100, + min: 10, + sum: 110, + } + ); assert.strictEqual(boundMeasure1, boundMeasure2); }); }); @@ -500,7 +512,7 @@ describe('Meter', () => { labelKeys: ['key'], }); assert.strictEqual(record[0].labels, labelSet); - const value = record[0].aggregator.value() as Sum; + const value = record[0].aggregator.toPoint().value as Sum; assert.strictEqual(value, 10.45); }); @@ -529,7 +541,7 @@ describe('Meter', () => { labelKeys: ['key'], }); assert.strictEqual(record[0].labels, labelSet); - const value = record[0].aggregator.value() as Sum; + const value = record[0].aggregator.toPoint().value as Sum; assert.strictEqual(value, 10); }); }); @@ -537,8 +549,14 @@ describe('Meter', () => { function ensureMetric(metric: MetricRecord) { assert.ok(metric.aggregator instanceof ObserverAggregator); - assert.ok(metric.aggregator.value() >= 0 && metric.aggregator.value() <= 1); - assert.ok(metric.aggregator.value() >= 0 && metric.aggregator.value() <= 1); + assert.ok( + metric.aggregator.toPoint().value >= 0 && + metric.aggregator.toPoint().value <= 1 + ); + assert.ok( + metric.aggregator.toPoint().value >= 0 && + metric.aggregator.toPoint().value <= 1 + ); const descriptor = metric.descriptor; assert.strictEqual(descriptor.name, 'name'); assert.strictEqual(descriptor.description, 'desc');