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

io.prometheus.client.Counter Maven / Gradle / Ivy

There is a newer version: 2.0.0
Show newest version
package io.prometheus.client;

import io.prometheus.client.exemplars.CounterExemplarSampler;
import io.prometheus.client.exemplars.Exemplar;
import io.prometheus.client.exemplars.ExemplarConfig;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicReference;

import static java.lang.Boolean.FALSE;
import static java.lang.Boolean.TRUE;

/**
 * Counter metric, to track counts of events or running totals.
 * 

* Example of Counters include: *

    *
  • Number of requests processed
  • *
  • Number of items that were inserted into a queue
  • *
  • Total amount of data a system has processed
  • *
* * Counters can only go up (and be reset), if your use case can go down you should use a {@link Gauge} instead. * Use the rate() function in Prometheus to calculate the rate of increase of a Counter. * By convention, the names of Counters are suffixed by _total. * *

* An example Counter: *

 * {@code
 *   class YourClass {
 *     static final Counter requests = Counter.build()
 *         .name("requests_total").help("Total requests.").register();
 *     static final Counter failedRequests = Counter.build()
 *         .name("requests_failed_total").help("Total failed requests.").register();
 *
 *     void processRequest() {
 *        requests.inc();
 *        try {
 *          // Your code here.
 *        } catch (Exception e) {
 *          failedRequests.inc();
 *          throw e;
 *        }
 *     }
 *   }
 * }
 * 
* *

* You can also use labels to track different types of metric: *

 * {@code
 *   class YourClass {
 *     static final Counter requests = Counter.build()
 *         .name("requests_total").help("Total requests.")
 *         .labelNames("method").register();
 *
 *     void processGetRequest() {
 *        requests.labels("get").inc();
 *        // Your code here.
 *     }
 *     void processPostRequest() {
 *        requests.labels("post").inc();
 *        // Your code here.
 *     }
 *   }
 * }
 * 
* These can be aggregated and processed together much more easily in the Prometheus * server than individual metrics for each labelset. * * If there is a suffix of _total on the metric name, it will be * removed. When exposing the time series for counter value, a * _total suffix will be added. This is for compatibility between * OpenMetrics and the Prometheus text format, as OpenMetrics requires the * _total suffix. */ public class Counter extends SimpleCollector implements Collector.Describable { private final Boolean exemplarsEnabled; // null means default from ExemplarConfig applies private final CounterExemplarSampler exemplarSampler; Counter(Builder b) { super(b); this.exemplarsEnabled = b.exemplarsEnabled; this.exemplarSampler = b.exemplarSampler; initializeNoLabelsChild(); } public static class Builder extends SimpleCollector.Builder { private Boolean exemplarsEnabled = null; private CounterExemplarSampler exemplarSampler = null; @Override public Counter create() { // Gracefully handle pre-OpenMetrics counters. if (name.endsWith("_total")) { name = name.substring(0, name.length() - 6); } dontInitializeNoLabelsChild = true; return new Counter(this); } /** * Enable exemplars and provide a custom {@link CounterExemplarSampler}. */ public Builder withExemplarSampler(CounterExemplarSampler exemplarSampler) { if (exemplarSampler == null) { throw new NullPointerException(); } this.exemplarSampler = exemplarSampler; return withExemplars(); } /** * Allow this counter to load exemplars from a {@link CounterExemplarSampler}. *

* If a specific exemplar sampler is configured for this counter that exemplar sampler is used * (see {@link #withExemplarSampler(CounterExemplarSampler)}). * Otherwise the default from {@link ExemplarConfig} is used. */ public Builder withExemplars() { this.exemplarsEnabled = TRUE; return this; } /** * Prevent this counter from loading exemplars from a {@link CounterExemplarSampler}. *

* You can still provide exemplars for explicitly individual observations, e.g. using * {@link #incWithExemplar(double, String...)}. */ public Builder withoutExemplars() { this.exemplarsEnabled = FALSE; return this; } } /** * Return a Builder to allow configuration of a new Counter. Ensures required fields are provided. * * @param name The name of the metric * @param help The help string of the metric */ public static Builder build(String name, String help) { return new Builder().name(name).help(help); } /** * Return a Builder to allow configuration of a new Counter. */ public static Builder build() { return new Builder(); } @Override protected Child newChild() { return new Child(exemplarsEnabled, exemplarSampler); } /** * The value of a single Counter. *

* Warning: References to a Child become invalid after using * {@link SimpleCollector#remove} or {@link SimpleCollector#clear}, */ public static class Child { private final DoubleAdder value = new DoubleAdder(); private final long created = System.currentTimeMillis(); private final Boolean exemplarsEnabled; private final CounterExemplarSampler exemplarSampler; private final AtomicReference exemplar = new AtomicReference(); public Child() { this(null, null); } public Child(Boolean exemplarsEnabled, CounterExemplarSampler exemplarSampler) { this.exemplarsEnabled = exemplarsEnabled; this.exemplarSampler = exemplarSampler; } /** * Increment the counter by 1. */ public void inc() { inc(1); } /** * Same as {@link #incWithExemplar(double, String...) incWithExemplar(1, exemplarLabels)}. */ public void incWithExemplar(String... exemplarLabels) { incWithExemplar(1, exemplarLabels); } /** * Same as {@link #incWithExemplar(double, Map) incWithExemplar(1, exemplarLabels)}. */ public void incWithExemplar(Map exemplarLabels) { incWithExemplar(1, exemplarLabels); } /** * Increment the counter by the given amount. * * @throws IllegalArgumentException If amt is negative. */ public void inc(double amt) { incWithExemplar(amt, (String[]) null); } /** * Like {@link #inc(double)}, but additionally creates an exemplar. *

* This exemplar takes precedence over any exemplar returned by the {@link CounterExemplarSampler} configured * in {@link ExemplarConfig}. *

* The exemplar will have {@code amt} as the value, {@code System.currentTimeMillis()} as the timestamp, * and the specified labels. * * @param amt same as in {@link #inc(double)} * @param exemplarLabels list of name/value pairs, as documented in {@link Exemplar#Exemplar(double, String...)}. * A commonly used name is {@code "trace_id"}. * Calling {@code incWithExemplar(amt)} means that an exemplar without labels will be created. * Calling {@code incWithExemplar(amt, (String[]) null)} is equivalent * to calling {@code inc(amt)}. */ public void incWithExemplar(double amt, String... exemplarLabels) { Exemplar exemplar = exemplarLabels == null ? null : new Exemplar(amt, System.currentTimeMillis(), exemplarLabels); if (amt < 0) { throw new IllegalArgumentException("Amount to increment must be non-negative."); } value.add(amt); updateExemplar(amt, exemplar); } /** * Same as {@link #incWithExemplar(double, String...)}, but the exemplar labels are passed as a {@link Map}. */ public void incWithExemplar(double amt, Map exemplarLabels) { incWithExemplar(amt, Exemplar.mapToArray(exemplarLabels)); } private void updateExemplar(double amt, Exemplar userProvidedExemplar) { Exemplar prev, next; do { prev = exemplar.get(); if (userProvidedExemplar == null) { next = sampleNextExemplar(amt, prev); } else { next = userProvidedExemplar; } if (next == null || next == prev) { return; } } while (!exemplar.compareAndSet(prev, next)); } private Exemplar sampleNextExemplar(double amt, Exemplar prev) { if (FALSE.equals(exemplarsEnabled)) { return null; } if (exemplarSampler != null) { return exemplarSampler.sample(amt, prev); } if (TRUE.equals(exemplarsEnabled) || ExemplarConfig.isExemplarsEnabled()) { CounterExemplarSampler exemplarSampler = ExemplarConfig.getCounterExemplarSampler(); if (exemplarSampler != null) { return exemplarSampler.sample(amt, prev); } } return null; } /** * Get the value of the counter. */ public double get() { return value.sum(); } private Exemplar getExemplar() { return exemplar.get(); } /** * Get the created time of the counter in milliseconds. */ public long created() { return created; } } // Convenience methods. /** * Increment the counter with no labels by 1. */ public void inc() { inc(1); } /** * Like {@link Child#incWithExemplar(String...)}, but for the counter without labels. */ public void incWithExemplar(String... exemplarLabels) { incWithExemplar(1, exemplarLabels); } /** * Like {@link Child#incWithExemplar(Map)}, but for the counter without labels. */ public void incWithExemplar(Map exemplarLabels) { incWithExemplar(1, exemplarLabels); } /** * Increment the counter with no labels by the given amount. * * @throws IllegalArgumentException If amt is negative. */ public void inc(double amt) { noLabelsChild.inc(amt); } /** * Like {@link Child#incWithExemplar(double, String...)}, but for the counter without labels. */ public void incWithExemplar(double amt, String... exemplarLabels) { noLabelsChild.incWithExemplar(amt, exemplarLabels); } /** * Like {@link Child#incWithExemplar(double, Map)}, but for the counter without labels. */ public void incWithExemplar(double amt, Map exemplarLabels) { noLabelsChild.incWithExemplar(amt, exemplarLabels); } /** * Get the value of the counter. */ public double get() { return noLabelsChild.get(); } @Override public List collect() { List samples = new ArrayList(children.size()); for(Map.Entry, Child> c: children.entrySet()) { samples.add(new MetricFamilySamples.Sample(fullname + "_total", labelNames, c.getKey(), c.getValue().get(), c.getValue().getExemplar())); samples.add(new MetricFamilySamples.Sample(fullname + "_created", labelNames, c.getKey(), c.getValue().created() / 1000.0)); } return familySamplesList(Type.COUNTER, samples); } @Override public List describe() { return Collections.singletonList(new CounterMetricFamily(fullname, help, labelNames)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy