xref: /openbmc/telemetry/src/metrics/collection_function.cpp (revision 583ba441654657bb4ba9d051b747144a7258c159)
1f7ea2997SKrzysztof Grobelny #include "metrics/collection_function.hpp"
2f7ea2997SKrzysztof Grobelny 
3f7ea2997SKrzysztof Grobelny #include <cmath>
4f7ea2997SKrzysztof Grobelny 
5f7ea2997SKrzysztof Grobelny namespace metrics
6f7ea2997SKrzysztof Grobelny {
7f7ea2997SKrzysztof Grobelny 
8f7ea2997SKrzysztof Grobelny class FunctionMinimum : public CollectionFunction
9f7ea2997SKrzysztof Grobelny {
10f7ea2997SKrzysztof Grobelny   public:
calculate(const std::vector<ReadingItem> & readings,Milliseconds) const11f7ea2997SKrzysztof Grobelny     double calculate(const std::vector<ReadingItem>& readings,
12f7ea2997SKrzysztof Grobelny                      Milliseconds) const override
13f7ea2997SKrzysztof Grobelny     {
14*583ba441SPatrick Williams         return std::min_element(
15*583ba441SPatrick Williams                    readings.begin(), readings.end(),
16f7ea2997SKrzysztof Grobelny                    [](const auto& left, const auto& right) {
17*583ba441SPatrick Williams                        return std::make_tuple(!std::isfinite(left.second),
18*583ba441SPatrick Williams                                               left.second) <
19*583ba441SPatrick Williams                               std::make_tuple(!std::isfinite(right.second),
20*583ba441SPatrick Williams                                               right.second);
21*583ba441SPatrick Williams                    })
22*583ba441SPatrick Williams             ->second;
23f7ea2997SKrzysztof Grobelny     }
24f7ea2997SKrzysztof Grobelny 
calculateForStartupInterval(std::vector<ReadingItem> & readings,Milliseconds timestamp) const25f7ea2997SKrzysztof Grobelny     double calculateForStartupInterval(std::vector<ReadingItem>& readings,
26f7ea2997SKrzysztof Grobelny                                        Milliseconds timestamp) const override
27f7ea2997SKrzysztof Grobelny     {
28f7ea2997SKrzysztof Grobelny         readings.assign(
29f7ea2997SKrzysztof Grobelny             {ReadingItem(timestamp, calculate(readings, timestamp))});
30f7ea2997SKrzysztof Grobelny         return readings.back().second;
31f7ea2997SKrzysztof Grobelny     }
32f7ea2997SKrzysztof Grobelny };
33f7ea2997SKrzysztof Grobelny 
34f7ea2997SKrzysztof Grobelny class FunctionMaximum : public CollectionFunction
35f7ea2997SKrzysztof Grobelny {
36f7ea2997SKrzysztof Grobelny   public:
calculate(const std::vector<ReadingItem> & readings,Milliseconds) const37f7ea2997SKrzysztof Grobelny     double calculate(const std::vector<ReadingItem>& readings,
38f7ea2997SKrzysztof Grobelny                      Milliseconds) const override
39f7ea2997SKrzysztof Grobelny     {
40*583ba441SPatrick Williams         return std::max_element(
41*583ba441SPatrick Williams                    readings.begin(), readings.end(),
42f7ea2997SKrzysztof Grobelny                    [](const auto& left, const auto& right) {
43*583ba441SPatrick Williams                        return std::make_tuple(std::isfinite(left.second),
44*583ba441SPatrick Williams                                               left.second) <
45*583ba441SPatrick Williams                               std::make_tuple(std::isfinite(right.second),
46*583ba441SPatrick Williams                                               right.second);
47*583ba441SPatrick Williams                    })
48*583ba441SPatrick Williams             ->second;
49f7ea2997SKrzysztof Grobelny     }
50f7ea2997SKrzysztof Grobelny 
calculateForStartupInterval(std::vector<ReadingItem> & readings,Milliseconds timestamp) const51f7ea2997SKrzysztof Grobelny     double calculateForStartupInterval(std::vector<ReadingItem>& readings,
52f7ea2997SKrzysztof Grobelny                                        Milliseconds timestamp) const override
53f7ea2997SKrzysztof Grobelny     {
54f7ea2997SKrzysztof Grobelny         readings.assign(
55f7ea2997SKrzysztof Grobelny             {ReadingItem(timestamp, calculate(readings, timestamp))});
56f7ea2997SKrzysztof Grobelny         return readings.back().second;
57f7ea2997SKrzysztof Grobelny     }
58f7ea2997SKrzysztof Grobelny };
59f7ea2997SKrzysztof Grobelny 
60f7ea2997SKrzysztof Grobelny class FunctionAverage : public CollectionFunction
61f7ea2997SKrzysztof Grobelny {
62f7ea2997SKrzysztof Grobelny   public:
calculate(const std::vector<ReadingItem> & readings,Milliseconds timestamp) const63f7ea2997SKrzysztof Grobelny     double calculate(const std::vector<ReadingItem>& readings,
64f7ea2997SKrzysztof Grobelny                      Milliseconds timestamp) const override
65f7ea2997SKrzysztof Grobelny     {
66f7ea2997SKrzysztof Grobelny         auto valueSum = 0.0;
67f7ea2997SKrzysztof Grobelny         auto timeSum = Milliseconds{0};
68f7ea2997SKrzysztof Grobelny         for (auto it = readings.begin(); it != std::prev(readings.end()); ++it)
69f7ea2997SKrzysztof Grobelny         {
70f7ea2997SKrzysztof Grobelny             if (std::isfinite(it->second))
71f7ea2997SKrzysztof Grobelny             {
72f7ea2997SKrzysztof Grobelny                 const auto kt = std::next(it);
73f7ea2997SKrzysztof Grobelny                 const auto duration = kt->first - it->first;
74f7ea2997SKrzysztof Grobelny                 valueSum += it->second * duration.count();
75f7ea2997SKrzysztof Grobelny                 timeSum += duration;
76f7ea2997SKrzysztof Grobelny             }
77f7ea2997SKrzysztof Grobelny         }
78f7ea2997SKrzysztof Grobelny 
79f7ea2997SKrzysztof Grobelny         const auto duration = timestamp - readings.back().first;
80f7ea2997SKrzysztof Grobelny         valueSum += readings.back().second * duration.count();
81f7ea2997SKrzysztof Grobelny         timeSum += duration;
82f7ea2997SKrzysztof Grobelny 
83f7ea2997SKrzysztof Grobelny         return valueSum / std::max(timeSum.count(), uint64_t{1u});
84f7ea2997SKrzysztof Grobelny     }
85f7ea2997SKrzysztof Grobelny 
calculateForStartupInterval(std::vector<ReadingItem> & readings,Milliseconds timestamp) const86f7ea2997SKrzysztof Grobelny     double calculateForStartupInterval(std::vector<ReadingItem>& readings,
87f7ea2997SKrzysztof Grobelny                                        Milliseconds timestamp) const override
88f7ea2997SKrzysztof Grobelny     {
89f7ea2997SKrzysztof Grobelny         auto result = calculate(readings, timestamp);
90f7ea2997SKrzysztof Grobelny         if (std::isfinite(result))
91f7ea2997SKrzysztof Grobelny         {
92f7ea2997SKrzysztof Grobelny             readings.assign({ReadingItem(readings.front().first, result),
93f7ea2997SKrzysztof Grobelny                              ReadingItem(timestamp, readings.back().second)});
94f7ea2997SKrzysztof Grobelny         }
95f7ea2997SKrzysztof Grobelny         return result;
96f7ea2997SKrzysztof Grobelny     }
97f7ea2997SKrzysztof Grobelny };
98f7ea2997SKrzysztof Grobelny 
99f7ea2997SKrzysztof Grobelny class FunctionSummation : public CollectionFunction
100f7ea2997SKrzysztof Grobelny {
101f7ea2997SKrzysztof Grobelny     using Multiplier = std::chrono::duration<double>;
102f7ea2997SKrzysztof Grobelny 
103f7ea2997SKrzysztof Grobelny   public:
calculate(const std::vector<ReadingItem> & readings,const Milliseconds timestamp) const104f7ea2997SKrzysztof Grobelny     double calculate(const std::vector<ReadingItem>& readings,
105f7ea2997SKrzysztof Grobelny                      const Milliseconds timestamp) const override
106f7ea2997SKrzysztof Grobelny     {
107f7ea2997SKrzysztof Grobelny         auto valueSum = 0.0;
108f7ea2997SKrzysztof Grobelny         for (auto it = readings.begin(); it != std::prev(readings.end()); ++it)
109f7ea2997SKrzysztof Grobelny         {
110f7ea2997SKrzysztof Grobelny             if (std::isfinite(it->second))
111f7ea2997SKrzysztof Grobelny             {
112f7ea2997SKrzysztof Grobelny                 const auto kt = std::next(it);
113f7ea2997SKrzysztof Grobelny                 const auto multiplier =
114f7ea2997SKrzysztof Grobelny                     calculateMultiplier(kt->first - it->first);
115f7ea2997SKrzysztof Grobelny                 valueSum += it->second * multiplier.count();
116f7ea2997SKrzysztof Grobelny             }
117f7ea2997SKrzysztof Grobelny         }
118f7ea2997SKrzysztof Grobelny 
119f7ea2997SKrzysztof Grobelny         const auto multiplier =
120f7ea2997SKrzysztof Grobelny             calculateMultiplier(timestamp - readings.back().first);
121f7ea2997SKrzysztof Grobelny         valueSum += readings.back().second * multiplier.count();
122f7ea2997SKrzysztof Grobelny 
123f7ea2997SKrzysztof Grobelny         return valueSum;
124f7ea2997SKrzysztof Grobelny     }
125f7ea2997SKrzysztof Grobelny 
calculateForStartupInterval(std::vector<ReadingItem> & readings,const Milliseconds timestamp) const126*583ba441SPatrick Williams     double calculateForStartupInterval(
127*583ba441SPatrick Williams         std::vector<ReadingItem>& readings,
128f7ea2997SKrzysztof Grobelny         const Milliseconds timestamp) const override
129f7ea2997SKrzysztof Grobelny     {
130f7ea2997SKrzysztof Grobelny         const auto result = calculate(readings, timestamp);
131f7ea2997SKrzysztof Grobelny         if (readings.size() > 2 && std::isfinite(result))
132f7ea2997SKrzysztof Grobelny         {
133f7ea2997SKrzysztof Grobelny             const auto multiplier =
134f7ea2997SKrzysztof Grobelny                 calculateMultiplier(timestamp - readings.front().first).count();
135f7ea2997SKrzysztof Grobelny             if (multiplier > 0.)
136f7ea2997SKrzysztof Grobelny             {
137f7ea2997SKrzysztof Grobelny                 const auto prevValue = result / multiplier;
138f7ea2997SKrzysztof Grobelny                 readings.assign(
139f7ea2997SKrzysztof Grobelny                     {ReadingItem(readings.front().first, prevValue),
140f7ea2997SKrzysztof Grobelny                      ReadingItem(timestamp, readings.back().second)});
141f7ea2997SKrzysztof Grobelny             }
142f7ea2997SKrzysztof Grobelny         }
143f7ea2997SKrzysztof Grobelny         return result;
144f7ea2997SKrzysztof Grobelny     }
145f7ea2997SKrzysztof Grobelny 
146f7ea2997SKrzysztof Grobelny   private:
calculateMultiplier(Milliseconds duration)147f7ea2997SKrzysztof Grobelny     static constexpr Multiplier calculateMultiplier(Milliseconds duration)
148f7ea2997SKrzysztof Grobelny     {
149f7ea2997SKrzysztof Grobelny         constexpr auto m = Multiplier{Seconds{1}};
150f7ea2997SKrzysztof Grobelny         return Multiplier{duration / m};
151f7ea2997SKrzysztof Grobelny     }
152f7ea2997SKrzysztof Grobelny };
153f7ea2997SKrzysztof Grobelny 
makeCollectionFunction(OperationType operationType)154*583ba441SPatrick Williams std::shared_ptr<CollectionFunction> makeCollectionFunction(
155*583ba441SPatrick Williams     OperationType operationType)
156f7ea2997SKrzysztof Grobelny {
157f7ea2997SKrzysztof Grobelny     using namespace std::string_literals;
158f7ea2997SKrzysztof Grobelny 
159f7ea2997SKrzysztof Grobelny     switch (operationType)
160f7ea2997SKrzysztof Grobelny     {
161f7ea2997SKrzysztof Grobelny         case OperationType::min:
162f7ea2997SKrzysztof Grobelny             return std::make_shared<FunctionMinimum>();
163f7ea2997SKrzysztof Grobelny         case OperationType::max:
164f7ea2997SKrzysztof Grobelny             return std::make_shared<FunctionMaximum>();
165f7ea2997SKrzysztof Grobelny         case OperationType::avg:
166f7ea2997SKrzysztof Grobelny             return std::make_shared<FunctionAverage>();
167f7ea2997SKrzysztof Grobelny         case OperationType::sum:
168f7ea2997SKrzysztof Grobelny             return std::make_shared<FunctionSummation>();
169f7ea2997SKrzysztof Grobelny         default:
170*583ba441SPatrick Williams             throw std::runtime_error(
171*583ba441SPatrick Williams                 "op: "s + utils::enumToString(operationType) +
172f7ea2997SKrzysztof Grobelny                 " is not supported"s);
173f7ea2997SKrzysztof Grobelny     }
174f7ea2997SKrzysztof Grobelny }
175f7ea2997SKrzysztof Grobelny 
176f7ea2997SKrzysztof Grobelny } // namespace metrics
177