blob: 9d358c02cf068488f258c54f68375e8df0312dfc [file] [log] [blame]
Jennifer Leec5d03ff2019-03-08 15:42:58 -08001/*
2// Copyright (c) 2019 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
Jennifer Leec5d03ff2019-03-08 15:42:58 -080016#pragma once
Ed Tanouse585b902021-11-02 15:50:54 -070017#ifndef BMCWEB_ENABLE_REDFISH_ONE_CHASSIS
Jennifer Leec5d03ff2019-03-08 15:42:58 -080018
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080019#include "async_resp.hpp"
20#include "dbus_utility.hpp"
21#include "error_messages.hpp"
22
George Liue99073f2022-12-09 11:06:16 +080023#include <boost/system/error_code.hpp>
Nan Zhoucf7eba02022-07-21 23:53:20 +000024#include <sdbusplus/asio/property.hpp>
25
George Liue99073f2022-12-09 11:06:16 +080026#include <array>
Nan Zhoucf7eba02022-07-21 23:53:20 +000027#include <charconv>
George Liue99073f2022-12-09 11:06:16 +080028#include <string_view>
Nan Zhoucf7eba02022-07-21 23:53:20 +000029
Jennifer Leec5d03ff2019-03-08 15:42:58 -080030namespace redfish
31{
32
Abhishek Patelb4bec662021-06-21 17:32:02 -050033enum NetworkProtocolUnitStructFields
34{
35 NET_PROTO_UNIT_NAME,
36 NET_PROTO_UNIT_DESC,
37 NET_PROTO_UNIT_LOAD_STATE,
38 NET_PROTO_UNIT_ACTIVE_STATE,
39 NET_PROTO_UNIT_SUB_STATE,
40 NET_PROTO_UNIT_DEVICE,
41 NET_PROTO_UNIT_OBJ_PATH,
42 NET_PROTO_UNIT_ALWAYS_0,
43 NET_PROTO_UNIT_ALWAYS_EMPTY,
44 NET_PROTO_UNIT_ALWAYS_ROOT_PATH
45};
46
47enum NetworkProtocolListenResponseElements
48{
49 NET_PROTO_LISTEN_TYPE,
50 NET_PROTO_LISTEN_STREAM
51};
52
53/**
54 * @brief D-Bus Unit structure returned in array from ListUnits Method
55 */
56using UnitStruct =
57 std::tuple<std::string, std::string, std::string, std::string, std::string,
58 std::string, sdbusplus::message::object_path, uint32_t,
59 std::string, sdbusplus::message::object_path>;
60
Jennifer Leec5d03ff2019-03-08 15:42:58 -080061template <typename CallbackFunc>
zhanghch058d1b46d2021-04-01 11:18:24 +080062void getMainChassisId(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
Jennifer Leec5d03ff2019-03-08 15:42:58 -080063 CallbackFunc&& callback)
64{
65 // Find managed chassis
George Liue99073f2022-12-09 11:06:16 +080066 constexpr std::array<std::string_view, 2> interfaces = {
67 "xyz.openbmc_project.Inventory.Item.Board",
68 "xyz.openbmc_project.Inventory.Item.Chassis"};
69 dbus::utility::getSubTree(
70 "/xyz/openbmc_project/inventory", 0, interfaces,
Jennifer Leec5d03ff2019-03-08 15:42:58 -080071 [callback,
George Liue99073f2022-12-09 11:06:16 +080072 asyncResp](const boost::system::error_code& ec,
Ed Tanousb9d36b42022-02-26 21:42:46 -080073 const dbus::utility::MapperGetSubTreeResponse& subtree) {
Ed Tanous002d39b2022-05-31 08:59:27 -070074 if (ec)
75 {
76 BMCWEB_LOG_ERROR << ec;
77 return;
78 }
79 if (subtree.empty())
80 {
81 BMCWEB_LOG_DEBUG << "Can't find chassis!";
82 return;
83 }
Jennifer Leec5d03ff2019-03-08 15:42:58 -080084
Ed Tanous002d39b2022-05-31 08:59:27 -070085 std::size_t idPos = subtree[0].first.rfind('/');
86 if (idPos == std::string::npos ||
87 (idPos + 1) >= subtree[0].first.size())
88 {
89 messages::internalError(asyncResp->res);
90 BMCWEB_LOG_DEBUG << "Can't parse chassis ID!";
91 return;
92 }
93 std::string chassisId = subtree[0].first.substr(idPos + 1);
94 BMCWEB_LOG_DEBUG << "chassisId = " << chassisId;
95 callback(chassisId, asyncResp);
George Liue99073f2022-12-09 11:06:16 +080096 });
Jennifer Leec5d03ff2019-03-08 15:42:58 -080097}
Abhishek Patelb4bec662021-06-21 17:32:02 -050098
99template <typename CallbackFunc>
100void getPortStatusAndPath(const std::string& serviceName,
101 CallbackFunc&& callback)
102{
103 crow::connections::systemBus->async_method_call(
Ed Tanousf94c4ec2022-01-06 12:44:41 -0800104 [serviceName, callback{std::forward<CallbackFunc>(callback)}](
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800105 const boost::system::error_code& ec,
Ed Tanousf94c4ec2022-01-06 12:44:41 -0800106 const std::vector<UnitStruct>& r) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700107 if (ec)
108 {
109 BMCWEB_LOG_ERROR << ec;
Abhishek Patelb4bec662021-06-21 17:32:02 -0500110 // return error code
Ed Tanous002d39b2022-05-31 08:59:27 -0700111 callback(ec, "", false);
112 return;
113 }
114
115 for (const UnitStruct& unit : r)
116 {
117 // Only traverse through <xyz>.socket units
118 const std::string& unitName = std::get<NET_PROTO_UNIT_NAME>(unit);
119
120 // find "." into unitsName
121 size_t lastCharPos = unitName.rfind('.');
122 if (lastCharPos == std::string::npos)
123 {
124 continue;
125 }
126
127 // is unitsName end with ".socket"
128 std::string unitNameEnd = unitName.substr(lastCharPos);
129 if (unitNameEnd != ".socket")
130 {
131 continue;
132 }
133
134 // find "@" into unitsName
135 if (size_t atCharPos = unitName.rfind('@');
136 atCharPos != std::string::npos)
137 {
138 lastCharPos = atCharPos;
139 }
140
141 // unitsName without "@eth(x).socket", only <xyz>
142 // unitsName without ".socket", only <xyz>
143 std::string unitNameStr = unitName.substr(0, lastCharPos);
144
145 // We are interested in services, which starts with
146 // mapped service name
147 if (unitNameStr != serviceName)
148 {
149 continue;
150 }
151
152 const std::string& socketPath =
153 std::get<NET_PROTO_UNIT_OBJ_PATH>(unit);
154 const std::string& unitState =
155 std::get<NET_PROTO_UNIT_SUB_STATE>(unit);
156
157 bool isProtocolEnabled =
158 ((unitState == "running") || (unitState == "listening"));
159 // We found service, return from inner loop.
160 callback(ec, socketPath, isProtocolEnabled);
161 return;
162 }
163
164 // no service foudn, throw error
165 boost::system::error_code ec1 = boost::system::errc::make_error_code(
166 boost::system::errc::no_such_process);
167 // return error code
168 callback(ec1, "", false);
169 BMCWEB_LOG_ERROR << ec1;
Abhishek Patelb4bec662021-06-21 17:32:02 -0500170 },
171 "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
172 "org.freedesktop.systemd1.Manager", "ListUnits");
173}
174
175template <typename CallbackFunc>
176void getPortNumber(const std::string& socketPath, CallbackFunc&& callback)
177{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700178 sdbusplus::asio::getProperty<
179 std::vector<std::tuple<std::string, std::string>>>(
180 *crow::connections::systemBus, "org.freedesktop.systemd1", socketPath,
181 "org.freedesktop.systemd1.Socket", "Listen",
Ed Tanousf94c4ec2022-01-06 12:44:41 -0800182 [callback{std::forward<CallbackFunc>(callback)}](
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800183 const boost::system::error_code& ec,
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700184 const std::vector<std::tuple<std::string, std::string>>& resp) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700185 if (ec)
186 {
187 BMCWEB_LOG_ERROR << ec;
188 callback(ec, 0);
189 return;
190 }
191 if (resp.empty())
192 {
193 // Network Protocol Listen Response Elements is empty
194 boost::system::error_code ec1 =
195 boost::system::errc::make_error_code(
196 boost::system::errc::bad_message);
197 // return error code
198 callback(ec1, 0);
199 BMCWEB_LOG_ERROR << ec1;
200 return;
201 }
202 const std::string& listenStream =
203 std::get<NET_PROTO_LISTEN_STREAM>(resp[0]);
204 const char* pa = &listenStream[listenStream.rfind(':') + 1];
205 int port{0};
206 if (auto [p, ec2] = std::from_chars(pa, nullptr, port);
207 ec2 != std::errc())
208 {
209 // there is only two possibility invalid_argument and
210 // result_out_of_range
211 boost::system::error_code ec3 =
212 boost::system::errc::make_error_code(
213 boost::system::errc::invalid_argument);
214 if (ec2 == std::errc::result_out_of_range)
Abhishek Patelb4bec662021-06-21 17:32:02 -0500215 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700216 ec3 = boost::system::errc::make_error_code(
217 boost::system::errc::result_out_of_range);
Abhishek Patelb4bec662021-06-21 17:32:02 -0500218 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700219 // return error code
220 callback(ec3, 0);
221 BMCWEB_LOG_ERROR << ec3;
222 }
223 callback(ec, port);
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700224 });
Abhishek Patelb4bec662021-06-21 17:32:02 -0500225}
226
Jennifer Leec5d03ff2019-03-08 15:42:58 -0800227} // namespace redfish
228#endif