xref: /openbmc/telemetry/src/utils/labeled_tuple.hpp (revision d22381949a8705da399107fcff71f8fbfe910f46)
1e2362796SWludzik, Jozef #pragma once
2e2362796SWludzik, Jozef 
3e2362796SWludzik, Jozef #include <nlohmann/json.hpp>
4e2362796SWludzik, Jozef #include <sdbusplus/message/types.hpp>
5e2362796SWludzik, Jozef 
6e2362796SWludzik, Jozef namespace utils
7e2362796SWludzik, Jozef {
8e2362796SWludzik, Jozef 
9e2362796SWludzik, Jozef inline void from_json(const nlohmann::json& j,
10e2362796SWludzik, Jozef                       sdbusplus::message::object_path& o)
11e2362796SWludzik, Jozef {
12e2362796SWludzik, Jozef     o = j.get<std::string>();
13e2362796SWludzik, Jozef }
14e2362796SWludzik, Jozef 
15e2362796SWludzik, Jozef inline void from_json(const nlohmann::json& j,
16e2362796SWludzik, Jozef                       std::vector<sdbusplus::message::object_path>& o)
17e2362796SWludzik, Jozef {
18e2362796SWludzik, Jozef     o.clear();
19e2362796SWludzik, Jozef     for (const nlohmann::json& item : j)
20e2362796SWludzik, Jozef     {
21e2362796SWludzik, Jozef         o.emplace_back(item.get<std::string>());
22e2362796SWludzik, Jozef     }
23e2362796SWludzik, Jozef }
24e2362796SWludzik, Jozef 
25e2362796SWludzik, Jozef namespace detail
26e2362796SWludzik, Jozef {
27e2362796SWludzik, Jozef 
28e2362796SWludzik, Jozef template <class T>
29e2362796SWludzik, Jozef struct has_utils_from_json
30e2362796SWludzik, Jozef {
31e2362796SWludzik, Jozef     template <class U>
32e2362796SWludzik, Jozef     static U& ref();
33e2362796SWludzik, Jozef 
34e2362796SWludzik, Jozef     template <class U>
35e2362796SWludzik, Jozef     static std::true_type check(
36e2362796SWludzik, Jozef         decltype(utils::from_json(ref<const nlohmann::json>(), ref<U>()))*);
37e2362796SWludzik, Jozef 
38e2362796SWludzik, Jozef     template <class>
39e2362796SWludzik, Jozef     static std::false_type check(...);
40e2362796SWludzik, Jozef 
41e2362796SWludzik, Jozef     static constexpr bool value =
42e2362796SWludzik, Jozef         decltype(check<std::decay_t<T>>(nullptr))::value;
43e2362796SWludzik, Jozef };
44e2362796SWludzik, Jozef 
45e2362796SWludzik, Jozef template <class T>
46e2362796SWludzik, Jozef constexpr bool has_utils_from_json_v = has_utils_from_json<T>::value;
47e2362796SWludzik, Jozef 
48e2362796SWludzik, Jozef } // namespace detail
49e2362796SWludzik, Jozef 
50e2362796SWludzik, Jozef template <class, class...>
51e2362796SWludzik, Jozef struct LabeledTuple;
52e2362796SWludzik, Jozef 
53e2362796SWludzik, Jozef template <class... Args, class... Labels>
54e2362796SWludzik, Jozef struct LabeledTuple<std::tuple<Args...>, Labels...>
55e2362796SWludzik, Jozef {
56e2362796SWludzik, Jozef     static_assert(sizeof...(Args) == sizeof...(Labels));
57e2362796SWludzik, Jozef 
58*d2238194SKrzysztof Grobelny     using tuple_type = std::tuple<Args...>;
59*d2238194SKrzysztof Grobelny 
60*d2238194SKrzysztof Grobelny     LabeledTuple() = default;
61*d2238194SKrzysztof Grobelny     LabeledTuple(const LabeledTuple&) = default;
62*d2238194SKrzysztof Grobelny     LabeledTuple(LabeledTuple&&) = default;
63*d2238194SKrzysztof Grobelny 
64*d2238194SKrzysztof Grobelny     LabeledTuple(tuple_type v) : value(std::move(v))
65*d2238194SKrzysztof Grobelny     {}
66*d2238194SKrzysztof Grobelny     LabeledTuple(Args... args) : value(std::move(args)...)
67*d2238194SKrzysztof Grobelny     {}
68*d2238194SKrzysztof Grobelny 
69*d2238194SKrzysztof Grobelny     LabeledTuple& operator=(const LabeledTuple&) = default;
70*d2238194SKrzysztof Grobelny     LabeledTuple& operator=(LabeledTuple&&) = default;
71*d2238194SKrzysztof Grobelny 
72*d2238194SKrzysztof Grobelny     nlohmann::json to_json() const
73e2362796SWludzik, Jozef     {
74e2362796SWludzik, Jozef         nlohmann::json j;
75*d2238194SKrzysztof Grobelny         to_json_all(j, std::make_index_sequence<sizeof...(Args)>());
76e2362796SWludzik, Jozef         return j;
77e2362796SWludzik, Jozef     }
78e2362796SWludzik, Jozef 
79*d2238194SKrzysztof Grobelny     void from_json(const nlohmann::json& j)
80e2362796SWludzik, Jozef     {
81*d2238194SKrzysztof Grobelny         from_json_all(j, std::make_index_sequence<sizeof...(Args)>());
82*d2238194SKrzysztof Grobelny     }
83*d2238194SKrzysztof Grobelny 
84*d2238194SKrzysztof Grobelny     template <size_t Idx>
85*d2238194SKrzysztof Grobelny     const auto& at_index() const
86*d2238194SKrzysztof Grobelny     {
87*d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
88*d2238194SKrzysztof Grobelny     }
89*d2238194SKrzysztof Grobelny 
90*d2238194SKrzysztof Grobelny     template <size_t Idx>
91*d2238194SKrzysztof Grobelny     auto& at_index()
92*d2238194SKrzysztof Grobelny     {
93*d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
94*d2238194SKrzysztof Grobelny     }
95*d2238194SKrzysztof Grobelny 
96*d2238194SKrzysztof Grobelny     template <class Label>
97*d2238194SKrzysztof Grobelny     const auto& at_label() const
98*d2238194SKrzysztof Grobelny     {
99*d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
100*d2238194SKrzysztof Grobelny     }
101*d2238194SKrzysztof Grobelny 
102*d2238194SKrzysztof Grobelny     template <class Label>
103*d2238194SKrzysztof Grobelny     auto& at_label()
104*d2238194SKrzysztof Grobelny     {
105*d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
106*d2238194SKrzysztof Grobelny     }
107*d2238194SKrzysztof Grobelny 
108*d2238194SKrzysztof Grobelny     bool operator==(const LabeledTuple& other) const
109*d2238194SKrzysztof Grobelny     {
110*d2238194SKrzysztof Grobelny         return value == other.value;
111*d2238194SKrzysztof Grobelny     }
112*d2238194SKrzysztof Grobelny 
113*d2238194SKrzysztof Grobelny     bool operator<(const LabeledTuple& other) const
114*d2238194SKrzysztof Grobelny     {
115*d2238194SKrzysztof Grobelny         return value < other.value;
116e2362796SWludzik, Jozef     }
117e2362796SWludzik, Jozef 
118e2362796SWludzik, Jozef   private:
119e2362796SWludzik, Jozef     template <size_t... Idx>
120*d2238194SKrzysztof Grobelny     void to_json_all(nlohmann::json& j, std::index_sequence<Idx...>) const
121e2362796SWludzik, Jozef     {
122*d2238194SKrzysztof Grobelny         (to_json_item<Idx>(j), ...);
123e2362796SWludzik, Jozef     }
124e2362796SWludzik, Jozef 
125e2362796SWludzik, Jozef     template <size_t Idx>
126*d2238194SKrzysztof Grobelny     void to_json_item(nlohmann::json& j) const
127e2362796SWludzik, Jozef     {
128e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
129*d2238194SKrzysztof Grobelny         j[Label::str()] = std::get<Idx>(value);
130e2362796SWludzik, Jozef     }
131e2362796SWludzik, Jozef 
132e2362796SWludzik, Jozef     template <size_t... Idx>
133*d2238194SKrzysztof Grobelny     void from_json_all(const nlohmann::json& j, std::index_sequence<Idx...>)
134e2362796SWludzik, Jozef     {
135*d2238194SKrzysztof Grobelny         (from_json_item<Idx>(j), ...);
136e2362796SWludzik, Jozef     }
137e2362796SWludzik, Jozef 
138e2362796SWludzik, Jozef     template <size_t Idx>
139*d2238194SKrzysztof Grobelny     void from_json_item(const nlohmann::json& j)
140e2362796SWludzik, Jozef     {
141e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
142*d2238194SKrzysztof Grobelny         using T = std::tuple_element_t<Idx, tuple_type>;
143e2362796SWludzik, Jozef         const nlohmann::json& item = j.at(Label::str());
144e2362796SWludzik, Jozef         if constexpr (detail::has_utils_from_json_v<T>)
145e2362796SWludzik, Jozef         {
146e2362796SWludzik, Jozef             T& v = std::get<Idx>(value);
147e2362796SWludzik, Jozef             utils::from_json(item, v);
148e2362796SWludzik, Jozef         }
149e2362796SWludzik, Jozef         else
150e2362796SWludzik, Jozef         {
151e2362796SWludzik, Jozef             std::get<Idx>(value) = item.get<T>();
152e2362796SWludzik, Jozef         }
153e2362796SWludzik, Jozef     }
154*d2238194SKrzysztof Grobelny 
155*d2238194SKrzysztof Grobelny     template <size_t Idx, class Label, class Self>
156*d2238194SKrzysztof Grobelny     static auto& find_item(Self& self)
157*d2238194SKrzysztof Grobelny     {
158*d2238194SKrzysztof Grobelny         if constexpr (std::is_same_v<Label, std::tuple_element_t<
159*d2238194SKrzysztof Grobelny                                                 Idx, std::tuple<Labels...>>>)
160*d2238194SKrzysztof Grobelny         {
161*d2238194SKrzysztof Grobelny             return std::get<Idx>(self.value);
162*d2238194SKrzysztof Grobelny         }
163*d2238194SKrzysztof Grobelny         else
164*d2238194SKrzysztof Grobelny         {
165*d2238194SKrzysztof Grobelny             return find_item<Idx + 1, Label>(self);
166*d2238194SKrzysztof Grobelny         }
167*d2238194SKrzysztof Grobelny     }
168*d2238194SKrzysztof Grobelny 
169*d2238194SKrzysztof Grobelny     tuple_type value;
170e2362796SWludzik, Jozef };
171e2362796SWludzik, Jozef 
172*d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
173*d2238194SKrzysztof Grobelny void to_json(nlohmann::json& json,
174*d2238194SKrzysztof Grobelny              const LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
175*d2238194SKrzysztof Grobelny {
176*d2238194SKrzysztof Grobelny     json = tuple.to_json();
177*d2238194SKrzysztof Grobelny }
178*d2238194SKrzysztof Grobelny 
179*d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
180*d2238194SKrzysztof Grobelny void from_json(const nlohmann::json& json,
181*d2238194SKrzysztof Grobelny                LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
182*d2238194SKrzysztof Grobelny {
183*d2238194SKrzysztof Grobelny     tuple.from_json(json);
184*d2238194SKrzysztof Grobelny }
185*d2238194SKrzysztof Grobelny 
186e2362796SWludzik, Jozef } // namespace utils
187