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

io.grpc.opentelemetry.OpenTelemetryMetricSink Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2024 The gRPC Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package io.grpc.opentelemetry;

import static com.google.common.base.Preconditions.checkNotNull;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import io.grpc.CallbackMetricInstrument;
import io.grpc.DoubleCounterMetricInstrument;
import io.grpc.DoubleHistogramMetricInstrument;
import io.grpc.LongCounterMetricInstrument;
import io.grpc.LongGaugeMetricInstrument;
import io.grpc.LongHistogramMetricInstrument;
import io.grpc.MetricInstrument;
import io.grpc.MetricSink;
import io.opentelemetry.api.common.Attributes;
import io.opentelemetry.api.common.AttributesBuilder;
import io.opentelemetry.api.metrics.BatchCallback;
import io.opentelemetry.api.metrics.DoubleCounter;
import io.opentelemetry.api.metrics.DoubleHistogram;
import io.opentelemetry.api.metrics.LongCounter;
import io.opentelemetry.api.metrics.LongHistogram;
import io.opentelemetry.api.metrics.Meter;
import io.opentelemetry.api.metrics.ObservableLongMeasurement;
import io.opentelemetry.api.metrics.ObservableMeasurement;
import java.util.ArrayList;
import java.util.BitSet;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;

final class OpenTelemetryMetricSink implements MetricSink {
  private static final Logger logger = Logger.getLogger(OpenTelemetryMetricSink.class.getName());
  private final Object lock = new Object();
  private final Meter openTelemetryMeter;
  private final Map enableMetrics;
  private final boolean disableDefaultMetrics;
  private final Set optionalLabels;
  private volatile List measures = new ArrayList<>();

  OpenTelemetryMetricSink(Meter meter, Map enableMetrics,
      boolean disableDefaultMetrics, List optionalLabels) {
    this.openTelemetryMeter = checkNotNull(meter, "meter");
    this.enableMetrics = ImmutableMap.copyOf(enableMetrics);
    this.disableDefaultMetrics = disableDefaultMetrics;
    this.optionalLabels = ImmutableSet.copyOf(optionalLabels);
  }

  @Override
  public Map getEnabledMetrics() {
    return enableMetrics;
  }

  @Override
  public Set getOptionalLabels() {
    return optionalLabels;
  }

  @Override
  public int getMeasuresSize() {
    return measures.size();
  }

  @VisibleForTesting
  List getMeasures() {
    synchronized (lock) {
      return Collections.unmodifiableList(measures);
    }
  }

  @Override
  public void addDoubleCounter(DoubleCounterMetricInstrument metricInstrument, double value,
      List requiredLabelValues, List optionalLabelValues) {
    MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
    if (instrumentData == null) {
      // Disabled metric
      return;
    }
    Attributes attributes = createAttributes(metricInstrument.getRequiredLabelKeys(),
        metricInstrument.getOptionalLabelKeys(), requiredLabelValues, optionalLabelValues,
        instrumentData.getOptionalLabelsBitSet());
    DoubleCounter counter = (DoubleCounter) instrumentData.getMeasure();
    counter.add(value, attributes);
  }

  @Override
  public void addLongCounter(LongCounterMetricInstrument metricInstrument, long value,
      List requiredLabelValues, List optionalLabelValues) {
    MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
    if (instrumentData == null) {
      // Disabled metric
      return;
    }
    Attributes attributes = createAttributes(metricInstrument.getRequiredLabelKeys(),
        metricInstrument.getOptionalLabelKeys(), requiredLabelValues, optionalLabelValues,
        instrumentData.getOptionalLabelsBitSet());
    LongCounter counter = (LongCounter) instrumentData.getMeasure();
    counter.add(value, attributes);
  }

  @Override
  public void recordDoubleHistogram(DoubleHistogramMetricInstrument metricInstrument, double value,
      List requiredLabelValues, List optionalLabelValues) {
    MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
    if (instrumentData == null) {
      // Disabled metric
      return;
    }
    Attributes attributes = createAttributes(metricInstrument.getRequiredLabelKeys(),
        metricInstrument.getOptionalLabelKeys(), requiredLabelValues, optionalLabelValues,
        instrumentData.getOptionalLabelsBitSet());
    DoubleHistogram histogram = (DoubleHistogram) instrumentData.getMeasure();
    histogram.record(value, attributes);
  }

  @Override
  public void recordLongHistogram(LongHistogramMetricInstrument metricInstrument, long value,
      List requiredLabelValues, List optionalLabelValues) {
    MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
    if (instrumentData == null) {
      // Disabled metric
      return;
    }
    Attributes attributes = createAttributes(metricInstrument.getRequiredLabelKeys(),
        metricInstrument.getOptionalLabelKeys(), requiredLabelValues, optionalLabelValues,
        instrumentData.getOptionalLabelsBitSet());
    LongHistogram histogram = (LongHistogram) instrumentData.getMeasure();
    histogram.record(value, attributes);
  }

  @Override
  public void recordLongGauge(LongGaugeMetricInstrument metricInstrument, long value,
      List requiredLabelValues, List optionalLabelValues) {
    MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
    if (instrumentData == null) {
      // Disabled metric
      return;
    }
    Attributes attributes = createAttributes(metricInstrument.getRequiredLabelKeys(),
        metricInstrument.getOptionalLabelKeys(), requiredLabelValues, optionalLabelValues,
        instrumentData.getOptionalLabelsBitSet());
    ObservableLongMeasurement gauge = (ObservableLongMeasurement) instrumentData.getMeasure();
    gauge.record(value, attributes);
  }

