mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-03-12 04:36:29 -07:00
* add header-only prometheus lib to ext * rename folder * Undo rename directory * prometheus simpleapi included on mac & linux * wip * wire up some controller stats * Get windows building with prometheus * bsd build flags for prometheus * Fix multiple network join from environment entrypoint.sh.release (#1961) * _bond_m guards _bond, not _paths_m (#1965) * Fix: warning: mutex '_aqm_m' is not held on every path through here [-Wthread-safety-analysis] (#1964) * Serve prom metrics from /metrics endpoint * Add prom metrics for Central controller specific things * reorganize metric initialization * testing out a labled gauge on Networks * increment error counter on throw * Consolidate metrics definitions Put all metric definitions into node/Metrics.hpp. Accessed as needed from there. * Revert "testing out a labled gauge on Networks" This reverts commit 499ed6d95e11452019cdf48e32ed4cd878c2705b. * still blows up but adding to the record for completeness right now * Fix runtime issues with metrics * Add metrics files to visual studio project * Missed an "extern" * add copyright headers to new files * Add metrics for sent/received bytes (total) * put /metrics endpoint behind auth * sendto returns int on Win32 --------- Co-authored-by: Leonardo Amaral <leleobhz@users.noreply.github.com> Co-authored-by: Brenton Bostick <bostick@gmail.com>
124 lines
4.5 KiB
C++
124 lines
4.5 KiB
C++
#pragma once
|
|
|
|
#include <map>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "prometheus/collectable.h"
|
|
#include "prometheus/family.h"
|
|
|
|
namespace prometheus {
|
|
|
|
/// \brief Manages the collection of a number of metrics.
|
|
///
|
|
/// The Registry is responsible to expose data to a class/method/function
|
|
/// "bridge", which returns the metrics in a format Prometheus supports.
|
|
///
|
|
/// The key class is the Collectable. This has a method - called Collect() -
|
|
/// that returns zero or more metrics and their samples. The metrics are
|
|
/// represented by the class Family<>, which implements the Collectable
|
|
/// interface. A new metric is registered with BuildCounter(), BuildGauge(),
|
|
/// BuildHistogram() or BuildSummary().
|
|
///
|
|
/// The class is thread-safe. No concurrent call to any API of this type causes
|
|
/// a data race.
|
|
class Registry : public Collectable {
|
|
|
|
public:
|
|
|
|
/// \brief How to deal with repeatedly added family names for a type.
|
|
///
|
|
/// Adding a family with the same name but different types is always an error
|
|
/// and will lead to an exception.
|
|
enum class InsertBehavior {
|
|
/// \brief If a family with the same name and labels already exists return
|
|
/// the existing one. If no family with that name exists create it.
|
|
/// Otherwise throw.
|
|
Merge,
|
|
/// \brief Throws if a family with the same name already exists.
|
|
Throw,
|
|
/// \brief Never merge and always create a new family. This violates the
|
|
/// prometheus specification but was the default behavior in earlier
|
|
/// versions
|
|
NonStandardAppend,
|
|
};
|
|
|
|
using FamilyPtr = std::unique_ptr<Family>;
|
|
using Families = std::vector<FamilyPtr>;
|
|
|
|
const InsertBehavior insert_behavior;
|
|
mutable std::mutex mutex;
|
|
Families families;
|
|
|
|
/// \brief name Create a new registry.
|
|
///
|
|
/// \param insert_behavior How to handle families with the same name.
|
|
Registry (InsertBehavior insert_behavior_ = InsertBehavior::Merge)
|
|
: insert_behavior(insert_behavior_) {}
|
|
|
|
/// \brief Returns a list of metrics and their samples.
|
|
///
|
|
/// Every time the Registry is scraped it calls each of the metrics Collect
|
|
/// function.
|
|
///
|
|
/// \return Zero or more metrics and their samples.
|
|
virtual MetricFamilies Collect() const {
|
|
|
|
std::lock_guard<std::mutex> lock{ mutex };
|
|
|
|
MetricFamilies results;
|
|
|
|
for (const FamilyPtr& family_ptr : families) {
|
|
MetricFamilies metrics = family_ptr->Collect();
|
|
results.insert(results.end(), std::make_move_iterator(metrics.begin()), std::make_move_iterator(metrics.end()));
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
template <typename CustomFamily>
|
|
CustomFamily& Add (const std::string& name, const std::string& help, const Family::Labels& labels) {
|
|
|
|
std::lock_guard<std::mutex> lock{ mutex };
|
|
|
|
bool found_one_but_not_merge = false;
|
|
for (const FamilyPtr& family_ptr : families) {
|
|
if (family_ptr->GetName() == name) {
|
|
|
|
if (family_ptr->type != CustomFamily::static_type) // found family with this name and with different type
|
|
throw std::invalid_argument("Family name already exists with different type");
|
|
|
|
else { // found family with this name and the same type
|
|
switch (insert_behavior) {
|
|
case InsertBehavior::Throw:
|
|
throw std::invalid_argument("Family name already exists");
|
|
case InsertBehavior::Merge:
|
|
if (family_ptr->GetConstantLabels() == labels)
|
|
return dynamic_cast<CustomFamily&>(*family_ptr);
|
|
else // this strange rule was in previos version prometheus cpp
|
|
found_one_but_not_merge = true;
|
|
case InsertBehavior::NonStandardAppend:
|
|
continue;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
if (found_one_but_not_merge) // this strange rule was in previos version prometheus cpp
|
|
throw std::invalid_argument("Family name already exists with different labels");
|
|
|
|
std::unique_ptr<CustomFamily> new_family_ptr (new CustomFamily(name, help, labels));
|
|
CustomFamily& new_family = *new_family_ptr;
|
|
families.push_back(std::move(new_family_ptr));
|
|
return new_family;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace prometheus
|