2025-01-14 14:42:39 +00:00
|
|
|
/* Statistic collection.
|
2016-10-22 20:42:05 -04:00
|
|
|
*
|
2025-01-14 14:42:39 +00:00
|
|
|
* Author: Steffen Vogel <post@steffenvogel.de>
|
|
|
|
* SPDX-FileCopyrightText: 2014-2023 Institute for Automation of Complex Power Systems, RWTH Aachen University
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2016-10-22 20:42:05 -04:00
|
|
|
|
2019-04-07 15:13:40 +02:00
|
|
|
#include <villas/hist.hpp>
|
2021-08-10 10:12:48 -04:00
|
|
|
#include <villas/node.hpp>
|
2025-01-14 14:42:39 +00:00
|
|
|
#include <villas/stats.hpp>
|
|
|
|
#include <villas/timing.hpp>
|
2019-04-23 13:09:50 +02:00
|
|
|
#include <villas/utils.hpp>
|
2016-10-22 20:42:05 -04:00
|
|
|
|
2019-06-04 16:55:38 +02:00
|
|
|
using namespace villas;
|
2021-08-10 10:12:48 -04:00
|
|
|
using namespace villas::node;
|
2019-06-04 16:55:38 +02:00
|
|
|
using namespace villas::utils;
|
|
|
|
|
2019-06-23 13:35:42 +02:00
|
|
|
std::unordered_map<Stats::Metric, Stats::MetricDescription> Stats::metrics = {
|
2025-01-14 14:42:39 +00:00
|
|
|
{Stats::Metric::SMPS_SKIPPED,
|
|
|
|
{"skipped", "samples", "Skipped samples and the distance between them"}},
|
|
|
|
{Stats::Metric::SMPS_REORDERED,
|
|
|
|
{"reordered", "samples",
|
|
|
|
"Reordered samples and the distance between them"}},
|
|
|
|
{Stats::Metric::GAP_SAMPLE,
|
|
|
|
{"gap_sent", "seconds", "Inter-message timestamps (as sent by remote)"}},
|
|
|
|
{Stats::Metric::GAP_RECEIVED,
|
|
|
|
{"gap_received", "seconds",
|
|
|
|
"Inter-message arrival time (as received by this instance)"}},
|
|
|
|
{Stats::Metric::OWD,
|
|
|
|
{"owd", "seconds", "One-way-delay (OWD) of received messages"}},
|
|
|
|
{Stats::Metric::AGE,
|
|
|
|
{"age", "seconds",
|
|
|
|
"Processing time of packets within the from receive to sent"}},
|
|
|
|
{Stats::Metric::SIGNAL_COUNT,
|
|
|
|
{"signal_cnt", "signals", "Number of signals per sample"}},
|
|
|
|
{Stats::Metric::RTP_LOSS_FRACTION,
|
|
|
|
{"rtp.loss_fraction", "percent", "Fraction lost since last RTP SR/RR."}},
|
|
|
|
{Stats::Metric::RTP_PKTS_LOST,
|
|
|
|
{"rtp.pkts_lost", "packets", "Cumulative number of packets lost"}},
|
|
|
|
{Stats::Metric::RTP_JITTER,
|
|
|
|
{"rtp.jitter", "seconds", "Interarrival jitter"}},
|
2019-02-15 09:40:38 +01:00
|
|
|
};
|
|
|
|
|
2019-06-23 13:35:42 +02:00
|
|
|
std::unordered_map<Stats::Type, Stats::TypeDescription> Stats::types = {
|
2025-01-14 14:42:39 +00:00
|
|
|
{Stats::Type::LAST, {"last", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::HIGHEST, {"highest", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::LOWEST, {"lowest", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::MEAN, {"mean", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::VAR, {"var", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::STDDEV, {"stddev", SignalType::FLOAT}},
|
|
|
|
{Stats::Type::TOTAL, {"total", SignalType::INTEGER}}};
|
2016-11-07 22:18:26 -05:00
|
|
|
|
2019-06-23 13:35:42 +02:00
|
|
|
std::vector<TableColumn> Stats::columns = {
|
2025-01-14 14:42:39 +00:00
|
|
|
{10, TableColumn::Alignment::LEFT, "Node", "%s"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Recv", "%ju", "pkts"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Sent", "%ju", "pkts"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Drop", "%ju", "pkts"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Skip", "%ju", "pkts"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "OWD last", "%lf", "secs"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "OWD mean", "%lf", "secs"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Rate last", "%lf", "pkt/sec"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Rate mean", "%lf", "pkt/sec"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Age mean", "%lf", "secs"},
|
|
|
|
{10, TableColumn::Alignment::RIGHT, "Age Max", "%lf", "sec"},
|
|
|
|
{8, TableColumn::Alignment::RIGHT, "Signals", "%ju", "signals"}};
|
|
|
|
|
|
|
|
enum Stats::Format Stats::lookupFormat(const std::string &str) {
|
|
|
|
if (str == "human")
|
|
|
|
return Format::HUMAN;
|
|
|
|
else if (str == "json")
|
|
|
|
return Format::JSON;
|
|
|
|
else if (str == "matlab")
|
|
|
|
return Format::MATLAB;
|
|
|
|
|
|
|
|
throw std::invalid_argument("Invalid format");
|
2017-07-28 18:11:52 +02:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
enum Stats::Metric Stats::lookupMetric(const std::string &str) {
|
|
|
|
for (auto m : metrics) {
|
|
|
|
if (str == m.second.name)
|
|
|
|
return m.first;
|
|
|
|
}
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
throw std::invalid_argument("Invalid stats metric");
|
2019-02-15 09:40:38 +01:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
enum Stats::Type Stats::lookupType(const std::string &str) {
|
|
|
|
for (auto t : types) {
|
|
|
|
if (str == t.second.name)
|
|
|
|
return t.first;
|
|
|
|
}
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
throw std::invalid_argument("Invalid stats type");
|
2019-02-15 09:40:38 +01:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
Stats::Stats(int buckets, int warmup) : logger(Log::get("stats")) {
|
|
|
|
for (auto m : metrics) {
|
|
|
|
histograms.emplace(std::piecewise_construct, std::forward_as_tuple(m.first),
|
|
|
|
std::forward_as_tuple(buckets, warmup));
|
|
|
|
}
|
2016-10-22 20:42:05 -04:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::update(enum Metric m, double val) { histograms[m].put(val); }
|
2016-11-07 22:18:26 -05:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::reset() {
|
|
|
|
for (auto m : metrics)
|
|
|
|
histograms[m.first].reset();
|
2016-10-22 20:42:05 -04:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
json_t *Stats::toJson() const {
|
|
|
|
json_t *obj = json_object();
|
2016-11-07 22:18:26 -05:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
for (auto m : metrics) {
|
|
|
|
const Hist &h = histograms.at(m.first);
|
2017-05-05 19:24:16 +00:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
json_object_set_new(obj, m.second.name, h.toJson());
|
|
|
|
}
|
2017-05-05 19:24:16 +00:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
return obj;
|
2016-10-22 20:42:05 -04:00
|
|
|
}
|
2017-03-20 09:15:54 -03:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::printHeader(enum Format fmt) {
|
|
|
|
switch (fmt) {
|
|
|
|
case Format::HUMAN:
|
|
|
|
setupTable();
|
|
|
|
table->header();
|
|
|
|
break;
|
2017-05-05 19:24:16 +00:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
default: {
|
|
|
|
}
|
|
|
|
}
|
2017-07-12 00:56:08 +02:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::setupTable() {
|
|
|
|
if (!table) {
|
|
|
|
auto logger = Log::get("stats");
|
|
|
|
table = std::make_shared<Table>(logger, columns);
|
|
|
|
}
|
2019-06-23 13:35:42 +02:00
|
|
|
}
|
2019-03-26 15:31:27 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::printPeriodic(FILE *f, enum Format fmt, node::Node *n) const {
|
|
|
|
switch (fmt) {
|
|
|
|
case Format::HUMAN:
|
|
|
|
setupTable();
|
|
|
|
table->row(11, n->getNameShort().c_str(),
|
|
|
|
(uintmax_t)histograms.at(Metric::OWD).getTotal(),
|
|
|
|
(uintmax_t)histograms.at(Metric::AGE).getTotal(),
|
|
|
|
(uintmax_t)histograms.at(Metric::SMPS_REORDERED).getTotal(),
|
|
|
|
(uintmax_t)histograms.at(Metric::SMPS_SKIPPED).getTotal(),
|
|
|
|
(double)histograms.at(Metric::OWD).getLast(),
|
|
|
|
(double)histograms.at(Metric::OWD).getMean(),
|
|
|
|
(double)1.0 / histograms.at(Metric::GAP_RECEIVED).getLast(),
|
|
|
|
(double)1.0 / histograms.at(Metric::GAP_RECEIVED).getMean(),
|
|
|
|
(double)histograms.at(Metric::AGE).getMean(),
|
|
|
|
(double)histograms.at(Metric::AGE).getHighest(),
|
|
|
|
(uintmax_t)histograms.at(Metric::SIGNAL_COUNT).getLast());
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Format::JSON: {
|
|
|
|
json_t *json_stats = json_pack(
|
|
|
|
"{ s: s, s: i, s: i, s: i, s: i, s: f, s: f, s: f, s: f, s: f, s: f, "
|
|
|
|
"s: i }",
|
|
|
|
"node", n->getNameShort().c_str(), "recv",
|
|
|
|
histograms.at(Metric::OWD).getTotal(), "sent",
|
|
|
|
histograms.at(Metric::AGE).getTotal(), "dropped",
|
|
|
|
histograms.at(Metric::SMPS_REORDERED).getTotal(), "skipped",
|
|
|
|
histograms.at(Metric::SMPS_SKIPPED).getTotal(), "owd_last",
|
|
|
|
1.0 / histograms.at(Metric::OWD).getLast(), "owd_mean",
|
|
|
|
1.0 / histograms.at(Metric::OWD).getMean(), "rate_last",
|
|
|
|
1.0 / histograms.at(Metric::GAP_SAMPLE).getLast(), "rate_mean",
|
|
|
|
1.0 / histograms.at(Metric::GAP_SAMPLE).getMean(), "age_mean",
|
|
|
|
histograms.at(Metric::AGE).getMean(), "age_max",
|
|
|
|
histograms.at(Metric::AGE).getHighest(), "signals",
|
|
|
|
histograms.at(Metric::SIGNAL_COUNT).getLast());
|
|
|
|
json_dumpf(json_stats, f, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: {
|
|
|
|
}
|
|
|
|
}
|
2016-10-22 20:42:05 -04:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
void Stats::print(FILE *f, enum Format fmt, int verbose) const {
|
|
|
|
switch (fmt) {
|
|
|
|
case Format::HUMAN:
|
|
|
|
for (auto m : metrics) {
|
|
|
|
logger->info("{}: {}", m.second.name, m.second.desc);
|
|
|
|
histograms.at(m.first).print(logger, verbose, " ");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Format::JSON:
|
|
|
|
json_dumpf(toJson(), f, 0);
|
|
|
|
fflush(f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: {
|
|
|
|
}
|
|
|
|
}
|
2016-10-22 20:42:05 -04:00
|
|
|
}
|
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
union SignalData Stats::getValue(enum Metric sm, enum Type st) const {
|
|
|
|
const Hist &h = histograms.at(sm);
|
|
|
|
union SignalData d;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
switch (st) {
|
|
|
|
case Type::TOTAL:
|
|
|
|
d.i = h.getTotal();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::LAST:
|
|
|
|
d.f = h.getLast();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::HIGHEST:
|
|
|
|
d.f = h.getHighest();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::LOWEST:
|
|
|
|
d.f = h.getLowest();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::MEAN:
|
|
|
|
d.f = h.getMean();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::STDDEV:
|
|
|
|
d.f = h.getStddev();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
case Type::VAR:
|
|
|
|
d.f = h.getVar();
|
|
|
|
break;
|
2019-02-15 09:40:38 +01:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
default:
|
|
|
|
d.f = -1;
|
|
|
|
}
|
2017-05-05 19:24:16 +00:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
return d;
|
2017-05-05 22:26:34 +00:00
|
|
|
}
|
2019-06-23 13:35:42 +02:00
|
|
|
|
2025-01-14 14:42:39 +00:00
|
|
|
const Hist &Stats::getHistogram(enum Metric sm) const {
|
|
|
|
return histograms.at(sm);
|
2019-06-23 13:35:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<Table> Stats::table = std::shared_ptr<Table>();
|