  @Override
  public Registration registerBatchCallback(Runnable callback,
      CallbackMetricInstrument... metricInstruments) {
    List measurements = new ArrayList<>(metricInstruments.length);
    for (CallbackMetricInstrument metricInstrument: metricInstruments) {
      MeasuresData instrumentData = measures.get(metricInstrument.getIndex());
      if (instrumentData == null) {
        // Disabled metric
        continue;
      }
      if (!(instrumentData.getMeasure() instanceof ObservableMeasurement)) {
        logger.log(Level.FINE, "Unsupported metric instrument type : {0} {1}",
            new Object[] {metricInstrument, instrumentData.getMeasure().getClass()});
        continue;
      }
      measurements.add((ObservableMeasurement) instrumentData.getMeasure());
    }
    if (measurements.isEmpty()) {
      return () -> { };
    }
    ObservableMeasurement first = measurements.get(0);
    measurements.remove(0);
    BatchCallback closeable = openTelemetryMeter.batchCallback(
        callback, first, measurements.toArray(new ObservableMeasurement[0]));
    return closeable::close;
  }

  @Override
  public void updateMeasures(List instruments) {
    synchronized (lock) {
      if (measures.size() >= instruments.size()) {
        // Already up-to-date
        return;
      }

      List newMeasures = new ArrayList<>(instruments.size());
      // Reuse existing measures
      newMeasures.addAll(measures);

      for (int i = measures.size(); i < instruments.size(); i++) {
        MetricInstrument instrument = instruments.get(i);
        // Check if the metric is disabled
        if (!shouldEnableMetric(instrument)) {
          // Adding null measure for disabled Metric
          newMeasures.add(null);
          continue;
        }

        BitSet bitSet = new BitSet(instrument.getOptionalLabelKeys().size());
        if (optionalLabels.isEmpty()) {
          // initialize an empty list
        } else {
          List labels = instrument.getOptionalLabelKeys();
          for (int j = 0; j < labels.size(); j++) {
            if (optionalLabels.contains(labels.get(j))) {
              bitSet.set(j);
            }
          }
        }

        int index = instrument.getIndex();
        String name = instrument.getName();
        String unit = instrument.getUnit();
        String description = instrument.getDescription();

        Object openTelemetryMeasure;
        if (instrument instanceof DoubleCounterMetricInstrument) {
          openTelemetryMeasure = openTelemetryMeter.counterBuilder(name)
              .setUnit(unit)
              .setDescription(description)
              .ofDoubles()
              .build();
        } else if (instrument instanceof LongCounterMetricInstrument) {
          openTelemetryMeasure = openTelemetryMeter.counterBuilder(name)
              .setUnit(unit)
              .setDescription(description)
              .build();
        } else if (instrument instanceof DoubleHistogramMetricInstrument) {
          openTelemetryMeasure = openTelemetryMeter.histogramBuilder(name)
              .setUnit(unit)
              .setDescription(description)
              .build();
        } else if (instrument instanceof LongHistogramMetricInstrument) {
          openTelemetryMeasure = openTelemetryMeter.histogramBuilder(name)
              .setUnit(unit)
              .setDescription(description)
              .ofLongs()
              .build();
        } else if (instrument instanceof LongGaugeMetricInstrument) {
          openTelemetryMeasure = openTelemetryMeter.gaugeBuilder(name)
              .setUnit(unit)
              .setDescription(description)
              .ofLongs()
              .buildObserver();
        } else {
          logger.log(Level.FINE, "Unsupported metric instrument type : {0}", instrument);
          openTelemetryMeasure = null;
        }
        newMeasures.add(index, new MeasuresData(bitSet, openTelemetryMeasure));
      }

      measures = newMeasures;
    }
  }

  private boolean shouldEnableMetric(MetricInstrument instrument) {
    Boolean explicitlyEnabled = enableMetrics.get(instrument.getName());
    if (explicitlyEnabled != null) {
      return explicitlyEnabled;
    }
    return instrument.isEnableByDefault() && !disableDefaultMetrics;
  }


  private Attributes createAttributes(List requiredLabelKeys,
      List optionalLabelKeys,
      List requiredLabelValues, List optionalLabelValues, BitSet bitSet) {
    AttributesBuilder builder = Attributes.builder();
    // Required Labels
    for (int i = 0; i < requiredLabelKeys.size(); i++) {
      builder.put(requiredLabelKeys.get(i), requiredLabelValues.get(i));
    }
    // Optional labels
    for (int i = bitSet.nextSetBit(0); i >= 0; i = bitSet.nextSetBit(i + 1)) {
      if (i == Integer.MAX_VALUE) {
        break; // or (i+1) would overflow
      }
      builder.put(optionalLabelKeys.get(i), optionalLabelValues.get(i));
    }
    return builder.build();
  }

  static final class MeasuresData {
    final BitSet optionalLabelsIndices;
    final Object measure;

    MeasuresData(BitSet optionalLabelsIndices, Object measure) {
      this.optionalLabelsIndices = optionalLabelsIndices;
      this.measure = measure;
    }

    public BitSet getOptionalLabelsBitSet() {
      return optionalLabelsIndices;
    }

    public Object getMeasure() {
      return measure;
    }
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy