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

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

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); synchronized (namesCollectorsLock) { for (String name : names) { if (namesToCollectors.containsKey(name)) { throw new IllegalArgumentException("Collector already registered that provides name: " + name); } } for (String name : names) { namesToCollectors.put(name, m); } collectorsToNames.put(m, names); } } /** * 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) { switch (family.type) { case SUMMARY: names.add(family.name + "_count"); names.add(family.name + "_sum"); names.add(family.name); break; case HISTOGRAM: names.add(family.name + "_count"); names.add(family.name + "_sum"); names.add(family.name + "_bucket"); names.add(family.name); break; default: names.add(family.name); } } 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(includedNames); } class MetricFamilySamplesEnumeration implements Enumeration { private final Iterator collectorIter; private Iterator metricFamilySamples; private Collector.MetricFamilySamples next; private Set includedNames; MetricFamilySamplesEnumeration(Set includedNames) { this.includedNames = includedNames; collectorIter = includedCollectorIterator(includedNames); findNextElement(); } private Iterator includedCollectorIterator(Set includedNames) { if (includedNames.isEmpty()) { return collectors().iterator(); } else { HashSet collectors = new HashSet(); synchronized (namesCollectorsLock) { for (Map.Entry entry : namesToCollectors.entrySet()) { if (includedNames.contains(entry.getKey())) { collectors.add(entry.getValue()); } } } return collectors.iterator(); } } MetricFamilySamplesEnumeration() { this(Collections.emptySet()); } private void findNextElement() { next = null; while (metricFamilySamples != null && metricFamilySamples.hasNext()) { next = filter(metricFamilySamples.next()); if (next != null) { return; } } if (next == null) { while (collectorIter.hasNext()) { metricFamilySamples = collectorIter.next().collect().iterator(); while (metricFamilySamples.hasNext()) { next = filter(metricFamilySamples.next()); if (next != null) { return; } } } } } private Collector.MetricFamilySamples filter(Collector.MetricFamilySamples next) { if (includedNames.isEmpty()) { return next; } else { Iterator it = next.samples.iterator(); while (it.hasNext()) { if (!includedNames.contains(it.next().name)) { it.remove(); } } if (next.samples.size() == 0) { return null; } return next; } } 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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy