如何使用 Spring Boot 和 Micrometer 创建自定义仪表?

Ale*_*ord 2 spring-boot prometheus micrometer

我编写了一个 Spring Boot 应用程序,我想使用Micrometer向 Prometheus 公开自定义指标。这是一个代码片段,我在其中递增计数器(有效)并尝试设置几个仪表(无效):

@Scheduled(cron = "*/2 * * * * *") // run every 2 seconds
private void logTemperature() throws UnknownHostException {

    // get measurement
    SensorReader sensorReader = new SensorReader();
    SensorReading sensorReading = sensorReader.getSensorReading();

    Metrics.counter("measurements").increment();
    Metrics.gauge("fahrenheit", sensorReading.getFahrenheit());
    Metrics.gauge("humidity", sensorReading.getHumidity());

    // do stuff with measurement
    // ...

}
Run Code Online (Sandbox Code Playgroud)

完整的源代码在这里:https : //github.com/alexwoolford/htu21d_logger/tree/master/htu21d-logger-spring

对象中的fahrenheit字段sensorReading是 a float。输出中存在仪表,但值是NaN(即不是数字):

# HELP measurements_total  
# TYPE measurements_total counter
measurements_total 82.0
...
# HELP humidity  
# TYPE humidity gauge
humidity NaN
...
# HELP fahrenheit  
# TYPE fahrenheit gauge
fahrenheit NaN
...
Run Code Online (Sandbox Code Playgroud)

我看过类似的帖子,例如千分尺/普罗米修斯如何防止仪表值变为 NaN?Micrometer - Prometheus Gauge 显示 NaN,但我无法从这些答案中弄清楚我需要做什么。

我还尝试创建一个变量并使用 lambda 更新它,例如

Metrics.gauge("humidity", humidity, humidity -> sensorReading.getHumidity());
Run Code Online (Sandbox Code Playgroud)

... 并且 guage 完全消失了(甚至没有NaN)。

这个问题与其他问题略有不同,因为我没有使用缓存,而是尝试使用全局注册表,这大概是最简单的选择。

che*_*tts 6

您正在使用的对象正在收集垃圾,因为 note 保留了这些值的引用。仪表默认使用弱引用。

如果您将其设置为使用强引用,则应避免使用NaNs。不幸的是,全局注册表助手不公开仪表构建器,因此您需要以稍微不同的方式创建仪表。

Gauge.builder("humidity", humidity, humidity -> sensorReading.getHumidity()).strongReference(true).register(Metrics.globalRegistry);
Run Code Online (Sandbox Code Playgroud)

作为旁注,我建议远离这种@Scheduled方法,除非传感器读数本身很慢。如果SensorReader是线程安全的,您可以删除@ScheduledSensorReader直接使用:

SensorReader sensorReader = new SensorReader();

Gauge.builder("humidity", sensorReader, sensorReader -> 
  sensorReader.getSensorReading().getHumidity())
  .strongReference(true)
  .register(Metrics.globalRegistry);
Gauge.builder("fahrenheit", sensorReader, sensorReader -> 
  sensorReader.getSensorReading().getFahrenheit())
  .strongReference(true)
  .register(Metrics.globalRegistry);

Run Code Online (Sandbox Code Playgroud)

这样,您将只查找传感器读数,即 Prometheus 抓取指标的速率。