xref: /openbmc/bmcweb/redfish-core/lib/redfish_util.hpp (revision 3f95a2772fc4fd57459fdfffbee52d8831fc4f35)
1 // SPDX-License-Identifier: Apache-2.0
2 // SPDX-FileCopyrightText: Copyright OpenBMC Authors
3 // SPDX-FileCopyrightText: Copyright 2019 Intel Corporation
4 #pragma once
5 
6 #include "async_resp.hpp"
7 #include "dbus_singleton.hpp"
8 #include "dbus_utility.hpp"
9 #include "error_messages.hpp"
10 #include "logging.hpp"
11 #include "utils/chassis_utils.hpp"
12 
13 #include <boost/system/errc.hpp>
14 #include <boost/system/error_code.hpp>
15 #include <sdbusplus/message/native_types.hpp>
16 
17 #include <algorithm>
18 #include <charconv>
19 #include <cstddef>
20 #include <cstdint>
21 #include <memory>
22 #include <ranges>
23 #include <span>
24 #include <string>
25 #include <string_view>
26 #include <system_error>
27 #include <tuple>
28 #include <utility>
29 #include <vector>
30 
31 namespace redfish
32 {
33 
34 enum NetworkProtocolUnitStructFields
35 {
36     NET_PROTO_UNIT_NAME,
37     NET_PROTO_UNIT_DESC,
38     NET_PROTO_UNIT_LOAD_STATE,
39     NET_PROTO_UNIT_ACTIVE_STATE,
40     NET_PROTO_UNIT_SUB_STATE,
41     NET_PROTO_UNIT_DEVICE,
42     NET_PROTO_UNIT_OBJ_PATH,
43     NET_PROTO_UNIT_ALWAYS_0,
44     NET_PROTO_UNIT_ALWAYS_EMPTY,
45     NET_PROTO_UNIT_ALWAYS_ROOT_PATH
46 };
47 
48 enum NetworkProtocolListenResponseElements
49 {
50     NET_PROTO_LISTEN_TYPE,
51     NET_PROTO_LISTEN_STREAM
52 };
53 
54 /**
55  * @brief D-Bus Unit structure returned in array from ListUnits Method
56  */
57 using UnitStruct =
58     std::tuple<std::string, std::string, std::string, std::string, std::string,
59                std::string, sdbusplus::message::object_path, uint32_t,
60                std::string, sdbusplus::message::object_path>;
61 
62 template <typename CallbackFunc>
getMainChassisId(const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,CallbackFunc && callback)63 void getMainChassisId(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
64                       CallbackFunc&& callback)
65 {
66     // Find managed chassis
67     dbus::utility::getSubTree(
68         "/xyz/openbmc_project/inventory", 0, chassisInterfaces,
69         [callback = std::forward<CallbackFunc>(callback),
70          asyncResp](const boost::system::error_code& ec,
71                     const dbus::utility::MapperGetSubTreeResponse& subtree) {
72             if (ec)
73             {
74                 BMCWEB_LOG_ERROR("{}", ec);
75                 return;
76             }
77             if (subtree.empty())
78             {
79                 BMCWEB_LOG_DEBUG("Can't find chassis!");
80                 return;
81             }
82 
83             std::size_t idPos = subtree[0].first.rfind('/');
84             if (idPos == std::string::npos ||
85                 (idPos + 1) >= subtree[0].first.size())
86             {
87                 messages::internalError(asyncResp->res);
88                 BMCWEB_LOG_DEBUG("Can't parse chassis ID!");
89                 return;
90             }
91             std::string chassisId = subtree[0].first.substr(idPos + 1);
92             BMCWEB_LOG_DEBUG("chassisId = {}", chassisId);
93             callback(chassisId, asyncResp);
94         });
95 }
96 
97 template <typename CallbackFunc>
getPortStatusAndPath(std::span<const std::pair<std::string_view,std::string_view>> protocolToDBus,CallbackFunc && callback)98 void getPortStatusAndPath(
99     std::span<const std::pair<std::string_view, std::string_view>>
100         protocolToDBus,
101     CallbackFunc&& callback)
102 {
103     crow::connections::systemBus->async_method_call(
104         [protocolToDBus, callback = std::forward<CallbackFunc>(callback)](
105             const boost::system::error_code& ec,
106             const std::vector<UnitStruct>& r) {
107             std::vector<std::tuple<std::string, std::string, bool>> socketData;
108             if (ec)
109             {
110                 BMCWEB_LOG_ERROR("{}", ec);
111                 // return error code
112                 callback(ec, socketData);
113                 return;
114             }
115 
116             // save all service output into vector
117             for (const UnitStruct& unit : r)
118             {
119                 // Only traverse through <xyz>.socket units
120                 const std::string& unitName =
121                     std::get<NET_PROTO_UNIT_NAME>(unit);
122 
123                 // find "." into unitsName
124                 size_t lastCharPos = unitName.rfind('.');
125                 if (lastCharPos == std::string::npos)
126                 {
127                     continue;
128                 }
129 
130                 // is unitsName end with ".socket"
131                 std::string unitNameEnd = unitName.substr(lastCharPos);
132                 if (unitNameEnd != ".socket")
133                 {
134                     continue;
135                 }
136 
137                 // find "@" into unitsName
138                 if (size_t atCharPos = unitName.rfind('@');
139                     atCharPos != std::string::npos)
140                 {
141                     lastCharPos = atCharPos;
142                 }
143 
144                 // unitsName without "@eth(x).socket", only <xyz>
145                 // unitsName without ".socket", only <xyz>
146                 std::string unitNameStr = unitName.substr(0, lastCharPos);
147 
148                 for (const auto& kv : protocolToDBus)
149                 {
150                     // We are interested in services, which starts with
151                     // mapped service name
152                     if (unitNameStr != kv.second)
153                     {
154                         continue;
155                     }
156 
157                     const std::string& socketPath =
158                         std::get<NET_PROTO_UNIT_OBJ_PATH>(unit);
159                     const std::string& unitState =
160                         std::get<NET_PROTO_UNIT_SUB_STATE>(unit);
161 
162                     bool isProtocolEnabled = ((unitState == "running") ||
163                                               (unitState == "listening"));
164 
165                     // Some protocols may have multiple services associated with
166                     // them (for example IPMI). Look to see if we've already
167                     // added an entry for the current protocol.
168                     auto find = std::ranges::find_if(
169                         socketData,
170                         [&kv](const std::tuple<std::string, std::string, bool>&
171                                   i) { return std::get<1>(i) == kv.first; });
172                     if (find != socketData.end())
173                     {
174                         // It only takes one enabled systemd service to consider
175                         // a protocol enabled so if the current entry already
176                         // has it enabled (or the new one is disabled) then just
177                         // continue, otherwise remove the current one and add
178                         // this new one.
179                         if (std::get<2>(*find) || !isProtocolEnabled)
180                         {
181                             // Already registered as enabled or current one is
182                             // not enabled, nothing to do
183                             BMCWEB_LOG_DEBUG(
184                                 "protocolName: {}, already true or current one is false: {}",
185                                 kv.first, isProtocolEnabled);
186                             break;
187                         }
188                         // Remove existing entry and replace with new one
189                         // (below)
190                         socketData.erase(find);
191                     }
192 
193                     socketData.emplace_back(socketPath, std::string(kv.first),
194                                             isProtocolEnabled);
195                     // We found service, return from inner loop.
196                     break;
197                 }
198             }
199 
200             callback(ec, socketData);
201         },
202         "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
203         "org.freedesktop.systemd1.Manager", "ListUnits");
204 }
205 
206 template <typename CallbackFunc>
getPortNumber(const std::string & socketPath,CallbackFunc && callback)207 void getPortNumber(const std::string& socketPath, CallbackFunc&& callback)
208 {
209     dbus::utility::getProperty<
210         std::vector<std::tuple<std::string, std::string>>>(
211         *crow::connections::systemBus, "org.freedesktop.systemd1", socketPath,
212         "org.freedesktop.systemd1.Socket", "Listen",
213         [callback = std::forward<CallbackFunc>(callback)](
214             const boost::system::error_code& ec,
215             const std::vector<std::tuple<std::string, std::string>>& resp) {
216             if (ec)
217             {
218                 BMCWEB_LOG_ERROR("{}", ec);
219                 callback(ec, 0);
220                 return;
221             }
222             if (resp.empty())
223             {
224                 // Network Protocol Listen Response Elements is empty
225                 boost::system::error_code ec1 =
226                     boost::system::errc::make_error_code(
227                         boost::system::errc::bad_message);
228                 // return error code
229                 callback(ec1, 0);
230                 BMCWEB_LOG_ERROR("{}", ec1);
231                 return;
232             }
233             const std::string& listenStream =
234                 std::get<NET_PROTO_LISTEN_STREAM>(resp[0]);
235             const char* pa = &listenStream[listenStream.rfind(':') + 1];
236             int port{0};
237             if (auto [p, ec2] = std::from_chars(pa, nullptr, port);
238                 ec2 != std::errc())
239             {
240                 // there is only two possibility invalid_argument and
241                 // result_out_of_range
242                 boost::system::error_code ec3 =
243                     boost::system::errc::make_error_code(
244                         boost::system::errc::invalid_argument);
245                 if (ec2 == std::errc::result_out_of_range)
246                 {
247                     ec3 = boost::system::errc::make_error_code(
248                         boost::system::errc::result_out_of_range);
249                 }
250                 // return error code
251                 callback(ec3, 0);
252                 BMCWEB_LOG_ERROR("{}", ec3);
253             }
254             callback(ec, port);
255         });
256 }
257 
258 } // namespace redfish
259