From bffc0483a9f32deccb5c83f8ccb070e88c9b5bbd Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Wed, 20 Mar 2024 21:43:30 -0700 Subject: [PATCH 1/6] feat(instrumentation): added synchronous gauge to SDK --- .../sdk-metrics/src/InstrumentDescriptor.ts | 1 + packages/sdk-metrics/src/Instruments.ts | 13 +++++++++++++ packages/sdk-metrics/src/Meter.ts | 19 +++++++++++++++++++ .../src/aggregator/ExponentialHistogram.ts | 1 + .../sdk-metrics/src/aggregator/Histogram.ts | 1 + packages/sdk-metrics/src/view/Aggregation.ts | 1 + .../test/export/ConsoleMetricExporter.test.ts | 1 + packages/sdk-metrics/test/export/utils.ts | 1 + 8 files changed, 38 insertions(+) diff --git a/packages/sdk-metrics/src/InstrumentDescriptor.ts b/packages/sdk-metrics/src/InstrumentDescriptor.ts index b8797e2c4d..ef09dda8bb 100644 --- a/packages/sdk-metrics/src/InstrumentDescriptor.ts +++ b/packages/sdk-metrics/src/InstrumentDescriptor.ts @@ -23,6 +23,7 @@ import { equalsCaseInsensitive } from './utils'; */ export enum InstrumentType { COUNTER = 'COUNTER', + GAUGE = 'GAUGE', HISTOGRAM = 'HISTOGRAM', UP_DOWN_COUNTER = 'UP_DOWN_COUNTER', OBSERVABLE_COUNTER = 'OBSERVABLE_COUNTER', diff --git a/packages/sdk-metrics/src/Instruments.ts b/packages/sdk-metrics/src/Instruments.ts index f665952f05..ce6a8c429a 100644 --- a/packages/sdk-metrics/src/Instruments.ts +++ b/packages/sdk-metrics/src/Instruments.ts @@ -22,6 +22,7 @@ import { ValueType, UpDownCounter, Counter, + Gauge, Histogram, Observable, ObservableCallback, @@ -110,6 +111,18 @@ export class CounterInstrument extends SyncInstrument implements Counter { } } +/** + * The class implements {@link Gauge} interface. + */ +export class GaugeInstrument extends SyncInstrument implements Gauge { + /** + * Records a measurement. + */ + record(value: number, attributes?: MetricAttributes, ctx?: Context): void { + this._record(value, attributes, ctx); + } +} + /** * The class implements {@link Histogram} interface. */ diff --git a/packages/sdk-metrics/src/Meter.ts b/packages/sdk-metrics/src/Meter.ts index b3d69d0dfd..1188ab5554 100644 --- a/packages/sdk-metrics/src/Meter.ts +++ b/packages/sdk-metrics/src/Meter.ts @@ -25,6 +25,8 @@ import { ObservableUpDownCounter, BatchObservableCallback, Observable, + Attributes, + Gauge, } from '@opentelemetry/api'; import { createInstrumentDescriptor, @@ -32,6 +34,7 @@ import { } from './InstrumentDescriptor'; import { CounterInstrument, + GaugeInstrument, HistogramInstrument, ObservableCounterInstrument, ObservableGaugeInstrument, @@ -46,6 +49,22 @@ import { MeterSharedState } from './state/MeterSharedState'; export class Meter implements IMeter { constructor(private _meterSharedState: MeterSharedState) {} + /** + * Create a {@link Gauge} instrument. + */ + createGauge( + name: string, + options?: MetricOptions + ): Gauge { + const descriptor = createInstrumentDescriptor( + name, + InstrumentType.GAUGE, + options + ); + const storage = this._meterSharedState.registerMetricStorage(descriptor); + return new GaugeInstrument(storage, descriptor); + } + /** * Create a {@link Histogram} instrument. */ diff --git a/packages/sdk-metrics/src/aggregator/ExponentialHistogram.ts b/packages/sdk-metrics/src/aggregator/ExponentialHistogram.ts index a0c438ee17..9eb296fb27 100644 --- a/packages/sdk-metrics/src/aggregator/ExponentialHistogram.ts +++ b/packages/sdk-metrics/src/aggregator/ExponentialHistogram.ts @@ -580,6 +580,7 @@ export class ExponentialHistogramAggregator // determine if instrument allows negative values. const allowsNegativeValues = + descriptor.type === InstrumentType.GAUGE || descriptor.type === InstrumentType.UP_DOWN_COUNTER || descriptor.type === InstrumentType.OBSERVABLE_GAUGE || descriptor.type === InstrumentType.OBSERVABLE_UP_DOWN_COUNTER; diff --git a/packages/sdk-metrics/src/aggregator/Histogram.ts b/packages/sdk-metrics/src/aggregator/Histogram.ts index 94cf906039..5c9390b81f 100644 --- a/packages/sdk-metrics/src/aggregator/Histogram.ts +++ b/packages/sdk-metrics/src/aggregator/Histogram.ts @@ -231,6 +231,7 @@ export class HistogramAggregator implements Aggregator { // determine if instrument allows negative values. const allowsNegativeValues = + descriptor.type === InstrumentType.GAUGE || descriptor.type === InstrumentType.UP_DOWN_COUNTER || descriptor.type === InstrumentType.OBSERVABLE_GAUGE || descriptor.type === InstrumentType.OBSERVABLE_UP_DOWN_COUNTER; diff --git a/packages/sdk-metrics/src/view/Aggregation.ts b/packages/sdk-metrics/src/view/Aggregation.ts index 2f74063ca2..f34bd1231a 100644 --- a/packages/sdk-metrics/src/view/Aggregation.ts +++ b/packages/sdk-metrics/src/view/Aggregation.ts @@ -182,6 +182,7 @@ export class DefaultAggregation extends Aggregation { case InstrumentType.OBSERVABLE_UP_DOWN_COUNTER: { return SUM_AGGREGATION; } + case InstrumentType.GAUGE: case InstrumentType.OBSERVABLE_GAUGE: { return LAST_VALUE_AGGREGATION; } diff --git a/packages/sdk-metrics/test/export/ConsoleMetricExporter.test.ts b/packages/sdk-metrics/test/export/ConsoleMetricExporter.test.ts index 6409f6aa04..258f0df00f 100644 --- a/packages/sdk-metrics/test/export/ConsoleMetricExporter.test.ts +++ b/packages/sdk-metrics/test/export/ConsoleMetricExporter.test.ts @@ -160,6 +160,7 @@ describe('ConsoleMetricExporter', () => { switch (instrumentType) { case InstrumentType.COUNTER: case InstrumentType.OBSERVABLE_COUNTER: + case InstrumentType.GAUGE: case InstrumentType.HISTOGRAM: case InstrumentType.OBSERVABLE_GAUGE: return AggregationTemporality.DELTA; diff --git a/packages/sdk-metrics/test/export/utils.ts b/packages/sdk-metrics/test/export/utils.ts index a5decfae41..a10021ee90 100644 --- a/packages/sdk-metrics/test/export/utils.ts +++ b/packages/sdk-metrics/test/export/utils.ts @@ -29,6 +29,7 @@ const instrumentTypes = [ InstrumentType.UP_DOWN_COUNTER, InstrumentType.OBSERVABLE_UP_DOWN_COUNTER, InstrumentType.HISTOGRAM, + InstrumentType.GAUGE, InstrumentType.OBSERVABLE_GAUGE, ]; From fc079f44bf2eb4703ff32b948a31414139323572 Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Sat, 23 Mar 2024 14:25:02 -0700 Subject: [PATCH 2/6] fixup! feat(instrumentation): added synchronous gauge to SDK --- packages/sdk-metrics/src/Instruments.ts | 2 +- packages/sdk-metrics/src/types.ts | 11 +++++++++++ 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/packages/sdk-metrics/src/Instruments.ts b/packages/sdk-metrics/src/Instruments.ts index ce6a8c429a..0113abd190 100644 --- a/packages/sdk-metrics/src/Instruments.ts +++ b/packages/sdk-metrics/src/Instruments.ts @@ -22,7 +22,6 @@ import { ValueType, UpDownCounter, Counter, - Gauge, Histogram, Observable, ObservableCallback, @@ -37,6 +36,7 @@ import { AsyncWritableMetricStorage, WritableMetricStorage, } from './state/WritableMetricStorage'; +import { Gauge } from './types'; export class SyncInstrument { constructor( diff --git a/packages/sdk-metrics/src/types.ts b/packages/sdk-metrics/src/types.ts index 89af1eea33..7c51c4152b 100644 --- a/packages/sdk-metrics/src/types.ts +++ b/packages/sdk-metrics/src/types.ts @@ -13,6 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ +import { Context, MetricAttributes } from '@opentelemetry/api'; export type CommonReaderOptions = { timeoutMillis?: number; @@ -23,3 +24,13 @@ export type CollectionOptions = CommonReaderOptions; export type ShutdownOptions = CommonReaderOptions; export type ForceFlushOptions = CommonReaderOptions; + +// NOTE: This is here because of the strange dependency on api? +export interface Gauge< + AttributesTypes extends MetricAttributes = MetricAttributes, +> { + /** + * Records a measurement. Value of the measurement must not be negative. + */ + record(value: number, attributes?: AttributesTypes, context?: Context): void; +} From 8bf5bfe119fa4a761f509808844e0a1b5ab67f72 Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Mon, 25 Mar 2024 08:43:36 -0700 Subject: [PATCH 3/6] fixup! feat(instrumentation): added synchronous gauge to SDK --- packages/sdk-metrics/src/Meter.ts | 2 +- packages/sdk-metrics/src/types.ts | 7 ++++++- 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/packages/sdk-metrics/src/Meter.ts b/packages/sdk-metrics/src/Meter.ts index 1188ab5554..da5ce6c005 100644 --- a/packages/sdk-metrics/src/Meter.ts +++ b/packages/sdk-metrics/src/Meter.ts @@ -26,7 +26,6 @@ import { BatchObservableCallback, Observable, Attributes, - Gauge, } from '@opentelemetry/api'; import { createInstrumentDescriptor, @@ -42,6 +41,7 @@ import { UpDownCounterInstrument, } from './Instruments'; import { MeterSharedState } from './state/MeterSharedState'; +import { Gauge } from './types'; /** * This class implements the {@link IMeter} interface. diff --git a/packages/sdk-metrics/src/types.ts b/packages/sdk-metrics/src/types.ts index 7c51c4152b..41c617ffc5 100644 --- a/packages/sdk-metrics/src/types.ts +++ b/packages/sdk-metrics/src/types.ts @@ -25,7 +25,12 @@ export type ShutdownOptions = CommonReaderOptions; export type ForceFlushOptions = CommonReaderOptions; -// NOTE: This is here because of the strange dependency on api? +/** + * @experimental + * + * This is intentionally not using the API's type as it's only available from @opentelemetry/api 1.9.0 and up. + * In SDK 2.0 we'll be able to bump the minimum API version and remove this workaround. + */ export interface Gauge< AttributesTypes extends MetricAttributes = MetricAttributes, > { From f497a292c704b4a6c50f1c353a1c11cef353143a Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Mon, 25 Mar 2024 19:17:12 -0700 Subject: [PATCH 4/6] fixup! feat(instrumentation): added synchronous gauge to SDK --- .../src/OTLPMetricExporterBase.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/experimental/packages/opentelemetry-exporter-metrics-otlp-http/src/OTLPMetricExporterBase.ts b/experimental/packages/opentelemetry-exporter-metrics-otlp-http/src/OTLPMetricExporterBase.ts index 66a5bfb596..91aaf67977 100644 --- a/experimental/packages/opentelemetry-exporter-metrics-otlp-http/src/OTLPMetricExporterBase.ts +++ b/experimental/packages/opentelemetry-exporter-metrics-otlp-http/src/OTLPMetricExporterBase.ts @@ -41,6 +41,7 @@ export const DeltaTemporalitySelector: AggregationTemporalitySelector = ( switch (instrumentType) { case InstrumentType.COUNTER: case InstrumentType.OBSERVABLE_COUNTER: + case InstrumentType.GAUGE: case InstrumentType.HISTOGRAM: case InstrumentType.OBSERVABLE_GAUGE: return AggregationTemporality.DELTA; @@ -57,6 +58,7 @@ export const LowMemoryTemporalitySelector: AggregationTemporalitySelector = ( case InstrumentType.COUNTER: case InstrumentType.HISTOGRAM: return AggregationTemporality.DELTA; + case InstrumentType.GAUGE: case InstrumentType.UP_DOWN_COUNTER: case InstrumentType.OBSERVABLE_UP_DOWN_COUNTER: case InstrumentType.OBSERVABLE_COUNTER: From 112ac6757a470d54779554ad13b3194ec6bf35dc Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Tue, 26 Mar 2024 10:20:32 -0700 Subject: [PATCH 5/6] fixup! feat(instrumentation): added synchronous gauge to SDK --- experimental/CHANGELOG.md | 1 + packages/sdk-metrics/src/Meter.ts | 1 + 2 files changed, 2 insertions(+) diff --git a/experimental/CHANGELOG.md b/experimental/CHANGELOG.md index 0b286f0bc9..7a4547ffc6 100644 --- a/experimental/CHANGELOG.md +++ b/experimental/CHANGELOG.md @@ -26,6 +26,7 @@ and AnyValueMap types [#4575](https://github.com/open-telemetry/opentelemetry-js ### :rocket: (Enhancement) +* feat(metrics): added synchronous gauge to SDK [#4565](https://github.com/open-telemetry/opentelemetry-js/pull/4565) @clintonb * feat(opentelemetry-instrumentation-xhr): optionally ignore network events [#4571](https://github.com/open-telemetry/opentelemetry-js/pull/4571/) @mustafahaddara * refactor(instr-http): use exported strings for semconv. [#4573](https://github.com/open-telemetry/opentelemetry-js/pull/4573/) @JamieDanielson * perf(instrumentation-http): remove obvious temp allocations [#4576](https://github.com/open-telemetry/opentelemetry-js/pull/4576) @Samuron diff --git a/packages/sdk-metrics/src/Meter.ts b/packages/sdk-metrics/src/Meter.ts index da5ce6c005..2af51dd82d 100644 --- a/packages/sdk-metrics/src/Meter.ts +++ b/packages/sdk-metrics/src/Meter.ts @@ -51,6 +51,7 @@ export class Meter implements IMeter { /** * Create a {@link Gauge} instrument. + * @experimental */ createGauge( name: string, From 512a359559ffba6c208bce6440fbc8772141261e Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Sun, 31 Mar 2024 09:47:52 -0700 Subject: [PATCH 6/6] fixup! feat(instrumentation): added synchronous gauge to SDK --- packages/sdk-metrics/test/Meter.test.ts | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/packages/sdk-metrics/test/Meter.test.ts b/packages/sdk-metrics/test/Meter.test.ts index e40d6aa507..875f202862 100644 --- a/packages/sdk-metrics/test/Meter.test.ts +++ b/packages/sdk-metrics/test/Meter.test.ts @@ -19,6 +19,7 @@ import * as assert from 'assert'; import * as sinon from 'sinon'; import { CounterInstrument, + GaugeInstrument, HistogramInstrument, ObservableCounterInstrument, ObservableGaugeInstrument, @@ -88,6 +89,18 @@ describe('Meter', () => { }); }); + describe('createGauge', () => { + testWithNames('Gauge', name => { + const meterSharedState = new MeterSharedState( + new MeterProviderSharedState(defaultResource), + defaultInstrumentationScope + ); + const meter = new Meter(meterSharedState); + const Gauge = meter.createGauge(name); + assert(Gauge instanceof GaugeInstrument); + }); + }); + describe('createObservableCounter', () => { testWithNames('ObservableCounter', name => { const meterSharedState = new MeterSharedState(