xref: /openbmc/bmcweb/redfish-core/lib/chassis.hpp (revision e610b3168321eee167271e4532c37fe1ed9c6f56)
1 /*
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 
18 #include "bmcweb_config.h"
19 
20 #include "app.hpp"
21 #include "dbus_utility.hpp"
22 #include "health.hpp"
23 #include "led.hpp"
24 #include "query.hpp"
25 #include "redfish_util.hpp"
26 #include "registries/privilege_registry.hpp"
27 #include "utils/collection.hpp"
28 #include "utils/dbus_utils.hpp"
29 #include "utils/json_utils.hpp"
30 
31 #include <boost/system/error_code.hpp>
32 #include <boost/url/format.hpp>
33 #include <sdbusplus/asio/property.hpp>
34 #include <sdbusplus/message.hpp>
35 #include <sdbusplus/unpack_properties.hpp>
36 
37 #include <array>
38 #include <ranges>
39 #include <string_view>
40 
41 namespace redfish
42 {
43 
44 /**
45  * @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  */
57 inline 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",
63         [asyncResp](const boost::system::error_code& ec,
64                     const std::vector<std::string>& storageList) {
65         if (ec)
66         {
67             BMCWEB_LOG_DEBUG("getStorageLink got DBUS response error");
68             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);
89     });
90 }
91 
92 /**
93  * @brief Retrieves chassis state properties over dbus
94  *
95  * @param[in] asyncResp - Shared pointer for completing asynchronous calls.
96  *
97  * @return None.
98  */
99 inline void getChassisState(std::shared_ptr<bmcweb::AsyncResp> asyncResp)
100 {
101     // 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",
106         [asyncResp{std::move(asyncResp)}](const boost::system::error_code& ec,
107                                           const std::string& chassisState) {
108         if (ec)
109         {
110             if (ec == boost::system::errc::host_unreachable)
111             {
112                 // Service not available, no error, just don't return
113                 // chassis state info
114                 BMCWEB_LOG_DEBUG("Service not available {}", ec);
115                 return;
116             }
117             BMCWEB_LOG_DEBUG("DBUS response error {}", ec);
118             messages::internalError(asyncResp->res);
119             return;
120         }
121 
122         BMCWEB_LOG_DEBUG("Chassis state: {}", chassisState);
123         // Verify Chassis State
124         if (chassisState == "xyz.openbmc_project.State.Chassis.PowerState.On")
125         {
126             asyncResp->res.jsonValue["PowerState"] = "On";
127             asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
128         }
129         else if (chassisState ==
130                  "xyz.openbmc_project.State.Chassis.PowerState.Off")
131         {
132             asyncResp->res.jsonValue["PowerState"] = "Off";
133             asyncResp->res.jsonValue["Status"]["State"] = "StandbyOffline";
134         }
135     });
136 }
137 
138 /**
139  * Retrieves physical security properties over dbus
140  */
141 inline void handlePhysicalSecurityGetSubTree(
142     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
143     const boost::system::error_code& ec,
144     const dbus::utility::MapperGetSubTreeResponse& subtree)
145 {
146     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())
157         {
158             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 
181             return;
182         }
183     }
184 }
185 
186 inline 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 
199     constexpr std::array<std::string_view, 2> interfaces{
200         "xyz.openbmc_project.Inventory.Item.Board",
201         "xyz.openbmc_project.Inventory.Item.Chassis"};
202     collection_util::getCollectionMembers(
203         asyncResp, boost::urls::url("/redfish/v1/Chassis"), interfaces,
204         "/xyz/openbmc_project/inventory");
205 }
206 
207 inline 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         {
216             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
217             messages::internalError(asyncResp->res);
218         }
219         return;
220     }
221     if (upstreamChassisPaths.empty())
222     {
223         return;
224     }
225     if (upstreamChassisPaths.size() > 1)
226     {
227         BMCWEB_LOG_ERROR("{} is contained by multiple chassis", chassisId);
228         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     {
237         BMCWEB_LOG_WARNING("Malformed upstream Chassis path {} on {}",
238                            upstreamChassisPath.str, chassisId);
239         return;
240     }
241 
242     asyncResp->res.jsonValue["Links"]["ContainedBy"]["@odata.id"] =
243         boost::urls::format("/redfish/v1/Chassis/{}", upstreamChassis);
244 }
245 
246 inline 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         {
255             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
256             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         {
276             BMCWEB_LOG_WARNING("Malformed downstream Chassis path {} on {}",
277                                downstreamChassisPath.str, chassisId);
278             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 
288 inline void
289     getChassisConnectivity(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
290                            const std::string& chassisId,
291                            const std::string& chassisPath)
292 {
293     BMCWEB_LOG_DEBUG("Get chassis connectivity");
294 
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 
304 /**
305  * ChassisCollection derived class for delivering Chassis Collection Schema
306  *  Functions triggers appropriate requests on DBus
307  */
308 inline void requestRoutesChassisCollection(App& app)
309 {
310     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/")
311         .privileges(redfish::privileges::getChassisCollection)
312         .methods(boost::beast::http::verb::get)(
313             std::bind_front(handleChassisCollectionGet, std::ref(app)));
314 }
315 
316 inline void
317     getChassisLocationCode(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
318                            const std::string& connectionName,
319                            const std::string& path)
320 {
321     sdbusplus::asio::getProperty<std::string>(
322         *crow::connections::systemBus, connectionName, path,
323         "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
324         [asyncResp](const boost::system::error_code& ec,
325                     const std::string& property) {
326         if (ec)
327         {
328             BMCWEB_LOG_ERROR("DBUS response error for Location");
329             messages::internalError(asyncResp->res);
330             return;
331         }
332 
333         asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
334             property;
335     });
336 }
337 
338 inline void getChassisUUID(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
339                            const std::string& connectionName,
340                            const std::string& path)
341 {
342     sdbusplus::asio::getProperty<std::string>(
343         *crow::connections::systemBus, connectionName, path,
344         "xyz.openbmc_project.Common.UUID", "UUID",
345         [asyncResp](const boost::system::error_code& ec,
346                     const std::string& chassisUUID) {
347         if (ec)
348         {
349             BMCWEB_LOG_ERROR("DBUS response error for UUID");
350             messages::internalError(asyncResp->res);
351             return;
352         }
353         asyncResp->res.jsonValue["UUID"] = chassisUUID;
354     });
355 }
356 
357 inline 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 
446 inline 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);
511         asyncResp->res.jsonValue["PCIeDevices"]["@odata.id"] =
512             "/redfish/v1/Systems/system/PCIeDevices";
513 
514         dbus::utility::getAssociationEndPoints(
515             path + "/drive",
516             [asyncResp, chassisId](const boost::system::error_code& ec3,
517                                    const dbus::utility::MapperEndPoints& resp) {
518             if (ec3 || resp.empty())
519             {
520                 return; // no drives = no failures
521             }
522 
523             nlohmann::json reference;
524             reference["@odata.id"] =
525                 boost::urls::format("/redfish/v1/Chassis/{}/Drives", chassisId);
526             asyncResp->res.jsonValue["Drives"] = std::move(reference);
527         });
528 
529         const std::string& connectionName = connectionNames[0].first;
530 
531         const std::vector<std::string>& interfaces2 = connectionNames[0].second;
532         const std::array<const char*, 2> hasIndicatorLed = {
533             "xyz.openbmc_project.Inventory.Item.Panel",
534             "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
535 
536         const std::string assetTagInterface =
537             "xyz.openbmc_project.Inventory.Decorator.AssetTag";
538         const std::string replaceableInterface =
539             "xyz.openbmc_project.Inventory.Decorator.Replaceable";
540         const std::string revisionInterface =
541             "xyz.openbmc_project.Inventory.Decorator.Revision";
542         for (const auto& interface : interfaces2)
543         {
544             if (interface == assetTagInterface)
545             {
546                 sdbusplus::asio::getProperty<std::string>(
547                     *crow::connections::systemBus, connectionName, path,
548                     assetTagInterface, "AssetTag",
549                     [asyncResp, chassisId](const boost::system::error_code& ec2,
550                                            const std::string& property) {
551                     if (ec2)
552                     {
553                         BMCWEB_LOG_ERROR("DBus response error for AssetTag: {}",
554                                          ec2);
555                         messages::internalError(asyncResp->res);
556                         return;
557                     }
558                     asyncResp->res.jsonValue["AssetTag"] = property;
559                 });
560             }
561             else if (interface == replaceableInterface)
562             {
563                 sdbusplus::asio::getProperty<bool>(
564                     *crow::connections::systemBus, connectionName, path,
565                     replaceableInterface, "HotPluggable",
566                     [asyncResp, chassisId](const boost::system::error_code& ec2,
567                                            const bool property) {
568                     if (ec2)
569                     {
570                         BMCWEB_LOG_ERROR(
571                             "DBus response error for HotPluggable: {}", ec2);
572                         messages::internalError(asyncResp->res);
573                         return;
574                     }
575                     asyncResp->res.jsonValue["HotPluggable"] = property;
576                 });
577             }
578             else if (interface == revisionInterface)
579             {
580                 sdbusplus::asio::getProperty<std::string>(
581                     *crow::connections::systemBus, connectionName, path,
582                     revisionInterface, "Version",
583                     [asyncResp, chassisId](const boost::system::error_code& ec2,
584                                            const std::string& property) {
585                     if (ec2)
586                     {
587                         BMCWEB_LOG_ERROR("DBus response error for Version: {}",
588                                          ec2);
589                         messages::internalError(asyncResp->res);
590                         return;
591                     }
592                     asyncResp->res.jsonValue["Version"] = property;
593                 });
594             }
595         }
596 
597         for (const char* interface : hasIndicatorLed)
598         {
599             if (std::ranges::find(interfaces2, interface) != interfaces2.end())
600             {
601                 getIndicatorLedState(asyncResp);
602                 getSystemLocationIndicatorActive(asyncResp);
603                 break;
604             }
605         }
606 
607         sdbusplus::asio::getAllProperties(
608             *crow::connections::systemBus, connectionName, path,
609             "xyz.openbmc_project.Inventory.Decorator.Asset",
610             [asyncResp, chassisId,
611              path](const boost::system::error_code&,
612                    const dbus::utility::DBusPropertiesMap& propertiesList) {
613             handleDecoratorAssetProperties(asyncResp, chassisId, path,
614                                            propertiesList);
615         });
616 
617         for (const auto& interface : interfaces2)
618         {
619             if (interface == "xyz.openbmc_project.Common.UUID")
620             {
621                 getChassisUUID(asyncResp, connectionName, path);
622             }
623             else if (interface ==
624                      "xyz.openbmc_project.Inventory.Decorator.LocationCode")
625             {
626                 getChassisLocationCode(asyncResp, connectionName, path);
627             }
628         }
629 
630         return;
631     }
632 
633     // Couldn't find an object with that name.  return an error
634     messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
635 }
636 
637 inline void
638     handleChassisGet(App& app, const crow::Request& req,
639                      const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
640                      const std::string& chassisId)
641 {
642     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
643     {
644         return;
645     }
646     constexpr std::array<std::string_view, 2> interfaces = {
647         "xyz.openbmc_project.Inventory.Item.Board",
648         "xyz.openbmc_project.Inventory.Item.Chassis"};
649 
650     dbus::utility::getSubTree(
651         "/xyz/openbmc_project/inventory", 0, interfaces,
652         std::bind_front(handleChassisGetSubTree, asyncResp, chassisId));
653 
654     constexpr std::array<std::string_view, 1> interfaces2 = {
655         "xyz.openbmc_project.Chassis.Intrusion"};
656 
657     dbus::utility::getSubTree(
658         "/xyz/openbmc_project", 0, interfaces2,
659         std::bind_front(handlePhysicalSecurityGetSubTree, asyncResp));
660 }
661 
662 inline void
663     handleChassisPatch(App& app, const crow::Request& req,
664                        const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
665                        const std::string& param)
666 {
667     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
668     {
669         return;
670     }
671     std::optional<bool> locationIndicatorActive;
672     std::optional<std::string> indicatorLed;
673 
674     if (param.empty())
675     {
676         return;
677     }
678 
679     if (!json_util::readJsonPatch(
680             req, asyncResp->res, "LocationIndicatorActive",
681             locationIndicatorActive, "IndicatorLED", indicatorLed))
682     {
683         return;
684     }
685 
686     // TODO (Gunnar): Remove IndicatorLED after enough time has passed
687     if (!locationIndicatorActive && !indicatorLed)
688     {
689         return; // delete this when we support more patch properties
690     }
691     if (indicatorLed)
692     {
693         asyncResp->res.addHeader(
694             boost::beast::http::field::warning,
695             "299 - \"IndicatorLED is deprecated. Use LocationIndicatorActive instead.\"");
696     }
697 
698     constexpr std::array<std::string_view, 2> interfaces = {
699         "xyz.openbmc_project.Inventory.Item.Board",
700         "xyz.openbmc_project.Inventory.Item.Chassis"};
701 
702     const std::string& chassisId = param;
703 
704     dbus::utility::getSubTree(
705         "/xyz/openbmc_project/inventory", 0, interfaces,
706         [asyncResp, chassisId, locationIndicatorActive,
707          indicatorLed](const boost::system::error_code& ec,
708                        const dbus::utility::MapperGetSubTreeResponse& subtree) {
709         if (ec)
710         {
711             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
712             messages::internalError(asyncResp->res);
713             return;
714         }
715 
716         // Iterate over all retrieved ObjectPaths.
717         for (const std::pair<
718                  std::string,
719                  std::vector<std::pair<std::string, std::vector<std::string>>>>&
720                  object : subtree)
721         {
722             const std::string& path = object.first;
723             const std::vector<std::pair<std::string, std::vector<std::string>>>&
724                 connectionNames = object.second;
725 
726             sdbusplus::message::object_path objPath(path);
727             if (objPath.filename() != chassisId)
728             {
729                 continue;
730             }
731 
732             if (connectionNames.empty())
733             {
734                 BMCWEB_LOG_ERROR("Got 0 Connection names");
735                 continue;
736             }
737 
738             const std::vector<std::string>& interfaces3 =
739                 connectionNames[0].second;
740 
741             const std::array<const char*, 2> hasIndicatorLed = {
742                 "xyz.openbmc_project.Inventory.Item.Panel",
743                 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
744             bool indicatorChassis = false;
745             for (const char* interface : hasIndicatorLed)
746             {
747                 if (std::ranges::find(interfaces3, interface) !=
748                     interfaces3.end())
749                 {
750                     indicatorChassis = true;
751                     break;
752                 }
753             }
754             if (locationIndicatorActive)
755             {
756                 if (indicatorChassis)
757                 {
758                     setSystemLocationIndicatorActive(asyncResp,
759                                                      *locationIndicatorActive);
760                 }
761                 else
762                 {
763                     messages::propertyUnknown(asyncResp->res,
764                                               "LocationIndicatorActive");
765                 }
766             }
767             if (indicatorLed)
768             {
769                 if (indicatorChassis)
770                 {
771                     setIndicatorLedState(asyncResp, *indicatorLed);
772                 }
773                 else
774                 {
775                     messages::propertyUnknown(asyncResp->res, "IndicatorLED");
776                 }
777             }
778             return;
779         }
780 
781         messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
782     });
783 }
784 
785 /**
786  * Chassis override class for delivering Chassis Schema
787  * Functions triggers appropriate requests on DBus
788  */
789 inline void requestRoutesChassis(App& app)
790 {
791     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
792         .privileges(redfish::privileges::getChassis)
793         .methods(boost::beast::http::verb::get)(
794             std::bind_front(handleChassisGet, std::ref(app)));
795 
796     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
797         .privileges(redfish::privileges::patchChassis)
798         .methods(boost::beast::http::verb::patch)(
799             std::bind_front(handleChassisPatch, std::ref(app)));
800 }
801 
802 /**
803  * Handle error responses from d-bus for chassis power cycles
804  */
805 inline void handleChassisPowerCycleError(const boost::system::error_code& ec,
806                                          const sdbusplus::message_t& eMsg,
807                                          crow::Response& res)
808 {
809     if (eMsg.get_error() == nullptr)
810     {
811         BMCWEB_LOG_ERROR("D-Bus response error: {}", ec);
812         messages::internalError(res);
813         return;
814     }
815     std::string_view errorMessage = eMsg.get_error()->name;
816 
817     // If operation failed due to BMC not being in Ready state, tell
818     // user to retry in a bit
819     if (errorMessage ==
820         std::string_view("xyz.openbmc_project.State.Chassis.Error.BMCNotReady"))
821     {
822         BMCWEB_LOG_DEBUG("BMC not ready, operation not allowed right now");
823         messages::serviceTemporarilyUnavailable(res, "10");
824         return;
825     }
826 
827     BMCWEB_LOG_ERROR("Chassis Power Cycle fail {} sdbusplus:{}", ec,
828                      errorMessage);
829     messages::internalError(res);
830 }
831 
832 inline void
833     doChassisPowerCycle(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
834 {
835     constexpr std::array<std::string_view, 1> interfaces = {
836         "xyz.openbmc_project.State.Chassis"};
837 
838     // Use mapper to get subtree paths.
839     dbus::utility::getSubTreePaths(
840         "/", 0, interfaces,
841         [asyncResp](
842             const boost::system::error_code& ec,
843             const dbus::utility::MapperGetSubTreePathsResponse& chassisList) {
844         if (ec)
845         {
846             BMCWEB_LOG_ERROR("[mapper] Bad D-Bus request error: {}", ec);
847             messages::internalError(asyncResp->res);
848             return;
849         }
850 
851         const char* processName = "xyz.openbmc_project.State.Chassis";
852         const char* interfaceName = "xyz.openbmc_project.State.Chassis";
853         const char* destProperty = "RequestedPowerTransition";
854         const std::string propertyValue =
855             "xyz.openbmc_project.State.Chassis.Transition.PowerCycle";
856         std::string objectPath = "/xyz/openbmc_project/state/chassis_system0";
857 
858         /* Look for system reset chassis path */
859         if ((std::ranges::find(chassisList, objectPath)) == chassisList.end())
860         {
861             /* We prefer to reset the full chassis_system, but if it doesn't
862              * exist on some platforms, fall back to a host-only power reset
863              */
864             objectPath = "/xyz/openbmc_project/state/chassis0";
865         }
866 
867         sdbusplus::asio::setProperty(
868             *crow::connections::systemBus, processName, objectPath,
869             interfaceName, destProperty, propertyValue,
870             [asyncResp](const boost::system::error_code& ec2,
871                         sdbusplus::message_t& sdbusErrMsg) {
872             // Use "Set" method to set the property value.
873             if (ec2)
874             {
875                 handleChassisPowerCycleError(ec2, sdbusErrMsg, asyncResp->res);
876 
877                 return;
878             }
879 
880             messages::success(asyncResp->res);
881         });
882     });
883 }
884 
885 inline void handleChassisResetActionInfoPost(
886     App& app, const crow::Request& req,
887     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
888     const std::string& /*chassisId*/)
889 {
890     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
891     {
892         return;
893     }
894     BMCWEB_LOG_DEBUG("Post Chassis Reset.");
895 
896     std::string resetType;
897 
898     if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", resetType))
899     {
900         return;
901     }
902 
903     if (resetType != "PowerCycle")
904     {
905         BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType);
906         messages::actionParameterNotSupported(asyncResp->res, resetType,
907                                               "ResetType");
908 
909         return;
910     }
911     doChassisPowerCycle(asyncResp);
912 }
913 
914 /**
915  * ChassisResetAction class supports the POST method for the Reset
916  * action.
917  * Function handles POST method request.
918  * Analyzes POST body before sending Reset request data to D-Bus.
919  */
920 
921 inline void requestRoutesChassisResetAction(App& app)
922 {
923     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Actions/Chassis.Reset/")
924         .privileges(redfish::privileges::postChassis)
925         .methods(boost::beast::http::verb::post)(
926             std::bind_front(handleChassisResetActionInfoPost, std::ref(app)));
927 }
928 
929 inline void handleChassisResetActionInfoGet(
930     App& app, const crow::Request& req,
931     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
932     const std::string& chassisId)
933 {
934     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
935     {
936         return;
937     }
938     asyncResp->res.jsonValue["@odata.type"] = "#ActionInfo.v1_1_2.ActionInfo";
939     asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
940         "/redfish/v1/Chassis/{}/ResetActionInfo", chassisId);
941     asyncResp->res.jsonValue["Name"] = "Reset Action Info";
942 
943     asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
944     nlohmann::json::array_t parameters;
945     nlohmann::json::object_t parameter;
946     parameter["Name"] = "ResetType";
947     parameter["Required"] = true;
948     parameter["DataType"] = "String";
949     nlohmann::json::array_t allowed;
950     allowed.emplace_back("PowerCycle");
951     parameter["AllowableValues"] = std::move(allowed);
952     parameters.emplace_back(std::move(parameter));
953 
954     asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
955 }
956 
957 /**
958  * ChassisResetActionInfo derived class for delivering Chassis
959  * ResetType AllowableValues using ResetInfo schema.
960  */
961 inline void requestRoutesChassisResetActionInfo(App& app)
962 {
963     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/ResetActionInfo/")
964         .privileges(redfish::privileges::getActionInfo)
965         .methods(boost::beast::http::verb::get)(
966             std::bind_front(handleChassisResetActionInfoGet, std::ref(app)));
967 }
968 
969 } // namespace redfish
970