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

com.github.marschall.micrometer.jfr.JfrFunctionTimer Maven / Gradle / Ivy

package com.github.marschall.micrometer.jfr;

import java.lang.ref.WeakReference;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.ToDoubleFunction;
import java.util.function.ToLongFunction;

import io.micrometer.core.instrument.Counter;
import io.micrometer.core.instrument.DistributionSummary;
import io.micrometer.core.instrument.FunctionCounter;
import io.micrometer.core.instrument.FunctionTimer;
import io.micrometer.core.instrument.Gauge;
import io.micrometer.core.instrument.LongTaskTimer;
import io.micrometer.core.instrument.Meter;
import io.micrometer.core.instrument.TimeGauge;
import io.micrometer.core.instrument.Timer;
import io.micrometer.core.instrument.util.TimeUtils;

final class JfrFunctionTimer extends AbstractJfrMeter implements FunctionTimer {

  private final WeakReference reference;
  private final ToLongFunction countFunction;
  private final ToDoubleFunction totalTimeFunction;
  private final TimeUnit totalTimeFunctionUnit;
  private final TimeUnit baseTimeUnit;
  private final Runnable hook;

  JfrFunctionTimer(Id id, T obj,
          ToLongFunction countFunction,
          ToDoubleFunction totalTimeFunction,
          TimeUnit totalTimeFunctionUnit,
          TimeUnit baseTimeUnit) {
    super(id, new FunctionTimerEventFactory(id, baseTimeUnit));
    this.reference = new WeakReference<>(obj);
    this.countFunction = countFunction;
    this.totalTimeFunction = totalTimeFunction;
    this.totalTimeFunctionUnit = totalTimeFunctionUnit;
    this.baseTimeUnit = baseTimeUnit;
    this.hook = () -> this.recordEvent();
    this.meterEventFactory.registerPeriodicEvent(this.jfrEventFactory, this.hook);
  }

  @Override
  public void close() {
    this.meterEventFactory.unregisterPeriodicEvent(this.hook);
    super.close();
  }

  void recordEvent() {
    double totalTime = this.totalTime(this.baseTimeUnit());
    double count = this.count();

    JfrFunctionTimerEvent event = this.newEmptyEvent();
    event.setTotalTimeAndCount(totalTime, count);
    event.commit();
  }

  @Override
  public double count() {
    T obj = this.reference.get();
    if (obj != null) {
      return this.countFunction.applyAsLong(obj);
    } else {
      return Double.NaN;
    }
  }

  @Override
  public double totalTime(TimeUnit destinationUnit) {
    T obj = this.reference.get();
    if (obj != null) {
      double totalTime = this.totalTimeFunction.applyAsDouble(obj);
      return TimeUtils.convert(totalTime, this.totalTimeFunctionUnit, destinationUnit);
    } else {
      return Double.NaN;
    }
  }

  @Override
  public double mean(TimeUnit destinationUnit) {
    T obj = this.reference.get();
    if (obj != null) {
      double count = this.countFunction.applyAsLong(obj);
      if (count == 0.0d) {
        return 0.0d;
      }
      double totalTime = this.totalTimeFunction.applyAsDouble(obj);
      double totalTimeInDestinationUnit = TimeUtils.convert(totalTime, this.totalTimeFunctionUnit, destinationUnit);
      return totalTimeInDestinationUnit / count;
    } else {
      return 0.0d;
    }
  }

  @Override
  public TimeUnit baseTimeUnit() {
    return this.baseTimeUnit;
  }

  @Override
  public  X match(Function visitGauge, Function visitCounter, Function visitTimer,
      Function visitSummary, Function visitLongTaskTimer,
      Function visitTimeGauge, Function visitFunctionCounter,
      Function visitFunctionTimer, Function visitMeter) {
    return visitFunctionTimer.apply(this);
  }

  @Override
  public void use(Consumer visitGauge, Consumer visitCounter, Consumer visitTimer,
      Consumer visitSummary, Consumer visitLongTaskTimer,
      Consumer visitTimeGauge, Consumer visitFunctionCounter,
      Consumer visitFunctionTimer, Consumer visitMeter) {
    visitFunctionTimer.accept(this);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy