io.prometheus.client.Collector Maven / Gradle / Ivy
Show all versions of simpleclient Show documentation
package io.prometheus.client;
import io.prometheus.client.exemplars.Exemplar;
import java.util.*;
import java.util.regex.Pattern;
/**
* A collector for a set of metrics.
*
* Normal users should use {@link Gauge}, {@link Counter}, {@link Summary} and {@link Histogram}.
*
* Subclasssing Collector is for advanced uses, such as proxying metrics from another monitoring system.
* It is it the responsibility of subclasses to ensure they produce valid metrics.
* @see Exposition formats.
*/
public abstract class Collector {
/**
* Return all metrics of this Collector.
*/
public abstract List collect();
/**
* Like {@link #collect()}, but the result should only contain {@code MetricFamilySamples} where
* {@code sampleNameFilter.test(name)} is {@code true} for at least one Sample name.
*
* The default implementation first collects all {@code MetricFamilySamples} and then discards the ones
* where {@code sampleNameFilter.test(name)} returns {@code false} for all names in
* {@link MetricFamilySamples#getNames()}.
* To improve performance, collector implementations should override this method to prevent
* {@code MetricFamilySamples} from being collected if they will be discarded anyways.
* See {@code ThreadExports} for an example.
*
* Note that the resulting List may contain {@code MetricFamilySamples} where some Sample names return
* {@code true} for {@code sampleNameFilter.test(name)} but some Sample names return {@code false}.
* This is ok, because before we produce the output format we will call
* {@link MetricFamilySamples#filter(Predicate)} to strip all Samples where {@code sampleNameFilter.test(name)}
* returns {@code false}.
*
* @param sampleNameFilter may be {@code null}, indicating that all metrics should be collected.
*/
public List collect(Predicate sampleNameFilter) {
List all = collect();
if (sampleNameFilter == null) {
return all;
}
List remaining = new ArrayList(all.size());
for (MetricFamilySamples mfs : all) {
for (String name : mfs.getNames()) {
if (sampleNameFilter.test(name)) {
remaining.add(mfs);
break;
}
}
}
return remaining;
}
public enum Type {
UNKNOWN, // This is untyped in Prometheus text format.
COUNTER,
GAUGE,
STATE_SET,
INFO,
HISTOGRAM,
GAUGE_HISTOGRAM,
SUMMARY,
}
/**
* A metric, and all of its samples.
*/
static public class MetricFamilySamples {
public final String name;
public final String unit;
public final Type type;
public final String help;
public final List samples; // this list is modified when samples are added/removed.
public MetricFamilySamples(String name, Type type, String help, List samples) {
this(name, "", type, help, samples);
}
public MetricFamilySamples(String name, String unit, Type type, String help, List samples) {
if (!unit.isEmpty() && !name.endsWith("_" + unit)) {
throw new IllegalArgumentException("Metric's unit is not the suffix of the metric name: " + name);
}
if ((type == Type.INFO || type == Type.STATE_SET) && !unit.isEmpty()) {
throw new IllegalArgumentException("Metric is of a type that cannot have a unit: " + name);
}
List mungedSamples = samples;
// Deal with _total from pre-OM automatically.
if (type == Type.COUNTER) {
if (name.endsWith("_total")) {
name = name.substring(0, name.length() - 6);
}
String withTotal = name + "_total";
mungedSamples = new ArrayList(samples.size());
for (Sample s: samples) {
String n = s.name;
if (name.equals(n)) {
n = withTotal;
}
mungedSamples.add(new Sample(n, s.labelNames, s.labelValues, s.value, s.exemplar, s.timestampMs));
}
}
this.name = name;
this.unit = unit;
this.type = type;
this.help = help;
this.samples = mungedSamples;
}
/**
* @param sampleNameFilter may be {@code null} indicating that the result contains the complete list of samples.
* @return A new MetricFamilySamples containing only the Samples matching the {@code sampleNameFilter},
* or {@code null} if no Sample matches.
*/
public MetricFamilySamples filter(Predicate sampleNameFilter) {
if (sampleNameFilter == null) {
return this;
}
List remainingSamples = new ArrayList(samples.size());
for (Sample sample : samples) {
if (sampleNameFilter.test(sample.name)) {
remainingSamples.add(sample);
}
}
if (remainingSamples.isEmpty()) {
return null;
}
return new MetricFamilySamples(name, unit, type, help, remainingSamples);
}
/**
* List of names that are reserved for Samples in these MetricsFamilySamples.
*
* This is used in two places:
*
* - To check potential name collisions in {@link CollectorRegistry#register(Collector)}.
*
- To check if a collector may contain metrics matching the metric name filter
* in {@link Collector#collect(Predicate)}.
*
* Note that {@code getNames()} always includes the name without suffix, even though some
* metrics types (like Counter) will not have a Sample with that name.
* The reason is that the name without suffix is used in the metadata comments ({@code # TYPE}, {@code # UNIT},
* {@code # HELP}), and as this name must be unique
* we include the name without suffix here as well.
*/
public String[] getNames() {
switch (type) {
case COUNTER:
return new String[]{
name + "_total",
name + "_created",
name
};
case SUMMARY:
return new String[]{
name + "_count",
name + "_sum",
name + "_created",
name
};
case HISTOGRAM:
return new String[]{
name + "_count",
name + "_sum",
name + "_bucket",
name + "_created",
name
};
case GAUGE_HISTOGRAM:
return new String[]{
name + "_gcount",
name + "_gsum",
name + "_bucket",
name
};
case INFO:
return new String[]{
name + "_info",
name
};
default:
return new String[]{name};
}
}
@Override
public boolean equals(Object obj) {
if (!(obj instanceof MetricFamilySamples)) {
return false;
}
MetricFamilySamples other = (MetricFamilySamples) obj;
return other.name.equals(name)
&& other.unit.equals(unit)
&& other.type.equals(type)
&& other.help.equals(help)
&& other.samples.equals(samples);
}
@Override
public int hashCode() {
int hash = 1;
hash = 37 * hash + name.hashCode();
hash = 37 * hash + unit.hashCode();
hash = 37 * hash + type.hashCode();
hash = 37 * hash + help.hashCode();
hash = 37 * hash + samples.hashCode();
return hash;
}
@Override
public String toString() {
return "Name: " + name + " Unit:" + unit + " Type: " + type + " Help: " + help +
" Samples: " + samples;
}
/**
* A single Sample, with a unique name and set of labels.
*/
public static class Sample {
public final String name;
public final List labelNames;
public final List labelValues; // Must have same length as labelNames.
public final double value;
public final Exemplar exemplar;
public final Long timestampMs; // It's an epoch format with milliseconds value included (this field is subject to change).
public Sample(String name, List labelNames, List labelValues, double value, Exemplar exemplar, Long timestampMs) {
this.name = name;
this.labelNames = labelNames;
this.labelValues = labelValues;
this.value = value;
this.exemplar = exemplar;
this.timestampMs = timestampMs;
}
public Sample(String name, List labelNames, List labelValues, double value, Long timestampMs) {
this(name, labelNames, labelValues, value, null, timestampMs);
}
public Sample(String name, List labelNames, List labelValues, double value, Exemplar exemplar) {
this(name, labelNames, labelValues, value, exemplar, null);
}
public Sample(String name, List labelNames, List labelValues, double value) {
this(name, labelNames, labelValues, value, null, null);
}
@Override
public boolean equals(Object obj) {
if (!(obj instanceof Sample)) {
return false;
}
Sample other = (Sample) obj;
return other.name.equals(name) &&
other.labelNames.equals(labelNames) &&
other.labelValues.equals(labelValues) &&
other.value == value &&
(exemplar == null && other.exemplar == null || other.exemplar != null && other.exemplar.equals(exemplar)) &&
(timestampMs == null && other.timestampMs == null || other.timestampMs != null && other.timestampMs.equals(timestampMs));
}
@Override
public int hashCode() {
int hash = 1;
hash = 37 * hash + name.hashCode();
hash = 37 * hash + labelNames.hashCode();
hash = 37 * hash + labelValues.hashCode();
long d = Double.doubleToLongBits(value);
hash = 37 * hash + (int)(d ^ (d >>> 32));
if (timestampMs != null) {
hash = 37 * hash + timestampMs.hashCode();
}
if (exemplar != null) {
hash = 37 * exemplar.hashCode();
}
return hash;
}
@Override
public String toString() {
return "Name: " + name + " LabelNames: " + labelNames + " labelValues: " + labelValues +
" Value: " + value + " TimestampMs: " + timestampMs;
}
}
}
/**
* Register the Collector with the default registry.
*/
public T register() {
return register(CollectorRegistry.defaultRegistry);
}
/**
* Register the Collector with the given registry.
*/
public T register(CollectorRegistry registry) {
registry.register(this);
return (T)this;
}
public interface Describable {
/**
* Provide a list of metric families this Collector is expected to return.
*
* These should exclude the samples. This is used by the registry to
* detect collisions and duplicate registrations.
*
* Usually custom collectors do not have to implement Describable. If
* Describable is not implemented and the CollectorRegistry was created
* with auto describe enabled (which is the case for the default registry)
* then {@link #collect} will be called at registration time instead of
* describe. If this could cause problems, either implement a proper
* describe, or if that's not practical have describe return an empty
* list.
*/
List describe();
}
/* Various utility functions for implementing Collectors. */
/**
* Number of nanoseconds in a second.
*/
public static final double NANOSECONDS_PER_SECOND = 1E9;
/**
* Number of milliseconds in a second.
*/
public static final double MILLISECONDS_PER_SECOND = 1E3;
private static final Pattern METRIC_NAME_RE = Pattern.compile("[a-zA-Z_:][a-zA-Z0-9_:]*");
private static final Pattern METRIC_LABEL_NAME_RE = Pattern.compile("[a-zA-Z_][a-zA-Z0-9_]*");
private static final Pattern RESERVED_METRIC_LABEL_NAME_RE = Pattern.compile("__.*");
/**
* Throw an exception if the metric name is invalid.
*/
protected static void checkMetricName(String name) {
if (!METRIC_NAME_RE.matcher(name).matches()) {
throw new IllegalArgumentException("Invalid metric name: " + name);
}
}
/**
* Sanitize metric name
*/
public static String sanitizeMetricName(String metricName) {
int length = metricName.length();
char[] sanitized = new char[length];
for(int i = 0; i < length; i++) {
char ch = metricName.charAt(i);
if(ch == ':' ||
(ch >= 'a' && ch <= 'z') ||
(ch >= 'A' && ch <= 'Z') ||
(i > 0 && ch >= '0' && ch <= '9')) {
sanitized[i] = ch;
} else {
sanitized[i] = '_';
}
}
return new String(sanitized);
}
/**
* Throw an exception if the metric label name is invalid.
*/
protected static void checkMetricLabelName(String name) {
if (!METRIC_LABEL_NAME_RE.matcher(name).matches()) {
throw new IllegalArgumentException("Invalid metric label name: " + name);
}
if (RESERVED_METRIC_LABEL_NAME_RE.matcher(name).matches()) {
throw new IllegalArgumentException("Invalid metric label name, reserved for internal use: " + name);
}
}
/**
* Convert a double to its string representation in Go.
*/
public static String doubleToGoString(double d) {
if (d == Double.POSITIVE_INFINITY) {
return "+Inf";
}
if (d == Double.NEGATIVE_INFINITY) {
return "-Inf";
}
return Double.toString(d);
}
}