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