io.prometheus.client.Histogram Maven / Gradle / Ivy
Show all versions of simpleclient Show documentation
package io.prometheus.client;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
/**
* 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();
* }
* }
* }
* }
*
*
* 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 {
double[] buckets;
Histogram(Builder b) {
super(b);
buckets = b.buckets;
initializeNoLabelsChild();
}
public static class Builder extends SimpleCollector.Builder {
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];
for (int i = 0; i < buckets.length; i++) {
tmp[i] = buckets[i];
}
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;
}
}
/**
* 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);
}
/**
* Represents an event being timed.
*/
public static class Timer {
Child child;
long start;
private Timer(Child child) {
this.child = child;
start = Child.timeProvider.nanoTime();
}
/**
* Observe the amount of time in seconds since {@link Child#startTimer} was called.
*/
public void observeDuration() {
child.observe((Child.timeProvider.nanoTime() - start) / NANOSECONDS_PER_SECOND);
}
}
/**
* 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 {
public static class Value {
private double sum;
private double[] buckets;
}
private Child(double[] buckets) {
upperBounds = buckets;
cumulativeCounts = new DoubleAdder[buckets.length];
for (int i = 0; i < buckets.length; ++i) {
cumulativeCounts[i] = new DoubleAdder();
}
}
private double[] upperBounds;
private DoubleAdder[] cumulativeCounts;
private DoubleAdder sum = new DoubleAdder();
static TimeProvider timeProvider = new TimeProvider();
/**
* Observe the given amount.
*/
public void observe(double amt) {
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);
break;
}
}
sum.add(amt);
}
/**
* 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);
}
/**
* Get the value of the Histogram.
*
* Warning: The definition of {@link Value} is subject to change.
*/
public Value get() {
Value v = new Value();
v.buckets = new double[cumulativeCounts.length];
double acc = 0;
for (int i = 0; i < cumulativeCounts.length; ++i) {
acc += cumulativeCounts[i].sum();
v.buckets[i] = acc;
}
v.sum = sum.sum();
return v;
}
}
// Convenience methods.
/**
* Observe the given amount on the histogram with no labels.
*/
public void observe(double amt) {
noLabelsChild.observe(amt);
}
/**
* 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();
}
@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]));
}
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));
}
MetricFamilySamples mfs = new MetricFamilySamples(fullname, Type.HISTOGRAM, help, samples);
List mfsList = new ArrayList();
mfsList.add(mfs);
return mfsList;
}
static class TimeProvider {
long nanoTime() {
return System.nanoTime();
}
}
}