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 
58d2238194SKrzysztof Grobelny     using tuple_type = std::tuple<Args...>;
59d2238194SKrzysztof Grobelny 
60d2238194SKrzysztof Grobelny     LabeledTuple() = default;
61d2238194SKrzysztof Grobelny     LabeledTuple(const LabeledTuple&) = default;
62d2238194SKrzysztof Grobelny     LabeledTuple(LabeledTuple&&) = default;
63d2238194SKrzysztof Grobelny 
64d2238194SKrzysztof Grobelny     LabeledTuple(tuple_type v) : value(std::move(v))
65d2238194SKrzysztof Grobelny     {}
66d2238194SKrzysztof Grobelny     LabeledTuple(Args... args) : value(std::move(args)...)
67d2238194SKrzysztof Grobelny     {}
68d2238194SKrzysztof Grobelny 
69d2238194SKrzysztof Grobelny     LabeledTuple& operator=(const LabeledTuple&) = default;
70d2238194SKrzysztof Grobelny     LabeledTuple& operator=(LabeledTuple&&) = default;
71d2238194SKrzysztof Grobelny 
72d2238194SKrzysztof Grobelny     nlohmann::json to_json() const
73e2362796SWludzik, Jozef     {
74e2362796SWludzik, Jozef         nlohmann::json j;
75d2238194SKrzysztof Grobelny         to_json_all(j, std::make_index_sequence<sizeof...(Args)>());
76e2362796SWludzik, Jozef         return j;
77e2362796SWludzik, Jozef     }
78e2362796SWludzik, Jozef 
79d2238194SKrzysztof Grobelny     void from_json(const nlohmann::json& j)
80e2362796SWludzik, Jozef     {
81d2238194SKrzysztof Grobelny         from_json_all(j, std::make_index_sequence<sizeof...(Args)>());
82d2238194SKrzysztof Grobelny     }
83d2238194SKrzysztof Grobelny 
843a617023SSzymon Dompke     std::string dump() const
853a617023SSzymon Dompke     {
863a617023SSzymon Dompke         return to_json().dump();
873a617023SSzymon Dompke     }
883a617023SSzymon Dompke 
89d2238194SKrzysztof Grobelny     template <size_t Idx>
90d2238194SKrzysztof Grobelny     const auto& at_index() const
91d2238194SKrzysztof Grobelny     {
92d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
93d2238194SKrzysztof Grobelny     }
94d2238194SKrzysztof Grobelny 
95d2238194SKrzysztof Grobelny     template <size_t Idx>
96d2238194SKrzysztof Grobelny     auto& at_index()
97d2238194SKrzysztof Grobelny     {
98d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
99d2238194SKrzysztof Grobelny     }
100d2238194SKrzysztof Grobelny 
101d2238194SKrzysztof Grobelny     template <class Label>
102d2238194SKrzysztof Grobelny     const auto& at_label() const
103d2238194SKrzysztof Grobelny     {
104d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
105d2238194SKrzysztof Grobelny     }
106d2238194SKrzysztof Grobelny 
107d2238194SKrzysztof Grobelny     template <class Label>
108d2238194SKrzysztof Grobelny     auto& at_label()
109d2238194SKrzysztof Grobelny     {
110d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
111d2238194SKrzysztof Grobelny     }
112d2238194SKrzysztof Grobelny 
113d2238194SKrzysztof Grobelny     bool operator==(const LabeledTuple& other) const
114d2238194SKrzysztof Grobelny     {
115d2238194SKrzysztof Grobelny         return value == other.value;
116d2238194SKrzysztof Grobelny     }
117d2238194SKrzysztof Grobelny 
118d2238194SKrzysztof Grobelny     bool operator<(const LabeledTuple& other) const
119d2238194SKrzysztof Grobelny     {
120d2238194SKrzysztof Grobelny         return value < other.value;
121e2362796SWludzik, Jozef     }
122e2362796SWludzik, Jozef 
123e2362796SWludzik, Jozef   private:
124e2362796SWludzik, Jozef     template <size_t... Idx>
125d2238194SKrzysztof Grobelny     void to_json_all(nlohmann::json& j, std::index_sequence<Idx...>) const
126e2362796SWludzik, Jozef     {
127d2238194SKrzysztof Grobelny         (to_json_item<Idx>(j), ...);
128e2362796SWludzik, Jozef     }
129e2362796SWludzik, Jozef 
130e2362796SWludzik, Jozef     template <size_t Idx>
131d2238194SKrzysztof Grobelny     void to_json_item(nlohmann::json& j) const
132e2362796SWludzik, Jozef     {
133e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
134d2238194SKrzysztof Grobelny         j[Label::str()] = std::get<Idx>(value);
135e2362796SWludzik, Jozef     }
136e2362796SWludzik, Jozef 
137e2362796SWludzik, Jozef     template <size_t... Idx>
138d2238194SKrzysztof Grobelny     void from_json_all(const nlohmann::json& j, std::index_sequence<Idx...>)
139e2362796SWludzik, Jozef     {
140d2238194SKrzysztof Grobelny         (from_json_item<Idx>(j), ...);
141e2362796SWludzik, Jozef     }
142e2362796SWludzik, Jozef 
143e2362796SWludzik, Jozef     template <size_t Idx>
144d2238194SKrzysztof Grobelny     void from_json_item(const nlohmann::json& j)
145e2362796SWludzik, Jozef     {
146e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
147d2238194SKrzysztof Grobelny         using T = std::tuple_element_t<Idx, tuple_type>;
148e2362796SWludzik, Jozef         const nlohmann::json& item = j.at(Label::str());
149e2362796SWludzik, Jozef         if constexpr (detail::has_utils_from_json_v<T>)
150e2362796SWludzik, Jozef         {
151e2362796SWludzik, Jozef             T& v = std::get<Idx>(value);
152e2362796SWludzik, Jozef             utils::from_json(item, v);
153e2362796SWludzik, Jozef         }
154e2362796SWludzik, Jozef         else
155e2362796SWludzik, Jozef         {
156e2362796SWludzik, Jozef             std::get<Idx>(value) = item.get<T>();
157e2362796SWludzik, Jozef         }
158e2362796SWludzik, Jozef     }
159d2238194SKrzysztof Grobelny 
160d2238194SKrzysztof Grobelny     template <size_t Idx, class Label, class Self>
161d2238194SKrzysztof Grobelny     static auto& find_item(Self& self)
162d2238194SKrzysztof Grobelny     {
163d2238194SKrzysztof Grobelny         if constexpr (std::is_same_v<Label, std::tuple_element_t<
164d2238194SKrzysztof Grobelny                                                 Idx, std::tuple<Labels...>>>)
165d2238194SKrzysztof Grobelny         {
166d2238194SKrzysztof Grobelny             return std::get<Idx>(self.value);
167d2238194SKrzysztof Grobelny         }
168d2238194SKrzysztof Grobelny         else
169d2238194SKrzysztof Grobelny         {
170*b8cc78ddSKrzysztof Grobelny             static_assert(Idx + 1 < sizeof...(Args),
171*b8cc78ddSKrzysztof Grobelny                           "Label not found in LabeledTuple");
172d2238194SKrzysztof Grobelny             return find_item<Idx + 1, Label>(self);
173d2238194SKrzysztof Grobelny         }
174d2238194SKrzysztof Grobelny     }
175d2238194SKrzysztof Grobelny 
176d2238194SKrzysztof Grobelny     tuple_type value;
177e2362796SWludzik, Jozef };
178e2362796SWludzik, Jozef 
179d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
180dcc4e193SKrzysztof Grobelny inline void to_json(nlohmann::json& json,
181d2238194SKrzysztof Grobelny                     const LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
182d2238194SKrzysztof Grobelny {
183d2238194SKrzysztof Grobelny     json = tuple.to_json();
184d2238194SKrzysztof Grobelny }
185d2238194SKrzysztof Grobelny 
186d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
187dcc4e193SKrzysztof Grobelny inline void from_json(const nlohmann::json& json,
188d2238194SKrzysztof Grobelny                       LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
189d2238194SKrzysztof Grobelny {
190d2238194SKrzysztof Grobelny     tuple.from_json(json);
191d2238194SKrzysztof Grobelny }
192d2238194SKrzysztof Grobelny 
193e2362796SWludzik, Jozef } // namespace utils
194