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

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

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

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

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

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

/**
 * Histogram metric, to track distributions of events.
 * 

* Example of uses for Histograms include: *

    *
  • Response latency
  • *
  • Request size
  • *
*

* Note: Each bucket is one timeseries. Many buckets and/or many dimensions with labels * can produce large amount of time series, that may cause performance problems. * *

* The default buckets are intended to cover a typical web/rpc request from milliseconds to seconds. *

* Example Histograms: *

 * {@code
 *   class YourClass {
 *     static final Histogram requestLatency = Histogram.build()
 *         .name("requests_latency_seconds").help("Request latency in seconds.").register();
 *
 *     void processRequest(Request req) {
 *        Histogram.Timer requestTimer = requestLatency.startTimer();
 *        try {
 *          // Your code here.
 *        } finally {
 *          requestTimer.observeDuration();
 *        }
 *     }
 *
 *     // Or if using Java 8 lambdas.
 *     void processRequestLambda(Request req) {
 *        requestLatency.time(() -> {
 *          // Your code here.
 *        });
 *     }
 *   }
 * }
 * 
*

* You can choose your own buckets: *

 * {@code
 *     static final Histogram requestLatency = Histogram.build()
 *         .buckets(.01, .02, .03, .04)
 *         .name("requests_latency_seconds").help("Request latency in seconds.").register();
 * }
 * 
* {@link Histogram.Builder#linearBuckets(double, double, int) linearBuckets} and * {@link Histogram.Builder#exponentialBuckets(double, double, int) exponentialBuckets} * offer easy ways to set common bucket patterns. */ public class Histogram extends SimpleCollector implements Collector.Describable { private final double[] buckets; private final Boolean exemplarsEnabled; // null means default from ExemplarConfig applies private final HistogramExemplarSampler exemplarSampler; Histogram(Builder b) { super(b); this.exemplarsEnabled = b.exemplarsEnabled; this.exemplarSampler = b.exemplarSampler; buckets = b.buckets; initializeNoLabelsChild(); } public static class Builder extends SimpleCollector.Builder { private Boolean exemplarsEnabled = null; private HistogramExemplarSampler exemplarSampler = null; private double[] buckets = new double[] { .005, .01, .025, .05, .075, .1, .25, .5, .75, 1, 2.5, 5, 7.5, 10 }; @Override public Histogram create() { for (int i = 0; i < buckets.length - 1; i++) { if (buckets[i] >= buckets[i + 1]) { throw new IllegalStateException("Histogram buckets must be in increasing order: " + buckets[i] + " >= " + buckets[i + 1]); } } if (buckets.length == 0) { throw new IllegalStateException("Histogram must have at least one bucket."); } for (String label : labelNames) { if (label.equals("le")) { throw new IllegalStateException("Histogram cannot have a label named 'le'."); } } // Append infinity bucket if it's not already there. if (buckets[buckets.length - 1] != Double.POSITIVE_INFINITY) { double[] tmp = new double[buckets.length + 1]; System.arraycopy(buckets, 0, tmp, 0, buckets.length); tmp[buckets.length] = Double.POSITIVE_INFINITY; buckets = tmp; } dontInitializeNoLabelsChild = true; return new Histogram(this); } /** * Set the upper bounds of buckets for the histogram. */ public Builder buckets(double... buckets) { this.buckets = buckets; return this; } /** * Set the upper bounds of buckets for the histogram with a linear sequence. */ public Builder linearBuckets(double start, double width, int count) { buckets = new double[count]; for (int i = 0; i < count; i++) { buckets[i] = start + i * width; } return this; } /** * Set the upper bounds of buckets for the histogram with an exponential sequence. */ public Builder exponentialBuckets(double start, double factor, int count) { buckets = new double[count]; for (int i = 0; i < count; i++) { buckets[i] = start * Math.pow(factor, i); } return this; } /** * Enable exemplars and provide a custom {@link HistogramExemplarSampler}. */ public Builder withExemplarSampler(HistogramExemplarSampler exemplarSampler) { if (exemplarSampler == null) { throw new NullPointerException(); } this.exemplarSampler = exemplarSampler; return withExemplars(); } /** * Allow this histogram to load exemplars from a {@link HistogramExemplarSampler}. *

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

* You can still provide exemplars for explicitly individual observations, e.g. using * {@link #observeWithExemplar(double, String...)}. */ public Builder withoutExemplars() { this.exemplarsEnabled = FALSE; return this; } } /** * Return a Builder to allow configuration of a new Histogram. 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 Histogram. */ public static Builder build() { return new Builder(); } @Override protected Child newChild() { return new Child(buckets, exemplarsEnabled, exemplarSampler); } /** * Represents an event being timed. */ public static class Timer implements Closeable { private final Child child; private final long start; private Timer(Child child, long start) { this.child = child; this.start = start; } /** * Observe the amount of time in seconds since {@link Child#startTimer} was called. * * @return Measured duration in seconds since {@link Child#startTimer} was called. */ public double observeDuration() { return observeDurationWithExemplar((String[]) null); } public double observeDurationWithExemplar(String... exemplarLabels) { double elapsed = SimpleTimer.elapsedSecondsFromNanos(start, SimpleTimer.defaultTimeProvider.nanoTime()); child.observeWithExemplar(elapsed, exemplarLabels); return elapsed; } public double observeDurationWithExemplar(Map exemplarLabels) { return observeDurationWithExemplar(Exemplar.mapToArray(exemplarLabels)); } /** * Equivalent to calling {@link #observeDuration()}. */ @Override public void close() { observeDuration(); } } /** * The value of a single Histogram. *

* Warning: References to a Child become invalid after using * {@link SimpleCollector#remove} or {@link SimpleCollector#clear}. */ public static class Child { /** * Executes runnable code (e.g. a Java 8 Lambda) and observes a duration of how long it took to run. * * @param timeable Code that is being timed * @return Measured duration in seconds for timeable to complete. */ public double time(Runnable timeable) { return timeWithExemplar(timeable, (String[]) null); } /** * Like {@link #time(Runnable)}, but additionally create an exemplar. *

* See {@link #observeWithExemplar(double, String...)} for documentation on the {@code exemplarLabels} parameter. */ public double timeWithExemplar(Runnable timeable, String... exemplarLabels) { Timer timer = startTimer(); double elapsed; try { timeable.run(); } finally { elapsed = timer.observeDurationWithExemplar(exemplarLabels); } return elapsed; } /** * Like {@link #time(Runnable)}, but additionally create an exemplar. *

* See {@link #observeWithExemplar(double, Map)} for documentation on the {@code exemplarLabels} parameter. */ public double timeWithExemplar(Runnable timeable, Map exemplarLabels) { return timeWithExemplar(timeable, Exemplar.mapToArray(exemplarLabels)); } /** * Executes callable code (e.g. a Java 8 Lambda) and observes a duration of how long it took to run. * * @param timeable Code that is being timed * @return Result returned by callable. */ public E time(Callable timeable) { return timeWithExemplar(timeable, (String[]) null); } /** * Like {@link #time(Callable)}, but additionally create an exemplar. *

* See {@link #observeWithExemplar(double, String...)} for documentation on the {@code exemplarLabels} parameter. */ public E timeWithExemplar(Callable timeable, String... exemplarLabels) { Timer timer = startTimer(); try { return timeable.call(); } catch (RuntimeException e) { throw e; } catch (Exception e) { throw new RuntimeException(e); } finally { timer.observeDurationWithExemplar(exemplarLabels); } } /** * Like {@link #time(Callable)}, but additionally create an exemplar. *

* See {@link #observeWithExemplar(double, Map)} for documentation on the {@code exemplarLabels} parameter. */ public E timeWithExemplar(Callable timeable, Map exemplarLabels) { return timeWithExemplar(timeable, Exemplar.mapToArray(exemplarLabels)); } public static class Value { public final double sum; public final double[] buckets; public final Exemplar[] exemplars; public final long created; public Value(double sum, double[] buckets, Exemplar[] exemplars, long created) { this.sum = sum; this.buckets = buckets; this.exemplars = exemplars; this.created = created; } } private Child(double[] buckets, Boolean exemplarsEnabled, HistogramExemplarSampler exemplarSampler) { upperBounds = buckets; this.exemplarsEnabled = exemplarsEnabled; this.exemplarSampler = exemplarSampler; exemplars = new ArrayList>(buckets.length); cumulativeCounts = new DoubleAdder[buckets.length]; for (int i = 0; i < buckets.length; ++i) { cumulativeCounts[i] = new DoubleAdder(); exemplars.add(new AtomicReference()); } } private final ArrayList> exemplars; private final Boolean exemplarsEnabled; private final HistogramExemplarSampler exemplarSampler; private final double[] upperBounds; private final DoubleAdder[] cumulativeCounts; private final DoubleAdder sum = new DoubleAdder(); private final long created = System.currentTimeMillis(); /** * Observe the given amount. * * @param amt in most cases amt should be >= 0. Negative values are supported, but you should read * * https://prometheus.io/docs/practices/histograms/#count-and-sum-of-observations for * implications and alternatives. */ public void observe(double amt) { observeWithExemplar(amt, (String[]) null); } /** * Like {@link #observe(double)}, but additionally creates an exemplar. *

* This exemplar takes precedence over any exemplar returned by the {@link HistogramExemplarSampler} 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 #observe(double)} (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 observeWithExemplar(amt)} means that an exemplar without labels is created. * Calling {@code observeWithExemplar(amt, (String[]) null)} is equivalent * to calling {@code observe(amt)}. */ public void observeWithExemplar(double amt, String... exemplarLabels) { Exemplar exemplar = exemplarLabels == null ? null : new Exemplar(amt, System.currentTimeMillis(), exemplarLabels); for (int i = 0; i < upperBounds.length; ++i) { // The last bucket is +Inf, so we always increment. if (amt <= upperBounds[i]) { cumulativeCounts[i].add(1); updateExemplar(amt, i, exemplar); break; } } sum.add(amt); } /** * Like {@link #observeWithExemplar(double, String...)}, but the exemplar labels are passed as a {@link Map}. */ public void observeWithExemplar(double amt, Map exemplarLabels) { observeWithExemplar(amt, Exemplar.mapToArray(exemplarLabels)); } private void updateExemplar(double amt, int i, Exemplar userProvidedExemplar) { AtomicReference exemplar = exemplars.get(i); double bucketFrom = i == 0 ? Double.NEGATIVE_INFINITY : upperBounds[i - 1]; double bucketTo = upperBounds[i]; Exemplar prev, next; do { prev = exemplar.get(); if (userProvidedExemplar != null) { next = userProvidedExemplar; } else { next = sampleNextExemplar(amt, bucketFrom, bucketTo, prev); } if (next == null || next == prev) { return; } } while (!exemplar.compareAndSet(prev, next)); } private Exemplar sampleNextExemplar(double amt, double bucketFrom, double bucketTo, Exemplar prev) { if (FALSE.equals(exemplarsEnabled)) { return null; } if (exemplarSampler != null) { return exemplarSampler.sample(amt, bucketFrom, bucketTo, prev); } if (TRUE.equals(exemplarsEnabled) || ExemplarConfig.isExemplarsEnabled()) { HistogramExemplarSampler exemplarSampler = ExemplarConfig.getHistogramExemplarSampler(); if (exemplarSampler != null) { return exemplarSampler.sample(amt, bucketFrom, bucketTo, prev); } } return null; } /** * Start a timer to track a duration. *

* Call {@link Timer#observeDuration} at the end of what you want to measure the duration of. */ public Timer startTimer() { return new Timer(this, SimpleTimer.defaultTimeProvider.nanoTime()); } /** * Get the value of the Histogram. *

* Warning: The definition of {@link Value} is subject to change. */ public Value get() { double[] buckets = new double[cumulativeCounts.length]; Exemplar[] exemplars = new Exemplar[cumulativeCounts.length]; double acc = 0; for (int i = 0; i < cumulativeCounts.length; ++i) { acc += cumulativeCounts[i].sum(); buckets[i] = acc; exemplars[i] = this.exemplars.get(i).get(); } return new Value(sum.sum(), buckets, exemplars, created); } } // Convenience methods. /** * Observe the given amount on the histogram with no labels. * * @param amt in most cases amt should be >= 0. Negative values are supported, but you should read * * https://prometheus.io/docs/practices/histograms/#count-and-sum-of-observations for * implications and alternatives. */ public void observe(double amt) { noLabelsChild.observe(amt); } /** * Like {@link Child#observeWithExemplar(double, String...)}, but for the histogram without labels. */ public void observeWithExemplar(double amt, String... exemplarLabels) { noLabelsChild.observeWithExemplar(amt, exemplarLabels); } /** * Like {@link Child#observeWithExemplar(double, Map)}, but for the histogram without labels. */ public void observeWithExemplar(double amt, Map exemplarLabels) { noLabelsChild.observeWithExemplar(amt, exemplarLabels); } /** * Start a timer to track a duration on the histogram with no labels. *

* Call {@link Timer#observeDuration} at the end of what you want to measure the duration of. */ public Timer startTimer() { return noLabelsChild.startTimer(); } /** * Executes runnable code (e.g. a Java 8 Lambda) and observes a duration of how long it took to run. * * @param timeable Code that is being timed * @return Measured duration in seconds for timeable to complete. */ public double time(Runnable timeable) { return noLabelsChild.time(timeable); } /** * Like {@link #time(Runnable)}, but additionally create an exemplar. *

* See {@link Child#observeWithExemplar(double, String...)} for documentation on the {@code exemplarLabels} parameter. */ public double timeWithExemplar(Runnable timeable, String... exemplarLabels) { return noLabelsChild.timeWithExemplar(timeable, exemplarLabels); } /** * Like {@link #time(Runnable)}, but additionally create an exemplar. *

* See {@link Child#observeWithExemplar(double, Map)} for documentation on the {@code exemplarLabels} parameter. */ public double timeWithExemplar(Runnable timeable, Map exemplarLabels) { return noLabelsChild.timeWithExemplar(timeable, exemplarLabels); } /** * Executes callable code (e.g. a Java 8 Lambda) and observes a duration of how long it took to run. * * @param timeable Code that is being timed * @return Result returned by callable. */ public E time(Callable timeable) { return noLabelsChild.time(timeable); } /** * Like {@link #time(Callable)}, but additionally create an exemplar. *

* See {@link Child#observeWithExemplar(double, String...)} for documentation on the {@code exemplarLabels} parameter. */ public E timeWithExemplar(Callable timeable, String... exemplarLabels) { return noLabelsChild.timeWithExemplar(timeable, exemplarLabels); } /** * Like {@link #time(Callable)}, but additionally create an exemplar. *

* See {@link Child#observeWithExemplar(double, Map)} for documentation on the {@code exemplarLabels} parameter. */ public E timeWithExemplar(Callable timeable, Map exemplarLabels) { return noLabelsChild.timeWithExemplar(timeable, exemplarLabels); } @Override public List collect() { List samples = new ArrayList(); for (Map.Entry, Child> c : children.entrySet()) { Child.Value v = c.getValue().get(); List labelNamesWithLe = new ArrayList(labelNames); labelNamesWithLe.add("le"); for (int i = 0; i < v.buckets.length; ++i) { List labelValuesWithLe = new ArrayList(c.getKey()); labelValuesWithLe.add(doubleToGoString(buckets[i])); samples.add(new MetricFamilySamples.Sample(fullname + "_bucket", labelNamesWithLe, labelValuesWithLe, v.buckets[i], v.exemplars[i])); } samples.add(new MetricFamilySamples.Sample(fullname + "_count", labelNames, c.getKey(), v.buckets[buckets.length-1])); samples.add(new MetricFamilySamples.Sample(fullname + "_sum", labelNames, c.getKey(), v.sum)); samples.add(new MetricFamilySamples.Sample(fullname + "_created", labelNames, c.getKey(), v.created / 1000.0)); } return familySamplesList(Type.HISTOGRAM, samples); } @Override public List describe() { return Collections.singletonList( new MetricFamilySamples(fullname, Type.HISTOGRAM, help, Collections.emptyList())); } double[] getBuckets() { return buckets; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy