io.prometheus.client.CollectorRegistry Maven / Gradle / Ivy
Show all versions of simpleclient Show documentation
package io.prometheus.client;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
/**
* A registry of Collectors.
*
* The majority of users should use the {@link #defaultRegistry}, rather than instantiating their own.
*
* Creating a registry other than the default is primarily useful for unittests, or
* pushing a subset of metrics to the Pushgateway
* from batch jobs.
*/
public class CollectorRegistry {
/**
* The default registry.
*/
public static final CollectorRegistry defaultRegistry = new CollectorRegistry(true);
private final Object namesCollectorsLock = new Object();
private final Map> collectorsToNames = new HashMap>();
private final Map namesToCollectors = new HashMap();
private final boolean autoDescribe;
public CollectorRegistry() {
this(false);
}
public CollectorRegistry(boolean autoDescribe) {
this.autoDescribe = autoDescribe;
}
/**
* Register a Collector.
*
* A collector can be registered to multiple CollectorRegistries.
*/
public void register(Collector m) {
List names = collectorNames(m);
assertNoDuplicateNames(m, names);
synchronized (namesCollectorsLock) {
for (String name : names) {
if (namesToCollectors.containsKey(name)) {
throw new IllegalArgumentException("Failed to register Collector of type " + m.getClass().getSimpleName()
+ ": " + name + " is already in use by another Collector of type "
+ namesToCollectors.get(name).getClass().getSimpleName());
}
}
for (String name : names) {
namesToCollectors.put(name, m);
}
collectorsToNames.put(m, names);
}
}
private void assertNoDuplicateNames(Collector m, List names) {
Set uniqueNames = new HashSet();
for (String name : names) {
if (!uniqueNames.add(name)) {
throw new IllegalArgumentException("Failed to register Collector of type " + m.getClass().getSimpleName()
+ ": The Collector exposes the same name multiple times: " + name);
}
}
}
/**
* Unregister a Collector.
*/
public void unregister(Collector m) {
synchronized (namesCollectorsLock) {
List names = collectorsToNames.remove(m);
for (String name : names) {
namesToCollectors.remove(name);
}
}
}
/**
* Unregister all Collectors.
*/
public void clear() {
synchronized (namesCollectorsLock) {
collectorsToNames.clear();
namesToCollectors.clear();
}
}
/**
* A snapshot of the current collectors.
*/
private Set collectors() {
synchronized (namesCollectorsLock) {
return new HashSet(collectorsToNames.keySet());
}
}
private List collectorNames(Collector m) {
List mfs;
if (m instanceof Collector.Describable) {
mfs = ((Collector.Describable) m).describe();
} else if (autoDescribe) {
mfs = m.collect();
} else {
mfs = Collections.emptyList();
}
List names = new ArrayList();
for (Collector.MetricFamilySamples family : mfs) {
names.addAll(Arrays.asList(family.getNames()));
}
return names;
}
/**
* Enumeration of metrics of all registered collectors.
*/
public Enumeration metricFamilySamples() {
return new MetricFamilySamplesEnumeration();
}
/**
* Enumeration of metrics matching the specified names.
*
* Note that the provided set of names will be matched against the time series
* name and not the metric name. For instance, to retrieve all samples from a
* histogram, you must include the '_count', '_sum' and '_bucket' names.
*/
public Enumeration filteredMetricFamilySamples(Set includedNames) {
return new MetricFamilySamplesEnumeration(new SampleNameFilter.Builder().nameMustBeEqualTo(includedNames).build());
}
/**
* Enumeration of metrics where {@code sampleNameFilter.test(name)} returns {@code true} for each {@code name} in
* {@link Collector.MetricFamilySamples#getNames()}.
* @param sampleNameFilter may be {@code null}, indicating that the enumeration should contain all metrics.
*/
public Enumeration filteredMetricFamilySamples(Predicate sampleNameFilter) {
return new MetricFamilySamplesEnumeration(sampleNameFilter);
}
class MetricFamilySamplesEnumeration implements Enumeration {
private final Iterator collectorIter;
private Iterator metricFamilySamples;
private Collector.MetricFamilySamples next;
private final Predicate sampleNameFilter;
MetricFamilySamplesEnumeration(Predicate sampleNameFilter) {
this.sampleNameFilter = sampleNameFilter;
this.collectorIter = filteredCollectorIterator();
findNextElement();
}
private Iterator filteredCollectorIterator() {
if (sampleNameFilter == null) {
return collectors().iterator();
} else {
HashSet collectors = new HashSet();
synchronized (namesCollectorsLock) {
for (Map.Entry> entry : collectorsToNames.entrySet()) {
List names = entry.getValue();
if (names.isEmpty()) {
collectors.add(entry.getKey());
} else {
for (String name : names) {
if (sampleNameFilter.test(name)) {
collectors.add(entry.getKey());
break;
}
}
}
}
}
return collectors.iterator();
}
}
MetricFamilySamplesEnumeration() {
this(null);
}
private void findNextElement() {
next = null;
while (metricFamilySamples != null && metricFamilySamples.hasNext()) {
next = metricFamilySamples.next().filter(sampleNameFilter);
if (next != null) {
return;
}
}
while (collectorIter.hasNext()) {
metricFamilySamples = collectorIter.next().collect(sampleNameFilter).iterator();
while (metricFamilySamples.hasNext()) {
next = metricFamilySamples.next().filter(sampleNameFilter);
if (next != null) {
return;
}
}
}
}
public Collector.MetricFamilySamples nextElement() {
Collector.MetricFamilySamples current = next;
if (current == null) {
throw new NoSuchElementException();
}
findNextElement();
return current;
}
public boolean hasMoreElements() {
return next != null;
}
}
/**
* Returns the given value, or null if it doesn't exist.
*
* This is inefficient, and intended only for use in unittests.
*/
public Double getSampleValue(String name) {
return getSampleValue(name, new String[]{}, new String[]{});
}
/**
* Returns the given value, or null if it doesn't exist.
*
* This is inefficient, and intended only for use in unittests.
*/
public Double getSampleValue(String name, String[] labelNames, String[] labelValues) {
for (Collector.MetricFamilySamples metricFamilySamples : Collections.list(metricFamilySamples())) {
for (Collector.MetricFamilySamples.Sample sample : metricFamilySamples.samples) {
if (sample.name.equals(name)
&& Arrays.equals(sample.labelNames.toArray(), labelNames)
&& Arrays.equals(sample.labelValues.toArray(), labelValues)) {
return sample.value;
}
}
}
return null;
}
}