All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.opentelemetry.sdk.metrics.internal.state.SdkObservableMeasurement Maven / Gradle / Ivy

There is a newer version: 1.44.1
Show newest version
/*
 * Copyright The OpenTelemetry Authors
 * SPDX-License-Identifier: Apache-2.0
 */

package io.opentelemetry.sdk.metrics.internal.state;

import static io.opentelemetry.sdk.metrics.internal.state.ImmutableMeasurement.createDouble;
import static io.opentelemetry.sdk.metrics.internal.state.ImmutableMeasurement.createLong;

import io.opentelemetry.api.common.Attributes;
import io.opentelemetry.api.metrics.ObservableDoubleMeasurement;
import io.opentelemetry.api.metrics.ObservableLongMeasurement;
import io.opentelemetry.sdk.common.InstrumentationScopeInfo;
import io.opentelemetry.sdk.common.export.MemoryMode;
import io.opentelemetry.sdk.internal.ThrottlingLogger;
import io.opentelemetry.sdk.metrics.internal.descriptor.InstrumentDescriptor;
import io.opentelemetry.sdk.metrics.internal.export.RegisteredReader;
import java.util.List;
import java.util.Objects;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.annotation.Nullable;

/**
 * Records values from asynchronous instruments to associated {@link AsynchronousMetricStorage}.
 *
 * 

This class is internal and is hence not for public use. Its APIs are unstable and can change * at any time. */ public final class SdkObservableMeasurement implements ObservableLongMeasurement, ObservableDoubleMeasurement { private static final Logger logger = Logger.getLogger(SdkObservableMeasurement.class.getName()); private final ThrottlingLogger throttlingLogger = new ThrottlingLogger(logger); private final InstrumentationScopeInfo instrumentationScopeInfo; private final InstrumentDescriptor instrumentDescriptor; private final List> storages; /** Only used when {@code activeReader}'s memoryMode is {@link MemoryMode#REUSABLE_DATA}. */ private final MutableMeasurement mutableMeasurement = new MutableMeasurement(); // These fields are set before invoking callbacks. They allow measurements to be recorded to the // storages for correct reader, and with the correct time. @Nullable private volatile RegisteredReader activeReader; private volatile long startEpochNanos; private volatile long epochNanos; private SdkObservableMeasurement( InstrumentationScopeInfo instrumentationScopeInfo, InstrumentDescriptor instrumentDescriptor, List> storages) { this.instrumentationScopeInfo = instrumentationScopeInfo; this.instrumentDescriptor = instrumentDescriptor; this.storages = storages; } /** * Create a {@link SdkObservableMeasurement}. * * @param instrumentationScopeInfo the instrumentation scope info of corresponding meter * @param instrumentDescriptor the instrument descriptor * @param storages the storages to record to * @return the observable measurement */ public static SdkObservableMeasurement create( InstrumentationScopeInfo instrumentationScopeInfo, InstrumentDescriptor instrumentDescriptor, List> storages) { return new SdkObservableMeasurement(instrumentationScopeInfo, instrumentDescriptor, storages); } /** Get the instrumentation scope info. */ public InstrumentationScopeInfo getInstrumentationScopeInfo() { return instrumentationScopeInfo; } /** * Set the active reader, and clock information. {@link #unsetActiveReader()} MUST be called * after. */ public void setActiveReader( RegisteredReader registeredReader, long startEpochNanos, long epochNanos) { this.activeReader = registeredReader; this.startEpochNanos = startEpochNanos; this.epochNanos = epochNanos; } /** * Unset the active reader. Called after {@link #setActiveReader(RegisteredReader, long, long)}. */ public void unsetActiveReader() { this.activeReader = null; } InstrumentDescriptor getInstrumentDescriptor() { return instrumentDescriptor; } List> getStorages() { return storages; } @Override public void record(long value) { record(value, Attributes.empty()); } @Override public void record(long value, Attributes attributes) { if (activeReader == null) { logNoActiveReader(); return; } Measurement measurement; MemoryMode memoryMode = activeReader.getReader().getMemoryMode(); if (Objects.requireNonNull(memoryMode) == MemoryMode.IMMUTABLE_DATA) { measurement = createLong(startEpochNanos, epochNanos, value, attributes); } else { MutableMeasurement.setLongMeasurement( mutableMeasurement, startEpochNanos, epochNanos, value, attributes); measurement = mutableMeasurement; } doRecord(measurement); } @Override public void record(double value) { record(value, Attributes.empty()); } @Override public void record(double value, Attributes attributes) { if (activeReader == null) { logNoActiveReader(); return; } if (Double.isNaN(value)) { logger.log( Level.FINE, "Instrument " + instrumentDescriptor.getName() + " has recorded measurement Not-a-Number (NaN) value with attributes " + attributes + ". Dropping measurement."); return; } Measurement measurement; MemoryMode memoryMode = activeReader.getReader().getMemoryMode(); if (Objects.requireNonNull(memoryMode) == MemoryMode.IMMUTABLE_DATA) { measurement = createDouble(startEpochNanos, epochNanos, value, attributes); } else { MutableMeasurement.setDoubleMeasurement( mutableMeasurement, startEpochNanos, epochNanos, value, attributes); measurement = mutableMeasurement; } doRecord(measurement); } private void doRecord(Measurement measurement) { RegisteredReader activeReader = this.activeReader; for (AsynchronousMetricStorage storage : storages) { if (storage.getRegisteredReader().equals(activeReader)) { storage.record(measurement); } } } private void logNoActiveReader() { throttlingLogger.log( Level.FINE, "Measurement recorded for instrument " + instrumentDescriptor.getName() + " outside callback registered to instrument. Dropping measurement."); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy