blob: ec905dd8612cca185285817f1670c7c066e6dbfd [file] [log] [blame]
Ed Tanous40e9b922024-09-10 13:50:16 -07001// SPDX-License-Identifier: Apache-2.0
2// SPDX-FileCopyrightText: Copyright OpenBMC Authors
3// SPDX-FileCopyrightText: Copyright 2019 Intel Corporation
Jennifer Leec5d03ff2019-03-08 15:42:58 -08004#pragma once
5
Ed Tanous3ccb3ad2023-01-13 17:40:03 -08006#include "async_resp.hpp"
Ed Tanous177612a2025-02-14 15:16:09 -08007#include "boost_formatters.hpp"
Ed Tanousd7857202025-01-28 15:32:26 -08008#include "dbus_singleton.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -08009#include "dbus_utility.hpp"
10#include "error_messages.hpp"
Ed Tanousd7857202025-01-28 15:32:26 -080011#include "logging.hpp"
Myung Bae3f95a272024-03-13 07:32:02 -070012#include "utils/chassis_utils.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080013
Ed Tanousd7857202025-01-28 15:32:26 -080014#include <boost/system/errc.hpp>
George Liue99073f2022-12-09 11:06:16 +080015#include <boost/system/error_code.hpp>
Ed Tanousd7857202025-01-28 15:32:26 -080016#include <sdbusplus/message/native_types.hpp>
Nan Zhoucf7eba02022-07-21 23:53:20 +000017
Ed Tanousd7857202025-01-28 15:32:26 -080018#include <algorithm>
Nan Zhoucf7eba02022-07-21 23:53:20 +000019#include <charconv>
Ed Tanousd7857202025-01-28 15:32:26 -080020#include <cstddef>
21#include <cstdint>
22#include <memory>
Ed Tanous3544d2a2023-08-06 18:12:20 -070023#include <ranges>
Ed Tanousd7857202025-01-28 15:32:26 -080024#include <span>
25#include <string>
George Liue99073f2022-12-09 11:06:16 +080026#include <string_view>
Ed Tanousd7857202025-01-28 15:32:26 -080027#include <system_error>
28#include <tuple>
29#include <utility>
30#include <vector>
Nan Zhoucf7eba02022-07-21 23:53:20 +000031
Jennifer Leec5d03ff2019-03-08 15:42:58 -080032namespace redfish
33{
34
Abhishek Patelb4bec662021-06-21 17:32:02 -050035enum NetworkProtocolUnitStructFields
36{
37 NET_PROTO_UNIT_NAME,
38 NET_PROTO_UNIT_DESC,
39 NET_PROTO_UNIT_LOAD_STATE,
40 NET_PROTO_UNIT_ACTIVE_STATE,
41 NET_PROTO_UNIT_SUB_STATE,
42 NET_PROTO_UNIT_DEVICE,
43 NET_PROTO_UNIT_OBJ_PATH,
44 NET_PROTO_UNIT_ALWAYS_0,
45 NET_PROTO_UNIT_ALWAYS_EMPTY,
46 NET_PROTO_UNIT_ALWAYS_ROOT_PATH
47};
48
49enum NetworkProtocolListenResponseElements
50{
51 NET_PROTO_LISTEN_TYPE,
52 NET_PROTO_LISTEN_STREAM
53};
54
55/**
56 * @brief D-Bus Unit structure returned in array from ListUnits Method
57 */
58using UnitStruct =
59 std::tuple<std::string, std::string, std::string, std::string, std::string,
60 std::string, sdbusplus::message::object_path, uint32_t,
61 std::string, sdbusplus::message::object_path>;
62
Jennifer Leec5d03ff2019-03-08 15:42:58 -080063template <typename CallbackFunc>
Ed Tanousdaadfb22024-12-20 09:25:54 -080064void getMainChassisId(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
Jennifer Leec5d03ff2019-03-08 15:42:58 -080065 CallbackFunc&& callback)
66{
67 // Find managed chassis
George Liue99073f2022-12-09 11:06:16 +080068 dbus::utility::getSubTree(
Myung Bae3f95a272024-03-13 07:32:02 -070069 "/xyz/openbmc_project/inventory", 0, chassisInterfaces,
Ed Tanous8cb2c022024-03-27 16:31:46 -070070 [callback = std::forward<CallbackFunc>(callback),
George Liue99073f2022-12-09 11:06:16 +080071 asyncResp](const boost::system::error_code& ec,
Ed Tanousb9d36b42022-02-26 21:42:46 -080072 const dbus::utility::MapperGetSubTreeResponse& subtree) {
Patrick Williamsbd79bce2024-08-16 15:22:20 -040073 if (ec)
74 {
75 BMCWEB_LOG_ERROR("{}", ec);
76 return;
77 }
78 if (subtree.empty())
79 {
80 BMCWEB_LOG_DEBUG("Can't find chassis!");
81 return;
82 }
Jennifer Leec5d03ff2019-03-08 15:42:58 -080083
Patrick Williamsbd79bce2024-08-16 15:22:20 -040084 std::size_t idPos = subtree[0].first.rfind('/');
85 if (idPos == std::string::npos ||
86 (idPos + 1) >= subtree[0].first.size())
87 {
88 messages::internalError(asyncResp->res);
89 BMCWEB_LOG_DEBUG("Can't parse chassis ID!");
90 return;
91 }
92 std::string chassisId = subtree[0].first.substr(idPos + 1);
93 BMCWEB_LOG_DEBUG("chassisId = {}", chassisId);
94 callback(chassisId, asyncResp);
95 });
Jennifer Leec5d03ff2019-03-08 15:42:58 -080096}
Abhishek Patelb4bec662021-06-21 17:32:02 -050097
98template <typename CallbackFunc>
Abhishek Patel5c3e9272021-06-24 10:11:33 -050099void getPortStatusAndPath(
100 std::span<const std::pair<std::string_view, std::string_view>>
101 protocolToDBus,
102 CallbackFunc&& callback)
Abhishek Patelb4bec662021-06-21 17:32:02 -0500103{
Ed Tanous177612a2025-02-14 15:16:09 -0800104 dbus::utility::async_method_call(
Ed Tanous8cb2c022024-03-27 16:31:46 -0700105 [protocolToDBus, callback = std::forward<CallbackFunc>(callback)](
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800106 const boost::system::error_code& ec,
Ed Tanousf94c4ec2022-01-06 12:44:41 -0800107 const std::vector<UnitStruct>& r) {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400108 std::vector<std::tuple<std::string, std::string, bool>> socketData;
109 if (ec)
Ed Tanous002d39b2022-05-31 08:59:27 -0700110 {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400111 BMCWEB_LOG_ERROR("{}", ec);
112 // return error code
113 callback(ec, socketData);
114 return;
Ed Tanous002d39b2022-05-31 08:59:27 -0700115 }
116
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400117 // save all service output into vector
118 for (const UnitStruct& unit : r)
Ed Tanous002d39b2022-05-31 08:59:27 -0700119 {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400120 // Only traverse through <xyz>.socket units
121 const std::string& unitName =
122 std::get<NET_PROTO_UNIT_NAME>(unit);
Ed Tanous002d39b2022-05-31 08:59:27 -0700123
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400124 // find "." into unitsName
125 size_t lastCharPos = unitName.rfind('.');
126 if (lastCharPos == std::string::npos)
Abhishek Patel5c3e9272021-06-24 10:11:33 -0500127 {
128 continue;
129 }
130
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400131 // is unitsName end with ".socket"
132 std::string unitNameEnd = unitName.substr(lastCharPos);
133 if (unitNameEnd != ".socket")
Andrew Geisslere6feae52023-04-06 10:27:41 -0600134 {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400135 continue;
Andrew Geisslere6feae52023-04-06 10:27:41 -0600136 }
137
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400138 // find "@" into unitsName
139 if (size_t atCharPos = unitName.rfind('@');
140 atCharPos != std::string::npos)
141 {
142 lastCharPos = atCharPos;
143 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700144
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400145 // unitsName without "@eth(x).socket", only <xyz>
146 // unitsName without ".socket", only <xyz>
147 std::string unitNameStr = unitName.substr(0, lastCharPos);
148
149 for (const auto& kv : protocolToDBus)
150 {
151 // We are interested in services, which starts with
152 // mapped service name
153 if (unitNameStr != kv.second)
154 {
155 continue;
156 }
157
158 const std::string& socketPath =
159 std::get<NET_PROTO_UNIT_OBJ_PATH>(unit);
160 const std::string& unitState =
161 std::get<NET_PROTO_UNIT_SUB_STATE>(unit);
162
163 bool isProtocolEnabled = ((unitState == "running") ||
164 (unitState == "listening"));
165
166 // Some protocols may have multiple services associated with
167 // them (for example IPMI). Look to see if we've already
168 // added an entry for the current protocol.
169 auto find = std::ranges::find_if(
170 socketData,
171 [&kv](const std::tuple<std::string, std::string, bool>&
172 i) { return std::get<1>(i) == kv.first; });
173 if (find != socketData.end())
174 {
175 // It only takes one enabled systemd service to consider
176 // a protocol enabled so if the current entry already
177 // has it enabled (or the new one is disabled) then just
178 // continue, otherwise remove the current one and add
179 // this new one.
180 if (std::get<2>(*find) || !isProtocolEnabled)
181 {
182 // Already registered as enabled or current one is
183 // not enabled, nothing to do
184 BMCWEB_LOG_DEBUG(
185 "protocolName: {}, already true or current one is false: {}",
186 kv.first, isProtocolEnabled);
187 break;
188 }
189 // Remove existing entry and replace with new one
190 // (below)
191 socketData.erase(find);
192 }
193
194 socketData.emplace_back(socketPath, std::string(kv.first),
195 isProtocolEnabled);
196 // We found service, return from inner loop.
197 break;
198 }
199 }
200
201 callback(ec, socketData);
202 },
Abhishek Patelb4bec662021-06-21 17:32:02 -0500203 "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
204 "org.freedesktop.systemd1.Manager", "ListUnits");
205}
206
207template <typename CallbackFunc>
208void getPortNumber(const std::string& socketPath, CallbackFunc&& callback)
209{
Ed Tanousdeae6a72024-11-11 21:58:57 -0800210 dbus::utility::getProperty<
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700211 std::vector<std::tuple<std::string, std::string>>>(
212 *crow::connections::systemBus, "org.freedesktop.systemd1", socketPath,
213 "org.freedesktop.systemd1.Socket", "Listen",
Ed Tanous8cb2c022024-03-27 16:31:46 -0700214 [callback = std::forward<CallbackFunc>(callback)](
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800215 const boost::system::error_code& ec,
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700216 const std::vector<std::tuple<std::string, std::string>>& resp) {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400217 if (ec)
Abhishek Patelb4bec662021-06-21 17:32:02 -0500218 {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400219 BMCWEB_LOG_ERROR("{}", ec);
220 callback(ec, 0);
221 return;
Abhishek Patelb4bec662021-06-21 17:32:02 -0500222 }
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400223 if (resp.empty())
224 {
225 // Network Protocol Listen Response Elements is empty
226 boost::system::error_code ec1 =
227 boost::system::errc::make_error_code(
228 boost::system::errc::bad_message);
229 // return error code
230 callback(ec1, 0);
231 BMCWEB_LOG_ERROR("{}", ec1);
232 return;
233 }
234 const std::string& listenStream =
235 std::get<NET_PROTO_LISTEN_STREAM>(resp[0]);
236 const char* pa = &listenStream[listenStream.rfind(':') + 1];
237 int port{0};
238 if (auto [p, ec2] = std::from_chars(pa, nullptr, port);
239 ec2 != std::errc())
240 {
241 // there is only two possibility invalid_argument and
242 // result_out_of_range
243 boost::system::error_code ec3 =
244 boost::system::errc::make_error_code(
245 boost::system::errc::invalid_argument);
246 if (ec2 == std::errc::result_out_of_range)
247 {
248 ec3 = boost::system::errc::make_error_code(
249 boost::system::errc::result_out_of_range);
250 }
251 // return error code
252 callback(ec3, 0);
253 BMCWEB_LOG_ERROR("{}", ec3);
254 }
255 callback(ec, port);
256 });
Abhishek Patelb4bec662021-06-21 17:32:02 -0500257}
258
Jennifer Leec5d03ff2019-03-08 15:42:58 -0800259} // namespace redfish