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 
84d2238194SKrzysztof Grobelny     template <size_t Idx>
85d2238194SKrzysztof Grobelny     const auto& at_index() const
86d2238194SKrzysztof Grobelny     {
87d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
88d2238194SKrzysztof Grobelny     }
89d2238194SKrzysztof Grobelny 
90d2238194SKrzysztof Grobelny     template <size_t Idx>
91d2238194SKrzysztof Grobelny     auto& at_index()
92d2238194SKrzysztof Grobelny     {
93d2238194SKrzysztof Grobelny         return std::get<Idx>(value);
94d2238194SKrzysztof Grobelny     }
95d2238194SKrzysztof Grobelny 
96d2238194SKrzysztof Grobelny     template <class Label>
97d2238194SKrzysztof Grobelny     const auto& at_label() const
98d2238194SKrzysztof Grobelny     {
99d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
100d2238194SKrzysztof Grobelny     }
101d2238194SKrzysztof Grobelny 
102d2238194SKrzysztof Grobelny     template <class Label>
103d2238194SKrzysztof Grobelny     auto& at_label()
104d2238194SKrzysztof Grobelny     {
105d2238194SKrzysztof Grobelny         return find_item<0, Label>(*this);
106d2238194SKrzysztof Grobelny     }
107d2238194SKrzysztof Grobelny 
108d2238194SKrzysztof Grobelny     bool operator==(const LabeledTuple& other) const
109d2238194SKrzysztof Grobelny     {
110d2238194SKrzysztof Grobelny         return value == other.value;
111d2238194SKrzysztof Grobelny     }
112d2238194SKrzysztof Grobelny 
113d2238194SKrzysztof Grobelny     bool operator<(const LabeledTuple& other) const
114d2238194SKrzysztof Grobelny     {
115d2238194SKrzysztof Grobelny         return value < other.value;
116e2362796SWludzik, Jozef     }
117e2362796SWludzik, Jozef 
118e2362796SWludzik, Jozef   private:
119e2362796SWludzik, Jozef     template <size_t... Idx>
120d2238194SKrzysztof Grobelny     void to_json_all(nlohmann::json& j, std::index_sequence<Idx...>) const
121e2362796SWludzik, Jozef     {
122d2238194SKrzysztof Grobelny         (to_json_item<Idx>(j), ...);
123e2362796SWludzik, Jozef     }
124e2362796SWludzik, Jozef 
125e2362796SWludzik, Jozef     template <size_t Idx>
126d2238194SKrzysztof Grobelny     void to_json_item(nlohmann::json& j) const
127e2362796SWludzik, Jozef     {
128e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
129d2238194SKrzysztof Grobelny         j[Label::str()] = std::get<Idx>(value);
130e2362796SWludzik, Jozef     }
131e2362796SWludzik, Jozef 
132e2362796SWludzik, Jozef     template <size_t... Idx>
133d2238194SKrzysztof Grobelny     void from_json_all(const nlohmann::json& j, std::index_sequence<Idx...>)
134e2362796SWludzik, Jozef     {
135d2238194SKrzysztof Grobelny         (from_json_item<Idx>(j), ...);
136e2362796SWludzik, Jozef     }
137e2362796SWludzik, Jozef 
138e2362796SWludzik, Jozef     template <size_t Idx>
139d2238194SKrzysztof Grobelny     void from_json_item(const nlohmann::json& j)
140e2362796SWludzik, Jozef     {
141e2362796SWludzik, Jozef         using Label = std::tuple_element_t<Idx, std::tuple<Labels...>>;
142d2238194SKrzysztof 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     }
154d2238194SKrzysztof Grobelny 
155d2238194SKrzysztof Grobelny     template <size_t Idx, class Label, class Self>
156d2238194SKrzysztof Grobelny     static auto& find_item(Self& self)
157d2238194SKrzysztof Grobelny     {
158d2238194SKrzysztof Grobelny         if constexpr (std::is_same_v<Label, std::tuple_element_t<
159d2238194SKrzysztof Grobelny                                                 Idx, std::tuple<Labels...>>>)
160d2238194SKrzysztof Grobelny         {
161d2238194SKrzysztof Grobelny             return std::get<Idx>(self.value);
162d2238194SKrzysztof Grobelny         }
163d2238194SKrzysztof Grobelny         else
164d2238194SKrzysztof Grobelny         {
165d2238194SKrzysztof Grobelny             return find_item<Idx + 1, Label>(self);
166d2238194SKrzysztof Grobelny         }
167d2238194SKrzysztof Grobelny     }
168d2238194SKrzysztof Grobelny 
169d2238194SKrzysztof Grobelny     tuple_type value;
170e2362796SWludzik, Jozef };
171e2362796SWludzik, Jozef 
172d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
173*dcc4e193SKrzysztof Grobelny inline void to_json(nlohmann::json& json,
174d2238194SKrzysztof Grobelny                     const LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
175d2238194SKrzysztof Grobelny {
176d2238194SKrzysztof Grobelny     json = tuple.to_json();
177d2238194SKrzysztof Grobelny }
178d2238194SKrzysztof Grobelny 
179d2238194SKrzysztof Grobelny template <class... Args, class... Labels>
180*dcc4e193SKrzysztof Grobelny inline void from_json(const nlohmann::json& json,
181d2238194SKrzysztof Grobelny                       LabeledTuple<std::tuple<Args...>, Labels...>& tuple)
182d2238194SKrzysztof Grobelny {
183d2238194SKrzysztof Grobelny     tuple.from_json(json);
184d2238194SKrzysztof Grobelny }
185d2238194SKrzysztof Grobelny 
186e2362796SWludzik, Jozef } // namespace utils
187