blob: 6d5e587b8e2279111739d1992f2779df42c6f663 [file] [log] [blame]
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +01001/*
2// Copyright (c) 2018 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*/
16#pragma once
17
Willy Tu13451e32023-05-24 16:08:18 -070018#include "bmcweb_config.h"
19
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080020#include "app.hpp"
George Liu7a1dbc42022-12-07 16:03:22 +080021#include "dbus_utility.hpp"
James Feistb49ac872019-05-21 15:12:01 -070022#include "health.hpp"
James Feist1c8fba92019-12-20 15:12:07 -080023#include "led.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080024#include "query.hpp"
Chau Ly7164bc62023-10-15 14:55:30 +000025#include "redfish_util.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080026#include "registries/privilege_registry.hpp"
27#include "utils/collection.hpp"
28#include "utils/dbus_utils.hpp"
Nan Zhoucf7eba02022-07-21 23:53:20 +000029#include "utils/json_utils.hpp"
Ed Tanous1abe55e2018-09-05 08:30:59 -070030
George Liue99073f2022-12-09 11:06:16 +080031#include <boost/system/error_code.hpp>
Ed Tanousef4c65b2023-04-24 15:28:50 -070032#include <boost/url/format.hpp>
Jonathan Doman1e1e5982021-06-11 09:36:17 -070033#include <sdbusplus/asio/property.hpp>
Andrew Geisslerfc903b32023-05-31 14:15:42 -040034#include <sdbusplus/message.hpp>
Krzysztof Grobelny86d89ed2022-08-29 14:49:20 +020035#include <sdbusplus/unpack_properties.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050036
George Liu7a1dbc42022-12-07 16:03:22 +080037#include <array>
Ed Tanous3544d2a2023-08-06 18:12:20 -070038#include <ranges>
George Liu7a1dbc42022-12-07 16:03:22 +080039#include <string_view>
40
Ed Tanous1abe55e2018-09-05 08:30:59 -070041namespace redfish
42{
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +010043
44/**
Willy Tu5e577bc2022-07-26 00:41:55 +000045 * @brief Retrieves resources over dbus to link to the chassis
46 *
47 * @param[in] asyncResp - Shared pointer for completing asynchronous
48 * calls
49 * @param[in] path - Chassis dbus path to look for the storage.
50 *
51 * Calls the Association endpoints on the path + "/storage" and add the link of
52 * json["Links"]["Storage@odata.count"] =
53 * {"@odata.id", "/redfish/v1/Storage/" + resourceId}
54 *
55 * @return None.
56 */
57inline void getStorageLink(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
58 const sdbusplus::message::object_path& path)
59{
60 sdbusplus::asio::getProperty<std::vector<std::string>>(
61 *crow::connections::systemBus, "xyz.openbmc_project.ObjectMapper",
62 (path / "storage").str, "xyz.openbmc_project.Association", "endpoints",
Willy Tud4b054c2023-06-12 15:18:45 -070063 [asyncResp](const boost::system::error_code& ec,
Willy Tu5e577bc2022-07-26 00:41:55 +000064 const std::vector<std::string>& storageList) {
65 if (ec)
66 {
Ed Tanous62598e32023-07-17 17:06:25 -070067 BMCWEB_LOG_DEBUG("getStorageLink got DBUS response error");
Willy Tu5e577bc2022-07-26 00:41:55 +000068 return;
69 }
70
71 nlohmann::json::array_t storages;
72 for (const std::string& storagePath : storageList)
73 {
74 std::string id =
75 sdbusplus::message::object_path(storagePath).filename();
76 if (id.empty())
77 {
78 continue;
79 }
80
81 nlohmann::json::object_t storage;
82 storage["@odata.id"] = boost::urls::format(
83 "/redfish/v1/Systems/system/Storage/{}", id);
84 storages.emplace_back(std::move(storage));
85 }
86 asyncResp->res.jsonValue["Links"]["Storage@odata.count"] =
87 storages.size();
88 asyncResp->res.jsonValue["Links"]["Storage"] = std::move(storages);
Patrick Williams5a39f772023-10-20 11:20:21 -050089 });
Willy Tu5e577bc2022-07-26 00:41:55 +000090}
91
92/**
Gunnar Millsbeeca0a2019-02-14 16:30:45 -060093 * @brief Retrieves chassis state properties over dbus
94 *
Ed Tanousac106bf2023-06-07 09:24:59 -070095 * @param[in] asyncResp - Shared pointer for completing asynchronous calls.
Gunnar Millsbeeca0a2019-02-14 16:30:45 -060096 *
97 * @return None.
98 */
Ed Tanousac106bf2023-06-07 09:24:59 -070099inline void getChassisState(std::shared_ptr<bmcweb::AsyncResp> asyncResp)
Gunnar Millsbeeca0a2019-02-14 16:30:45 -0600100{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700101 // crow::connections::systemBus->async_method_call(
102 sdbusplus::asio::getProperty<std::string>(
103 *crow::connections::systemBus, "xyz.openbmc_project.State.Chassis",
104 "/xyz/openbmc_project/state/chassis0",
105 "xyz.openbmc_project.State.Chassis", "CurrentPowerState",
Ed Tanousac106bf2023-06-07 09:24:59 -0700106 [asyncResp{std::move(asyncResp)}](const boost::system::error_code& ec,
107 const std::string& chassisState) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700108 if (ec)
109 {
110 if (ec == boost::system::errc::host_unreachable)
Gunnar Millsbeeca0a2019-02-14 16:30:45 -0600111 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700112 // Service not available, no error, just don't return
113 // chassis state info
Ed Tanous62598e32023-07-17 17:06:25 -0700114 BMCWEB_LOG_DEBUG("Service not available {}", ec);
Gunnar Millsbeeca0a2019-02-14 16:30:45 -0600115 return;
116 }
Ed Tanous62598e32023-07-17 17:06:25 -0700117 BMCWEB_LOG_DEBUG("DBUS response error {}", ec);
Ed Tanousac106bf2023-06-07 09:24:59 -0700118 messages::internalError(asyncResp->res);
Ed Tanous002d39b2022-05-31 08:59:27 -0700119 return;
120 }
Gunnar Millsbeeca0a2019-02-14 16:30:45 -0600121
Ed Tanous62598e32023-07-17 17:06:25 -0700122 BMCWEB_LOG_DEBUG("Chassis state: {}", chassisState);
Ed Tanous002d39b2022-05-31 08:59:27 -0700123 // Verify Chassis State
124 if (chassisState == "xyz.openbmc_project.State.Chassis.PowerState.On")
125 {
Ed Tanousac106bf2023-06-07 09:24:59 -0700126 asyncResp->res.jsonValue["PowerState"] = "On";
127 asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
Ed Tanous002d39b2022-05-31 08:59:27 -0700128 }
129 else if (chassisState ==
130 "xyz.openbmc_project.State.Chassis.PowerState.Off")
131 {
Ed Tanousac106bf2023-06-07 09:24:59 -0700132 asyncResp->res.jsonValue["PowerState"] = "Off";
133 asyncResp->res.jsonValue["Status"]["State"] = "StandbyOffline";
Ed Tanous002d39b2022-05-31 08:59:27 -0700134 }
Patrick Williams5a39f772023-10-20 11:20:21 -0500135 });
Gunnar Millsbeeca0a2019-02-14 16:30:45 -0600136}
137
Qiang XUc1819422019-02-27 13:51:32 +0800138/**
139 * Retrieves physical security properties over dbus
140 */
Chau Ly7164bc62023-10-15 14:55:30 +0000141inline void handlePhysicalSecurityGetSubTree(
142 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
143 const boost::system::error_code& ec,
144 const dbus::utility::MapperGetSubTreeResponse& subtree)
Qiang XUc1819422019-02-27 13:51:32 +0800145{
Chau Ly7164bc62023-10-15 14:55:30 +0000146 if (ec)
147 {
148 // do not add err msg in redfish response, because this is not
149 // mandatory property
150 BMCWEB_LOG_INFO("DBUS error: no matched iface {}", ec);
151 return;
152 }
153 // Iterate over all retrieved ObjectPaths.
154 for (const auto& object : subtree)
155 {
156 if (!object.second.empty())
Ed Tanous002d39b2022-05-31 08:59:27 -0700157 {
Chau Ly7164bc62023-10-15 14:55:30 +0000158 const auto service = object.second.front();
159
160 BMCWEB_LOG_DEBUG("Get intrusion status by service ");
161
162 sdbusplus::asio::getProperty<std::string>(
163 *crow::connections::systemBus, service.first, object.first,
164 "xyz.openbmc_project.Chassis.Intrusion", "Status",
165 [asyncResp](const boost::system::error_code& ec1,
166 const std::string& value) {
167 if (ec1)
168 {
169 // do not add err msg in redfish response, because this is
170 // not
171 // mandatory property
172 BMCWEB_LOG_ERROR("DBUS response error {}", ec1);
173 return;
174 }
175 asyncResp->res
176 .jsonValue["PhysicalSecurity"]["IntrusionSensorNumber"] = 1;
177 asyncResp->res
178 .jsonValue["PhysicalSecurity"]["IntrusionSensor"] = value;
179 });
180
Ed Tanous002d39b2022-05-31 08:59:27 -0700181 return;
182 }
Chau Ly7164bc62023-10-15 14:55:30 +0000183 }
Qiang XUc1819422019-02-27 13:51:32 +0800184}
185
Nan Zhoucf7eba02022-07-21 23:53:20 +0000186inline void handleChassisCollectionGet(
187 App& app, const crow::Request& req,
188 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
189{
190 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
191 {
192 return;
193 }
194 asyncResp->res.jsonValue["@odata.type"] =
195 "#ChassisCollection.ChassisCollection";
196 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Chassis";
197 asyncResp->res.jsonValue["Name"] = "Chassis Collection";
198
George Liu7a1dbc42022-12-07 16:03:22 +0800199 constexpr std::array<std::string_view, 2> interfaces{
200 "xyz.openbmc_project.Inventory.Item.Board",
201 "xyz.openbmc_project.Inventory.Item.Chassis"};
Nan Zhoucf7eba02022-07-21 23:53:20 +0000202 collection_util::getCollectionMembers(
Lakshmi Yadlapati36b5f1e2023-09-26 23:53:28 -0500203 asyncResp, boost::urls::url("/redfish/v1/Chassis"), interfaces,
204 "/xyz/openbmc_project/inventory");
Nan Zhoucf7eba02022-07-21 23:53:20 +0000205}
206
Jie Yanga5617492021-06-29 12:59:14 -0700207inline void getChassisContainedBy(
208 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
209 const std::string& chassisId, const boost::system::error_code& ec,
210 const dbus::utility::MapperEndPoints& upstreamChassisPaths)
211{
212 if (ec)
213 {
214 if (ec.value() != EBADR)
215 {
Ed Tanous62598e32023-07-17 17:06:25 -0700216 BMCWEB_LOG_ERROR("DBUS response error {}", ec);
Jie Yanga5617492021-06-29 12:59:14 -0700217 messages::internalError(asyncResp->res);
218 }
219 return;
220 }
221 if (upstreamChassisPaths.empty())
222 {
223 return;
224 }
225 if (upstreamChassisPaths.size() > 1)
226 {
Ed Tanous8ece0e42024-01-02 13:16:50 -0800227 BMCWEB_LOG_ERROR("{} is contained by multiple chassis", chassisId);
Jie Yanga5617492021-06-29 12:59:14 -0700228 messages::internalError(asyncResp->res);
229 return;
230 }
231
232 sdbusplus::message::object_path upstreamChassisPath(
233 upstreamChassisPaths[0]);
234 std::string upstreamChassis = upstreamChassisPath.filename();
235 if (upstreamChassis.empty())
236 {
Ed Tanous62598e32023-07-17 17:06:25 -0700237 BMCWEB_LOG_WARNING("Malformed upstream Chassis path {} on {}",
238 upstreamChassisPath.str, chassisId);
Jie Yanga5617492021-06-29 12:59:14 -0700239 return;
240 }
241
242 asyncResp->res.jsonValue["Links"]["ContainedBy"]["@odata.id"] =
243 boost::urls::format("/redfish/v1/Chassis/{}", upstreamChassis);
244}
245
246inline void getChassisContains(
247 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
248 const std::string& chassisId, const boost::system::error_code& ec,
249 const dbus::utility::MapperEndPoints& downstreamChassisPaths)
250{
251 if (ec)
252 {
253 if (ec.value() != EBADR)
254 {
Ed Tanous62598e32023-07-17 17:06:25 -0700255 BMCWEB_LOG_ERROR("DBUS response error {}", ec);
Jie Yanga5617492021-06-29 12:59:14 -0700256 messages::internalError(asyncResp->res);
257 }
258 return;
259 }
260 if (downstreamChassisPaths.empty())
261 {
262 return;
263 }
264 nlohmann::json& jValue = asyncResp->res.jsonValue["Links"]["Contains"];
265 if (!jValue.is_array())
266 {
267 // Create the array if it was empty
268 jValue = nlohmann::json::array();
269 }
270 for (const auto& p : downstreamChassisPaths)
271 {
272 sdbusplus::message::object_path downstreamChassisPath(p);
273 std::string downstreamChassis = downstreamChassisPath.filename();
274 if (downstreamChassis.empty())
275 {
Ed Tanous62598e32023-07-17 17:06:25 -0700276 BMCWEB_LOG_WARNING("Malformed downstream Chassis path {} on {}",
277 downstreamChassisPath.str, chassisId);
Jie Yanga5617492021-06-29 12:59:14 -0700278 continue;
279 }
280 nlohmann::json link;
281 link["@odata.id"] = boost::urls::format("/redfish/v1/Chassis/{}",
282 downstreamChassis);
283 jValue.push_back(std::move(link));
284 }
285 asyncResp->res.jsonValue["Links"]["Contains@odata.count"] = jValue.size();
286}
287
288inline void
289 getChassisConnectivity(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
290 const std::string& chassisId,
291 const std::string& chassisPath)
292{
Ed Tanous62598e32023-07-17 17:06:25 -0700293 BMCWEB_LOG_DEBUG("Get chassis connectivity");
Jie Yanga5617492021-06-29 12:59:14 -0700294
295 dbus::utility::getAssociationEndPoints(
296 chassisPath + "/contained_by",
297 std::bind_front(getChassisContainedBy, asyncResp, chassisId));
298
299 dbus::utility::getAssociationEndPoints(
300 chassisPath + "/containing",
301 std::bind_front(getChassisContains, asyncResp, chassisId));
302}
303
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100304/**
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100305 * ChassisCollection derived class for delivering Chassis Collection Schema
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700306 * Functions triggers appropriate requests on DBus
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100307 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700308inline void requestRoutesChassisCollection(App& app)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700309{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700310 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/")
Ed Tanoused398212021-06-09 17:05:54 -0700311 .privileges(redfish::privileges::getChassisCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700312 .methods(boost::beast::http::verb::get)(
Nan Zhoucf7eba02022-07-21 23:53:20 +0000313 std::bind_front(handleChassisCollectionGet, std::ref(app)));
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700314}
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100315
Willy Tu308f70c2021-09-28 20:24:52 -0700316inline void
317 getChassisLocationCode(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
318 const std::string& connectionName,
319 const std::string& path)
320{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700321 sdbusplus::asio::getProperty<std::string>(
322 *crow::connections::systemBus, connectionName, path,
323 "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800324 [asyncResp](const boost::system::error_code& ec,
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700325 const std::string& property) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700326 if (ec)
327 {
Ed Tanous62598e32023-07-17 17:06:25 -0700328 BMCWEB_LOG_ERROR("DBUS response error for Location");
Ed Tanous002d39b2022-05-31 08:59:27 -0700329 messages::internalError(asyncResp->res);
330 return;
331 }
Willy Tu308f70c2021-09-28 20:24:52 -0700332
Ed Tanous002d39b2022-05-31 08:59:27 -0700333 asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
334 property;
Patrick Williams5a39f772023-10-20 11:20:21 -0500335 });
Willy Tu308f70c2021-09-28 20:24:52 -0700336}
337
338inline void getChassisUUID(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
339 const std::string& connectionName,
340 const std::string& path)
341{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700342 sdbusplus::asio::getProperty<std::string>(
343 *crow::connections::systemBus, connectionName, path,
344 "xyz.openbmc_project.Common.UUID", "UUID",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800345 [asyncResp](const boost::system::error_code& ec,
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700346 const std::string& chassisUUID) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700347 if (ec)
348 {
Ed Tanous62598e32023-07-17 17:06:25 -0700349 BMCWEB_LOG_ERROR("DBUS response error for UUID");
Ed Tanous002d39b2022-05-31 08:59:27 -0700350 messages::internalError(asyncResp->res);
351 return;
352 }
353 asyncResp->res.jsonValue["UUID"] = chassisUUID;
Patrick Williams5a39f772023-10-20 11:20:21 -0500354 });
Willy Tu308f70c2021-09-28 20:24:52 -0700355}
356
Chau Ly7164bc62023-10-15 14:55:30 +0000357inline void handleDecoratorAssetProperties(
358 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
359 const std::string& chassisId, const std::string& path,
360 const dbus::utility::DBusPropertiesMap& propertiesList)
361{
362 const std::string* partNumber = nullptr;
363 const std::string* serialNumber = nullptr;
364 const std::string* manufacturer = nullptr;
365 const std::string* model = nullptr;
366 const std::string* sparePartNumber = nullptr;
367
368 const bool success = sdbusplus::unpackPropertiesNoThrow(
369 dbus_utils::UnpackErrorPrinter(), propertiesList, "PartNumber",
370 partNumber, "SerialNumber", serialNumber, "Manufacturer", manufacturer,
371 "Model", model, "SparePartNumber", sparePartNumber);
372
373 if (!success)
374 {
375 messages::internalError(asyncResp->res);
376 return;
377 }
378
379 if (partNumber != nullptr)
380 {
381 asyncResp->res.jsonValue["PartNumber"] = *partNumber;
382 }
383
384 if (serialNumber != nullptr)
385 {
386 asyncResp->res.jsonValue["SerialNumber"] = *serialNumber;
387 }
388
389 if (manufacturer != nullptr)
390 {
391 asyncResp->res.jsonValue["Manufacturer"] = *manufacturer;
392 }
393
394 if (model != nullptr)
395 {
396 asyncResp->res.jsonValue["Model"] = *model;
397 }
398
399 // SparePartNumber is optional on D-Bus
400 // so skip if it is empty
401 if (sparePartNumber != nullptr && !sparePartNumber->empty())
402 {
403 asyncResp->res.jsonValue["SparePartNumber"] = *sparePartNumber;
404 }
405
406 asyncResp->res.jsonValue["Name"] = chassisId;
407 asyncResp->res.jsonValue["Id"] = chassisId;
408#ifdef BMCWEB_ALLOW_DEPRECATED_POWER_THERMAL
409 asyncResp->res.jsonValue["Thermal"]["@odata.id"] =
410 boost::urls::format("/redfish/v1/Chassis/{}/Thermal", chassisId);
411 // Power object
412 asyncResp->res.jsonValue["Power"]["@odata.id"] =
413 boost::urls::format("/redfish/v1/Chassis/{}/Power", chassisId);
414#endif
415#ifdef BMCWEB_NEW_POWERSUBSYSTEM_THERMALSUBSYSTEM
416 asyncResp->res.jsonValue["ThermalSubsystem"]["@odata.id"] =
417 boost::urls::format("/redfish/v1/Chassis/{}/ThermalSubsystem",
418 chassisId);
419 asyncResp->res.jsonValue["PowerSubsystem"]["@odata.id"] =
420 boost::urls::format("/redfish/v1/Chassis/{}/PowerSubsystem", chassisId);
421 asyncResp->res.jsonValue["EnvironmentMetrics"]["@odata.id"] =
422 boost::urls::format("/redfish/v1/Chassis/{}/EnvironmentMetrics",
423 chassisId);
424#endif
425 // SensorCollection
426 asyncResp->res.jsonValue["Sensors"]["@odata.id"] =
427 boost::urls::format("/redfish/v1/Chassis/{}/Sensors", chassisId);
428 asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
429
430 nlohmann::json::array_t computerSystems;
431 nlohmann::json::object_t system;
432 system["@odata.id"] = "/redfish/v1/Systems/system";
433 computerSystems.emplace_back(std::move(system));
434 asyncResp->res.jsonValue["Links"]["ComputerSystems"] =
435 std::move(computerSystems);
436
437 nlohmann::json::array_t managedBy;
438 nlohmann::json::object_t manager;
439 manager["@odata.id"] = "/redfish/v1/Managers/bmc";
440 managedBy.emplace_back(std::move(manager));
441 asyncResp->res.jsonValue["Links"]["ManagedBy"] = std::move(managedBy);
442 getChassisState(asyncResp);
443 getStorageLink(asyncResp, path);
444}
445
446inline void handleChassisGetSubTree(
447 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
448 const std::string& chassisId, const boost::system::error_code& ec,
449 const dbus::utility::MapperGetSubTreeResponse& subtree)
450{
451 if (ec)
452 {
453 BMCWEB_LOG_ERROR("DBUS response error {}", ec);
454 messages::internalError(asyncResp->res);
455 return;
456 }
457 // Iterate over all retrieved ObjectPaths.
458 for (const std::pair<
459 std::string,
460 std::vector<std::pair<std::string, std::vector<std::string>>>>&
461 object : subtree)
462 {
463 const std::string& path = object.first;
464 const std::vector<std::pair<std::string, std::vector<std::string>>>&
465 connectionNames = object.second;
466
467 sdbusplus::message::object_path objPath(path);
468 if (objPath.filename() != chassisId)
469 {
470 continue;
471 }
472
473 getChassisConnectivity(asyncResp, chassisId, path);
474
475 auto health = std::make_shared<HealthPopulate>(asyncResp);
476
477 if constexpr (bmcwebEnableHealthPopulate)
478 {
479 dbus::utility::getAssociationEndPoints(
480 path + "/all_sensors",
481 [health](const boost::system::error_code& ec2,
482 const dbus::utility::MapperEndPoints& resp) {
483 if (ec2)
484 {
485 return; // no sensors = no failures
486 }
487 health->inventory = resp;
488 });
489
490 health->populate();
491 }
492
493 if (connectionNames.empty())
494 {
495 BMCWEB_LOG_ERROR("Got 0 Connection names");
496 continue;
497 }
498
499 asyncResp->res.jsonValue["@odata.type"] = "#Chassis.v1_22_0.Chassis";
500 asyncResp->res.jsonValue["@odata.id"] =
501 boost::urls::format("/redfish/v1/Chassis/{}", chassisId);
502 asyncResp->res.jsonValue["Name"] = "Chassis Collection";
503 asyncResp->res.jsonValue["ChassisType"] = "RackMount";
504 asyncResp->res.jsonValue["Actions"]["#Chassis.Reset"]["target"] =
505 boost::urls::format("/redfish/v1/Chassis/{}/Actions/Chassis.Reset",
506 chassisId);
507 asyncResp->res
508 .jsonValue["Actions"]["#Chassis.Reset"]["@Redfish.ActionInfo"] =
509 boost::urls::format("/redfish/v1/Chassis/{}/ResetActionInfo",
510 chassisId);
Chau Ly7164bc62023-10-15 14:55:30 +0000511 dbus::utility::getAssociationEndPoints(
512 path + "/drive",
513 [asyncResp, chassisId](const boost::system::error_code& ec3,
514 const dbus::utility::MapperEndPoints& resp) {
515 if (ec3 || resp.empty())
516 {
517 return; // no drives = no failures
518 }
519
520 nlohmann::json reference;
521 reference["@odata.id"] =
522 boost::urls::format("/redfish/v1/Chassis/{}/Drives", chassisId);
523 asyncResp->res.jsonValue["Drives"] = std::move(reference);
524 });
525
526 const std::string& connectionName = connectionNames[0].first;
527
528 const std::vector<std::string>& interfaces2 = connectionNames[0].second;
529 const std::array<const char*, 2> hasIndicatorLed = {
530 "xyz.openbmc_project.Inventory.Item.Panel",
531 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
532
533 const std::string assetTagInterface =
534 "xyz.openbmc_project.Inventory.Decorator.AssetTag";
535 const std::string replaceableInterface =
536 "xyz.openbmc_project.Inventory.Decorator.Replaceable";
Carson Labradob4d593f2024-02-16 22:34:32 +0000537 const std::string revisionInterface =
538 "xyz.openbmc_project.Inventory.Decorator.Revision";
Chau Ly7164bc62023-10-15 14:55:30 +0000539 for (const auto& interface : interfaces2)
540 {
541 if (interface == assetTagInterface)
542 {
543 sdbusplus::asio::getProperty<std::string>(
544 *crow::connections::systemBus, connectionName, path,
545 assetTagInterface, "AssetTag",
546 [asyncResp, chassisId](const boost::system::error_code& ec2,
547 const std::string& property) {
548 if (ec2)
549 {
550 BMCWEB_LOG_ERROR("DBus response error for AssetTag: {}",
551 ec2);
552 messages::internalError(asyncResp->res);
553 return;
554 }
555 asyncResp->res.jsonValue["AssetTag"] = property;
556 });
557 }
558 else if (interface == replaceableInterface)
559 {
560 sdbusplus::asio::getProperty<bool>(
561 *crow::connections::systemBus, connectionName, path,
562 replaceableInterface, "HotPluggable",
563 [asyncResp, chassisId](const boost::system::error_code& ec2,
564 const bool property) {
565 if (ec2)
566 {
567 BMCWEB_LOG_ERROR(
568 "DBus response error for HotPluggable: {}", ec2);
569 messages::internalError(asyncResp->res);
570 return;
571 }
572 asyncResp->res.jsonValue["HotPluggable"] = property;
573 });
574 }
Carson Labradob4d593f2024-02-16 22:34:32 +0000575 else if (interface == revisionInterface)
576 {
577 sdbusplus::asio::getProperty<std::string>(
578 *crow::connections::systemBus, connectionName, path,
579 revisionInterface, "Version",
580 [asyncResp, chassisId](const boost::system::error_code& ec2,
581 const std::string& property) {
582 if (ec2)
583 {
584 BMCWEB_LOG_ERROR("DBus response error for Version: {}",
585 ec2);
586 messages::internalError(asyncResp->res);
587 return;
588 }
589 asyncResp->res.jsonValue["Version"] = property;
590 });
591 }
Chau Ly7164bc62023-10-15 14:55:30 +0000592 }
593
594 for (const char* interface : hasIndicatorLed)
595 {
596 if (std::ranges::find(interfaces2, interface) != interfaces2.end())
597 {
598 getIndicatorLedState(asyncResp);
599 getSystemLocationIndicatorActive(asyncResp);
600 break;
601 }
602 }
603
604 sdbusplus::asio::getAllProperties(
605 *crow::connections::systemBus, connectionName, path,
606 "xyz.openbmc_project.Inventory.Decorator.Asset",
607 [asyncResp, chassisId,
608 path](const boost::system::error_code&,
609 const dbus::utility::DBusPropertiesMap& propertiesList) {
610 handleDecoratorAssetProperties(asyncResp, chassisId, path,
611 propertiesList);
612 });
613
614 for (const auto& interface : interfaces2)
615 {
616 if (interface == "xyz.openbmc_project.Common.UUID")
617 {
618 getChassisUUID(asyncResp, connectionName, path);
619 }
620 else if (interface ==
621 "xyz.openbmc_project.Inventory.Decorator.LocationCode")
622 {
623 getChassisLocationCode(asyncResp, connectionName, path);
624 }
625 }
626
627 return;
628 }
629
630 // Couldn't find an object with that name. return an error
631 messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
632}
633
Nan Zhoucf7eba02022-07-21 23:53:20 +0000634inline void
635 handleChassisGet(App& app, const crow::Request& req,
636 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
637 const std::string& chassisId)
638{
639 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
640 {
641 return;
642 }
George Liue99073f2022-12-09 11:06:16 +0800643 constexpr std::array<std::string_view, 2> interfaces = {
Nan Zhoucf7eba02022-07-21 23:53:20 +0000644 "xyz.openbmc_project.Inventory.Item.Board",
645 "xyz.openbmc_project.Inventory.Item.Chassis"};
646
George Liue99073f2022-12-09 11:06:16 +0800647 dbus::utility::getSubTree(
648 "/xyz/openbmc_project/inventory", 0, interfaces,
Chau Ly7164bc62023-10-15 14:55:30 +0000649 std::bind_front(handleChassisGetSubTree, asyncResp, chassisId));
Nan Zhoucf7eba02022-07-21 23:53:20 +0000650
Chau Ly7164bc62023-10-15 14:55:30 +0000651 constexpr std::array<std::string_view, 1> interfaces2 = {
652 "xyz.openbmc_project.Chassis.Intrusion"};
Nan Zhoucf7eba02022-07-21 23:53:20 +0000653
Chau Ly7164bc62023-10-15 14:55:30 +0000654 dbus::utility::getSubTree(
655 "/xyz/openbmc_project", 0, interfaces2,
656 std::bind_front(handlePhysicalSecurityGetSubTree, asyncResp));
Nan Zhoucf7eba02022-07-21 23:53:20 +0000657}
658
659inline void
660 handleChassisPatch(App& app, const crow::Request& req,
661 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
662 const std::string& param)
663{
664 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
665 {
666 return;
667 }
668 std::optional<bool> locationIndicatorActive;
669 std::optional<std::string> indicatorLed;
670
671 if (param.empty())
672 {
673 return;
674 }
675
676 if (!json_util::readJsonPatch(
677 req, asyncResp->res, "LocationIndicatorActive",
678 locationIndicatorActive, "IndicatorLED", indicatorLed))
679 {
680 return;
681 }
682
683 // TODO (Gunnar): Remove IndicatorLED after enough time has passed
684 if (!locationIndicatorActive && !indicatorLed)
685 {
686 return; // delete this when we support more patch properties
687 }
688 if (indicatorLed)
689 {
690 asyncResp->res.addHeader(
691 boost::beast::http::field::warning,
692 "299 - \"IndicatorLED is deprecated. Use LocationIndicatorActive instead.\"");
693 }
694
George Liue99073f2022-12-09 11:06:16 +0800695 constexpr std::array<std::string_view, 2> interfaces = {
Nan Zhoucf7eba02022-07-21 23:53:20 +0000696 "xyz.openbmc_project.Inventory.Item.Board",
697 "xyz.openbmc_project.Inventory.Item.Chassis"};
698
699 const std::string& chassisId = param;
700
George Liue99073f2022-12-09 11:06:16 +0800701 dbus::utility::getSubTree(
702 "/xyz/openbmc_project/inventory", 0, interfaces,
Nan Zhoucf7eba02022-07-21 23:53:20 +0000703 [asyncResp, chassisId, locationIndicatorActive,
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800704 indicatorLed](const boost::system::error_code& ec,
Nan Zhoucf7eba02022-07-21 23:53:20 +0000705 const dbus::utility::MapperGetSubTreeResponse& subtree) {
706 if (ec)
707 {
Ed Tanous62598e32023-07-17 17:06:25 -0700708 BMCWEB_LOG_ERROR("DBUS response error {}", ec);
Nan Zhoucf7eba02022-07-21 23:53:20 +0000709 messages::internalError(asyncResp->res);
710 return;
711 }
712
713 // Iterate over all retrieved ObjectPaths.
714 for (const std::pair<
715 std::string,
716 std::vector<std::pair<std::string, std::vector<std::string>>>>&
717 object : subtree)
718 {
719 const std::string& path = object.first;
720 const std::vector<std::pair<std::string, std::vector<std::string>>>&
721 connectionNames = object.second;
722
723 sdbusplus::message::object_path objPath(path);
724 if (objPath.filename() != chassisId)
725 {
726 continue;
727 }
728
729 if (connectionNames.empty())
730 {
Ed Tanous62598e32023-07-17 17:06:25 -0700731 BMCWEB_LOG_ERROR("Got 0 Connection names");
Nan Zhoucf7eba02022-07-21 23:53:20 +0000732 continue;
733 }
734
735 const std::vector<std::string>& interfaces3 =
736 connectionNames[0].second;
737
738 const std::array<const char*, 2> hasIndicatorLed = {
739 "xyz.openbmc_project.Inventory.Item.Panel",
740 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
741 bool indicatorChassis = false;
742 for (const char* interface : hasIndicatorLed)
743 {
Ed Tanous3544d2a2023-08-06 18:12:20 -0700744 if (std::ranges::find(interfaces3, interface) !=
745 interfaces3.end())
Nan Zhoucf7eba02022-07-21 23:53:20 +0000746 {
747 indicatorChassis = true;
748 break;
749 }
750 }
751 if (locationIndicatorActive)
752 {
753 if (indicatorChassis)
754 {
George Liu59a17e42022-10-08 09:27:47 +0800755 setSystemLocationIndicatorActive(asyncResp,
756 *locationIndicatorActive);
Nan Zhoucf7eba02022-07-21 23:53:20 +0000757 }
758 else
759 {
760 messages::propertyUnknown(asyncResp->res,
761 "LocationIndicatorActive");
762 }
763 }
764 if (indicatorLed)
765 {
766 if (indicatorChassis)
767 {
768 setIndicatorLedState(asyncResp, *indicatorLed);
769 }
770 else
771 {
772 messages::propertyUnknown(asyncResp->res, "IndicatorLED");
773 }
774 }
775 return;
776 }
777
Jiaqing Zhaod8a5d5d2022-08-05 16:21:51 +0800778 messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
Patrick Williams5a39f772023-10-20 11:20:21 -0500779 });
Nan Zhoucf7eba02022-07-21 23:53:20 +0000780}
781
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100782/**
783 * Chassis override class for delivering Chassis Schema
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700784 * Functions triggers appropriate requests on DBus
Rapkiewicz, Pawele37f8452018-03-09 13:49:50 +0100785 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700786inline void requestRoutesChassis(App& app)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700787{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700788 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -0700789 .privileges(redfish::privileges::getChassis)
Ed Tanous002d39b2022-05-31 08:59:27 -0700790 .methods(boost::beast::http::verb::get)(
Nan Zhoucf7eba02022-07-21 23:53:20 +0000791 std::bind_front(handleChassisGet, std::ref(app)));
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700792
793 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -0700794 .privileges(redfish::privileges::patchChassis)
Ed Tanous002d39b2022-05-31 08:59:27 -0700795 .methods(boost::beast::http::verb::patch)(
Nan Zhoucf7eba02022-07-21 23:53:20 +0000796 std::bind_front(handleChassisPatch, std::ref(app)));
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700797}
P.K. Leedd99e042020-06-17 19:43:16 +0800798
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400799/**
800 * Handle error responses from d-bus for chassis power cycles
801 */
802inline void handleChassisPowerCycleError(const boost::system::error_code& ec,
803 const sdbusplus::message_t& eMsg,
804 crow::Response& res)
805{
806 if (eMsg.get_error() == nullptr)
807 {
Ed Tanous62598e32023-07-17 17:06:25 -0700808 BMCWEB_LOG_ERROR("D-Bus response error: {}", ec);
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400809 messages::internalError(res);
810 return;
811 }
812 std::string_view errorMessage = eMsg.get_error()->name;
813
814 // If operation failed due to BMC not being in Ready state, tell
815 // user to retry in a bit
816 if (errorMessage ==
817 std::string_view("xyz.openbmc_project.State.Chassis.Error.BMCNotReady"))
818 {
Ed Tanous62598e32023-07-17 17:06:25 -0700819 BMCWEB_LOG_DEBUG("BMC not ready, operation not allowed right now");
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400820 messages::serviceTemporarilyUnavailable(res, "10");
821 return;
822 }
823
Ed Tanous62598e32023-07-17 17:06:25 -0700824 BMCWEB_LOG_ERROR("Chassis Power Cycle fail {} sdbusplus:{}", ec,
825 errorMessage);
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400826 messages::internalError(res);
827}
828
zhanghch058d1b46d2021-04-01 11:18:24 +0800829inline void
830 doChassisPowerCycle(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
P.K. Leedd99e042020-06-17 19:43:16 +0800831{
George Liu7a1dbc42022-12-07 16:03:22 +0800832 constexpr std::array<std::string_view, 1> interfaces = {
Vijay Khemkac3b3c922020-09-22 23:00:12 -0700833 "xyz.openbmc_project.State.Chassis"};
834
835 // Use mapper to get subtree paths.
George Liu7a1dbc42022-12-07 16:03:22 +0800836 dbus::utility::getSubTreePaths(
837 "/", 0, interfaces,
Ed Tanousb9d36b42022-02-26 21:42:46 -0800838 [asyncResp](
George Liu7a1dbc42022-12-07 16:03:22 +0800839 const boost::system::error_code& ec,
Ed Tanousb9d36b42022-02-26 21:42:46 -0800840 const dbus::utility::MapperGetSubTreePathsResponse& chassisList) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700841 if (ec)
842 {
Ed Tanous62598e32023-07-17 17:06:25 -0700843 BMCWEB_LOG_ERROR("[mapper] Bad D-Bus request error: {}", ec);
Ed Tanous002d39b2022-05-31 08:59:27 -0700844 messages::internalError(asyncResp->res);
845 return;
846 }
847
848 const char* processName = "xyz.openbmc_project.State.Chassis";
849 const char* interfaceName = "xyz.openbmc_project.State.Chassis";
850 const char* destProperty = "RequestedPowerTransition";
851 const std::string propertyValue =
852 "xyz.openbmc_project.State.Chassis.Transition.PowerCycle";
853 std::string objectPath = "/xyz/openbmc_project/state/chassis_system0";
854
855 /* Look for system reset chassis path */
Ed Tanous3544d2a2023-08-06 18:12:20 -0700856 if ((std::ranges::find(chassisList, objectPath)) == chassisList.end())
Ed Tanous002d39b2022-05-31 08:59:27 -0700857 {
858 /* We prefer to reset the full chassis_system, but if it doesn't
859 * exist on some platforms, fall back to a host-only power reset
860 */
861 objectPath = "/xyz/openbmc_project/state/chassis0";
862 }
863
George Liu9ae226f2023-06-21 17:56:46 +0800864 sdbusplus::asio::setProperty(
865 *crow::connections::systemBus, processName, objectPath,
866 interfaceName, destProperty, propertyValue,
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400867 [asyncResp](const boost::system::error_code& ec2,
868 sdbusplus::message_t& sdbusErrMsg) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700869 // Use "Set" method to set the property value.
Ed Tanous8a592812022-06-04 09:06:59 -0700870 if (ec2)
P.K. Leedd99e042020-06-17 19:43:16 +0800871 {
Andrew Geisslerfc903b32023-05-31 14:15:42 -0400872 handleChassisPowerCycleError(ec2, sdbusErrMsg, asyncResp->res);
873
P.K. Leedd99e042020-06-17 19:43:16 +0800874 return;
875 }
876
Ed Tanous002d39b2022-05-31 08:59:27 -0700877 messages::success(asyncResp->res);
George Liu7a1dbc42022-12-07 16:03:22 +0800878 });
Patrick Williams5a39f772023-10-20 11:20:21 -0500879 });
P.K. Leedd99e042020-06-17 19:43:16 +0800880}
881
Nan Zhoucf7eba02022-07-21 23:53:20 +0000882inline void handleChassisResetActionInfoPost(
883 App& app, const crow::Request& req,
884 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
885 const std::string& /*chassisId*/)
886{
887 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
888 {
889 return;
890 }
Ed Tanous62598e32023-07-17 17:06:25 -0700891 BMCWEB_LOG_DEBUG("Post Chassis Reset.");
Nan Zhoucf7eba02022-07-21 23:53:20 +0000892
893 std::string resetType;
894
895 if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", resetType))
896 {
897 return;
898 }
899
900 if (resetType != "PowerCycle")
901 {
Ed Tanous62598e32023-07-17 17:06:25 -0700902 BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType);
Nan Zhoucf7eba02022-07-21 23:53:20 +0000903 messages::actionParameterNotSupported(asyncResp->res, resetType,
904 "ResetType");
905
906 return;
907 }
908 doChassisPowerCycle(asyncResp);
909}
910
P.K. Leedd99e042020-06-17 19:43:16 +0800911/**
912 * ChassisResetAction class supports the POST method for the Reset
913 * action.
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700914 * Function handles POST method request.
915 * Analyzes POST body before sending Reset request data to D-Bus.
P.K. Leedd99e042020-06-17 19:43:16 +0800916 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700917
918inline void requestRoutesChassisResetAction(App& app)
P.K. Leedd99e042020-06-17 19:43:16 +0800919{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700920 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Actions/Chassis.Reset/")
Ed Tanoused398212021-06-09 17:05:54 -0700921 .privileges(redfish::privileges::postChassis)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700922 .methods(boost::beast::http::verb::post)(
Nan Zhoucf7eba02022-07-21 23:53:20 +0000923 std::bind_front(handleChassisResetActionInfoPost, std::ref(app)));
924}
P.K. Leedd99e042020-06-17 19:43:16 +0800925
Nan Zhoucf7eba02022-07-21 23:53:20 +0000926inline void handleChassisResetActionInfoGet(
927 App& app, const crow::Request& req,
928 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
929 const std::string& chassisId)
930{
931 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
932 {
933 return;
934 }
935 asyncResp->res.jsonValue["@odata.type"] = "#ActionInfo.v1_1_2.ActionInfo";
Ed Tanousef4c65b2023-04-24 15:28:50 -0700936 asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
937 "/redfish/v1/Chassis/{}/ResetActionInfo", chassisId);
Nan Zhoucf7eba02022-07-21 23:53:20 +0000938 asyncResp->res.jsonValue["Name"] = "Reset Action Info";
P.K. Leedd99e042020-06-17 19:43:16 +0800939
Nan Zhoucf7eba02022-07-21 23:53:20 +0000940 asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
941 nlohmann::json::array_t parameters;
942 nlohmann::json::object_t parameter;
943 parameter["Name"] = "ResetType";
944 parameter["Required"] = true;
945 parameter["DataType"] = "String";
946 nlohmann::json::array_t allowed;
Patrick Williamsad539542023-05-12 10:10:08 -0500947 allowed.emplace_back("PowerCycle");
Nan Zhoucf7eba02022-07-21 23:53:20 +0000948 parameter["AllowableValues"] = std::move(allowed);
Patrick Williamsad539542023-05-12 10:10:08 -0500949 parameters.emplace_back(std::move(parameter));
P.K. Leedd99e042020-06-17 19:43:16 +0800950
Nan Zhoucf7eba02022-07-21 23:53:20 +0000951 asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700952}
AppaRao Puli1cb1a9e2020-07-17 23:38:57 +0530953
954/**
955 * ChassisResetActionInfo derived class for delivering Chassis
956 * ResetType AllowableValues using ResetInfo schema.
957 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700958inline void requestRoutesChassisResetActionInfo(App& app)
AppaRao Puli1cb1a9e2020-07-17 23:38:57 +0530959{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700960 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/ResetActionInfo/")
Ed Tanoused398212021-06-09 17:05:54 -0700961 .privileges(redfish::privileges::getActionInfo)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700962 .methods(boost::beast::http::verb::get)(
Nan Zhoucf7eba02022-07-21 23:53:20 +0000963 std::bind_front(handleChassisResetActionInfoGet, std::ref(app)));
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700964}
AppaRao Puli1cb1a9e2020-07-17 23:38:57 +0530965
Ed Tanous1abe55e2018-09-05 08:30:59 -0700966} // namespace redfish